]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
slab: fix kmalloc_nolock() context check for PREEMPT_RT
authorSwaraj Gaikwad <swarajgaikwad1925@gmail.com>
Tue, 13 Jan 2026 15:06:39 +0000 (20:36 +0530)
committerVlastimil Babka <vbabka@suse.cz>
Wed, 21 Jan 2026 10:34:18 +0000 (11:34 +0100)
On PREEMPT_RT kernels, local_lock becomes a sleeping lock. The current
check in kmalloc_nolock() only verifies we're not in NMI or hard IRQ
context, but misses the case where preemption is disabled.

When a BPF program runs from a tracepoint with preemption disabled
(preempt_count > 0), kmalloc_nolock() proceeds to call
local_lock_irqsave() which attempts to acquire a sleeping lock,
triggering:

  BUG: sleeping function called from invalid context
  in_atomic(): 1, irqs_disabled(): 0, non_block: 0, pid: 6128
  preempt_count: 2, expected: 0

Fix this by checking !preemptible() on PREEMPT_RT, which directly
expresses the constraint that we cannot take a sleeping lock when
preemption is disabled. This encompasses the previous checks for NMI
and hard IRQ contexts while also catching cases where preemption is
disabled.

Fixes: af92793e52c3 ("slab: Introduce kmalloc_nolock() and kfree_nolock().")
Reported-by: syzbot+b1546ad4a95331b2101e@syzkaller.appspotmail.com
Closes: https://syzkaller.appspot.com/bug?extid=b1546ad4a95331b2101e
Signed-off-by: Swaraj Gaikwad <swarajgaikwad1925@gmail.com>
Acked-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Alexei Starovoitov <ast@kernel.org>
Acked-by: Harry Yoo <harry.yoo@oracle.com>
Link: https://patch.msgid.link/20260113150639.48407-1-swarajgaikwad1925@gmail.co
Cc: <stable@vger.kernel.org>
Signed-off-by: Vlastimil Babka <vbabka@suse.cz>
mm/slub.c

index 861592ac54257b9d148ff921e6d8f62aced607b3..f77b7407c51bca0712df0fea402936b98bfb3d0b 100644 (file)
--- a/mm/slub.c
+++ b/mm/slub.c
@@ -5694,8 +5694,12 @@ void *kmalloc_nolock_noprof(size_t size, gfp_t gfp_flags, int node)
        if (unlikely(!size))
                return ZERO_SIZE_PTR;
 
-       if (IS_ENABLED(CONFIG_PREEMPT_RT) && (in_nmi() || in_hardirq()))
-               /* kmalloc_nolock() in PREEMPT_RT is not supported from irq */
+       if (IS_ENABLED(CONFIG_PREEMPT_RT) && !preemptible())
+               /*
+                * kmalloc_nolock() in PREEMPT_RT is not supported from
+                * non-preemptible context because local_lock becomes a
+                * sleeping lock on RT.
+                */
                return NULL;
 retry:
        if (unlikely(size > KMALLOC_MAX_CACHE_SIZE))