Re: [PATCH 4/6] genhd: Fix use after free in __blkdev_get()

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Jan,

On 2018/2/7 0:05, Jan Kara wrote:
> When two blkdev_open() calls race with device removal and recreation,
> __blkdev_get() can use looked up gendisk after it is freed:
> 
> CPU0				CPU1			CPU2
> 							del_gendisk(disk);
> 							  bdev_unhash_inode(inode);
> blkdev_open()			blkdev_open()
>   bdev = bd_acquire(inode);
>     - creates and returns new inode
> 				  bdev = bd_acquire(inode);
> 				    - returns the same inode
>   __blkdev_get(devt)		  __blkdev_get(devt)
>     disk = get_gendisk(devt);
>       - got structure of device going away
> 							<finish device removal>
> 							<new device gets
> 							 created under the same
> 							 device number>
> 				  disk = get_gendisk(devt);
> 				    - got new device structure
> 				  if (!bdev->bd_openers) {
> 				    does the first open
> 				  }
>     if (!bdev->bd_openers)
>       - false
>     } else {
>       put_disk_and_module(disk)
>         - remember this was old device - this was last ref and disk is
>           now freed
>     }
>     disk_unblock_events(disk); -> oops
> 
> Fix the problem by making sure we drop reference to disk in
> __blkdev_get() only after we are really done with it.
> 
> Reported-by: Hou Tao <houtao1@xxxxxxxxxx>
> Signed-off-by: Jan Kara <jack@xxxxxxx>
> ---
>  fs/block_dev.c | 7 +++++--
>  1 file changed, 5 insertions(+), 2 deletions(-)
> 

After applying the patch, the use-after-free problem is gone, so

Tested-by: Hou Tao <houtao1@xxxxxxxxxx>

> diff --git a/fs/block_dev.c b/fs/block_dev.c
> index 1dbbf847911a..fe41a76769fa 100644
> --- a/fs/block_dev.c
> +++ b/fs/block_dev.c
> @@ -1409,6 +1409,7 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
>  	int ret;
>  	int partno;
>  	int perm = 0;
> +	bool first_open = false;
>  
>  	if (mode & FMODE_READ)
>  		perm |= MAY_READ;
> @@ -1435,6 +1436,7 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
>  	disk_block_events(disk);
>  	mutex_lock_nested(&bdev->bd_mutex, for_part);
>  	if (!bdev->bd_openers) {
> +		first_open = true;
>  		bdev->bd_disk = disk;
>  		bdev->bd_queue = disk->queue;
>  		bdev->bd_contains = bdev;
> @@ -1520,14 +1522,15 @@ static int __blkdev_get(struct block_device *bdev, fmode_t mode, int for_part)
>  			if (ret)
>  				goto out_unlock_bdev;
>  		}
> -		/* only one opener holds refs to the module and disk */
> -		put_disk_and_module(disk);
>  	}
>  	bdev->bd_openers++;
>  	if (for_part)
>  		bdev->bd_part_count++;
>  	mutex_unlock(&bdev->bd_mutex);
>  	disk_unblock_events(disk);
> +	/* only one opener holds refs to the module and disk */
> +	if (!first_open)
> +		put_disk_and_module(disk);
>  	return 0;
>  
>   out_clear:
> 




[Index of Archives]     [Linux RAID]     [Linux SCSI]     [Linux ATA RAID]     [IDE]     [Linux Wireless]     [Linux Kernel]     [ATH6KL]     [Linux Bluetooth]     [Linux Netdev]     [Kernel Newbies]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Device Mapper]

  Powered by Linux