Patch "bpf: Free dynamically allocated bits in bpf_iter_bits_destroy()" has been added to the 6.11-stable tree

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



This is a note to let you know that I've just added the patch titled

    bpf: Free dynamically allocated bits in bpf_iter_bits_destroy()

to the 6.11-stable tree which can be found at:
    http://www.kernel.org/git/?p=linux/kernel/git/stable/stable-queue.git;a=summary

The filename of the patch is:
     bpf-free-dynamically-allocated-bits-in-bpf_iter_bits.patch
and it can be found in the queue-6.11 subdirectory.

If you, or anyone else, feels it should not be added to the stable tree,
please let <stable@xxxxxxxxxxxxxxx> know about it.



commit 78ff4912dee5a49329760c31205a4c0ad120bf6c
Author: Hou Tao <houtao1@xxxxxxxxxx>
Date:   Wed Oct 30 18:05:12 2024 +0800

    bpf: Free dynamically allocated bits in bpf_iter_bits_destroy()
    
    [ Upstream commit 101ccfbabf4738041273ce64e2b116cf440dea13 ]
    
    bpf_iter_bits_destroy() uses "kit->nr_bits <= 64" to check whether the
    bits are dynamically allocated. However, the check is incorrect and may
    cause a kmemleak as shown below:
    
    unreferenced object 0xffff88812628c8c0 (size 32):
      comm "swapper/0", pid 1, jiffies 4294727320
      hex dump (first 32 bytes):
            b0 c1 55 f5 81 88 ff ff f0 f0 f0 f0 f0 f0 f0 f0  ..U...........
            f0 f0 f0 f0 f0 f0 f0 f0 00 00 00 00 00 00 00 00  ..............
      backtrace (crc 781e32cc):
            [<00000000c452b4ab>] kmemleak_alloc+0x4b/0x80
            [<0000000004e09f80>] __kmalloc_node_noprof+0x480/0x5c0
            [<00000000597124d6>] __alloc.isra.0+0x89/0xb0
            [<000000004ebfffcd>] alloc_bulk+0x2af/0x720
            [<00000000d9c10145>] prefill_mem_cache+0x7f/0xb0
            [<00000000ff9738ff>] bpf_mem_alloc_init+0x3e2/0x610
            [<000000008b616eac>] bpf_global_ma_init+0x19/0x30
            [<00000000fc473efc>] do_one_initcall+0xd3/0x3c0
            [<00000000ec81498c>] kernel_init_freeable+0x66a/0x940
            [<00000000b119f72f>] kernel_init+0x20/0x160
            [<00000000f11ac9a7>] ret_from_fork+0x3c/0x70
            [<0000000004671da4>] ret_from_fork_asm+0x1a/0x30
    
    That is because nr_bits will be set as zero in bpf_iter_bits_next()
    after all bits have been iterated.
    
    Fix the issue by setting kit->bit to kit->nr_bits instead of setting
    kit->nr_bits to zero when the iteration completes in
    bpf_iter_bits_next(). In addition, use "!nr_bits || bits >= nr_bits" to
    check whether the iteration is complete and still use "nr_bits > 64" to
    indicate whether bits are dynamically allocated. The "!nr_bits" check is
    necessary because bpf_iter_bits_new() may fail before setting
    kit->nr_bits, and this condition will stop the iteration early instead
    of accessing the zeroed or freed kit->bits.
    
    Considering the initial value of kit->bits is -1 and the type of
    kit->nr_bits is unsigned int, change the type of kit->nr_bits to int.
    The potential overflow problem will be handled in the following patch.
    
    Fixes: 4665415975b0 ("bpf: Add bits iterator")
    Acked-by: Yafang Shao <laoar.shao@xxxxxxxxx>
    Signed-off-by: Hou Tao <houtao1@xxxxxxxxxx>
    Link: https://lore.kernel.org/r/20241030100516.3633640-2-houtao@xxxxxxxxxxxxxxx
    Signed-off-by: Alexei Starovoitov <ast@xxxxxxxxxx>
    Signed-off-by: Sasha Levin <sashal@xxxxxxxxxx>

diff --git a/kernel/bpf/helpers.c b/kernel/bpf/helpers.c
index cc8c00864a680..a4521d2606b6b 100644
--- a/kernel/bpf/helpers.c
+++ b/kernel/bpf/helpers.c
@@ -2835,7 +2835,7 @@ struct bpf_iter_bits_kern {
 		unsigned long *bits;
 		unsigned long bits_copy;
 	};
-	u32 nr_bits;
+	int nr_bits;
 	int bit;
 } __aligned(8);
 
@@ -2909,17 +2909,16 @@ bpf_iter_bits_new(struct bpf_iter_bits *it, const u64 *unsafe_ptr__ign, u32 nr_w
 __bpf_kfunc int *bpf_iter_bits_next(struct bpf_iter_bits *it)
 {
 	struct bpf_iter_bits_kern *kit = (void *)it;
-	u32 nr_bits = kit->nr_bits;
+	int bit = kit->bit, nr_bits = kit->nr_bits;
 	const unsigned long *bits;
-	int bit;
 
-	if (nr_bits == 0)
+	if (!nr_bits || bit >= nr_bits)
 		return NULL;
 
 	bits = nr_bits == 64 ? &kit->bits_copy : kit->bits;
-	bit = find_next_bit(bits, nr_bits, kit->bit + 1);
+	bit = find_next_bit(bits, nr_bits, bit + 1);
 	if (bit >= nr_bits) {
-		kit->nr_bits = 0;
+		kit->bit = bit;
 		return NULL;
 	}
 




[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux