免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 4248 | 回复: 6
打印 上一主题 下一主题

[算法] 马上放假,爽,再发我实现的一个内存池算法 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-04-30 14:00 |只看该作者 |倒序浏览
这是我的开源程序库的一部分(http://gforge.osdn.net.cn/frs/?group_id=86),
原实现在 ylib*/young/youngc/yc_memory.c 文件内。
源码如下:

用到的定义:

  1. typedef  unsigned char  ylib_byte_t;
  2. typedef  unsigned int   ylib_word_t;

  3. #ifndef  YLIB_WORD_MAX
  4. #define  YLIB_WORD_MAX  UINT_MAX
  5. #endif

  6. enum YOUNG_LIBRARY_COMMON_CONSTANT
  7. {
  8.     BYTE_BITS = CHAR_BIT,
  9.     YLIB_BYTE_MAX = UCHAR_MAX,
  10.     CHARACTER_SIZE_MAX = MB_LEN_MAX
  11. };
复制代码


  1. 函数声明:

  2. size_t get_pool_alloc_count( void );

  3. size_t get_pool_dealloc_count( void );

  4. void set_pool_lock( void (*lock)(size_t index) );

  5. void set_pool_unlock( void (*unlock)(size_t index) );

  6. size_t memalign( size_t element_size, bool memory_type );

  7. void pool_print( void );

  8. void* pool_alloc( size_t bytes );

  9. /* 只将内存块回收至内存池 */
  10. void pool_dealloc( void* ptr, size_t bytes );

  11. /* 先将内存块回收至内存池,再将内存池中空闲的内存页释放 */
  12. void pool_free( void* ptr, size_t bytes );
复制代码


  1. 内存池实现:
  2. /*
  3. * 内存池由 MEMORY_POOL_LISTS 个链表组成,每个链表分别处理不同大小的内存块,
  4. * 每个链表处理的内存块大小 = (索引 + 1) * MEMORY_POOL_ALIGN
  5. *
  6. * 内存池实现原理图:
  7. * -------------------------------------
  8. * |  0  |  1  |  2  |  3  |    ......              | MEMORY_POOL_LISTS - 1 |
  9. * -------------------------------------
  10. *    |
  11. *    |
  12. *    |     ----
  13. *     -> | next |--
  14. *          |---|   |
  15. *          | use  |   |
  16. *          |---|   |
  17. *          |  页  |   |
  18. *          ----   |
  19. *                     |
  20. *  ----------
  21. *  |
  22. *  |   ----
  23. *  ->| next |-->NULL
  24. *      |---|
  25. *      | use  | 注:使用该字的二进制位来记录缓存页中各内存块的使用状态
  26. *      |---|
  27. *      |  页  | 注:内存页大小 = (索引 + 1) * MEMORY_POOL_ALIGN * use的位数
  28. *      ----
  29. */

  30. /*****************************************************************************/
  31. /*****************************************************************************/
  32. #include <stdio.h>
  33. #include <stdlib.h>
  34. #include "yc_memory.h"

  35. #ifdef __cplusplus
  36.     namespace youngc {  extern "C" {
  37. #endif
  38. /*****************************************************************************/
  39. /*****************************************************************************/

  40. enum YOUNG_LIBRARY_MEMORY_CONSTANT
  41. {
  42.     MEMORY_ALIGN_SIZE = sizeof(int),
  43.     MEMORY_POOL_ALIGN = 8,
  44.     MEMORY_POOL_MAX_BYTES = 128,
  45.     MEMORY_POOL_BLOCKS = sizeof(ylib_word_t) * BYTE_BITS,
  46.     MEMORY_POOL_LISTS = MEMORY_POOL_MAX_BYTES / MEMORY_POOL_ALIGN
  47. };

  48. typedef struct memory_page
  49. {
  50.     struct memory_page*  next; /* 下一个内存页 */
  51.     ylib_word_t          use;  /* 位为0表示未被分配,位为1表示已被分配 */
  52. } page_t;

  53. #define  LIST_INDEX( bytes ) \
  54.          ( ((bytes) + MEMORY_POOL_ALIGN - 1) / MEMORY_POOL_ALIGN - 1 )

  55. /*****************************************************************************/
  56. /*****************************************************************************/

  57. static void (*pool_lock)( size_t ) = NULL;
  58. static void (*pool_unlock)( size_t ) = NULL;
  59. static size_t pool_alloc_count = 0;
  60. static size_t pool_dealloc_count = 0;
  61. static page_t* pool[MEMORY_POOL_LISTS] = { NULL };

  62. /*****************************************************************************/
  63. /*****************************************************************************/

  64. size_t get_pool_alloc_count( void )
  65. {
  66.     return pool_alloc_count;
  67. }

  68. size_t get_pool_dealloc_count( void )
  69. {
  70.     return pool_dealloc_count;
  71. }

  72. void set_pool_lock( void (*lock)(size_t) )
  73. {
  74.     pool_lock = lock;
  75. }

  76. void set_pool_unlock( void (*unlock)(size_t) )
  77. {
  78.     pool_unlock = unlock;
  79. }

  80. /*****************************************************************************/

  81. size_t memalign( size_t element_size, bool memory_type )
  82. {
  83.     size_t residue = MEMORY_ALIGN_SIZE - 1;
  84.     size_t alignsize = element_size & ~residue;

  85.     if( element_size & residue )
  86.         alignsize += MEMORY_ALIGN_SIZE;

  87.     if( memory_type == true && alignsize < sizeof(ylib_inner_t) )
  88.         alignsize = sizeof(ylib_inner_t);

  89.     return alignsize;
  90. }

  91. /*****************************************************************************/

  92. static void word_to_bits( ylib_word_t val, char* str_last )
  93. {
  94.     int i;
  95.     ylib_word_t mask = 1;

  96.     for( i = 0; i < MEMORY_POOL_BLOCKS; ++i, mask <<= 1 )
  97.         *--str_last = val & mask ? '1' : '0';
  98. }

  99. void pool_print( void )
  100. {
  101.     int i;
  102.     page_t* curr;
  103.     char bits[MEMORY_POOL_BLOCKS + 1] = { '\0' };

  104.     for( i = 0; i < MEMORY_POOL_LISTS; ++i )
  105.     {
  106.         curr = pool[i];

  107.         printf( "\npool[%d:%d] = %p: ", i,
  108.                 (i + 1) * MEMORY_POOL_ALIGN, pool[i] );

  109.         while( curr )
  110.         {
  111.             word_to_bits( curr->use, bits + MEMORY_POOL_BLOCKS );
  112.             printf( "\n\t" );
  113.             printf( "next = %p | use = %s", curr->next, bits );
  114.             curr = curr->next;
  115.         }
  116.     }
  117. }

  118. /*****************************************************************************/

  119. void* pool_alloc( size_t bytes )
  120. {
  121.     void* ptr = NULL;
  122.     page_t* pg = NULL;

  123.     if( bytes > MEMORY_POOL_MAX_BYTES )
  124.     {
  125.         ptr = malloc( bytes );
  126.     }
  127.     else
  128.     {
  129.         size_t index = LIST_INDEX( bytes );
  130.         size_t block_size = (index + 1) * MEMORY_POOL_ALIGN;

  131.         if( pool_lock )
  132.             pool_lock( index );

  133.         if( pool[index] )
  134.         {
  135.             page_t* curr = pool[index];

  136.             while( curr ) /* 遍历链中的所有内存页寻找空闲的内存块 */
  137.             {
  138.                 if( curr->use == YLIB_WORD_MAX ) /* 该页中没有空闲块 */
  139.                     curr = curr->next; /* 进入下一个内存页 */
  140.                 else /* 该页中有空闲块 */
  141.                 {
  142.                     ylib_word_t i, j, mask = 1;
  143.                     /* 先找到第一个空闲块所在的字节 */
  144.                     for( i = 0; i < sizeof(ylib_word_t); ++i )
  145.                     {
  146.                         ylib_byte_t val = (ylib_byte_t)(curr->use
  147.                                                         >> (BYTE_BITS * i));
  148.                         if( val != YLIB_BYTE_MAX )
  149.                         {
  150.                             /* 再查找空闲块所在的位 */
  151.                             mask <<= BYTE_BITS * i;
  152.                             for( j = 0; j < BYTE_BITS; ++j, mask <<= 1 )
  153.                             {
  154.                                 if( !(curr->use & mask) ) /* 当前块未使用 */
  155.                                 {
  156.                                     curr->use |= mask; /* 设置使用标志 */
  157.                                     ptr = (ylib_byte_t*)curr + sizeof(page_t)
  158.                                           + (i * BYTE_BITS + j) * block_size;
  159.                                     ++pool_alloc_count;
  160.                                     if( pool_unlock )
  161.                                         pool_unlock( index );
  162.                                     return ptr;
  163.                                 }
  164.                             } /* end for j */
  165.                         } /* end if nbits */
  166.                     } /* end for i */
  167.                 } /* end else */
  168.             } /* end while */
  169.         } /* end if( pool[index] ) */

  170.         /* 该链下未分配内存页或无空闲块,此时需增加新的内存页 */
  171.         pg = (page_t*)malloc( sizeof(page_t) + (index + 1)
  172.                               * MEMORY_POOL_ALIGN * MEMORY_POOL_BLOCKS );
  173.         if( pg )
  174.         {
  175.             pg->next = pool[index];
  176.             pool[index] = pg;
  177.             pg->use = 1;
  178.             ptr = (ylib_byte_t*)pool[index] + sizeof(page_t);
  179.         }

  180.         if( pool_unlock )
  181.             pool_unlock( index );
  182.     } /* end else */

  183.     if( ptr )
  184.         ++pool_alloc_count;

  185.     return ptr;
  186. }

  187. /*****************************************************************************/

  188. void pool_dealloc( void* ptr, size_t bytes )
  189. {
  190.     if( !ptr )
  191.         return;

  192.     if( bytes <= MEMORY_POOL_MAX_BYTES )
  193.     {
  194.         size_t index = LIST_INDEX( bytes );
  195.         size_t pagesize = (index + 1) * MEMORY_POOL_ALIGN * MEMORY_POOL_BLOCKS;
  196.         page_t *prev = NULL, *curr = pool[index];
  197.         ylib_byte_t *begin, *end, *blk = (ylib_byte_t*)ptr;

  198.         if( pool_lock )
  199.             pool_lock( index );

  200.         while( curr )
  201.         {
  202.             begin = (ylib_byte_t*)curr + sizeof(page_t);
  203.             end = begin + pagesize;
  204.             if( blk < begin || end <= blk ) /* 判断ptr是否在当前内存页内 */
  205.             {
  206.                 prev = curr;
  207.                 curr = curr->next;
  208.             }
  209.             else
  210.             {
  211.                 /* 检查ptr是否正确 */
  212.                 if( (blk - begin) % ((index + 1) * MEMORY_POOL_ALIGN) == 0 )
  213.                 {
  214.                     ylib_word_t mask = 1;
  215.                     size_t blk_index = (blk - begin)
  216.                                        / ((index + 1) * MEMORY_POOL_ALIGN);
  217.                     mask <<= blk_index;
  218.                     curr->use &= (~mask);
  219.                     /* 如果当前内存页已空闲并且非链首,则将当前内存页移至链首 */
  220.                     if( curr->use == 0 && prev )
  221.                     {
  222.                         prev->next = curr->next;
  223.                         curr->next = pool[index];
  224.                         pool[index] = curr;
  225.                     }
  226.                     ++pool_dealloc_count;
  227.                 }
  228.                 if( pool_unlock )
  229.                     pool_unlock( index );
  230.                 return;
  231.             }
  232.         } /* end while */
  233.     } /* end if */

  234.     /* ptr不是由内存池分配 */
  235.     free( ptr );
  236.     ++pool_dealloc_count;
  237. }

  238. /*****************************************************************************/

  239. void pool_free( void* ptr, size_t bytes )
  240. {
  241.     if( bytes > MEMORY_POOL_MAX_BYTES )
  242.     {
  243.         if( ptr )
  244.         {
  245.             free( ptr );
  246.             ++pool_dealloc_count;
  247.         }
  248.     }
  249.     else
  250.     {
  251.         size_t index = LIST_INDEX( bytes );
  252.         page_t *erase = NULL, *prev = NULL, *curr = pool[index];

  253.         if( ptr )
  254.             pool_dealloc( ptr, bytes );

  255.         if( pool_lock )
  256.             pool_lock( index );

  257.         /* 释放空闲页 */
  258.         while( curr )
  259.         {
  260.             if( curr->use != 0 )
  261.             {
  262.                 prev = curr;
  263.                 curr = curr->next;
  264.             }
  265.             else
  266.             {
  267.                 erase = curr;
  268.                 if( prev )
  269.                     prev->next = curr->next;
  270.                 else
  271.                     pool[index] = curr->next;
  272.                 curr = curr->next;
  273.                 free( erase );
  274.             }
  275.         } /* end while */

  276.         if( pool_unlock )
  277.             pool_unlock( index );
  278.     } /* end else */
  279. }

  280. /*****************************************************************************/
  281. /*****************************************************************************/
  282. #ifdef __cplusplus
  283.     }  }
  284. #endif
  285. /*****************************************************************************/
  286. /*****************************************************************************/
复制代码


其中的 pool_lock 和  pool_unlock 函数指针是由使用者传递的供并发使用时调用的加锁和解锁函数,
因为内存池的每个链可以使用不同的锁,所以我给这两个函数指针类型指定了一个参数,用一传递当前
链的索引号。

[ 本帖最后由 phoneix 于 2007-4-30 16:45 编辑 ]

论坛徽章:
0
2 [报告]
发表于 2007-04-30 14:14 |只看该作者
楼主能不能把格式整理一下,再用论坛的[code][/code]标记发出来,这样简直没法看。
顺便讲一讲你的实现的功能吧,每个函数头给出函数的用途。这样应该是一篇不错的帖子。

论坛徽章:
0
3 [报告]
发表于 2007-04-30 14:25 |只看该作者
呵呵,我是兴之所至才发的,我的计划是从去年写一个系列的文章来介绍我的程序库的实现原理和实现细节,
结果这半年太忙了,写了个开头后面就没有了,既然楼上的朋友提出了要求,那我晚上回家整理一下,重新
发一篇新的文章,希望到时候砖头不要砸的太厉害了。

论坛徽章:
0
4 [报告]
发表于 2007-04-30 15:13 |只看该作者
楼主还考虑了对内存池进行管理的时候,链表遍历造成的performance的影响?

论坛徽章:
0
5 [报告]
发表于 2007-04-30 15:55 |只看该作者
看上面的图解,似乎跟SGI STL的空间配置器实现相似。
代码我没有看

论坛徽章:
0
6 [报告]
发表于 2007-04-30 16:06 |只看该作者
原帖由 jaffaz 于 2007-4-30 15:55 发表
看上面的图解,似乎跟SGI STL的空间配置器实现相似。
代码我没有看

不好意思,看错了。
这个比sgi stl的稍微复杂些,多了个页的概念。再碎片控制上更好,但也增加了时间开销。

论坛徽章:
0
7 [报告]
发表于 2007-04-30 16:47 |只看该作者
呵呵,其实SGI STL比我这个要复杂的,SGI STL分配的速度比这个内存池应该是要快的。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP