码迷,mamicode.com
首页 > 其他好文 > 详细

Kernel那些事儿之内存管理(8) --- Slab(中)

时间:2015-07-16 22:31:02      阅读:273      评论:0      收藏:0      [点我收藏+]

标签:内存管理   linux kernel   

上篇讲了Slab中的数据结构,这篇该讲Slab中的操作了。


既然是内存管理,那操作无非就两点:allocate 和 free。


1. 申请一个object

在Slab中,申请一个object是通过函数 kmem_cache_alloc() 来完成的。

3618 void *kmem_cache_alloc(struct kmem_cache *cachep, gfp_t flags)
3619 {
3620     return __cache_alloc(cachep, flags, __builtin_return_address(0));
3621 }

该函数有两个参数:

  • cachep指定了要从哪个cache中申请object.

  • flags: 如果该cache中所有的slab都已经满了,那就不得不向buddy system求救。flags指定了此时从buddy system中申请page frame时要使用的标志。


其实函数 kmem_cache_alloc() 只是一个wapper,经过层层调用,除去必要的锁操作,最终真正干活的是函数 ____cache_alloc()。

3190 static inline void *____cache_alloc(struct kmem_cache *cachep, gfp_t flags)
3191 {
3192     void *objp;
3193     struct array_cache *ac;
3194
3195     check_irq_off();
3196
3197     ac = cpu_cache_get(cachep);
3198     if (likely(ac->avail)) {
3199         STATS_INC_ALLOCHIT(cachep);
3200         ac->touched = 1;
3201         objp = ac->entry[--ac->avail];
3202     } else {
3203         STATS_INC_ALLOCMISS(cachep);
3204         objp = cache_alloc_refill(cachep, flags);
3205     }
3206     return objp;
3207 }

该函数看起来唬人,其实其武功招式总结起来就三招,人称“分配三板斧”。


第一板斧:per-cpu object cache.

3197 ~ 3201: 首先找到当前CPU的object cache,然后检查cache中有没有可用的object,有的话就拿出一个来。然后完活收工。


定位object cache是通过以下函数完成的:

 760 static inline struct array_cache *cpu_cache_get(struct kmem_cache *cachep)
 761 {
 762     return cachep->array[smp_processor_id()];
 763 }


上篇讲过,object cache实际上是一个LIFO的栈结构,其中成员变量 avail 即代表了可用object的个数,又作为索引值指出了栈顶位置。

所以从object cache中拿出一个object的操作,可以通过 3201 行的语句来完成。

3201         objp = ac->entry[--ac->avail];


第二板斧:Slab中的空闲object

如果object cache中没有可用的object,那就要先填充object cache。如果不考虑 shared object cache,填充cache的object来自于Slab页面。

该操作由函数 cache_alloc_refill() 完成。

2957 static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags)
2958 {
2959     int batchcount;
2960     struct kmem_list3 *l3;
2961     struct array_cache *ac;
2962     int node;
2963
2964     node = numa_node_id();
2965
2966     check_irq_off();
2967     ac = cpu_cache_get(cachep);
2968 retry:
2969     batchcount = ac->batchcount;

2978     l3 = cachep->nodelists[node];
2979
2980     BUG_ON(ac->avail > 0 || !l3);
2981     spin_lock(&l3->list_lock);
2982
2983     /* See if we can refill from the shared array */
2984     if (l3->shared && transfer_objects(ac, l3->shared, batchcount))
2985         goto alloc_done;
2986
2987     while (batchcount > 0) {
2988         struct list_head *entry;
2989         struct slab *slabp;
2990         /* Get slab alloc is to come from. */
2991         entry = l3->slabs_partial.next;
2992         if (entry == &l3->slabs_partial) {
2993             l3->free_touched = 1;
2994             entry = l3->slabs_free.next;
2995             if (entry == &l3->slabs_free)
2996                 goto must_grow;
2997         }
2998
2999         slabp = list_entry(entry, struct slab, list);
3000         check_slabp(cachep, slabp);
3001         check_spinlock_acquired(cachep);
3002
3003         /*
3004          * The slab was either on partial or free list so
3005          * there must be at least one object available for
3006          * allocation.
3007          */
3008         BUG_ON(slabp->inuse < 0 || slabp->inuse >= cachep->num);
3009
3010         while (slabp->inuse < cachep->num && batchcount--) {
3011             STATS_INC_ALLOCED(cachep);
3012             STATS_INC_ACTIVE(cachep);
3013             STATS_SET_HIGH(cachep);
3014
3015             ac->entry[ac->avail++] = slab_get_obj(cachep, slabp,
3016                                 node);
3017         }
3018         check_slabp(cachep, slabp);
3019
3020         /* move slabp to correct slabp list: */
3021         list_del(&slabp->list);
3022         if (slabp->free == BUFCTL_END)
3023             list_add(&slabp->list, &l3->slabs_full);
3024         else
3025             list_add(&slabp->list, &l3->slabs_partial);
3026     }
3027
3028 must_grow:
3029     l3->free_objects -= ac->avail;
3030 alloc_done:
3031     spin_unlock(&l3->list_lock);
3032
3033     if (unlikely(!ac->avail)) {
3034         int x;
3035         x = cache_grow(cachep, flags | GFP_THISNODE, node, NULL);
3036
3037         /* cache_grow can reenable interrupts, then ac could change. */
3038         ac = cpu_cache_get(cachep);
3039         if (!x && ac->avail == 0)   /* no objects in sight? abort */
3040             return NULL;
3041
3042         if (!ac->avail)     /* objects refilled by interrupt? */
3043             goto retry;
3044     }
3045     ac->touched = 1;
3046     return ac->entry[--ac->avail];
3047 }

