Skip to content

Commit 3534e5a

Browse files
Luo MengMike Snitzer
authored andcommitted
dm thin: fix use-after-free crash in dm_sm_register_threshold_callback
Fault inject on pool metadata device reports: BUG: KASAN: use-after-free in dm_pool_register_metadata_threshold+0x40/0x80 Read of size 8 at addr ffff8881b9d50068 by task dmsetup/950 CPU: 7 PID: 950 Comm: dmsetup Tainted: G W 5.19.0-rc6 #1 Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS 1.14.0-1.fc33 04/01/2014 Call Trace: <TASK> dump_stack_lvl+0x34/0x44 print_address_description.constprop.0.cold+0xeb/0x3f4 kasan_report.cold+0xe6/0x147 dm_pool_register_metadata_threshold+0x40/0x80 pool_ctr+0xa0a/0x1150 dm_table_add_target+0x2c8/0x640 table_load+0x1fd/0x430 ctl_ioctl+0x2c4/0x5a0 dm_ctl_ioctl+0xa/0x10 __x64_sys_ioctl+0xb3/0xd0 do_syscall_64+0x35/0x80 entry_SYSCALL_64_after_hwframe+0x46/0xb0 This can be easily reproduced using: echo offline > /sys/block/sda/device/state dd if=/dev/zero of=/dev/mapper/thin bs=4k count=10 dmsetup load pool --table "0 20971520 thin-pool /dev/sda /dev/sdb 128 0 0" If a metadata commit fails, the transaction will be aborted and the metadata space maps will be destroyed. If a DM table reload then happens for this failed thin-pool, a use-after-free will occur in dm_sm_register_threshold_callback (called from dm_pool_register_metadata_threshold). Fix this by in dm_pool_register_metadata_threshold() by returning the -EINVAL error if the thin-pool is in fail mode. Also fail pool_ctr() with a new error message: "Error registering metadata threshold". Fixes: ac8c3f3 ("dm thin: generate event when metadata threshold passed") Cc: [email protected] Reported-by: Hulk Robot <[email protected]> Signed-off-by: Luo Meng <[email protected]> Signed-off-by: Mike Snitzer <[email protected]>
1 parent 2ee73ef commit 3534e5a

File tree

2 files changed

+8
-3
lines changed

2 files changed

+8
-3
lines changed

drivers/md/dm-thin-metadata.c

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2045,10 +2045,13 @@ int dm_pool_register_metadata_threshold(struct dm_pool_metadata *pmd,
20452045
dm_sm_threshold_fn fn,
20462046
void *context)
20472047
{
2048-
int r;
2048+
int r = -EINVAL;
20492049

20502050
pmd_write_lock_in_core(pmd);
2051-
r = dm_sm_register_threshold_callback(pmd->metadata_sm, threshold, fn, context);
2051+
if (!pmd->fail_io) {
2052+
r = dm_sm_register_threshold_callback(pmd->metadata_sm,
2053+
threshold, fn, context);
2054+
}
20522055
pmd_write_unlock(pmd);
20532056

20542057
return r;

drivers/md/dm-thin.c

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3375,8 +3375,10 @@ static int pool_ctr(struct dm_target *ti, unsigned argc, char **argv)
33753375
calc_metadata_threshold(pt),
33763376
metadata_low_callback,
33773377
pool);
3378-
if (r)
3378+
if (r) {
3379+
ti->error = "Error registering metadata threshold";
33793380
goto out_flags_changed;
3381+
}
33803382

33813383
dm_pool_register_pre_commit_callback(pool->pmd,
33823384
metadata_pre_commit_callback, pool);

0 commit comments

Comments
 (0)