summaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorJan Kara <jack@suse.cz>2017-03-22 20:36:53 -0400
committerJens Axboe <axboe@fb.com>2017-03-22 22:11:22 -0400
commit03e262798884b0a5f948b17433afd80606cb3497 (patch)
tree6f173a691e73dfdd8ba35c08269ff793277d2df8 /fs
parenta83b576c9c25cf771fb0b15ec5eb2e7510ec2f5a (diff)
block: Fix bdi assignment to bdev inode when racing with disk delete
When disk->fops->open() in __blkdev_get() returns -ERESTARTSYS, we restart the process of opening the block device. However we forget to switch bdev->bd_bdi back to noop_backing_dev_info and as a result bdev inode will be pointing to a stale bdi. Fix the problem by setting bdev->bd_bdi later when __blkdev_get() is already guaranteed to succeed. Acked-by: Tejun Heo <tj@kernel.org> Reviewed-by: Hannes Reinecke <hare@suse.com> Signed-off-by: Jan Kara <jack@suse.cz> Signed-off-by: Jens Axboe <axboe@fb.com>
Diffstat (limited to 'fs')
-rw-r--r--fs/block_dev.c7
1 files changed, 3 insertions, 4 deletions
diff --git a/fs/block_dev.c b/fs/block_dev.c
index 2eca00ec4370..53e2389ae4d4 100644
--- a/fs/block_dev.c
+++ b/fs/block_dev.c
@@ -1556,8 +1556,6 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
1556 bdev->bd_disk = disk; 1556 bdev->bd_disk = disk;
1557 bdev->bd_queue = disk->queue; 1557 bdev->bd_queue = disk->queue;
1558 bdev->bd_contains = bdev; 1558 bdev->bd_contains = bdev;
1559 if (bdev->bd_bdi == &noop_backing_dev_info)
1560 bdev->bd_bdi = bdi_get(disk->queue->backing_dev_info);
1561 1559
1562 if (!partno) { 1560 if (!partno) {
1563 ret = -ENXIO; 1561 ret = -ENXIO;
@@ -1622,6 +1620,9 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
1622 } 1620 }
1623 bd_set_size(bdev, (loff_t)bdev->bd_part->nr_sects << 9); 1621 bd_set_size(bdev, (loff_t)bdev->bd_part->nr_sects << 9);
1624 } 1622 }
1623
1624 if (bdev->bd_bdi == &noop_backing_dev_info)
1625 bdev->bd_bdi = bdi_get(disk->queue->backing_dev_info);
1625 } else { 1626 } else {
1626 if (bdev->bd_contains == bdev) { 1627 if (bdev->bd_contains == bdev) {
1627 ret = 0; 1628 ret = 0;
@@ -1653,8 +1654,6 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
1653 bdev->bd_disk = NULL; 1654 bdev->bd_disk = NULL;
1654 bdev->bd_part = NULL; 1655 bdev->bd_part = NULL;
1655 bdev->bd_queue = NULL; 1656 bdev->bd_queue = NULL;
1656 bdi_put(bdev->bd_bdi);
1657 bdev->bd_bdi = &noop_backing_dev_info;
1658 if (bdev != bdev->bd_contains) 1657 if (bdev != bdev->bd_contains)
1659 __blkdev_put(bdev->bd_contains, mode, 1); 1658 __blkdev_put(bdev->bd_contains, mode, 1);
1660 bdev->bd_contains = NULL; 1659 bdev->bd_contains = NULL;