2984行先检查一下能否从shared object cache中拿一些object填充到我们的object cache中。该操作其实就是一个memory copy的过程。


2987 ~ 3026 是从Slab页面中填充object cache的操作。

该操作首先遍历所有partial的slab,然后再遍历所有free的slab。对每个slab,拿出其中的free objects,填充到object cache中。

从slab中拿出一个free object的操作由函数 slab_get_obj() 完成。

2695 static void *slab_get_obj(struct kmem_cache *cachep, struct slab *slabp,
2696                 int nodeid)
2697 {
2698     void *objp = index_to_obj(cachep, slabp, slabp->free);
2699     kmem_bufctl_t next;
2700
2701     slabp->inuse++;
2702     next = slab_bufctl(slabp)[slabp->free];

2707     slabp->free = next;
2708
2709     return objp;
2710 }

我们上篇讲过,slab中所有空闲的object 的描述符组成了一个简单的链表。该函数从这个链表中取下头部的一个空闲object。


把拿出的空闲object填充到object cache的操作是由3015行完成的。

3015             ac->entry[ac->avail++] = slab_get_obj(cachep, slabp,
3016                                 node);


第三板斧:buddy system

如果所有的slab都满了,即每个slab都没有空闲object了,那怎么办?这个时候就得求助于buddy system了。

从buddy system中申请新的page frame,创建新的slab,从而创建出一批新的object。该操作由函数 cache_grow() 完成。


