diff mbox series

[-next] ubi: block: Fix use-after-free in ubiblock_cleanup

Message ID 20230921020142.2562687-1-wangzhaolong1@huawei.com
State Not Applicable
Headers show
Series [-next] ubi: block: Fix use-after-free in ubiblock_cleanup | expand

Commit Message

ZhaoLong Wang Sept. 21, 2023, 2:01 a.m. UTC
The following BUG is reported when a ubiblock is removed:

 ==================================================================
 BUG: KASAN: slab-use-after-free in ubiblock_cleanup+0x88/0xa0 [ubi]
 Read of size 4 at addr ffff88810c8f3804 by task ubiblock/1716

 CPU: 5 PID: 1716 Comm: ubiblock Not tainted 6.6.0-rc2+ #135
 Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS ?-20190727_073836-buildvm-ppc64le-16.ppc.fedoraproject.org-3.fc31 04/01/2014
 Call Trace:
  <TASK>
  dump_stack_lvl+0x37/0x50
  print_report+0xd0/0x620
  kasan_report+0xb6/0xf0
  ubiblock_cleanup+0x88/0xa0 [ubi]
  ubiblock_remove+0x121/0x190 [ubi]
  vol_cdev_ioctl+0x355/0x630 [ubi]
  __x64_sys_ioctl+0xc7/0x100
  do_syscall_64+0x3f/0x90
  entry_SYSCALL_64_after_hwframe+0x6e/0xd8
 RIP: 0033:0x7f08d7445577
 Code: b3 66 90 48 8b 05 11 89 2c 00 64 c7 00 26 00 00 00 48 c7 c0 ff ff ff ff c3 66 2e 0f 1f 84 00 00 00 00 00 b8 10 00 00 00 0f 05 <48> 3d 01 f0 ff ff 73 01 c3 48 8b 0d e1 8
 RSP: 002b:00007ffde05a3018 EFLAGS: 00000206 ORIG_RAX: 0000000000000010
 RAX: ffffffffffffffda RBX: 00000000ffffffff RCX: 00007f08d7445577
 RDX: 0000000000000000 RSI: 0000000000004f08 RDI: 0000000000000003
 RBP: 0000000000816010 R08: 00000000008163a7 R09: 0000000000000000
 R10: 0000000000000003 R11: 0000000000000206 R12: 0000000000000003
 R13: 00007ffde05a3130 R14: 0000000000000000 R15: 0000000000000000
  </TASK>

 Allocated by task 1715:
  kasan_save_stack+0x22/0x50
  kasan_set_track+0x25/0x30
  __kasan_kmalloc+0x7f/0x90
  __alloc_disk_node+0x40/0x2b0
  __blk_mq_alloc_disk+0x3e/0xb0
  ubiblock_create+0x2ba/0x620 [ubi]
  vol_cdev_ioctl+0x581/0x630 [ubi]
  __x64_sys_ioctl+0xc7/0x100
  do_syscall_64+0x3f/0x90
  entry_SYSCALL_64_after_hwframe+0x6e/0xd8

 Freed by task 0:
  kasan_save_stack+0x22/0x50
  kasan_set_track+0x25/0x30
  kasan_save_free_info+0x2b/0x50
  __kasan_slab_free+0x10e/0x190
  __kmem_cache_free+0x96/0x220
  bdev_free_inode+0xa4/0xf0
  rcu_core+0x496/0xec0
  __do_softirq+0xeb/0x384

 The buggy address belongs to the object at ffff88810c8f3800
  which belongs to the cache kmalloc-1k of size 1024
 The buggy address is located 4 bytes inside of
  freed 1024-byte region [ffff88810c8f3800, ffff88810c8f3c00)

 The buggy address belongs to the physical page:
 page:00000000d03de848 refcount:1 mapcount:0 mapping:0000000000000000 index:0x0 pfn:0x10c8f0
 head:00000000d03de848 order:3 entire_mapcount:0 nr_pages_mapped:0 pincount:0
 flags: 0x200000000000840(slab|head|node=0|zone=2)
 page_type: 0xffffffff()
 raw: 0200000000000840 ffff888100042dc0 ffffea0004244400 dead000000000002
 raw: 0000000000000000 0000000080100010 00000001ffffffff 0000000000000000
 page dumped because: kasan: bad access detected

 Memory state around the buggy address:
  ffff88810c8f3700: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
  ffff88810c8f3780: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
 >ffff88810c8f3800: fa fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
                    ^
  ffff88810c8f3880: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
  ffff88810c8f3900: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
 ==================================================================

