diff options
author | Jan Kara <jack@suse.cz> | 2017-03-22 20:36:53 -0400 |
---|---|---|
committer | Jens Axboe <axboe@fb.com> | 2017-03-22 22:11:22 -0400 |
commit | 03e262798884b0a5f948b17433afd80606cb3497 (patch) | |
tree | 6f173a691e73dfdd8ba35c08269ff793277d2df8 /fs | |
parent | a83b576c9c25cf771fb0b15ec5eb2e7510ec2f5a (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.c | 7 |
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; |