免费注册 查看新帖 |

Chinaunix

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

linux启动内存分配器2。。。。。。 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-01-05 20:16 |只看该作者 |倒序浏览
linux启动内存分配器2。。。。。。











介绍了上面的初始化流程,对于分配和释放就简单了,分配就是将分配器映射位图中对应的位置1,释放过程相反。




view plaincopy to clipboard
  1. 01./*分配size大小的空间*/  
  2. 02.static void * __init alloc_bootmem_core(struct bootmem_data *bdata,  
  3. 03.                    unsigned long size, unsigned long align,  
  4. 04.                    unsigned long goal, unsigned long limit)  
  5. 05.{  
  6. 06.    unsigned long fallback = 0;  
  7. 07.    unsigned long min, max, start, sidx, midx, step;  
  8. 08.  
  9. 09.    bdebug("nid=%td size=%lx [%lu pages] align=%lx goal=%lx limit=%lx\n",  
  10. 10.        bdata - bootmem_node_data, size, PAGE_ALIGN(size) >> PAGE_SHIFT,  
  11. 11.        align, goal, limit);  
  12. 12.  
  13. 13.    BUG_ON(!size);  
  14. 14.    BUG_ON(align & (align - 1));  
  15. 15.    BUG_ON(limit && goal + size > limit);  
  16. 16.    /*如果没有映射位图返回空,分配失败*/  
  17. 17.    if (!bdata->node_bootmem_map)  
  18. 18.        return NULL;  
  19. 19.  
  20. 20.    min = bdata->node_min_pfn;  
  21. 21.    max = bdata->node_low_pfn;  
  22. 22.  
  23. 23.    goal >>= PAGE_SHIFT;  
  24. 24.    limit >>= PAGE_SHIFT;  
  25. 25.  
  26. 26.    if (limit && max > limit)  
  27. 27.        max = limit;  
  28. 28.    if (max <= min)  
  29. 29.        return NULL;  
  30. 30.    /*step为需要对齐于页面数*/  
  31. 31.    step = max(align >> PAGE_SHIFT, 1UL);  
  32. 32.    /*计算起始页面*/  
  33. 33.    if (goal && min < goal && goal < max)  
  34. 34.        start = ALIGN(goal, step);  
  35. 35.    else  
  36. 36.        start = ALIGN(min, step);  
  37. 37.    /*计算分配页面区间*/  
  38. 38.    sidx = start - bdata->node_min_pfn;  
  39. 39.    midx = max - bdata->node_min_pfn;  
  40. 40.    /*前一次分配的页号比这次开始分配的页面号大
  41. 41.    那么,如果第一次没有分配到,回退到这次的
  42. 42.    开始重新试,因为第一次分配是从上一次分配
  43. 43.    的位置开始的*/  
  44. 44.    if (bdata->hint_idx > sidx) {      
  45. 45.         * Handle the valid case of sidx being zero and still  
  46. 46.         * catch the fallback below.  
  47. 47.         */  
  48. 48.        fallback = sidx + 1;  
  49. 49.        /*从上一次分配的位置开始,对齐与页面*/  
  50. 50.        sidx = align_idx(bdata, bdata->hint_idx, step);  
  51. 51.    }  
  52. 52.  
  53. 53.    while (1) {  
  54. 54.        int merge;  
  55. 55.        void *region;  
  56. 56.        unsigned long eidx, i, start_off, end_off;  
  57. 57.find_block:  
  58. 58.        /*查找第一个为0的位*/  
  59. 59.        sidx = find_next_zero_bit(bdata->node_bootmem_map, midx, sidx);  
  60. 60.        sidx = align_idx(bdata, sidx, step);  
  61. 61.        eidx = sidx + PFN_UP(size);/*结束位置*/  
  62. 62.  
  63. 63.        if (sidx >= midx || eidx > midx)/*找到结束了*/  
  64. 64.            break;  
  65. 65.  
  66. 66.        for (i = sidx; i < eidx; i++)/*检查这段区域是否空闲*/  
  67. 67.            if (test_bit(i, bdata->node_bootmem_map)) {/*如果不是,将跳过这段继续查找*/  
  68. 68.                sidx = align_idx(bdata, i, step);  
  69. 69.                if (sidx == i)  
  70. 70.                    sidx += step;  
  71. 71.                goto find_block;  
  72. 72.            }  
  73. 73.  
  74. 74.        if (bdata->last_end_off & (PAGE_SIZE - 1) &&/*如果为相邻的页面,也就是说上次分配的页面和这次分配的开始页面为相邻的*/  
  75. 75.                PFN_DOWN(bdata->last_end_off) + 1 == sidx)  
  76. 76.            start_off = align_off(bdata, bdata->last_end_off, align);  
  77. 77.        else  
  78. 78.            start_off = PFN_PHYS(sidx);  
  79. 79.  
  80. 80.        /*merge==1表示上次结束和这次开始不在同一个页面上*/  
  81. 81.        merge = PFN_DOWN(start_off) < sidx;  
  82. 82.        end_off = start_off + size;  
  83. 83.        /*更新数据*/  
  84. 84.        bdata->last_end_off = end_off;  
  85. 85.        bdata->hint_idx = PFN_UP(end_off);  
  86. 86.  
  87. 87.        /*
  88. 88.         * Reserve the area now:
  89. 89.         */  
  90. 90.         /*设定新加入的页面为保留,就是将对应的映射位置1*/  
  91. 91.        if (__reserve(bdata, PFN_DOWN(start_off) + merge,  
  92. 92.                PFN_UP(end_off), BOOTMEM_EXCLUSIVE))  
  93. 93.            BUG();  
  94. 94.        /*对应开始地址的虚拟地址返回*/  
  95. 95.        region = phys_to_virt(PFN_PHYS(bdata->node_min_pfn) +  
  96. 96.                start_off);  
  97. 97.        memset(region, 0, size);/*分配的大小*/  
  98. 98.        /*
  99. 99.         * The min_count is set to 0 so that bootmem allocated blocks
  100. 100.         * are never reported as leaks.
  101. 101.         */  
  102. 102.         /*调试用*/  
  103. 103.        kmemleak_alloc(region, size, 0, 0);  
  104. 104.        return region;  
  105. 105.    }  
  106. 106.  
  107. 107.    if (fallback) {/*回退,重新查看*/  
  108. 108.        sidx = align_idx(bdata, fallback - 1, step);  
  109. 109.        fallback = 0;  
  110. 110.        goto find_block;  
  111. 111.    }  
  112. 112.  
  113. 113.    return NULL;  
  114. 114.}  
复制代码

论坛徽章:
0
2 [报告]
发表于 2012-01-05 20:25 |只看该作者
谢谢分享
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP