malloc.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. #include "malloc.h"
  2. //////////////////////////////////////////////////////////////////////////////////
  3. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  4. //ALIENTEK STM32F407开发板
  5. //内存管理 驱动代码
  6. //正点原子@ALIENTEK
  7. //技术论坛:www.openedv.com
  8. //创建日期:2014/5/15
  9. //版本:V1.0
  10. //版权所有,盗版必究。
  11. //Copyright(C) 广州市星翼电子科技有限公司 2014-2024
  12. //All rights reserved
  13. //////////////////////////////////////////////////////////////////////////////////
  14. //内存池(4字节对齐)
  15. __align(4) u8 mem1base[MEM1_MAX_SIZE];
  16. __align(4) u8 mem2base[MEM2_MAX_SIZE] __attribute__((at(0x68000000))); //外部SRAM内存池
  17. __align(4) u8 mem3base[MEM3_MAX_SIZE] __attribute__((at(0x10000000))); //内部CMM内存池
  18. //内存管理表
  19. u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]; //内部SRAM内存池MAP
  20. u16 mem2mapbase[MEM2_ALLOC_TABLE_SIZE] __attribute__((at(0X68000000+MEM2_MAX_SIZE))); //外部SRAM内存池MAP
  21. u16 mem3mapbase[MEM3_ALLOC_TABLE_SIZE] __attribute__((at(0X10000000+MEM3_MAX_SIZE))); //内部CCM内存池MAP
  22. //内存管理参数
  23. const u32 memtblsize[SRAMBANK]={MEM1_ALLOC_TABLE_SIZE,MEM2_ALLOC_TABLE_SIZE,MEM3_ALLOC_TABLE_SIZE}; //内存表大小
  24. const u32 memblksize[SRAMBANK]={MEM1_BLOCK_SIZE,MEM2_BLOCK_SIZE,MEM3_BLOCK_SIZE}; //内存分块大小
  25. const u32 memsize[SRAMBANK]={MEM1_MAX_SIZE,MEM2_MAX_SIZE,MEM3_MAX_SIZE}; //内存总大小
  26. //内存管理控制器
  27. struct _m_mallco_dev mallco_dev=
  28. {
  29. mem_init, //内存初始化
  30. mem_perused, //内存使用率
  31. mem1base,mem2base,mem3base, //内存池
  32. mem1mapbase,mem2mapbase,mem3mapbase,//内存管理状态表
  33. 0,0,0, //内存管理未就绪
  34. };
  35. //复制内存
  36. //*des:目的地址
  37. //*src:源地址
  38. //n:需要复制的内存长度(字节为单位)
  39. void mymemcpy(void *des,void *src,u32 n)
  40. {
  41. u8 *xdes = des;
  42. u8 *xsrc = src;
  43. while(n--) *xdes++ = *xsrc++;
  44. }
  45. //设置内存
  46. //*s:内存首地址
  47. //c :要设置的值
  48. //count:需要设置的内存大小(字节为单位)
  49. void mymemset(void*s,u8 c,u32 count)
  50. {
  51. u8 *xs = s;
  52. while(count--) *xs++=c;
  53. }
  54. //内存管理初始化
  55. //memx:所属内存块
  56. void mem_init(u8 memx)
  57. {
  58. mymemset(mallco_dev.memmap[memx],0,memtblsize[memx]*2); //内存状态表清零
  59. mymemset(mallco_dev.membase[memx], 0,memsize[memx]); //内存池所有数据清零
  60. mallco_dev.memrdy[memx]=1; //内存管理初始化OK
  61. }
  62. //获取内存使用率
  63. //memx:所属内存块
  64. //返回值:使用率(0~100)
  65. u8 mem_perused(u8 memx)
  66. {
  67. u32 used=0;
  68. u32 i;
  69. for(i=0;i<memtblsize[memx];i++)
  70. {
  71. if(mallco_dev.memmap[memx][i])used++;
  72. }
  73. return (used*100)/(memtblsize[memx]);
  74. }
  75. //内存分配(内部调用)
  76. //memx:所属内存块
  77. //size:要分配的内存大小(字节)
  78. //返回值:0XFFFFFFFF,代表错误;其他,内存偏移地址
  79. u32 mem_malloc(u8 memx,u32 size)
  80. {
  81. signed long offset=0;
  82. u16 nmemb; //需要的内存块数
  83. u16 cmemb=0;//连续空内存块数
  84. u32 i;
  85. if(!mallco_dev.memrdy[memx])mallco_dev.init(memx);//未初始化,先执行初始化
  86. if(size==0)return 0XFFFFFFFF;//不需要分配
  87. nmemb=size/memblksize[memx]; //获取需要分配的连续内存块数
  88. if(size%memblksize[memx])nmemb++;
  89. for(offset=memtblsize[memx]-1;offset>=0;offset--)//搜索整个内存控制区
  90. {
  91. if(!mallco_dev.memmap[memx][offset])cmemb++;//连续空内存块数增加
  92. else cmemb=0; //连续内存块清零
  93. if(cmemb==nmemb) //找到了连续nmemb个空内存块
  94. {
  95. for(i=0;i<nmemb;i++) //标注内存块非空
  96. {
  97. mallco_dev.memmap[memx][offset+i]=nmemb;
  98. }
  99. return (offset*memblksize[memx]);//返回偏移地址
  100. }
  101. }
  102. return 0XFFFFFFFF;//未找到符合分配条件的内存块
  103. }
  104. //释放内存(内部调用)
  105. //memx:所属内存块
  106. //offset:内存地址偏移
  107. //返回值:0,释放成功;1,释放失败;
  108. u8 mem_free(u8 memx,u32 offset)
  109. {
  110. int i;
  111. if(!mallco_dev.memrdy[memx])//未初始化,先执行初始化
  112. {
  113. mallco_dev.init(memx);
  114. return 1;//未初始化
  115. }
  116. if(offset<memsize[memx])//偏移在内存池内.
  117. {
  118. int index=offset/memblksize[memx]; //偏移所在内存块号码
  119. int nmemb=mallco_dev.memmap[memx][index]; //内存块数量
  120. for(i=0;i<nmemb;i++) //内存块清零
  121. {
  122. mallco_dev.memmap[memx][index+i]=0;
  123. }
  124. return 0;
  125. }else return 2;//偏移超区了.
  126. }
  127. //释放内存(外部调用)
  128. //memx:所属内存块
  129. //ptr:内存首地址
  130. void myfree(u8 memx,void *ptr)
  131. {
  132. u32 offset;
  133. if(ptr==NULL)return;//地址为0.
  134. offset=(u32)ptr-(u32)mallco_dev.membase[memx];
  135. mem_free(memx,offset);//释放内存
  136. }
  137. //分配内存(外部调用)
  138. //memx:所属内存块
  139. //size:内存大小(字节)
  140. //返回值:分配到的内存首地址.
  141. void *mymalloc(u8 memx,u32 size)
  142. {
  143. u32 offset;
  144. offset=mem_malloc(memx,size);
  145. if(offset==0XFFFFFFFF)return NULL;
  146. else return (void*)((u32)mallco_dev.membase[memx]+offset);
  147. }
  148. //重新分配内存(外部调用)
  149. //memx:所属内存块
  150. //*ptr:旧内存首地址
  151. //size:要分配的内存大小(字节)
  152. //返回值:新分配到的内存首地址.
  153. void *myrealloc(u8 memx,void *ptr,u32 size)
  154. {
  155. u32 offset;
  156. offset=mem_malloc(memx,size);
  157. if(offset==0XFFFFFFFF)return NULL;
  158. else
  159. {
  160. mymemcpy((void*)((u32)mallco_dev.membase[memx]+offset),ptr,size); //拷贝旧内存内容到新内存
  161. myfree(memx,ptr); //释放旧内存
  162. return (void*)((u32)mallco_dev.membase[memx]+offset); //返回新内存首地址
  163. }
  164. }