2760 static int cache_grow(struct kmem_cache *cachep,
2761         gfp_t flags, int nodeid, void *objp)
2762 {
2763     struct slab *slabp;
2764     size_t offset;
2765     gfp_t local_flags;
2766     struct kmem_list3 *l3;
2767
2768     /*
2769      * Be lazy and only check for valid flags here,  keeping it out of the
2770      * critical path in kmem_cache_alloc().
2771      */
2772     BUG_ON(flags & GFP_SLAB_BUG_MASK);
2773     local_flags = flags & (GFP_CONSTRAINT_MASK|GFP_RECLAIM_MASK);
2774
2775     /* Take the l3 list lock to change the colour_next on this node */
2776     check_irq_off();
2777     l3 = cachep->nodelists[nodeid];
2778     spin_lock(&l3->list_lock);
2779
2780     /* Get colour for the slab, and cal the next value. */
2781     offset = l3->colour_next;
2782     l3->colour_next++;
2783     if (l3->colour_next >= cachep->colour)
2784         l3->colour_next = 0;
2785     spin_unlock(&l3->list_lock);
2786
2787     offset *= cachep->colour_off;

2780 ~ 2787,该函数首先确定新建slab的color值。

上篇讲过,结构体kmem_list3的成员变量colour_next,保存了下一个要使用的color值。cache描述符中的colour,保存了可用的color 数。cache描述符中的colour_off, 保存了对齐字节数。

2781 ~ 2784行把colour_next的值赋值给变量offset,然后更新colour_next的值。如果colour_next的值达到了最大值,则重置为0。

2787行用color值乘以字节对齐数,算出来的就是slab开始位置的偏移量。


确定了color值后,下一步就是要向buddy system申请page frame了。

2804     if (!objp)
2805         objp = kmem_getpages(cachep, local_flags, nodeid);
2806     if (!objp)
2807         goto failed;


为新slab申请page frame的操作由函数 kmem_getpages() 完成。

1653 static void *kmem_getpages(struct kmem_cache *cachep, gfp_t flags, int nodeid)
1654 {
1655     struct page *page;
1656     int nr_pages;
1657     int i;

1667     flags |= cachep->gfpflags;
1668     if (cachep->flags & SLAB_RECLAIM_ACCOUNT)
1669         flags |= __GFP_RECLAIMABLE;
1670
1671     page = alloc_pages_node(nodeid, flags, cachep->gfporder);
1672     if (!page)
1673         return NULL;
1674
1675     nr_pages = (1 << cachep->gfporder);

1682     for (i = 0; i < nr_pages; i++)
1683         __SetPageSlab(page + i);
1684     return page_address(page);
1685 }

函数 kmem_getpages() 首先确定分配flags。至此,调用 kmem_cache_alloc() 时传递的参数 flags 终于被用到了。但是这个flags还要再加上cache本身的flag,才会被用来申请page frame。

然后1671行调用了我们前面讲的alloc_pages函数,从buddy system中申请page frame。申请的page frame order为cachep->gfporder。

1682 ~ 1683行,为每个申请到的page frame的描述符设置 PG_slab 标志。

最后返回第一个page frame的线性地址。


好,申请到page frame了,那下一步做什么?让我们回到函数 cache_grow() 继续跟踪。

2810     slabp = alloc_slabmgmt(cachep, objp, offset,
2811             local_flags & ~GFP_CONSTRAINT_MASK, nodeid);
2812     if (!slabp)
2813         goto opps1;

2810 ~ 2813为新建的slab 分配并初始化slab描述符,由函数 alloc_slabmgmt() 完成。

2609 static struct slab *alloc_slabmgmt(struct kmem_cache *cachep, void *objp,
2610                    int colour_off, gfp_t local_flags,
2611                    int nodeid)
2612 {
2613     struct slab *slabp;
2614
2615     if (OFF_SLAB(cachep)) {
2616         /* Slab management obj is off-slab. */
2617         slabp = kmem_cache_alloc_node(cachep->slabp_cache,
2618                           local_flags & ~GFP_THISNODE, nodeid);
2619         if (!slabp)
2620             return NULL;
2621     } else {
2622         slabp = objp + colour_off;
2623         colour_off += cachep->slab_size;
2624     }
2625     slabp->inuse = 0;
2626     slabp->colouroff = colour_off;
2627     slabp->s_mem = objp + colour_off;
2628     slabp->nodeid = nodeid;
2629     return slabp;
2630 }

2615 ~ 2620:针对external slab,从cachep->slabp_cache中分配slab描述符(以及object描述符)。

2622 ~ 2623:针对internal slab,在slab页面内指定slab描述符的地址。考虑进去slab coloring, slab描述符放在 (color*aln) 位置处。


2625 ~ 2628:初始化新建的slab描述符。正如上篇讲到的,第一个object的偏移量定义为(color * aln) + dsize。


有了slab描述符后,让我们回到函数 cache_grow() 继续跟踪。

2816     slab_map_pages(cachep, slabp, objp);
2817
2818     cache_init_objs(cachep, slabp);
2819


2816行建立page frame与cache和slab描述符之间的关联。

2737 static void slab_map_pages(struct kmem_cache *cache, struct slab *slab,
2738                void *addr)
2739 {
2740     int nr_pages;
2741     struct page *page;
2742
2743     page = virt_to_page(addr);
2744
2745     nr_pages = 1;
2746     if (likely(!PageCompound(page)))
2747         nr_pages <<= cache->gfporder;
2748
2749     do {
2750         page_set_cache(page, cache);
2751         page_set_slab(page, slab);
2752         page++;
2753     } while (--nr_pages);
2754 }

建立关联的方式就是重用了page描述符中的成员变量lru。

 583 static inline void page_set_cache(struct page *page, struct kmem_cache *cache)
 584 {
 585     page->lru.next = (struct list_head *)cache;
 586 }


 595 static inline void page_set_slab(struct page *page, struct slab *slab)
 596 {
 597     page->lru.prev = (struct list_head *)slab;
 598 }


2818行初始化新建slab中的objects.

2637 static void cache_init_objs(struct kmem_cache *cachep,
2638                 struct slab *slabp)
2639 {
2640     int i;
2641
2642     for (i = 0; i < cachep->num; i++) {
2643         void *objp = index_to_obj(cachep, slabp, i);

2676         if (cachep->ctor)
2677             cachep->ctor(cachep, objp);

2679         slab_bufctl(slabp)[i] = i + 1;
2680     }
2681     slab_bufctl(slabp)[i - 1] = BUFCTL_END;
2682     slabp->free = 0;
2683 }

这个初始化做了两件事:

  1. 如果该cache定义了构造函数,则对每个新建的object执行构造函数。

  2. 初始化object描述符,构造出空闲object链表。


建立了关联,初始化了object,继续跟踪 cache_grow()。

2823     spin_lock(&l3->list_lock);
2824
2825     /* Make slab active. */
2826     list_add_tail(&slabp->list, &(l3->slabs_free));
2827     STATS_INC_GROWN(cachep);
2828     l3->free_objects += cachep->num;
2829     spin_unlock(&l3->list_lock);
2830     return 1;
2831 opps1:
2832     kmem_freepages(cachep, objp);
2833 failed:
2834     if (local_flags & __GFP_WAIT)
2835         local_irq_disable();
2836     return 0;
2837 }

