]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
mm/slab: Add kvfree_atomic() helper
authorUladzislau Rezki (Sony) <urezki@gmail.com>
Tue, 28 Apr 2026 16:14:18 +0000 (18:14 +0200)
committerHerbert Xu <herbert@gondor.apana.org.au>
Tue, 5 May 2026 08:12:07 +0000 (16:12 +0800)
kvmalloc() now supports non-sleeping GFP flags, including
the vmalloc fallback path. This means it may return vmalloc
memory even for GFP_ATOMIC and GFP_NOWAIT allocations.

Freeing such memory with kvfree() may then end up calling
vfree(), which is not safe for non-sleeping contexts.

Introduce kvfree_atomic() helper for such cases. It mirrors
kvfree(), but uses vfree_atomic() for vmalloced memory.

Signed-off-by: Uladzislau Rezki (Sony) <urezki@gmail.com>
Acked-by: Vlastimil Babka (SUSE) <vbabka@kernel.org>
Acked-by: Harry Yoo (Oracle) <harry@kernel.org>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
include/linux/slab.h
mm/slub.c

index 15a60b501b95b6c3f5b7099f4aab67689be312ab..2b5ab488e96b07102660747c11b560ceb675e159 100644 (file)
@@ -1234,6 +1234,9 @@ void *kvrealloc_node_align_noprof(const void *p, size_t size, unsigned long alig
 extern void kvfree(const void *addr);
 DEFINE_FREE(kvfree, void *, if (!IS_ERR_OR_NULL(_T)) kvfree(_T))
 
+extern void kvfree_atomic(const void *addr);
+DEFINE_FREE(kvfree_atomic, void *, if (!IS_ERR_OR_NULL(_T)) kvfree_atomic(_T))
+
 extern void kvfree_sensitive(const void *addr, size_t len);
 
 unsigned int kmem_cache_size(struct kmem_cache *s);
index 0baa906f39ab840fa79bf7f3744d5deac8891ca8..8f900453672968469460be9027758c5a8b2b8444 100644 (file)
--- a/mm/slub.c
+++ b/mm/slub.c
@@ -6882,6 +6882,22 @@ void kvfree(const void *addr)
 }
 EXPORT_SYMBOL(kvfree);
 
+/**
+ * kvfree_atomic() - Free memory.
+ * @addr: Pointer to allocated memory.
+ *
+ * Same as kvfree(), but uses vfree_atomic() for vmalloc
+ * backed memory. Must not be called from NMI context.
+ */
+void kvfree_atomic(const void *addr)
+{
+       if (is_vmalloc_addr(addr))
+               vfree_atomic(addr);
+       else
+               kfree(addr);
+}
+EXPORT_SYMBOL(kvfree_atomic);
+
 /**
  * kvfree_sensitive - Free a data object containing sensitive information.
  * @addr: address of the data object to be freed.