Commit 130d4df5 authored by Vlastimil Babka's avatar Vlastimil Babka

mm/sl[au]b: rearrange struct slab fields to allow larger rcu_head

Joel reports [1] that increasing the rcu_head size for debugging
purposes used to work before struct slab was split from struct page, but
now runs into the various SLAB_MATCH() sanity checks of the layout.

This is because the rcu_head in struct page is in union with large
sub-structures and has space to grow without exceeding their size, while
in struct slab (for SLAB and SLUB) it's in union only with a list_head.

On closer inspection (and after the previous patch) we can put all
fields except slab_cache to a union with rcu_head, as slab_cache is
sufficient for the rcu freeing callbacks to work and the rest can be
overwritten by rcu_head without causing issues.

This is only somewhat complicated by the need to keep SLUB's
freelist+counters aligned for cmpxchg_double. As a result the fields
need to be reordered so that slab_cache is first (after page flags) and
the union with rcu_head follows. For consistency, do that for SLAB as
well, although not necessary there.

As a result, the rcu_head field in struct page and struct slab is no
longer at the same offset, but that doesn't matter as there is no
casting that would rely on that in the slab freeing callbacks, so we can
just drop the respective SLAB_MATCH() check.

Also we need to update the SLAB_MATCH() for compound_head to reflect the
new ordering.

While at it, also add a static_assert to check the alignment needed for
cmpxchg_double so mistakes are found sooner than a runtime GPF.

[1] https://lore.kernel.org/all/85afd876-d8bb-0804-b2c5-48ed3055e702@joelfernandes.org/Reported-by: default avatarJoel Fernandes <joel@joelfernandes.org>
Signed-off-by: default avatarVlastimil Babka <vbabka@suse.cz>
Acked-by: default avatarHyeonggon Yoo <42.hyeyoo@gmail.com>
parent 8b881763
...@@ -11,20 +11,24 @@ struct slab { ...@@ -11,20 +11,24 @@ struct slab {
#if defined(CONFIG_SLAB) #if defined(CONFIG_SLAB)
struct kmem_cache *slab_cache;
union { union {
struct {
struct list_head slab_list; struct list_head slab_list;
struct rcu_head rcu_head;
};
struct kmem_cache *slab_cache;
void *freelist; /* array of free object indexes */ void *freelist; /* array of free object indexes */
void *s_mem; /* first object */ void *s_mem; /* first object */
};
struct rcu_head rcu_head;
};
unsigned int active; unsigned int active;
#elif defined(CONFIG_SLUB) #elif defined(CONFIG_SLUB)
struct kmem_cache *slab_cache;
union {
struct {
union { union {
struct list_head slab_list; struct list_head slab_list;
struct rcu_head rcu_head;
#ifdef CONFIG_SLUB_CPU_PARTIAL #ifdef CONFIG_SLUB_CPU_PARTIAL
struct { struct {
struct slab *next; struct slab *next;
...@@ -32,7 +36,6 @@ struct slab { ...@@ -32,7 +36,6 @@ struct slab {
}; };
#endif #endif
}; };
struct kmem_cache *slab_cache;
/* Double-word boundary */ /* Double-word boundary */
void *freelist; /* first free object */ void *freelist; /* first free object */
union { union {
...@@ -43,6 +46,9 @@ struct slab { ...@@ -43,6 +46,9 @@ struct slab {
unsigned frozen:1; unsigned frozen:1;
}; };
}; };
};
struct rcu_head rcu_head;
};
unsigned int __unused; unsigned int __unused;
#elif defined(CONFIG_SLOB) #elif defined(CONFIG_SLOB)
...@@ -66,9 +72,10 @@ struct slab { ...@@ -66,9 +72,10 @@ struct slab {
#define SLAB_MATCH(pg, sl) \ #define SLAB_MATCH(pg, sl) \
static_assert(offsetof(struct page, pg) == offsetof(struct slab, sl)) static_assert(offsetof(struct page, pg) == offsetof(struct slab, sl))
SLAB_MATCH(flags, __page_flags); SLAB_MATCH(flags, __page_flags);
SLAB_MATCH(compound_head, slab_list); /* Ensure bit 0 is clear */
#ifndef CONFIG_SLOB #ifndef CONFIG_SLOB
SLAB_MATCH(rcu_head, rcu_head); SLAB_MATCH(compound_head, slab_cache); /* Ensure bit 0 is clear */
#else
SLAB_MATCH(compound_head, slab_list); /* Ensure bit 0 is clear */
#endif #endif
SLAB_MATCH(_refcount, __page_refcount); SLAB_MATCH(_refcount, __page_refcount);
#ifdef CONFIG_MEMCG #ifdef CONFIG_MEMCG
...@@ -76,6 +83,9 @@ SLAB_MATCH(memcg_data, memcg_data); ...@@ -76,6 +83,9 @@ SLAB_MATCH(memcg_data, memcg_data);
#endif #endif
#undef SLAB_MATCH #undef SLAB_MATCH
static_assert(sizeof(struct slab) <= sizeof(struct page)); static_assert(sizeof(struct slab) <= sizeof(struct page));
#if defined(CONFIG_HAVE_CMPXCHG_DOUBLE) && defined(CONFIG_SLUB)
static_assert(IS_ALIGNED(offsetof(struct slab, freelist), 2*sizeof(void *)));
#endif
/** /**
* folio_slab - Converts from folio to slab. * folio_slab - Converts from folio to slab.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment