swmem.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381
  1. /************************************************************************
  2. * AUTHOR: NiuJiuRu
  3. * FILENAME: swmem.c
  4. * DESCRIPTION: 可分块/可跟踪内存泄漏位置的内存管理
  5. * NOTE:
  6. * HISTORY:
  7. * 1, [2010-09-06] created by NiuJiuRu
  8. * 2, [2013-05-30] modified by NiuJiuRu, 优化"sw_mem_alloc()"函数, 修改
  9. * 分配顺序(整个缓冲全部分配完毕后,再在节点空隙间分配),
  10. * 提高内存分配的速度
  11. ***********************************************************************/
  12. #include "swapi.h"
  13. #include "swdir.h"
  14. #include "swmutex.h"
  15. #include "swmem.h"
  16. #if defined _DEBUG && !defined _MEM_DEBUG_DETAIL
  17. #define _MEM_DEBUG_DETAIL
  18. #endif
  19. // 判断对齐粒度是否有效(2的n次方 n >= 0)
  20. #define ALIGN_VALID(align) ((align) > 0 ? (((align) & (~(align)+1)) == (align) ? true : false) : false)
  21. // 整数对齐, align = 1, 2, 4, 8, 16, ...
  22. #define ALIGN_SIZE(size, align) (((size)+((align)-1)) & ~((align)-1))
  23. // 指针对齐, align = 1, 2, 4, 8, 16, ...
  24. #define ALIGN_PTR(ptr, size, align) (((long)(ptr)+(size)+((align)-1)) & ~((align)-1))
  25. typedef struct _MemNode_
  26. {
  27. #ifdef _MEM_DEBUG_DETAIL
  28. char filename[MAX_PATH_CHARS];
  29. int line;
  30. #endif
  31. int size;
  32. struct _MemNode_ *next;
  33. } SMemNode;
  34. typedef struct
  35. {
  36. // 工作缓冲区
  37. char *buf;
  38. // 工作缓冲区大小
  39. int size;
  40. // 内存对齐字节数
  41. int align;
  42. // 访问工作缓冲区的互斥锁
  43. void *hMutex;
  44. // 空间分配链表
  45. SMemNode *list;
  46. // 最前的查找块结点(遍历分配内存使用)
  47. SMemNode *first;
  48. // 最后的查找块结点(遍历分配内存使用)
  49. SMemNode *tail;
  50. // 历史上最大分配过的内存块大小(字节)
  51. int history_maxsize;
  52. } SMemHeader;
  53. /* 创建内存管理模块, align = 1, 2, 4, 8, 16, ... */
  54. void *sw_mem_create(char *buf, int size, int align)
  55. {
  56. SMemHeader *xm = NULL;
  57. if(!buf || size <= 0 || !ALIGN_VALID(align)) goto ErrorP;
  58. xm = (SMemHeader *)ALIGN_PTR(buf, 0, align);
  59. memset(xm, 0, sizeof(SMemHeader));
  60. xm->align = align;
  61. xm->buf = (char *)ALIGN_PTR(xm, sizeof(SMemHeader), xm->align);
  62. xm->size = buf+size-xm->buf;
  63. xm->hMutex = sw_mutex_create();
  64. if(!xm->hMutex) goto ErrorP;
  65. return xm;
  66. ErrorP:
  67. sw_mem_destroy(xm);
  68. return NULL;
  69. }
  70. /* 销毁内存管理模块 */
  71. void sw_mem_destroy(void *hMem)
  72. {
  73. SMemHeader *xm = (SMemHeader *)hMem;
  74. if(!xm) return;
  75. if(xm->hMutex)
  76. {
  77. sw_mutex_destroy(xm->hMutex);
  78. }
  79. memset(xm, 0, sizeof(SMemHeader));
  80. }
  81. /* 复位内存管理模块 */
  82. void sw_mem_reset(void *hMem)
  83. {
  84. SMemHeader *xm = (SMemHeader *)hMem;
  85. if(!xm) return;
  86. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  87. xm->list = NULL;
  88. xm->first = NULL;
  89. xm->tail = NULL;
  90. xm->history_maxsize = 0;
  91. sw_mutex_unlock(xm->hMutex); // unlock
  92. }
  93. /* 检查内存节点, 并返回历史上最大消耗过的内存数 */
  94. int sw_mem_check(const void *hMem, PMemCheckCallback proc, void *lpParameter)
  95. {
  96. SMemHeader *xm = (SMemHeader *)hMem;
  97. SMemNode *node = NULL;
  98. char *ptr = NULL;
  99. if(!xm) return -1;
  100. node = xm->list;
  101. while(node && proc)
  102. {
  103. ptr = (char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align);
  104. #ifdef _MEM_DEBUG_DETAIL
  105. proc(xGetPathFileName(node->filename), node->line, ptr, node->size, lpParameter);
  106. #else
  107. proc("", -1, ptr, node->size, lpParameter);
  108. #endif
  109. node = node->next;
  110. }
  111. return xm->history_maxsize;
  112. }
  113. /* 获得总的内存大小 */
  114. int sw_mem_getTotalSize(const void *hMem)
  115. {
  116. SMemHeader *xm = (SMemHeader *)hMem;
  117. if(!xm) return -1;
  118. return xm->size;
  119. }
  120. // 查找内存节点
  121. static SMemNode *sw_mem_find(const void *hMem, void *ptr)
  122. {
  123. SMemHeader *xm = (SMemHeader *)hMem;
  124. SMemNode *node = NULL;
  125. if(!xm || !ptr) return NULL;
  126. node = xm->list;
  127. while(node)
  128. {
  129. if(ptr == (void *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)) break;
  130. node = node->next;
  131. }
  132. return node;
  133. }
  134. // 扫描现有的内存节点链表, 查看是否有符合申请条件的节点间空隙-遍历分配
  135. static void *sw_mem_visitAllocGap(void *hMem, int size, const char *filename, int line)
  136. {
  137. SMemHeader *xm = (SMemHeader *)hMem;
  138. SMemNode *node = NULL, *new_node = NULL;
  139. char *ptr = NULL;
  140. int gapLen = 0, firstGapLen = 0;
  141. if(!xm) return NULL;
  142. // 1, 检查链表头和buf之间的空隙
  143. if(xm->first == NULL)
  144. {
  145. if(xm->list == NULL) return NULL;
  146. node = xm->list; // 首节点
  147. firstGapLen = ((char *)node) - (char *)ALIGN_PTR(xm->buf, sizeof(SMemNode), xm->align);
  148. if(size <= firstGapLen)
  149. { // 找到了符合申请要求的内存空间
  150. new_node = (SMemNode *)xm->buf;
  151. memset(new_node, 0, sizeof(SMemNode));
  152. #ifdef _MEM_DEBUG_DETAIL
  153. if(filename)
  154. {
  155. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  156. new_node->line = line;
  157. }
  158. #endif
  159. new_node->size = size;
  160. xm->list = new_node;
  161. new_node->next = node;
  162. xm->first = new_node;
  163. return (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align);
  164. }
  165. }
  166. else
  167. {
  168. node = xm->first; // 查找空闲内存, 从"xm->first"处开始查找, 而不是从"xm->list"处
  169. firstGapLen = 0;
  170. }
  171. // 2, 检查相临两块之间的空隙
  172. while(node->next != NULL)
  173. {
  174. ptr = (char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align) + ALIGN_SIZE(node->size, xm->align);
  175. gapLen = ((char*)node->next) - (char *)ALIGN_PTR(ptr, sizeof(SMemNode), xm->align);
  176. if(firstGapLen < xm->align)
  177. { // 如果第一间隔空隙太小, 不够最低分配字节数, 就修改"xm->first"指针
  178. xm->first = node;
  179. firstGapLen = gapLen;
  180. }
  181. if(size <= gapLen)
  182. { // 找到了符合申请要求的内存空间
  183. new_node = (SMemNode *)ptr;
  184. memset(new_node, 0, sizeof(SMemNode));
  185. #ifdef _MEM_DEBUG_DETAIL
  186. if(filename)
  187. {
  188. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  189. new_node->line = line;
  190. }
  191. #endif
  192. new_node->size = size;
  193. new_node->next = node->next;
  194. node->next = new_node;
  195. return (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align);
  196. }
  197. node = node->next;
  198. }
  199. return NULL;
  200. }
  201. /* 分配内存 */
  202. void *sw_mem_alloc(void *hMem, int size, const char *filename, int line)
  203. {
  204. SMemHeader *xm = (SMemHeader *)hMem;
  205. SMemNode *new_node = NULL;
  206. char *ptr = NULL;
  207. if(!xm || size < 0) return NULL;
  208. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  209. // 1, 新建内存节点
  210. if(xm->list == NULL) new_node = (SMemNode *)xm->buf; // 第一次申请
  211. else new_node = (SMemNode *)((char *)ALIGN_PTR(xm->tail, sizeof(SMemNode), xm->align) + ALIGN_SIZE(xm->tail->size, xm->align));
  212. ptr = (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align); // 指向新节点buf的开始位置
  213. // 2, 剩余内存空间不足时, 尝试在现有内存节点间的空隙中分配
  214. if((int)(xm->buf+xm->size-ptr) < size)
  215. {
  216. ptr = NULL;
  217. if(!ptr) ptr = (char *)sw_mem_visitAllocGap(xm, size, filename, line); // 遍历分配
  218. if(ptr)
  219. {
  220. if(xm->history_maxsize < (ptr-xm->buf)+size) xm->history_maxsize = (ptr-xm->buf)+size;
  221. sw_mutex_unlock(xm->hMutex); // unlock
  222. return ptr; // return a void pointer to the allocated space
  223. }
  224. sw_log_fatal("sorry, alloc memory fail, because %p memory container is full!!!", xm);
  225. sw_mutex_unlock(xm->hMutex); // unlock
  226. return NULL;
  227. }
  228. // 3, 填充该内存节点
  229. memset(new_node, 0, sizeof(SMemNode));
  230. #ifdef _MEM_DEBUG_DETAIL
  231. if(filename)
  232. {
  233. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  234. new_node->line = line;
  235. }
  236. #endif
  237. new_node->size = size;
  238. if(xm->list == NULL) xm->list = new_node;
  239. else xm->tail->next = new_node;
  240. xm->tail = new_node;
  241. if(xm->history_maxsize < (ptr-xm->buf)+size) xm->history_maxsize = (ptr-xm->buf)+size;
  242. sw_mutex_unlock(xm->hMutex); // unlock
  243. // 4, return a void pointer to the allocated space
  244. return ptr;
  245. }
  246. /* 分配内存并初始化为"0x00" */
  247. void *sw_mem_calloc(void *hMem, int size, const char *filename, int line)
  248. {
  249. SMemHeader *xm = (SMemHeader *)hMem;
  250. char *ptr = NULL;
  251. ptr = sw_mem_alloc(xm, size, filename, line);
  252. if(ptr) memset(ptr, 0, size);
  253. return ptr;
  254. }
  255. /* 复制字符串 */
  256. char *sw_mem_strdup(void *hMem, const char *str, const char *filename, int line)
  257. {
  258. SMemHeader *xm = (SMemHeader *)hMem;
  259. char *ptr = NULL;
  260. if(str) ptr = sw_mem_alloc(xm, strlen(str)+1, filename, line);
  261. if(ptr) memcpy(ptr, str, strlen(str)+1);
  262. return ptr;
  263. }
  264. /* 重新申请内存 */
  265. void *sw_mem_realloc(void *hMem, void *ptr, int size, const char *filename, int line)
  266. {
  267. SMemHeader *xm = (SMemHeader *)hMem;
  268. SMemNode *node = NULL;
  269. char *new_ptr = NULL;
  270. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  271. node = sw_mem_find(xm, ptr);
  272. if(node && (size <= node->size || \
  273. (long)((char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)+size) <= (long)(node->next ? node->next : (SMemNode *)(xm->buf+xm->size))))
  274. {
  275. node->size = size;
  276. sw_mutex_unlock(xm->hMutex); // unlock
  277. return ptr;
  278. }
  279. sw_mutex_unlock(xm->hMutex); // unlock
  280. new_ptr = sw_mem_alloc(xm, size, filename, line);
  281. if(new_ptr && ptr)
  282. {
  283. memcpy(new_ptr, ptr, node ? node->size : size);
  284. sw_mem_free(xm, ptr, filename, line);
  285. }
  286. return new_ptr;
  287. }
  288. /* 释放内存 */
  289. void sw_mem_free(void *hMem, void *ptr, const char *filename, int line)
  290. {
  291. SMemHeader *xm = (SMemHeader *)hMem;
  292. SMemNode *node = NULL, *previous = NULL;
  293. if(!xm) return;
  294. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  295. // 寻找该节点
  296. for(node = xm->list; node; node = node->next)
  297. {
  298. if(ptr == (void *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)) break;
  299. previous = node;
  300. }
  301. if(!node) goto EndP;
  302. // 在内存节点链上断开该节点
  303. if(node == xm->list) xm->list = node->next; // 释放的为首节点
  304. else previous->next = node->next;
  305. // 调整"最前的查找块结点"和"最后的查找块结点"指针
  306. if(xm->first >= node) xm->first = previous;
  307. if(xm->tail == node) xm->tail = previous;
  308. EndP:
  309. sw_mutex_unlock(xm->hMutex); // unlock
  310. }