Commit d57a964e authored by Andrey Konovalov's avatar Andrey Konovalov Committed by Linus Torvalds

kasan, mm: integrate slab init_on_free with HW_TAGS

This change uses the previously added memory initialization feature of
HW_TAGS KASAN routines for slab memory when init_on_free is enabled.

With this change, memory initialization memset() is no longer called when
both HW_TAGS KASAN and init_on_free are enabled.  Instead, memory is
initialized in KASAN runtime.

For SLUB, the memory initialization memset() is moved into
slab_free_hook() that currently directly follows the initialization loop.
A new argument is added to slab_free_hook() that indicates whether to
initialize the memory or not.

To avoid discrepancies with which memory gets initialized that can be
caused by future changes, both KASAN hook and initialization memset() are
put together and a warning comment is added.

Combining setting allocation tags with memory initialization improves
HW_TAGS KASAN performance when init_on_free is enabled.

Link: https://lkml.kernel.org/r/190fd15c1886654afdec0d19ebebd5ade665b601.1615296150.git.andreyknvl@google.comSigned-off-by: default avatarAndrey Konovalov <andreyknvl@google.com>
Reviewed-by: default avatarMarco Elver <elver@google.com>
Cc: Alexander Potapenko <glider@google.com>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Branislav Rankov <Branislav.Rankov@arm.com>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Cc: Christoph Lameter <cl@linux.com>
Cc: David Rientjes <rientjes@google.com>
Cc: Dmitry Vyukov <dvyukov@google.com>
Cc: Evgenii Stepanov <eugenis@google.com>
Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Kevin Brodsky <kevin.brodsky@arm.com>
Cc: Pekka Enberg <penberg@kernel.org>
Cc: Peter Collingbourne <pcc@google.com>
Cc: Vincenzo Frascino <vincenzo.frascino@arm.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Will Deacon <will.deacon@arm.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent da844b78
...@@ -203,11 +203,13 @@ static __always_inline void * __must_check kasan_init_slab_obj( ...@@ -203,11 +203,13 @@ static __always_inline void * __must_check kasan_init_slab_obj(
return (void *)object; return (void *)object;
} }
bool __kasan_slab_free(struct kmem_cache *s, void *object, unsigned long ip); bool __kasan_slab_free(struct kmem_cache *s, void *object,
static __always_inline bool kasan_slab_free(struct kmem_cache *s, void *object) unsigned long ip, bool init);
static __always_inline bool kasan_slab_free(struct kmem_cache *s,
void *object, bool init)
{ {
if (kasan_enabled()) if (kasan_enabled())
return __kasan_slab_free(s, object, _RET_IP_); return __kasan_slab_free(s, object, _RET_IP_, init);
return false; return false;
} }
...@@ -313,7 +315,7 @@ static inline void *kasan_init_slab_obj(struct kmem_cache *cache, ...@@ -313,7 +315,7 @@ static inline void *kasan_init_slab_obj(struct kmem_cache *cache,
{ {
return (void *)object; return (void *)object;
} }
static inline bool kasan_slab_free(struct kmem_cache *s, void *object) static inline bool kasan_slab_free(struct kmem_cache *s, void *object, bool init)
{ {
return false; return false;
} }
......
...@@ -322,8 +322,8 @@ void * __must_check __kasan_init_slab_obj(struct kmem_cache *cache, ...@@ -322,8 +322,8 @@ void * __must_check __kasan_init_slab_obj(struct kmem_cache *cache,
return (void *)object; return (void *)object;
} }
static inline bool ____kasan_slab_free(struct kmem_cache *cache, static inline bool ____kasan_slab_free(struct kmem_cache *cache, void *object,
void *object, unsigned long ip, bool quarantine) unsigned long ip, bool quarantine, bool init)
{ {
u8 tag; u8 tag;
void *tagged_object; void *tagged_object;
...@@ -351,7 +351,7 @@ static inline bool ____kasan_slab_free(struct kmem_cache *cache, ...@@ -351,7 +351,7 @@ static inline bool ____kasan_slab_free(struct kmem_cache *cache,
} }
kasan_poison(object, round_up(cache->object_size, KASAN_GRANULE_SIZE), kasan_poison(object, round_up(cache->object_size, KASAN_GRANULE_SIZE),
KASAN_KMALLOC_FREE, false); KASAN_KMALLOC_FREE, init);
if ((IS_ENABLED(CONFIG_KASAN_GENERIC) && !quarantine)) if ((IS_ENABLED(CONFIG_KASAN_GENERIC) && !quarantine))
return false; return false;
...@@ -362,9 +362,10 @@ static inline bool ____kasan_slab_free(struct kmem_cache *cache, ...@@ -362,9 +362,10 @@ static inline bool ____kasan_slab_free(struct kmem_cache *cache,
return kasan_quarantine_put(cache, object); return kasan_quarantine_put(cache, object);
} }
bool __kasan_slab_free(struct kmem_cache *cache, void *object, unsigned long ip) bool __kasan_slab_free(struct kmem_cache *cache, void *object,
unsigned long ip, bool init)
{ {
return ____kasan_slab_free(cache, object, ip, true); return ____kasan_slab_free(cache, object, ip, true, init);
} }
static inline bool ____kasan_kfree_large(void *ptr, unsigned long ip) static inline bool ____kasan_kfree_large(void *ptr, unsigned long ip)
...@@ -409,7 +410,7 @@ void __kasan_slab_free_mempool(void *ptr, unsigned long ip) ...@@ -409,7 +410,7 @@ void __kasan_slab_free_mempool(void *ptr, unsigned long ip)
return; return;
kasan_poison(ptr, page_size(page), KASAN_FREE_PAGE, false); kasan_poison(ptr, page_size(page), KASAN_FREE_PAGE, false);
} else { } else {
____kasan_slab_free(page->slab_cache, ptr, ip, false); ____kasan_slab_free(page->slab_cache, ptr, ip, false, false);
} }
} }
......
...@@ -3425,17 +3425,24 @@ static void cache_flusharray(struct kmem_cache *cachep, struct array_cache *ac) ...@@ -3425,17 +3425,24 @@ static void cache_flusharray(struct kmem_cache *cachep, struct array_cache *ac)
static __always_inline void __cache_free(struct kmem_cache *cachep, void *objp, static __always_inline void __cache_free(struct kmem_cache *cachep, void *objp,
unsigned long caller) unsigned long caller)
{ {
bool init;
if (is_kfence_address(objp)) { if (is_kfence_address(objp)) {
kmemleak_free_recursive(objp, cachep->flags); kmemleak_free_recursive(objp, cachep->flags);
__kfence_free(objp); __kfence_free(objp);
return; return;
} }
if (unlikely(slab_want_init_on_free(cachep))) /*
* As memory initialization might be integrated into KASAN,
* kasan_slab_free and initialization memset must be
* kept together to avoid discrepancies in behavior.
*/
init = slab_want_init_on_free(cachep);
if (init && !kasan_has_integrated_init())
memset(objp, 0, cachep->object_size); memset(objp, 0, cachep->object_size);
/* KASAN might put objp into memory quarantine, delaying its reuse. */
/* Put the object into the quarantine, don't touch it for now. */ if (kasan_slab_free(cachep, objp, init))
if (kasan_slab_free(cachep, objp))
return; return;
/* Use KCSAN to help debug racy use-after-free. */ /* Use KCSAN to help debug racy use-after-free. */
......
...@@ -1533,7 +1533,8 @@ static __always_inline void kfree_hook(void *x) ...@@ -1533,7 +1533,8 @@ static __always_inline void kfree_hook(void *x)
kasan_kfree_large(x); kasan_kfree_large(x);
} }
static __always_inline bool slab_free_hook(struct kmem_cache *s, void *x) static __always_inline bool slab_free_hook(struct kmem_cache *s,
void *x, bool init)
{ {
kmemleak_free_recursive(x, s->flags); kmemleak_free_recursive(x, s->flags);
...@@ -1559,8 +1560,25 @@ static __always_inline bool slab_free_hook(struct kmem_cache *s, void *x) ...@@ -1559,8 +1560,25 @@ static __always_inline bool slab_free_hook(struct kmem_cache *s, void *x)
__kcsan_check_access(x, s->object_size, __kcsan_check_access(x, s->object_size,
KCSAN_ACCESS_WRITE | KCSAN_ACCESS_ASSERT); KCSAN_ACCESS_WRITE | KCSAN_ACCESS_ASSERT);
/* KASAN might put x into memory quarantine, delaying its reuse */ /*
return kasan_slab_free(s, x); * As memory initialization might be integrated into KASAN,
* kasan_slab_free and initialization memset's must be
* kept together to avoid discrepancies in behavior.
*
* The initialization memset's clear the object and the metadata,
* but don't touch the SLAB redzone.
*/
if (init) {
int rsize;
if (!kasan_has_integrated_init())
memset(kasan_reset_tag(x), 0, s->object_size);
rsize = (s->flags & SLAB_RED_ZONE) ? s->red_left_pad : 0;
memset((char *)kasan_reset_tag(x) + s->inuse, 0,
s->size - s->inuse - rsize);
}
/* KASAN might put x into memory quarantine, delaying its reuse. */
return kasan_slab_free(s, x, init);
} }
static inline bool slab_free_freelist_hook(struct kmem_cache *s, static inline bool slab_free_freelist_hook(struct kmem_cache *s,
...@@ -1570,10 +1588,9 @@ static inline bool slab_free_freelist_hook(struct kmem_cache *s, ...@@ -1570,10 +1588,9 @@ static inline bool slab_free_freelist_hook(struct kmem_cache *s,
void *object; void *object;
void *next = *head; void *next = *head;
void *old_tail = *tail ? *tail : *head; void *old_tail = *tail ? *tail : *head;
int rsize;
if (is_kfence_address(next)) { if (is_kfence_address(next)) {
slab_free_hook(s, next); slab_free_hook(s, next, false);
return true; return true;
} }
...@@ -1585,20 +1602,8 @@ static inline bool slab_free_freelist_hook(struct kmem_cache *s, ...@@ -1585,20 +1602,8 @@ static inline bool slab_free_freelist_hook(struct kmem_cache *s,
object = next; object = next;
next = get_freepointer(s, object); next = get_freepointer(s, object);
if (slab_want_init_on_free(s)) {
/*
* Clear the object and the metadata, but don't touch
* the redzone.
*/
memset(kasan_reset_tag(object), 0, s->object_size);
rsize = (s->flags & SLAB_RED_ZONE) ? s->red_left_pad
: 0;
memset((char *)kasan_reset_tag(object) + s->inuse, 0,
s->size - s->inuse - rsize);
}
/* If object's reuse doesn't have to be delayed */ /* If object's reuse doesn't have to be delayed */
if (!slab_free_hook(s, object)) { if (!slab_free_hook(s, object, slab_want_init_on_free(s))) {
/* Move object to the new freelist */ /* Move object to the new freelist */
set_freepointer(s, object, *head); set_freepointer(s, object, *head);
*head = object; *head = object;
...@@ -3236,7 +3241,7 @@ int build_detached_freelist(struct kmem_cache *s, size_t size, ...@@ -3236,7 +3241,7 @@ int build_detached_freelist(struct kmem_cache *s, size_t size,
} }
if (is_kfence_address(object)) { if (is_kfence_address(object)) {
slab_free_hook(df->s, object); slab_free_hook(df->s, object, false);
__kfence_free(object); __kfence_free(object);
p[size] = NULL; /* mark object processed */ p[size] = NULL; /* mark object processed */
return size; return size;
......
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