把新建的slab添加到cache的free slab链表中,更新kmem_list3中free_objects的值,然后完活收工。


第三板斧使出来之后,该cache新增了一slab的free object。于是重新使用第二板斧。


2. 释放一个object

释放一个object的API是函数 kmem_cache_free(). 

3760 void kmem_cache_free(struct kmem_cache *cachep, void *objp)
3761 {
3762     unsigned long flags;
3763
3764     local_irq_save(flags);
3765     debug_check_no_locks_freed(objp, obj_size(cachep));
3766     __cache_free(cachep, objp);
3767     local_irq_restore(flags);
3768 }


该函数也是一个wrapper,最终干活的是函数  __cache_free()。

3582 static inline void __cache_free(struct kmem_cache *cachep, void *objp)
3583 {
3584     struct array_cache *ac = cpu_cache_get(cachep);

3598
3599     if (likely(ac->avail < ac->limit)) {
3600         STATS_INC_FREEHIT(cachep);
3601         ac->entry[ac->avail++] = objp;
3602         return;
3603     } else {
3604         STATS_INC_FREEMISS(cachep);
3605         cache_flusharray(cachep, ac);
3606         ac->entry[ac->avail++] = objp;
3607     }
3608 }

相对应的,该函数也是三板斧,人称“释放三板斧”,而且这三板斧与“申请三板斧”一一对应,真是一对生死冤家。


第一板斧:per-cpu object cache

3599 ~ 3602,如果当前CPU的object cache没有满,则直接把要释放的object放到object cache中,然后完活收工。


第二板斧:Slab页面

如果object cache满了,则要刷一批object到slab页面中。该操作由函数 cache_flusharray() 完成。

3527 static void cache_flusharray(struct kmem_cache *cachep, struct array_cache *ac)
3528 {
3529     int batchcount;
3530     struct kmem_list3 *l3;
3531     int node = numa_node_id();
3532
3533     batchcount = ac->batchcount;

3537     check_irq_off();
3538     l3 = cachep->nodelists[node];
3539     spin_lock(&l3->list_lock);
3540     if (l3->shared) {
3541         struct array_cache *shared_array = l3->shared;
3542         int max = shared_array->limit - shared_array->avail;
3543         if (max) {
3544             if (batchcount > max)
3545                 batchcount = max;
3546             memcpy(&(shared_array->entry[shared_array->avail]),
3547                    ac->entry, sizeof(void *) * batchcount);
3548             shared_array->avail += batchcount;
3549             goto free_done;
3550         }
3551     }
3552
3553     free_block(cachep, ac->entry, batchcount, node);
3554 free_done:

3573     spin_unlock(&l3->list_lock);
3574     ac->avail -= batchcount;
3575     memmove(ac->entry, &(ac->entry[batchcount]), sizeof(void *)*ac->avail);
3576 }

3540 ~ 3551:如果shared object cache没有满额,则转移一批object到shared cache中。

3553行:利用函数 free_block() 刷一批object到slab页面中。

3575行:把object cache中剩下的object上移。为了保证object cache是LIFO的结构,会把最早放入object cache的那些object刷掉,因此需要把剩下的object向上移动。


