You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

168 line
5.2 KiB

  1. #include "malloc.h"
  2. //////////////////////////////////////////////////////////////////////////////////
  3. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  4. //ALIENTEK STM32F7开发板
  5. //内存管理 驱动代码
  6. //正点原子@ALIENTEK
  7. //技术论坛:www.openedv.com
  8. //创建日期:2015/12/12
  9. //版本:V1.0
  10. //版权所有,盗版必究。
  11. //Copyright(C) 广州市星翼电子科技有限公司 2014-2024
  12. //All rights reserved
  13. //////////////////////////////////////////////////////////////////////////////////
  14. //内存池(32字节对齐)
  15. _Pragma("data_alignment=32") u8 mem1base[MEM1_MAX_SIZE]; //内部SRAM内存池
  16. _Pragma("data_alignment=32") u8 mem2base[MEM2_MAX_SIZE] @(0XC01F4000); //外部SDRAM内存池,前面2M给LTDC用了(1280*800*2)
  17. _Pragma("data_alignment=32") u8 mem3base[MEM3_MAX_SIZE] @(0X20000000); //内部DTCM内存池
  18. //内存管理表
  19. u32 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]; //内部SRAM内存池MAP
  20. u32 mem2mapbase[MEM2_ALLOC_TABLE_SIZE] @(0XC01F4000+MEM2_MAX_SIZE); //外部SRAM内存池MAP
  21. u32 mem3mapbase[MEM3_ALLOC_TABLE_SIZE] @(0X20000000+MEM3_MAX_SIZE); //内部DTCM内存池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. my_mem_init, //内存初始化
  30. my_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 my_mem_init(u8 memx)
  57. {
  58. mymemset(mallco_dev.memmap[memx],0,memtblsize[memx]*4); //内存状态表数据清零
  59. mallco_dev.memrdy[memx]=1; //内存管理初始化OK
  60. }
  61. //获取内存使用率
  62. //memx:所属内存块
  63. //返回值:使用率(扩大了10倍,0~1000,代表0.0%~100.0%)
  64. u16 my_mem_perused(u8 memx)
  65. {
  66. u32 used=0;
  67. u32 i;
  68. for(i=0;i<memtblsize[memx];i++)
  69. {
  70. if(mallco_dev.memmap[memx][i])used++;
  71. }
  72. return (used*1000)/(memtblsize[memx]);
  73. }
  74. //内存分配(内部调用)
  75. //memx:所属内存块
  76. //size:要分配的内存大小(字节)
  77. //返回值:0XFFFFFFFF,代表错误;其他,内存偏移地址
  78. u32 my_mem_malloc(u8 memx,u32 size)
  79. {
  80. signed long offset=0;
  81. u32 nmemb; //需要的内存块数
  82. u32 cmemb=0;//连续空内存块数
  83. u32 i;
  84. if(!mallco_dev.memrdy[memx])mallco_dev.init(memx);//未初始化,先执行初始化
  85. if(size==0)return 0XFFFFFFFF;//不需要分配
  86. nmemb=size/memblksize[memx]; //获取需要分配的连续内存块数
  87. if(size%memblksize[memx])nmemb++;
  88. for(offset=memtblsize[memx]-1;offset>=0;offset--)//搜索整个内存控制区
  89. {
  90. if(!mallco_dev.memmap[memx][offset])cmemb++;//连续空内存块数增加
  91. else cmemb=0; //连续内存块清零
  92. if(cmemb==nmemb) //找到了连续nmemb个空内存块
  93. {
  94. for(i=0;i<nmemb;i++) //标注内存块非空
  95. {
  96. mallco_dev.memmap[memx][offset+i]=nmemb;
  97. }
  98. return (offset*memblksize[memx]);//返回偏移地址
  99. }
  100. }
  101. return 0XFFFFFFFF;//未找到符合分配条件的内存块
  102. }
  103. //释放内存(内部调用)
  104. //memx:所属内存块
  105. //offset:内存地址偏移
  106. //返回值:0,释放成功;1,释放失败;
  107. u8 my_mem_free(u8 memx,u32 offset)
  108. {
  109. int i;
  110. if(!mallco_dev.memrdy[memx])//未初始化,先执行初始化
  111. {
  112. mallco_dev.init(memx);
  113. return 1;//未初始化
  114. }
  115. if(offset<memsize[memx])//偏移在内存池内.
  116. {
  117. int index=offset/memblksize[memx]; //偏移所在内存块号码
  118. int nmemb=mallco_dev.memmap[memx][index]; //内存块数量
  119. for(i=0;i<nmemb;i++) //内存块清零
  120. {
  121. mallco_dev.memmap[memx][index+i]=0;
  122. }
  123. return 0;
  124. }else return 2;//偏移超区了.
  125. }
  126. //释放内存(外部调用)
  127. //memx:所属内存块
  128. //ptr:内存首地址
  129. void myfree(u8 memx,void *ptr)
  130. {
  131. u32 offset;
  132. if(ptr==NULL)return;//地址为0.
  133. offset=(u32)ptr-(u32)mallco_dev.membase[memx];
  134. my_mem_free(memx,offset); //释放内存
  135. }
  136. //分配内存(外部调用)
  137. //memx:所属内存块
  138. //size:内存大小(字节)
  139. //返回值:分配到的内存首地址.
  140. void *mymalloc(u8 memx,u32 size)
  141. {
  142. u32 offset;
  143. offset=my_mem_malloc(memx,size);
  144. if(offset==0XFFFFFFFF)return NULL;
  145. else return (void*)((u32)mallco_dev.membase[memx]+offset);
  146. }
  147. //重新分配内存(外部调用)
  148. //memx:所属内存块
  149. //*ptr:旧内存首地址
  150. //size:要分配的内存大小(字节)
  151. //返回值:新分配到的内存首地址.
  152. void *myrealloc(u8 memx,void *ptr,u32 size)
  153. {
  154. u32 offset;
  155. offset=my_mem_malloc(memx,size);
  156. if(offset==0XFFFFFFFF)return NULL;
  157. else
  158. {
  159. mymemcpy((void*)((u32)mallco_dev.membase[memx]+offset),ptr,size); //拷贝旧内存内容到新内存
  160. myfree(memx,ptr); //释放旧内存
  161. return (void*)((u32)mallco_dev.membase[memx]+offset); //返回新内存首地址
  162. }
  163. }