Skip to content

Commit 89f9749

Browse files
htejunJens Axboe
authored andcommitted
block: fix __blkdev_get() for removable devices
Commit 0762b8b moved disk_get_part() in front of recursive get on the whole disk, which caused removable devices to try disk_get_part() before rescanning after a new media is inserted, which might fail legit open attempts or give the old partition. This patch fixes the problem by moving disk_get_part() after __blkdev_get() on the whole disk. This problem was spotted by Borislav Petkov. Signed-off-by: Tejun Heo <tj@kernel.org> Tested-by: Borislav Petkov <petkovbb@gmail.com> Signed-off-by: Jens Axboe <jens.axboe@oracle.com>
1 parent 561920a commit 89f9749

File tree

1 file changed

+11
-12
lines changed

1 file changed

+11
-12
lines changed

fs/block_dev.c

Lines changed: 11 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -986,7 +986,6 @@ static int __blkdev_put(struct block_device *bdev, fmode_t mode, int for_part);
986986
static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
987987
{
988988
struct gendisk *disk;
989-
struct hd_struct *part = NULL;
990989
int ret;
991990
int partno;
992991
int perm = 0;
@@ -1004,24 +1003,25 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
10041003
return ret;
10051004
}
10061005

1007-
ret = -ENXIO;
1008-
10091006
lock_kernel();
10101007

1008+
ret = -ENXIO;
10111009
disk = get_gendisk(bdev->bd_dev, &partno);
10121010
if (!disk)
10131011
goto out_unlock_kernel;
1014-
part = disk_get_part(disk, partno);
1015-
if (!part)
1016-
goto out_unlock_kernel;
10171012

10181013
mutex_lock_nested(&bdev->bd_mutex, for_part);
10191014
if (!bdev->bd_openers) {
10201015
bdev->bd_disk = disk;
1021-
bdev->bd_part = part;
10221016
bdev->bd_contains = bdev;
10231017
if (!partno) {
10241018
struct backing_dev_info *bdi;
1019+
1020+
ret = -ENXIO;
1021+
bdev->bd_part = disk_get_part(disk, partno);
1022+
if (!bdev->bd_part)
1023+
goto out_clear;
1024+
10251025
if (disk->fops->open) {
10261026
ret = disk->fops->open(bdev, mode);
10271027
if (ret)
@@ -1049,18 +1049,17 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
10491049
bdev->bd_contains = whole;
10501050
bdev->bd_inode->i_data.backing_dev_info =
10511051
whole->bd_inode->i_data.backing_dev_info;
1052+
bdev->bd_part = disk_get_part(disk, partno);
10521053
if (!(disk->flags & GENHD_FL_UP) ||
1053-
!part || !part->nr_sects) {
1054+
!bdev->bd_part || !bdev->bd_part->nr_sects) {
10541055
ret = -ENXIO;
10551056
goto out_clear;
10561057
}
1057-
bd_set_size(bdev, (loff_t)part->nr_sects << 9);
1058+
bd_set_size(bdev, (loff_t)bdev->bd_part->nr_sects << 9);
10581059
}
10591060
} else {
1060-
disk_put_part(part);
10611061
put_disk(disk);
10621062
module_put(disk->fops->owner);
1063-
part = NULL;
10641063
disk = NULL;
10651064
if (bdev->bd_contains == bdev) {
10661065
if (bdev->bd_disk->fops->open) {
@@ -1080,6 +1079,7 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
10801079
return 0;
10811080

10821081
out_clear:
1082+
disk_put_part(bdev->bd_part);
10831083
bdev->bd_disk = NULL;
10841084
bdev->bd_part = NULL;
10851085
bdev->bd_inode->i_data.backing_dev_info = &default_backing_dev_info;
@@ -1091,7 +1091,6 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
10911091
out_unlock_kernel:
10921092
unlock_kernel();
10931093

1094-
disk_put_part(part);
10951094
if (disk)
10961095
module_put(disk->fops->owner);
10971096
put_disk(disk);

0 commit comments

Comments
 (0)