dm: requeue IO if mapping table not yet available
[ 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>
This commit is contained in:
		 Mike Snitzer
					Mike Snitzer
				
			
				
					committed by
					
						 Greg Kroah-Hartman
						Greg Kroah-Hartman
					
				
			
			
				
	
			
			
			 Greg Kroah-Hartman
						Greg Kroah-Hartman
					
				
			
						parent
						
							71c8df33fd
						
					
				
				
					commit
					2f2f017ea8
				
			| @@ -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); | ||||
| 	} | ||||
|   | ||||
| @@ -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) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user