Fix it by using a local variable to record the gendisk ID.

Fixes: 77567b25ab9f ("ubi: use blk_mq_alloc_disk and blk_cleanup_disk")
Signed-off-by: ZhaoLong Wang <wangzhaolong1@huawei.com>
---
 drivers/mtd/ubi/block.c | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

Comments

Zhihao Cheng Sept. 21, 2023, 2:39 a.m. UTC | #1
在 2023/9/21 10:01, ZhaoLong Wang 写道:
> The following BUG is reported when a ubiblock is removed:
> 
>   ==================================================================
>   BUG: KASAN: slab-use-after-free in ubiblock_cleanup+0x88/0xa0 [ubi]
>   Read of size 4 at addr ffff88810c8f3804 by task ubiblock/1716
> 
>   CPU: 5 PID: 1716 Comm: ubiblock Not tainted 6.6.0-rc2+ #135
>   Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS ?-20190727_073836-buildvm-ppc64le-16.ppc.fedoraproject.org-3.fc31 04/01/2014
>   Call Trace:
>    <TASK>
>    dump_stack_lvl+0x37/0x50
>    print_report+0xd0/0x620
>    kasan_report+0xb6/0xf0
>    ubiblock_cleanup+0x88/0xa0 [ubi]
>    ubiblock_remove+0x121/0x190 [ubi]
>    vol_cdev_ioctl+0x355/0x630 [ubi]
>    __x64_sys_ioctl+0xc7/0x100
>    do_syscall_64+0x3f/0x90
>    entry_SYSCALL_64_after_hwframe+0x6e/0xd8
>   RIP: 0033:0x7f08d7445577
>   Code: b3 66 90 48 8b 05 11 89 2c 00 64 c7 00 26 00 00 00 48 c7 c0 ff ff ff ff c3 66 2e 0f 1f 84 00 00 00 00 00 b8 10 00 00 00 0f 05 <48> 3d 01 f0 ff ff 73 01 c3 48 8b 0d e1 8
>   RSP: 002b:00007ffde05a3018 EFLAGS: 00000206 ORIG_RAX: 0000000000000010
>   RAX: ffffffffffffffda RBX: 00000000ffffffff RCX: 00007f08d7445577
>   RDX: 0000000000000000 RSI: 0000000000004f08 RDI: 0000000000000003
>   RBP: 0000000000816010 R08: 00000000008163a7 R09: 0000000000000000
>   R10: 0000000000000003 R11: 0000000000000206 R12: 0000000000000003
>   R13: 00007ffde05a3130 R14: 0000000000000000 R15: 0000000000000000
>    </TASK>
> 
>   Allocated by task 1715:
>    kasan_save_stack+0x22/0x50
>    kasan_set_track+0x25/0x30
>    __kasan_kmalloc+0x7f/0x90
>    __alloc_disk_node+0x40/0x2b0
>    __blk_mq_alloc_disk+0x3e/0xb0
>    ubiblock_create+0x2ba/0x620 [ubi]
>    vol_cdev_ioctl+0x581/0x630 [ubi]
>    __x64_sys_ioctl+0xc7/0x100
>    do_syscall_64+0x3f/0x90
>    entry_SYSCALL_64_after_hwframe+0x6e/0xd8
> 
>   Freed by task 0:
>    kasan_save_stack+0x22/0x50
>    kasan_set_track+0x25/0x30
>    kasan_save_free_info+0x2b/0x50
>    __kasan_slab_free+0x10e/0x190
>    __kmem_cache_free+0x96/0x220
>    bdev_free_inode+0xa4/0xf0
>    rcu_core+0x496/0xec0
>    __do_softirq+0xeb/0x384
> 
>   The buggy address belongs to the object at ffff88810c8f3800
>    which belongs to the cache kmalloc-1k of size 1024
>   The buggy address is located 4 bytes inside of
>    freed 1024-byte region [ffff88810c8f3800, ffff88810c8f3c00)
> 
>   The buggy address belongs to the physical page:
>   page:00000000d03de848 refcount:1 mapcount:0 mapping:0000000000000000 index:0x0 pfn:0x10c8f0
>   head:00000000d03de848 order:3 entire_mapcount:0 nr_pages_mapped:0 pincount:0
>   flags: 0x200000000000840(slab|head|node=0|zone=2)
>   page_type: 0xffffffff()
>   raw: 0200000000000840 ffff888100042dc0 ffffea0004244400 dead000000000002
>   raw: 0000000000000000 0000000080100010 00000001ffffffff 0000000000000000
>   page dumped because: kasan: bad access detected
> 
>   Memory state around the buggy address:
>    ffff88810c8f3700: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
>    ffff88810c8f3780: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
>   >ffff88810c8f3800: fa fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
>                      ^
>    ffff88810c8f3880: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
>    ffff88810c8f3900: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
>   ==================================================================
> 
> Fix it by using a local variable to record the gendisk ID.
> 
> Fixes: 77567b25ab9f ("ubi: use blk_mq_alloc_disk and blk_cleanup_disk")
> Signed-off-by: ZhaoLong Wang <wangzhaolong1@huawei.com>
> ---
>   drivers/mtd/ubi/block.c | 4 +++-
>   1 file changed, 3 insertions(+), 1 deletion(-)
> 

