内核提供了计算一个Slab缓存的大小的函数,Slab缓存的内存布局分成以下两种:
1. Slab缓存管理区在Slab内,即同对象存放在一起。
2. Slab缓存管理区存放在单独的一片区域,即同对象分别存放。
通过宏CFLGS_OFF_SLAB来区分。
参数介绍:
gfporder: Slab缓存大小为2^gfporder个页面
buffer_size: 每个对象的大小
align:对齐
flags:CFLGS_OFF_SLAB表示Slab管理区在缓存外,否则表示在缓存内。
left_over:回传参数,在一个Slab中被浪费的大小。
num: 回传参数,一个Slab中,对象的数量。
static void cache_estimate(unsigned long gfporder, size_t buffer_size,
size_t align, int flags, size_t *left_over,
unsigned int *num)
{
int nr_objs;
size_t mgmt_size;
计算Slab的大小
size_t slab_size = PAGE_SIZE << gfporder;
如果Slab管理区与Slab对象分开存放
if (flags & CFLGS_OFF_SLAB) {
mgmt_size = 0;
对象数量就为Slab总大小除以每个对象的大小
nr_objs = slab_size / buffer_size;
如果在一个Slab中对象数量,超过一定数量,将对象数重置为最大值
if (nr_objs > SLAB_LIMIT)
nr_objs = SLAB_LIMIT;
} else {
如果Slab管理区同对象存放在一起,那么对象数量就是Slab总大小减去
Slab占去的大小,然后除以每个对象占用的大小,包括每个对象都有一个
4字节大小的buffer用于记录对象在一个Slab中的偏移值。
nr_objs = (slab_size - sizeof(struct slab)) /
(buffer_size + sizeof(kmem_bufctl_t));
根据Slab的大小,调整Slab对象的数量
if (slab_mgmt_size(nr_objs, align) + nr_objs*buffer_size
> slab_size)
nr_objs--;
if (nr_objs > SLAB_LIMIT)
nr_objs = SLAB_LIMIT;
计算Slab管理区所暂用的大小
mgmt_size = slab_mgmt_size(nr_objs, align);
}
*num = nr_objs;
*left_over = slab_size - nr_objs*buffer_size - mgmt_size;
}