From 7c349b12a595387affbaf203dec6aa3e78efb695 Mon Sep 17 00:00:00 2001 From: Thibaut Sautereau Date: Fri, 29 Nov 2019 16:27:14 +0100 Subject: [PATCH] slub: Extend init_on_alloc to slab caches with constructors Signed-off-by: Thibaut Sautereau Signed-off-by: Levente Polyak --- mm/slab.h | 2 ++ mm/slub.c | 16 ++++++++++++++-- 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/mm/slab.h b/mm/slab.h index d28687a4b45a5..975a75b7230c5 100644 --- a/mm/slab.h +++ b/mm/slab.h @@ -683,8 +683,10 @@ static inline void cache_random_seq_destroy(struct kmem_cache *cachep) { } static inline bool slab_want_init_on_alloc(gfp_t flags, struct kmem_cache *c) { if (static_branch_unlikely(&init_on_alloc)) { +#ifndef CONFIG_SLUB if (c->ctor) return false; +#endif if (c->flags & (SLAB_TYPESAFE_BY_RCU | SLAB_POISON)) return flags & __GFP_ZERO; return true; diff --git a/mm/slub.c b/mm/slub.c index cc1741cc7a643..58a9a9bcb10f7 100644 --- a/mm/slub.c +++ b/mm/slub.c @@ -2843,8 +2843,14 @@ static __always_inline void *slab_alloc_node(struct kmem_cache *s, if (s->ctor) s->ctor(object); kasan_poison_object_data(s, object); - } else if (unlikely(slab_want_init_on_alloc(gfpflags, s)) && object) + } else if (unlikely(slab_want_init_on_alloc(gfpflags, s)) && object) { memset(object, 0, s->object_size); + if (s->ctor) { + kasan_unpoison_object_data(s, object); + s->ctor(object); + kasan_poison_object_data(s, object); + } + } if (object) { check_canary(s, object, s->random_inactive); @@ -3288,8 +3294,14 @@ int kmem_cache_alloc_bulk(struct kmem_cache *s, gfp_t flags, size_t size, } else if (unlikely(slab_want_init_on_alloc(flags, s))) { int j; - for (j = 0; j < i; j++) + for (j = 0; j < i; j++) { memset(p[j], 0, s->object_size); + if (s->ctor) { + kasan_unpoison_object_data(s, p[j]); + s->ctor(p[j]); + kasan_poison_object_data(s, p[j]); + } + } } for (k = 0; k < i; k++) {