]> git.baikalelectronics.ru Git - kernel.git/commit
dm: fix dm-raid crash if md_handle_request() splits bio
authorMike Snitzer <snitzer@kernel.org>
Wed, 20 Jul 2022 17:58:04 +0000 (13:58 -0400)
committerMike Snitzer <snitzer@kernel.org>
Thu, 28 Jul 2022 21:36:30 +0000 (17:36 -0400)
commit729f85c9980ec0071ed6296b99642eb2b7dfdf89
treeeb4a62bec57415ffc4a735e7683ec5d7f77c3be1
parentf331362555c21fc889865a5e833c4628dbe3e464
dm: fix dm-raid crash if md_handle_request() splits bio

Commit 2f1bfe407e35b ("dm: pass NULL bdev to bio_alloc_clone")
introduced the optimization to _not_ perform bio_associate_blkg()'s
relatively costly work when DM core clones its bio. But in doing so it
exposed the possibility for DM's cloned bio to alter DM target
behavior (e.g. crash) if a target were to issue IO without first
calling bio_set_dev().

The DM raid target can trigger an MD crash due to its need to split
the DM bio that is passed to md_handle_request(). The split will
recurse to submit_bio_noacct() using a bio with an uninitialized
->bi_blkg. This NULL bio->bi_blkg causes blk_throtl_bio() to
dereference a NULL blkg_to_tg(bio->bi_blkg).

Fix this in DM core by adding a new 'needs_bio_set_dev' target flag that
will make alloc_tio() call bio_set_dev() on behalf of the target.
dm-raid is the only target that requires this flag. bio_set_dev()
initializes the DM cloned bio's ->bi_blkg, using bio_associate_blkg,
before passing the bio to md_handle_request().

Long-term fix would be to audit and refactor MD code to rely on DM to
split its bio, using dm_accept_partial_bio(), but there are MD raid
personalities (e.g. raid1 and raid10) whose implementation are tightly
coupled to handling the bio splitting inline.

Fixes: 2f1bfe407e35b ("dm: pass NULL bdev to bio_alloc_clone")
Cc: stable@vger.kernel.org
Signed-off-by: Mike Snitzer <snitzer@kernel.org>
drivers/md/dm-raid.c
drivers/md/dm.c
include/linux/device-mapper.h
include/uapi/linux/dm-ioctl.h