swmem.c 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  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. #ifdef LINUX
  79. free(xm->hMutex);
  80. #endif
  81. }
  82. memset(xm, 0, sizeof(SMemHeader));
  83. }
  84. /* 复位内存管理模块 */
  85. void sw_mem_reset(void *hMem)
  86. {
  87. SMemHeader *xm = (SMemHeader *)hMem;
  88. if(!xm) return;
  89. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  90. xm->list = NULL;
  91. xm->first = NULL;
  92. xm->tail = NULL;
  93. xm->history_maxsize = 0;
  94. sw_mutex_unlock(xm->hMutex); // unlock
  95. }
  96. /* 检查内存节点, 并返回历史上最大消耗过的内存数 */
  97. int sw_mem_check(const void *hMem, PMemCheckCallback proc, void *lpParameter)
  98. {
  99. SMemHeader *xm = (SMemHeader *)hMem;
  100. SMemNode *node = NULL;
  101. char *ptr = NULL;
  102. if(!xm) return -1;
  103. node = xm->list;
  104. while(node && proc)
  105. {
  106. ptr = (char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align);
  107. #ifdef _MEM_DEBUG_DETAIL
  108. proc(xGetPathFileName(node->filename), node->line, ptr, node->size, lpParameter);
  109. #else
  110. proc("", -1, ptr, node->size, lpParameter);
  111. #endif
  112. node = node->next;
  113. }
  114. return xm->history_maxsize;
  115. }
  116. /* 获得总的内存大小 */
  117. int sw_mem_getTotalSize(const void *hMem)
  118. {
  119. SMemHeader *xm = (SMemHeader *)hMem;
  120. if(!xm) return -1;
  121. return xm->size;
  122. }
  123. // 查找内存节点
  124. static SMemNode *sw_mem_find(const void *hMem, void *ptr)
  125. {
  126. SMemHeader *xm = (SMemHeader *)hMem;
  127. SMemNode *node = NULL;
  128. if(!xm || !ptr) return NULL;
  129. node = xm->list;
  130. while(node)
  131. {
  132. if(ptr == (void *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)) break;
  133. node = node->next;
  134. }
  135. return node;
  136. }
  137. // 扫描现有的内存节点链表, 查看是否有符合申请条件的节点间空隙-遍历分配
  138. static void *sw_mem_visitAllocGap(void *hMem, int size, const char *filename, int line)
  139. {
  140. SMemHeader *xm = (SMemHeader *)hMem;
  141. SMemNode *node = NULL, *new_node = NULL;
  142. char *ptr = NULL;
  143. int gapLen = 0, firstGapLen = 0;
  144. if(!xm) return NULL;
  145. // 1, 检查链表头和buf之间的空隙
  146. if(xm->first == NULL)
  147. {
  148. if(xm->list == NULL) return NULL;
  149. node = xm->list; // 首节点
  150. firstGapLen = ((char *)node) - (char *)ALIGN_PTR(xm->buf, sizeof(SMemNode), xm->align);
  151. if(size <= firstGapLen)
  152. { // 找到了符合申请要求的内存空间
  153. new_node = (SMemNode *)xm->buf;
  154. memset(new_node, 0, sizeof(SMemNode));
  155. #ifdef _MEM_DEBUG_DETAIL
  156. if(filename)
  157. {
  158. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  159. new_node->line = line;
  160. }
  161. #endif
  162. new_node->size = size;
  163. xm->list = new_node;
  164. new_node->next = node;
  165. xm->first = new_node;
  166. return (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align);
  167. }
  168. }
  169. else
  170. {
  171. node = xm->first; // 查找空闲内存, 从"xm->first"处开始查找, 而不是从"xm->list"处
  172. firstGapLen = 0;
  173. }
  174. // 2, 检查相临两块之间的空隙
  175. while(node->next != NULL)
  176. {
  177. ptr = (char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align) + ALIGN_SIZE(node->size, xm->align);
  178. gapLen = ((char*)node->next) - (char *)ALIGN_PTR(ptr, sizeof(SMemNode), xm->align);
  179. if(firstGapLen < xm->align)
  180. { // 如果第一间隔空隙太小, 不够最低分配字节数, 就修改"xm->first"指针
  181. xm->first = node;
  182. firstGapLen = gapLen;
  183. }
  184. if(size <= gapLen)
  185. { // 找到了符合申请要求的内存空间
  186. new_node = (SMemNode *)ptr;
  187. memset(new_node, 0, sizeof(SMemNode));
  188. #ifdef _MEM_DEBUG_DETAIL
  189. if(filename)
  190. {
  191. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  192. new_node->line = line;
  193. }
  194. #endif
  195. new_node->size = size;
  196. new_node->next = node->next;
  197. node->next = new_node;
  198. return (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align);
  199. }
  200. node = node->next;
  201. }
  202. return NULL;
  203. }
  204. /* 分配内存 */
  205. void *sw_mem_alloc(void *hMem, int size, const char *filename, int line)
  206. {
  207. SMemHeader *xm = (SMemHeader *)hMem;
  208. SMemNode *new_node = NULL;
  209. char *ptr = NULL;
  210. if(!xm || size < 0) return NULL;
  211. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  212. // 1, 新建内存节点
  213. if(xm->list == NULL) new_node = (SMemNode *)xm->buf; // 第一次申请
  214. else new_node = (SMemNode *)((char *)ALIGN_PTR(xm->tail, sizeof(SMemNode), xm->align) + ALIGN_SIZE(xm->tail->size, xm->align));
  215. ptr = (char *)ALIGN_PTR(new_node, sizeof(SMemNode), xm->align); // 指向新节点buf的开始位置
  216. // 2, 剩余内存空间不足时, 尝试在现有内存节点间的空隙中分配
  217. if((int)(xm->buf+xm->size-ptr) < size)
  218. {
  219. ptr = NULL;
  220. if(!ptr) ptr = (char *)sw_mem_visitAllocGap(xm, size, filename, line); // 遍历分配
  221. if(ptr)
  222. {
  223. if(xm->history_maxsize < (ptr-xm->buf)+size) xm->history_maxsize = (ptr-xm->buf)+size;
  224. sw_mutex_unlock(xm->hMutex); // unlock
  225. return ptr; // return a void pointer to the allocated space
  226. }
  227. sw_log_fatal("sorry, alloc memory fail, because %p memory container is full!!!", xm);
  228. sw_mutex_unlock(xm->hMutex); // unlock
  229. return NULL;
  230. }
  231. // 3, 填充该内存节点
  232. memset(new_node, 0, sizeof(SMemNode));
  233. #ifdef _MEM_DEBUG_DETAIL
  234. if(filename)
  235. {
  236. strncpy(new_node->filename, filename, sizeof(new_node->filename)-1);
  237. new_node->line = line;
  238. }
  239. #endif
  240. new_node->size = size;
  241. if(xm->list == NULL) xm->list = new_node;
  242. else xm->tail->next = new_node;
  243. xm->tail = new_node;
  244. if(xm->history_maxsize < (ptr-xm->buf)+size) xm->history_maxsize = (ptr-xm->buf)+size;
  245. sw_mutex_unlock(xm->hMutex); // unlock
  246. // 4, return a void pointer to the allocated space
  247. return ptr;
  248. }
  249. /* 分配内存并初始化为"0x00" */
  250. void *sw_mem_calloc(void *hMem, int size, const char *filename, int line)
  251. {
  252. SMemHeader *xm = (SMemHeader *)hMem;
  253. char *ptr = NULL;
  254. ptr = sw_mem_alloc(xm, size, filename, line);
  255. if(ptr) memset(ptr, 0, size);
  256. return ptr;
  257. }
  258. /* 复制字符串 */
  259. char *sw_mem_strdup(void *hMem, const char *str, const char *filename, int line)
  260. {
  261. SMemHeader *xm = (SMemHeader *)hMem;
  262. char *ptr = NULL;
  263. if(str) ptr = sw_mem_alloc(xm, strlen(str)+1, filename, line);
  264. if(ptr) memcpy(ptr, str, strlen(str)+1);
  265. return ptr;
  266. }
  267. /* 重新申请内存 */
  268. void *sw_mem_realloc(void *hMem, void *ptr, int size, const char *filename, int line)
  269. {
  270. SMemHeader *xm = (SMemHeader *)hMem;
  271. SMemNode *node = NULL;
  272. char *new_ptr = NULL;
  273. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  274. node = sw_mem_find(xm, ptr);
  275. if(node && (size <= node->size || \
  276. (long)((char *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)+size) <= (long)(node->next ? node->next : (SMemNode *)(xm->buf+xm->size))))
  277. {
  278. node->size = size;
  279. sw_mutex_unlock(xm->hMutex); // unlock
  280. return ptr;
  281. }
  282. sw_mutex_unlock(xm->hMutex); // unlock
  283. new_ptr = sw_mem_alloc(xm, size, filename, line);
  284. if(new_ptr && ptr)
  285. {
  286. memcpy(new_ptr, ptr, node ? node->size : size);
  287. sw_mem_free(xm, ptr, filename, line);
  288. }
  289. return new_ptr;
  290. }
  291. /* 释放内存 */
  292. void sw_mem_free(void *hMem, void *ptr, const char *filename, int line)
  293. {
  294. SMemHeader *xm = (SMemHeader *)hMem;
  295. SMemNode *node = NULL, *previous = NULL;
  296. if(!xm) return;
  297. sw_mutex_lock(xm->hMutex, WAIT_FOREVER); // lock
  298. // 寻找该节点
  299. for(node = xm->list; node; node = node->next)
  300. {
  301. if(ptr == (void *)ALIGN_PTR(node, sizeof(SMemNode), xm->align)) break;
  302. previous = node;
  303. }
  304. if(!node) goto EndP;
  305. // 在内存节点链上断开该节点
  306. if(node == xm->list) xm->list = node->next; // 释放的为首节点
  307. else previous->next = node->next;
  308. // 调整"最前的查找块结点"和"最后的查找块结点"指针
  309. if(xm->first >= node) xm->first = previous;
  310. if(xm->tail == node) xm->tail = previous;
  311. EndP:
  312. sw_mutex_unlock(xm->hMutex); // unlock
  313. }