From: Christoph Lameter Subject: Slub: No need for slab counters if !SLUB_DEBUG Counters are used mainly for showing data through the sysfs API. If that API is not compiled in then there is no point in keeping track of this data. Disable counters for the number of slabs and the number of total slabs if !SLUB_DEBUG. This also affects SLABINFO support. It now must depends on SLUB_DEBUG (which is on by default). Signed-off-by: Christoph Lameter --- include/linux/slub_def.h | 4 ++-- init/Kconfig | 2 +- mm/slub.c | 13 ++++++++++++- 3 files changed, 15 insertions(+), 4 deletions(-) Index: linux-2.6/mm/slub.c =================================================================== --- linux-2.6.orig/mm/slub.c 2008-03-26 12:09:48.610417784 -0700 +++ linux-2.6/mm/slub.c 2008-03-26 12:10:39.937436933 -0700 @@ -1118,10 +1118,12 @@ static struct page *new_slab(struct kmem goto out; n = get_node(s, page_to_nid(page)); +#ifdef CONFIG_SLUB_DEBUG if (n) { atomic_long_inc(&n->nr_slabs); atomic_long_add(page->objects, &n->total_objects); } +#endif page->slab = s; page->flags |= 1 << PG_slab; if (s->flags & (SLAB_DEBUG_FREE | SLAB_RED_ZONE | SLAB_POISON | @@ -1196,10 +1198,12 @@ static void free_slab(struct kmem_cache static void discard_slab(struct kmem_cache *s, struct page *page) { +#ifdef CONFIG_SLUB_DEBUG struct kmem_cache_node *n = get_node(s, page_to_nid(page)); atomic_long_dec(&n->nr_slabs); atomic_long_sub(page->objects, &n->total_objects); +#endif free_slab(s, page); } @@ -1916,10 +1920,11 @@ static void init_kmem_cache_cpu(struct k static void init_kmem_cache_node(struct kmem_cache_node *n) { n->nr_partial = 0; - atomic_long_set(&n->nr_slabs, 0); spin_lock_init(&n->list_lock); INIT_LIST_HEAD(&n->partial); #ifdef CONFIG_SLUB_DEBUG + atomic_long_set(&n->nr_slabs, 0); + atomic_long_set(&n->total_objects, 0); INIT_LIST_HEAD(&n->full); #endif } @@ -2088,7 +2093,9 @@ static struct kmem_cache_node *early_kme init_tracking(kmalloc_caches, n); #endif init_kmem_cache_node(n); +#ifdef CONFIG_SLUB_DEBUG atomic_long_inc(&n->nr_slabs); +#endif /* * lockdep requires consistent irq usage for each lock @@ -2395,8 +2402,10 @@ static inline int kmem_cache_close(struc struct kmem_cache_node *n = get_node(s, node); n->nr_partial -= free_list(s, n, &n->partial); +#ifdef CONFIG_SLUB_DEBUG if (atomic_long_read(&n->nr_slabs)) return 1; +#endif } free_kmem_cache_nodes(s); return 0; @@ -2846,6 +2855,7 @@ static void slab_mem_offline_callback(vo list_for_each_entry(s, &slab_caches, list) { n = get_node(s, offline_node); if (n) { +#ifdef CONFIG_SLUB_DEBUG /* * if n->nr_slabs > 0, slabs still exist on the node * that is going down. We were unable to free them, @@ -2853,6 +2863,7 @@ static void slab_mem_offline_callback(vo * callback. So, we must fail. */ BUG_ON(atomic_long_read(&n->nr_slabs)); +#endif s->node[offline_node] = NULL; kmem_cache_free(kmalloc_caches, n); Index: linux-2.6/include/linux/slub_def.h =================================================================== --- linux-2.6.orig/include/linux/slub_def.h 2008-03-26 12:09:48.640417391 -0700 +++ linux-2.6/include/linux/slub_def.h 2008-03-26 12:10:39.949178700 -0700 @@ -46,10 +46,10 @@ struct kmem_cache_cpu { struct kmem_cache_node { spinlock_t list_lock; /* Protect partial list and nr_partial */ unsigned long nr_partial; - atomic_long_t nr_slabs; - atomic_long_t total_objects; struct list_head partial; #ifdef CONFIG_SLUB_DEBUG + atomic_long_t nr_slabs; + atomic_long_t total_objects; struct list_head full; #endif }; Index: linux-2.6/init/Kconfig =================================================================== --- linux-2.6.orig/init/Kconfig 2008-03-26 12:09:48.650417272 -0700 +++ linux-2.6/init/Kconfig 2008-03-26 12:10:39.959179016 -0700 @@ -763,7 +763,7 @@ endmenu # General setup config SLABINFO bool depends on PROC_FS - depends on SLAB || SLUB + depends on SLAB || (SLUB && SLUB_DEBUG) default y config RT_MUTEXES