aboutsummaryrefslogtreecommitdiffstats
path: root/mm
diff options
context:
space:
mode:
Diffstat (limited to 'mm')
-rw-r--r--mm/slub.c76
1 files changed, 51 insertions, 25 deletions
diff --git a/mm/slub.c b/mm/slub.c
index 67f7d6068934..0a220df5ed7c 100644
--- a/mm/slub.c
+++ b/mm/slub.c
@@ -341,6 +341,26 @@ static inline int slab_index(void *p, struct kmem_cache *s, void *addr)
341 return (p - addr) / s->size; 341 return (p - addr) / s->size;
342} 342}
343 343
344static inline struct kmem_cache_order_objects oo_make(int order,
345 unsigned long size)
346{
347 struct kmem_cache_order_objects x = {
348 (order << 16) + (PAGE_SIZE << order) / size
349 };
350
351 return x;
352}
353
354static inline int oo_order(struct kmem_cache_order_objects x)
355{
356 return x.x >> 16;
357}
358
359static inline int oo_objects(struct kmem_cache_order_objects x)
360{
361 return x.x & ((1 << 16) - 1);
362}
363
344#ifdef CONFIG_SLUB_DEBUG 364#ifdef CONFIG_SLUB_DEBUG
345/* 365/*
346 * Debug settings: 366 * Debug settings:
@@ -665,7 +685,7 @@ static int slab_pad_check(struct kmem_cache *s, struct page *page)
665 return 1; 685 return 1;
666 686
667 start = page_address(page); 687 start = page_address(page);
668 length = (PAGE_SIZE << s->order); 688 length = (PAGE_SIZE << compound_order(page));
669 end = start + length; 689 end = start + length;
670 remainder = length % s->size; 690 remainder = length % s->size;
671 if (!remainder) 691 if (!remainder)
@@ -1090,19 +1110,21 @@ static inline void dec_slabs_node(struct kmem_cache *s, int node) {}
1090static struct page *allocate_slab(struct kmem_cache *s, gfp_t flags, int node) 1110static struct page *allocate_slab(struct kmem_cache *s, gfp_t flags, int node)
1091{ 1111{
1092 struct page *page; 1112 struct page *page;
1093 int pages = 1 << s->order; 1113 struct kmem_cache_order_objects oo = s->oo;
1114 int order = oo_order(oo);
1115 int pages = 1 << order;
1094 1116
1095 flags |= s->allocflags; 1117 flags |= s->allocflags;
1096 1118
1097 if (node == -1) 1119 if (node == -1)
1098 page = alloc_pages(flags, s->order); 1120 page = alloc_pages(flags, order);
1099 else 1121 else
1100 page = alloc_pages_node(node, flags, s->order); 1122 page = alloc_pages_node(node, flags, order);
1101 1123
1102 if (!page) 1124 if (!page)
1103 return NULL; 1125 return NULL;
1104 1126
1105 page->objects = s->objects; 1127 page->objects = oo_objects(oo);
1106 mod_zone_page_state(page_zone(page), 1128 mod_zone_page_state(page_zone(page),
1107 (s->flags & SLAB_RECLAIM_ACCOUNT) ? 1129 (s->flags & SLAB_RECLAIM_ACCOUNT) ?
1108 NR_SLAB_RECLAIMABLE : NR_SLAB_UNRECLAIMABLE, 1130 NR_SLAB_RECLAIMABLE : NR_SLAB_UNRECLAIMABLE,
@@ -1143,7 +1165,7 @@ static struct page *new_slab(struct kmem_cache *s, gfp_t flags, int node)
1143 start = page_address(page); 1165 start = page_address(page);
1144 1166
1145 if (unlikely(s->flags & SLAB_POISON)) 1167 if (unlikely(s->flags & SLAB_POISON))
1146 memset(start, POISON_INUSE, PAGE_SIZE << s->order); 1168 memset(start, POISON_INUSE, PAGE_SIZE << compound_order(page));
1147 1169
1148 last = start; 1170 last = start;
1149 for_each_object(p, s, start, page->objects) { 1171 for_each_object(p, s, start, page->objects) {
@@ -1162,7 +1184,8 @@ out:
1162 1184
1163static void __free_slab(struct kmem_cache *s, struct page *page) 1185static void __free_slab(struct kmem_cache *s, struct page *page)
1164{ 1186{
1165 int pages = 1 << s->order; 1187 int order = compound_order(page);
1188 int pages = 1 << order;
1166 1189
1167 if (unlikely(SlabDebug(page))) { 1190 if (unlikely(SlabDebug(page))) {
1168 void *p; 1191 void *p;
@@ -1181,7 +1204,7 @@ static void __free_slab(struct kmem_cache *s, struct page *page)
1181 1204
1182 __ClearPageSlab(page); 1205 __ClearPageSlab(page);
1183 reset_page_mapcount(page); 1206 reset_page_mapcount(page);
1184 __free_pages(page, s->order); 1207 __free_pages(page, order);
1185} 1208}
1186 1209
1187static void rcu_free_slab(struct rcu_head *h) 1210static void rcu_free_slab(struct rcu_head *h)
@@ -2202,6 +2225,7 @@ static int calculate_sizes(struct kmem_cache *s)
2202 unsigned long flags = s->flags; 2225 unsigned long flags = s->flags;
2203 unsigned long size = s->objsize; 2226 unsigned long size = s->objsize;
2204 unsigned long align = s->align; 2227 unsigned long align = s->align;
2228 int order;
2205 2229
2206 /* 2230 /*
2207 * Round up object size to the next word boundary. We can only 2231 * Round up object size to the next word boundary. We can only
@@ -2294,17 +2318,17 @@ static int calculate_sizes(struct kmem_cache *s)
2294 * page allocator order 0 allocs so take a reasonably large 2318 * page allocator order 0 allocs so take a reasonably large
2295 * order that will allows us a good number of objects. 2319 * order that will allows us a good number of objects.
2296 */ 2320 */
2297 s->order = max(slub_max_order, PAGE_ALLOC_COSTLY_ORDER); 2321 order = max(slub_max_order, PAGE_ALLOC_COSTLY_ORDER);
2298 s->flags |= __PAGE_ALLOC_FALLBACK; 2322 s->flags |= __PAGE_ALLOC_FALLBACK;
2299 s->allocflags |= __GFP_NOWARN; 2323 s->allocflags |= __GFP_NOWARN;
2300 } else 2324 } else
2301 s->order = calculate_order(size); 2325 order = calculate_order(size);
2302 2326
2303 if (s->order < 0) 2327 if (order < 0)
2304 return 0; 2328 return 0;
2305 2329
2306 s->allocflags = 0; 2330 s->allocflags = 0;
2307 if (s->order) 2331 if (order)
2308 s->allocflags |= __GFP_COMP; 2332 s->allocflags |= __GFP_COMP;
2309 2333
2310 if (s->flags & SLAB_CACHE_DMA) 2334 if (s->flags & SLAB_CACHE_DMA)
@@ -2316,9 +2340,9 @@ static int calculate_sizes(struct kmem_cache *s)
2316 /* 2340 /*
2317 * Determine the number of objects per slab 2341 * Determine the number of objects per slab
2318 */ 2342 */
2319 s->objects = (PAGE_SIZE << s->order) / size; 2343 s->oo = oo_make(order, size);
2320 2344
2321 return !!s->objects; 2345 return !!oo_objects(s->oo);
2322 2346
2323} 2347}
2324 2348
@@ -2351,7 +2375,7 @@ error:
2351 if (flags & SLAB_PANIC) 2375 if (flags & SLAB_PANIC)
2352 panic("Cannot create slab %s size=%lu realsize=%u " 2376 panic("Cannot create slab %s size=%lu realsize=%u "
2353 "order=%u offset=%u flags=%lx\n", 2377 "order=%u offset=%u flags=%lx\n",
2354 s->name, (unsigned long)size, s->size, s->order, 2378 s->name, (unsigned long)size, s->size, oo_order(s->oo),
2355 s->offset, flags); 2379 s->offset, flags);
2356 return 0; 2380 return 0;
2357} 2381}
@@ -2789,8 +2813,9 @@ int kmem_cache_shrink(struct kmem_cache *s)
2789 struct kmem_cache_node *n; 2813 struct kmem_cache_node *n;
2790 struct page *page; 2814 struct page *page;
2791 struct page *t; 2815 struct page *t;
2816 int objects = oo_objects(s->oo);
2792 struct list_head *slabs_by_inuse = 2817 struct list_head *slabs_by_inuse =
2793 kmalloc(sizeof(struct list_head) * s->objects, GFP_KERNEL); 2818 kmalloc(sizeof(struct list_head) * objects, GFP_KERNEL);
2794 unsigned long flags; 2819 unsigned long flags;
2795 2820
2796 if (!slabs_by_inuse) 2821 if (!slabs_by_inuse)
@@ -2803,7 +2828,7 @@ int kmem_cache_shrink(struct kmem_cache *s)
2803 if (!n->nr_partial) 2828 if (!n->nr_partial)
2804 continue; 2829 continue;
2805 2830
2806 for (i = 0; i < s->objects; i++) 2831 for (i = 0; i < objects; i++)
2807 INIT_LIST_HEAD(slabs_by_inuse + i); 2832 INIT_LIST_HEAD(slabs_by_inuse + i);
2808 2833
2809 spin_lock_irqsave(&n->list_lock, flags); 2834 spin_lock_irqsave(&n->list_lock, flags);
@@ -2835,7 +2860,7 @@ int kmem_cache_shrink(struct kmem_cache *s)
2835 * Rebuild the partial list with the slabs filled up most 2860 * Rebuild the partial list with the slabs filled up most
2836 * first and the least used slabs at the end. 2861 * first and the least used slabs at the end.
2837 */ 2862 */
2838 for (i = s->objects - 1; i >= 0; i--) 2863 for (i = objects - 1; i >= 0; i--)
2839 list_splice(slabs_by_inuse + i, n->partial.prev); 2864 list_splice(slabs_by_inuse + i, n->partial.prev);
2840 2865
2841 spin_unlock_irqrestore(&n->list_lock, flags); 2866 spin_unlock_irqrestore(&n->list_lock, flags);
@@ -3351,7 +3376,7 @@ static long validate_slab_cache(struct kmem_cache *s)
3351{ 3376{
3352 int node; 3377 int node;
3353 unsigned long count = 0; 3378 unsigned long count = 0;
3354 unsigned long *map = kmalloc(BITS_TO_LONGS(s->objects) * 3379 unsigned long *map = kmalloc(BITS_TO_LONGS(oo_objects(s->oo)) *
3355 sizeof(unsigned long), GFP_KERNEL); 3380 sizeof(unsigned long), GFP_KERNEL);
3356 3381
3357 if (!map) 3382 if (!map)
@@ -3719,7 +3744,7 @@ static ssize_t show_slab_objects(struct kmem_cache *s,
3719 - n->nr_partial; 3744 - n->nr_partial;
3720 3745
3721 if (flags & SO_OBJECTS) 3746 if (flags & SO_OBJECTS)
3722 x = full_slabs * s->objects; 3747 x = full_slabs * oo_objects(s->oo);
3723 else 3748 else
3724 x = full_slabs; 3749 x = full_slabs;
3725 total += x; 3750 total += x;
@@ -3798,13 +3823,13 @@ SLAB_ATTR_RO(object_size);
3798 3823
3799static ssize_t objs_per_slab_show(struct kmem_cache *s, char *buf) 3824static ssize_t objs_per_slab_show(struct kmem_cache *s, char *buf)
3800{ 3825{
3801 return sprintf(buf, "%d\n", s->objects); 3826 return sprintf(buf, "%d\n", oo_objects(s->oo));
3802} 3827}
3803SLAB_ATTR_RO(objs_per_slab); 3828SLAB_ATTR_RO(objs_per_slab);
3804 3829
3805static ssize_t order_show(struct kmem_cache *s, char *buf) 3830static ssize_t order_show(struct kmem_cache *s, char *buf)
3806{ 3831{
3807 return sprintf(buf, "%d\n", s->order); 3832 return sprintf(buf, "%d\n", oo_order(s->oo));
3808} 3833}
3809SLAB_ATTR_RO(order); 3834SLAB_ATTR_RO(order);
3810 3835
@@ -4451,11 +4476,12 @@ static int s_show(struct seq_file *m, void *p)
4451 nr_inuse += count_partial(n); 4476 nr_inuse += count_partial(n);
4452 } 4477 }
4453 4478
4454 nr_objs = nr_slabs * s->objects; 4479 nr_objs = nr_slabs * oo_objects(s->oo);
4455 nr_inuse += (nr_slabs - nr_partials) * s->objects; 4480 nr_inuse += (nr_slabs - nr_partials) * oo_objects(s->oo);
4456 4481
4457 seq_printf(m, "%-17s %6lu %6lu %6u %4u %4d", s->name, nr_inuse, 4482 seq_printf(m, "%-17s %6lu %6lu %6u %4u %4d", s->name, nr_inuse,
4458 nr_objs, s->size, s->objects, (1 << s->order)); 4483 nr_objs, s->size, oo_objects(s->oo),
4484 (1 << oo_order(s->oo)));
4459 seq_printf(m, " : tunables %4u %4u %4u", 0, 0, 0); 4485 seq_printf(m, " : tunables %4u %4u %4u", 0, 0, 0);
4460 seq_printf(m, " : slabdata %6lu %6lu %6lu", nr_slabs, nr_slabs, 4486 seq_printf(m, " : slabdata %6lu %6lu %6lu", nr_slabs, nr_slabs,
4461 0UL); 4487 0UL);