Reviewed-by: Zhihao Cheng <chengzhihao1@huawei.com>

> diff --git a/drivers/mtd/ubi/block.c b/drivers/mtd/ubi/block.c
> index 437c5b83ffe5..309a42aeaa4c 100644
> --- a/drivers/mtd/ubi/block.c
> +++ b/drivers/mtd/ubi/block.c
> @@ -447,13 +447,15 @@ int ubiblock_create(struct ubi_volume_info *vi)
>   
>   static void ubiblock_cleanup(struct ubiblock *dev)
>   {
> +	int id = dev->gd->first_minor;
> +
>   	/* Stop new requests to arrive */
>   	del_gendisk(dev->gd);
>   	/* Finally destroy the blk queue */
>   	dev_info(disk_to_dev(dev->gd), "released");
>   	put_disk(dev->gd);
>   	blk_mq_free_tag_set(&dev->tag_set);
> -	idr_remove(&ubiblock_minor_idr, dev->gd->first_minor);
> +	idr_remove(&ubiblock_minor_idr, id);
>   }
>   
>   int ubiblock_remove(struct ubi_volume_info *vi)
>
diff mbox series

Patch

diff --git a/drivers/mtd/ubi/block.c b/drivers/mtd/ubi/block.c
index 437c5b83ffe5..309a42aeaa4c 100644
--- a/drivers/mtd/ubi/block.c
+++ b/drivers/mtd/ubi/block.c
@@ -447,13 +447,15 @@  int ubiblock_create(struct ubi_volume_info *vi)
 
 static void ubiblock_cleanup(struct ubiblock *dev)
 {
+	int id = dev->gd->first_minor;
+
 	/* Stop new requests to arrive */
 	del_gendisk(dev->gd);
 	/* Finally destroy the blk queue */
 	dev_info(disk_to_dev(dev->gd), "released");
 	put_disk(dev->gd);
 	blk_mq_free_tag_set(&dev->tag_set);
-	idr_remove(&ubiblock_minor_idr, dev->gd->first_minor);
+	idr_remove(&ubiblock_minor_idr, id);
 }
 
 int ubiblock_remove(struct ubi_volume_info *vi)