Chinaunix首页 | 论坛 | 博客
  • 博客访问: 233617
  • 博文数量: 37
  • 博客积分: 933
  • 博客等级: 军士长
  • 技术积分: 511
  • 用 户 组: 普通用户
  • 注册时间: 2011-03-16 10:15
文章分类
文章存档

2012年(1)

2011年(36)

分类: LINUX

2011-03-16 10:24:13

Linux的内存管理由最初启动时的bootmem进行管理,到后来才转化为Buddy system(伙伴系统)的。
下面是伙伴系统最为常用的两个操作。
1)分配块
2)释放块
 580 static struct page *__rmqueue(struct zone *zone, unsigned int order)
 581 {
 582         struct free_area * area;
 583         unsigned int current_order;
 584         struct page *page;
 585
 586         for (current_order = order; current_order < MAX_ORDER; ++current_order) {
 587                 area = zone->free_area + current_order;
 588                 if (list_empty(&area->free_list))
 589                         continue;
 590
 591                 page = list_entry(area->free_list.next, struct page, lru);
 592                 list_del(&page->lru);
 593                 rmv_page_order(page);
 594                 area->nr_free--;
 595                 zone->free_pages -= 1UL << order;
 596                 expand(zone, page, order, current_order, area);
 597                 return page;
 598         }
 599
 600         return NULL;
 601 }
这段代码还是很简单的,首先函数接受二个参数:
1)zone:指向要分配的块所在的管理区。确切的说是指向管理区描述符地址的指针;
2)order:要分配的块的阶数。e.g:order=2 就是分配连续内存数为4页的内存。order=3 8页。
内核定义了一系列的宏:
 #ifndef CONFIG_FORCE_MAX_ZONEORDER
  #define MAX_ORDER 11
  #else
  #define MAX_ORDER CONFIG_FORCE_MAX_ZONEORDER
  #endif
这段宏给我们提供了一个可以自己配置MAX_ORDER大小的选项,如果没有定义,MAX_ORDER=11
也就是说order的最大取值为11,更进一步的讲:内核可以分配的最大的连续内存页为2^10=1024页
1024*4K=4M。
函数体内执行了一个for循环,从给定order的链表开始向上查询。
如果这个链表为空list_empty(&area->free_list)那么继续下一次循环continue
如果这个链表不为空,也就是说此链表中具有空闲的2^order个连续页面,则执行下面的操作:
1) page = list_entry(area->free_list.next, struct page, lru);
获取这个链表的第一个实例,使page指向满足要求的连续内存页(这些页组成了块)的第一页
2) list_del(&page->lru);
从当前链表中删除这些页
3)rmv_page_order(page);
这个函数其实很简单:
 250 static inline void rmv_page_order(struct page *page)
     /*  */
 251 {
 252         __ClearPageBuddy(page);
 253         set_page_private(page, 0);
 254 }
清楚这个页的PG_buddy标志并设置这个页的private字段为0;
4) 594                 area->nr_free--;
    595                 zone->free_pages -= 1UL << order;
空闲块数减1,管理区所有空闲页面数减去1<5) expand(zone, page, order, current_order, area);这个函数负责完成空闲区域链表的重组
 515 static inline void expand(struct zone *zone, struct page *page,
 516         int low, int high, struct free_area *area)
 517 {
 518         unsigned long size = 1 << high;
 519
 520         while (high > low) {
 521                 area--;
 522                 high--;
 523                 size >>= 1;
 524                 BUG_ON(bad_range(zone, &page[size]));
 525                 list_add(&page[size].lru, &area->free_list);
 526                 area->nr_free++;
 527                 set_page_order(&page[size], high);
 528         }
 529 }
current order 大于order时,也就是说没有在请求块大小的链表中找到空闲块,而是在上一级的链表中找到
举个例子会更好的理解:
你请求的块大小为4个页面也就是说order=2 可是这时操作系统发现 4个连续页的块都用完了,这个就让你去order=3
(连续页面数为8的块)中去找,找到了,要进行拆分,把剩下的6个连续页面分为4个和2个,连入到对应的空闲区链表
中去。
 
阅读(3143) | 评论(0) | 转发(0) |
0

上一篇:没有了

下一篇:__alloc_pages 浅淡

给主人留下些什么吧!~~