dm: requeue IO if mapping table not yet available
authorMike Snitzer <snitzer@redhat.com>
Tue, 22 Feb 2022 18:28:12 +0000 (13:28 -0500)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 13 Apr 2022 19:00:57 +0000 (21:00 +0200)
[ Upstream commit fa247089de9936a46e290d4724cb5f0b845600f5 ]

Update both bio-based and request-based DM to requeue IO if the
mapping table not available.

This race of IO being submitted before the DM device ready is so
narrow, yet possible for initial table load given that the DM device's
request_queue is created prior, that it best to requeue IO to handle
this unlikely case.

Reported-by: Zhang Yi <yi.zhang@huawei.com>
Signed-off-by: Mike Snitzer <snitzer@redhat.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/md/dm-rq.c
drivers/md/dm.c

index b1e867feb4f6b38b029707f60dd6341d958963a6..4833f4b20b2c74138b6aede230c6ef0992ad282b 100644 (file)
@@ -492,8 +492,13 @@ static blk_status_t dm_mq_queue_rq(struct blk_mq_hw_ctx *hctx,
 
        if (unlikely(!ti)) {
                int srcu_idx;
-               struct dm_table *map = dm_get_live_table(md, &srcu_idx);
+               struct dm_table *map;
 
+               map = dm_get_live_table(md, &srcu_idx);
+               if (unlikely(!map)) {
+                       dm_put_live_table(md, srcu_idx);
+                       return BLK_STS_RESOURCE;
+               }
                ti = dm_table_find_target(map, 0);
                dm_put_live_table(md, srcu_idx);
        }
index 6030cba5b0382fe4b9c18a0569a48994f4e9a148..2836d44094aba74b81c684276f9ebaf1b39d682b 100644 (file)
@@ -1692,15 +1692,10 @@ static blk_qc_t dm_submit_bio(struct bio *bio)
        struct dm_table *map;
 
        map = dm_get_live_table(md, &srcu_idx);
-       if (unlikely(!map)) {
-               DMERR_LIMIT("%s: mapping table unavailable, erroring io",
-                           dm_device_name(md));
-               bio_io_error(bio);
-               goto out;
-       }
 
-       /* If suspended, queue this IO for later */
-       if (unlikely(test_bit(DMF_BLOCK_IO_FOR_SUSPEND, &md->flags))) {
+       /* If suspended, or map not yet available, queue this IO for later */
+       if (unlikely(test_bit(DMF_BLOCK_IO_FOR_SUSPEND, &md->flags)) ||
+           unlikely(!map)) {
                if (bio->bi_opf & REQ_NOWAIT)
                        bio_wouldblock_error(bio);
                else if (bio->bi_opf & REQ_RAHEAD)