3483 static void free_block(struct kmem_cache *cachep, void **objpp, int nr_objects,
3484                int node)
3485 {
3486     int i;
3487     struct kmem_list3 *l3;
3488
3489     for (i = 0; i < nr_objects; i++) {
3490         void *objp = objpp[i];
3491         struct slab *slabp;
3492
3493         slabp = virt_to_slab(objp);
3494         l3 = cachep->nodelists[node];
3495         list_del(&slabp->list);
3496         check_spinlock_acquired_node(cachep, node);
3497         check_slabp(cachep, slabp);
3498         slab_put_obj(cachep, slabp, objp, node);
3499         STATS_DEC_ACTIVE(cachep);
3500         l3->free_objects++;
3501         check_slabp(cachep, slabp);
3502
3503         /* fixup slab chains */
3504         if (slabp->inuse == 0) {
3505             if (l3->free_objects > l3->free_limit) {
3506                 l3->free_objects -= cachep->num;
3507                 /* No need to drop any previously held
3508                  * lock here, even if we have a off-slab slab
3509                  * descriptor it is guaranteed to come from
3510                  * a different cache, refer to comments before
3511                  * alloc_slabmgmt.
3512                  */
3513                 slab_destroy(cachep, slabp);
3514             } else {
3515                 list_add(&slabp->list, &l3->slabs_free);
3516             }
3517         } else {
3518             /* Unconditionally move a slab to the end of the
3519              * partial list on free - maximum time for the
3520              * other objects to be freed, too.
3521              */
3522             list_add_tail(&slabp->list, &l3->slabs_partial);
3523         }
3524     }
3525 }


3493行根据object的地址,确定出slab描述符的地址。这里就用到我们在创建slab时建立的关联了。

 612 static inline struct slab *virt_to_slab(const void *obj)
 613 {
 614     struct page *page = virt_to_head_page(obj);
 615     return page_get_slab(page);
 616 }
 600 static inline struct slab *page_get_slab(struct page *page)
 601 {
 602     BUG_ON(!PageSlab(page));
 603     return (struct slab *)page->lru.prev;
 604 }


3498行把一个object放回到slab中。

2712 static void slab_put_obj(struct kmem_cache *cachep, struct slab *slabp,
2713                 void *objp, int nodeid)
2714 {
2715     unsigned int objnr = obj_to_index(cachep, slabp, objp);

2727     slab_bufctl(slabp)[objnr] = slabp->free;
2728     slabp->free = objnr;
2729     slabp->inuse--;
2730 }


3504 ~ 3523:放回一个object后,就要检查该slab的状态:

  1. partial:把slab挂到slabs_partial链表上。

  2. free:检查cache中free object的数量有没有超过限制。

    1. 没有超过:把slab挂到slabs_free链表中。

    2. 超过了:删掉该slab,把其page frame还给buddy system。


第三板斧:buddy system

删除一个slab的操作是由函数 slab_destroy() 完成的。

1944 static void slab_destroy(struct kmem_cache *cachep, struct slab *slabp)
1945 {
1946     void *addr = slabp->s_mem - slabp->colouroff;
1947

1957     kmem_freepages(cachep, addr);
1958     if (OFF_SLAB(cachep))
1959         kmem_cache_free(cachep->slabp_cache, slabp);

1961 }

该函数首先释放slab所用的page frame,然后对于external slab,把slab描述符占用的空间释放掉。


释放slab所用page frame的操作是由函数 kmem_freepages() 完成的。

1690 static void kmem_freepages(struct kmem_cache *cachep, void *addr)
1691 {
1692     unsigned long i = (1 << cachep->gfporder);
1693     struct page *page = virt_to_page(addr);

1702     while (i--) {
1703         BUG_ON(!PageSlab(page));
1704         __ClearPageSlab(page);
1705         page++;
1706     }

1709     free_pages((unsigned long)addr, cachep->gfporder);
1710 }

1702 ~ 1706:对slab所用的每个page frame,清除其描述符中的 PG_slab 标志。

1709:调用函数 free_pages() 把slab所用的page frame还给buddy system。


本文出自 “内核部落格” 博客,请务必保留此出处http://richardguo.blog.51cto.com/9343720/1675411

Kernel那些事儿之内存管理(8) --- Slab(中)

标签:内存管理   linux kernel   

原文地址:http://richardguo.blog.51cto.com/9343720/1675411

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!