sys_malloc.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. #include "sys_malloc.h"
  2. //////////////////////////////////////////////////////////////////////////////////
  3. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  4. //ALIENTEK STM32H7开发板
  5. //内存管理 驱动代码
  6. //正点原子@ALIENTEK
  7. //技术论坛:www.openedv.com
  8. //创建日期:2018/7/30
  9. //版本:V1.0
  10. //版权所有,盗版必究。
  11. //Copyright(C) 广州市星翼电子科技有限公司 2014-2024
  12. //All rights reserved
  13. //********************************************************************************
  14. //修改说明
  15. //无
  16. //////////////////////////////////////////////////////////////////////////////////
  17. //内存池(64字节对齐)
  18. __align(64) u8 mem1base[MEM1_MAX_SIZE]; //内部SRAM内存池
  19. //内存管理表
  20. u32 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]; //内部SRAM内存池MAP
  21. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  22. __align(64) u8 mem2base[MEM2_MAX_SIZE] __attribute__((at(0XC01F4000))); //外部SDRAM内存池,前面2M给LTDC用了(1280*800*2)
  23. u32 mem2mapbase[MEM2_ALLOC_TABLE_SIZE] __attribute__((at(0XC01F4000+MEM2_MAX_SIZE))); //外部SDRAM内存池MAP
  24. #endif
  25. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  26. __align(64) u8 mem3base[MEM3_MAX_SIZE] __attribute__((at(0x30000000))); //内部SRAM1+SRAM2内存池
  27. u32 mem3mapbase[MEM3_ALLOC_TABLE_SIZE] __attribute__((at(0x30000000+MEM3_MAX_SIZE))); //内部SRAM1+SRAM2内存池MAP
  28. #endif
  29. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  30. __align(64) u8 mem4base[MEM4_MAX_SIZE] __attribute__((at(0x38000000))); //内部SRAM4内存池
  31. u32 mem4mapbase[MEM4_ALLOC_TABLE_SIZE] __attribute__((at(0x38000000+MEM4_MAX_SIZE))); //内部SRAM4内存池MAP
  32. #endif
  33. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  34. __align(64) u8 mem5base[MEM5_MAX_SIZE] __attribute__((at(0x20000000))); //内部DTCM内存池
  35. u32 mem5mapbase[MEM5_ALLOC_TABLE_SIZE] __attribute__((at(0x20000000+MEM5_MAX_SIZE))); //内部DTCM内存池MAP
  36. #endif
  37. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  38. __align(64) u8 mem6base[MEM6_MAX_SIZE] __attribute__((at(0x00000000))); //内部ITCM内存池
  39. u32 mem6mapbase[MEM6_ALLOC_TABLE_SIZE] __attribute__((at(0x00000000+MEM6_MAX_SIZE))); //内部ITCM内存池MAP
  40. #endif
  41. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  42. #endif
  43. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  44. #endif
  45. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  46. #endif
  47. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  48. #endif
  49. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  50. #endif
  51. //内存管理参数
  52. const u32 memtblsize[SRAMBANK]={
  53. MEM1_ALLOC_TABLE_SIZE,
  54. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  55. MEM2_ALLOC_TABLE_SIZE,
  56. #endif
  57. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  58. MEM3_ALLOC_TABLE_SIZE,
  59. #endif
  60. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  61. MEM4_ALLOC_TABLE_SIZE,
  62. #endif
  63. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  64. MEM5_ALLOC_TABLE_SIZE,
  65. #endif
  66. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  67. MEM6_ALLOC_TABLE_SIZE
  68. #endif
  69. }; //内存表大小
  70. const u32 memblksize[SRAMBANK]={
  71. MEM1_BLOCK_SIZE,
  72. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  73. MEM2_BLOCK_SIZE,
  74. #endif
  75. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  76. MEM3_BLOCK_SIZE,
  77. #endif
  78. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  79. MEM4_BLOCK_SIZE,
  80. #endif
  81. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  82. MEM5_BLOCK_SIZE,
  83. #endif
  84. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  85. MEM6_BLOCK_SIZE
  86. #endif
  87. }; //内存分块大小
  88. const u32 memsize[SRAMBANK] ={
  89. MEM1_MAX_SIZE,
  90. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  91. MEM2_MAX_SIZE,
  92. #endif
  93. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  94. MEM3_MAX_SIZE,
  95. #endif
  96. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  97. MEM4_MAX_SIZE,
  98. #endif
  99. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  100. MEM5_MAX_SIZE,
  101. #endif
  102. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  103. MEM6_MAX_SIZE
  104. #endif
  105. }; //内存总大小
  106. //内存管理控制器
  107. struct _m_mallco_dev mallco_dev=
  108. {
  109. .init = my_mem_init, //内存初始化
  110. .perused = my_mem_perused, //内存使用率
  111. .membase[SRAMIN] = mem1base, //内存池
  112. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  113. .membase[SRAMEX] = mem2base,
  114. #endif
  115. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  116. mem3base,
  117. #endif
  118. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  119. mem4base,
  120. #endif
  121. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  122. mem5base,
  123. #endif
  124. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  125. mem6base,
  126. #endif
  127. .memmap[SRAMIN] = mem1mapbase,//内存管理状态表
  128. #ifdef SRAMEX //1 //外部内存池(SDRAM),SDRAM共32MB
  129. mem2mapbase,
  130. #endif
  131. #ifdef SRAM12 //2 //SRAM1/2/3内存池,SRAM1+SRAM2,共256KB
  132. mem3mapbase,
  133. #endif
  134. #ifdef SRAM4 //3 //SRAM4内存池,SRAM4共64KB
  135. mem4mapbase,
  136. #endif
  137. #ifdef SRAMDTCM //4 //DTCM内存池,DTCM共128KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  138. mem5mapbase,
  139. #endif
  140. #ifdef SRAMITCM //5 //ITCM内存池,DTCM共64 KB,此部分内存仅CPU和MDMA(通过AHBS)可以访问!!!!
  141. mem6mapbase
  142. #endif
  143. .memrdy= {0,0,0,0,0,0,} //内存管理未就绪
  144. };
  145. //复制内存
  146. //*des:目的地址
  147. //*src:源地址
  148. //n:需要复制的内存长度(字节为单位)
  149. void mymemcpy(void *des,void *src,u32 n)
  150. {
  151. u8 *xdes=des;
  152. u8 *xsrc=src;
  153. while(n--)*xdes++=*xsrc++;
  154. }
  155. //设置内存
  156. //*s:内存首地址
  157. //c :要设置的值
  158. //count:需要设置的内存大小(字节为单位)
  159. void mymemset(void *s,u8 c,u32 count)
  160. {
  161. u8 *xs = s;
  162. while(count--)*xs++=c;
  163. }
  164. //内存管理初始化
  165. //memx:所属内存块
  166. void my_mem_init(u8 memx)
  167. {
  168. mymemset(mallco_dev.memmap[memx],0,memtblsize[memx]*4); //内存状态表数据清零
  169. mallco_dev.memrdy[memx]=1; //内存管理初始化OK
  170. }
  171. //获取内存使用率
  172. //memx:所属内存块
  173. //返回值:使用率(扩大了10倍,0~1000,代表0.0%~100.0%)
  174. u16 my_mem_perused(u8 memx)
  175. {
  176. u32 used=0;
  177. u32 i;
  178. for(i=0;i<memtblsize[memx];i++)
  179. {
  180. if(mallco_dev.memmap[memx][i])used++;
  181. }
  182. return (used*1000)/(memtblsize[memx]);
  183. }
  184. //内存分配(内部调用)
  185. //memx:所属内存块
  186. //size:要分配的内存大小(字节)
  187. //返回值:0XFFFFFFFF,代表错误;其他,内存偏移地址
  188. u32 my_mem_malloc(u8 memx,u32 size)
  189. {
  190. signed long offset=0;
  191. u32 nmemb; //需要的内存块数
  192. u32 cmemb=0;//连续空内存块数
  193. u32 i;
  194. if(!mallco_dev.memrdy[memx])mallco_dev.init(memx);//未初始化,先执行初始化
  195. if(size==0)return 0XFFFFFFFF;//不需要分配
  196. nmemb=size/memblksize[memx]; //获取需要分配的连续内存块数
  197. if(size%memblksize[memx])nmemb++;
  198. for(offset=memtblsize[memx]-1;offset>=0;offset--)//搜索整个内存控制区
  199. {
  200. if(!mallco_dev.memmap[memx][offset])cmemb++;//连续空内存块数增加
  201. else cmemb=0; //连续内存块清零
  202. if(cmemb==nmemb) //找到了连续nmemb个空内存块
  203. {
  204. for(i=0;i<nmemb;i++) //标注内存块非空
  205. {
  206. mallco_dev.memmap[memx][offset+i]=nmemb;
  207. }
  208. return (offset*memblksize[memx]);//返回偏移地址
  209. }
  210. }
  211. return 0XFFFFFFFF;//未找到符合分配条件的内存块
  212. }
  213. //释放内存(内部调用)
  214. //memx:所属内存块
  215. //offset:内存地址偏移
  216. //返回值:0,释放成功;1,释放失败;
  217. u8 my_mem_free(u8 memx,u32 offset)
  218. {
  219. int i;
  220. if(!mallco_dev.memrdy[memx])//未初始化,先执行初始化
  221. {
  222. mallco_dev.init(memx);
  223. return 1;//未初始化
  224. }
  225. if(offset<memsize[memx])//偏移在内存池内.
  226. {
  227. int index=offset/memblksize[memx]; //偏移所在内存块号码
  228. int nmemb=mallco_dev.memmap[memx][index]; //内存块数量
  229. for(i=0;i<nmemb;i++) //内存块清零
  230. {
  231. mallco_dev.memmap[memx][index+i]=0;
  232. }
  233. return 0;
  234. }else return 2;//偏移超区了.
  235. }
  236. //释放内存(外部调用)
  237. //memx:所属内存块
  238. //ptr:内存首地址
  239. void myfree(u8 memx,void *ptr)
  240. {
  241. u32 offset;
  242. if(ptr==NULL)return;//地址为0.
  243. offset=(u32)ptr-(u32)mallco_dev.membase[memx];
  244. my_mem_free(memx,offset); //释放内存
  245. }
  246. //分配内存(外部调用)
  247. //memx:所属内存块
  248. //size:内存大小(字节)
  249. //返回值:分配到的内存首地址.
  250. void *mymalloc(u8 memx,u32 size)
  251. {
  252. u32 offset;
  253. offset=my_mem_malloc(memx,size);
  254. if(offset==0XFFFFFFFF)return NULL;
  255. else return (void*)((u32)mallco_dev.membase[memx]+offset);
  256. }
  257. //重新分配内存(外部调用)
  258. //memx:所属内存块
  259. //*ptr:旧内存首地址
  260. //size:要分配的内存大小(字节)
  261. //返回值:新分配到的内存首地址.
  262. void *myrealloc(u8 memx,void *ptr,u32 size)
  263. {
  264. u32 offset;
  265. offset=my_mem_malloc(memx,size);
  266. if(offset==0XFFFFFFFF)return NULL;
  267. else
  268. {
  269. mymemcpy((void*)((u32)mallco_dev.membase[memx]+offset),ptr,size); //拷贝旧内存内容到新内存
  270. myfree(memx,ptr); //释放旧内存
  271. return (void*)((u32)mallco_dev.membase[memx]+offset); //返回新内存首地址
  272. }
  273. }