94aa5a9
From 6051b3759d3d4b70d33a7be70ab6b86ed3559224 Mon Sep 17 00:00:00 2001
94aa5a9
From: Paolo Bonzini <pbonzini@redhat.com>
94aa5a9
Date: Mon, 16 Jan 2012 17:12:58 +0100
94aa5a9
Subject: [PATCH 1/3] block: add and use scsi_blk_cmd_ioctl
94aa5a9
94aa5a9
Introduce a wrapper around scsi_cmd_ioctl that takes a block device.
94aa5a9
94aa5a9
The function will then be enhanced to detect partition block devices
94aa5a9
and, in that case, subject the ioctls to whitelisting.
94aa5a9
94aa5a9
[ Cherry picked from 6ad62f051ef784a48a6103af289f91b5c472e955 ]
94aa5a9
94aa5a9
Cc: stable@kernel.org
94aa5a9
Cc: linux-scsi@vger.kernel.org
94aa5a9
Cc: Jens Axboe <axboe@kernel.dk>
94aa5a9
Cc: James Bottomley <JBottomley@parallels.com>
94aa5a9
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
94aa5a9
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
94aa5a9
---
94aa5a9
 block/scsi_ioctl.c             |    7 +++++++
94aa5a9
 drivers/block/cciss.c          |    6 +++---
94aa5a9
 drivers/block/ub.c             |    3 +--
94aa5a9
 drivers/block/virtio_blk.c     |    4 ++--
94aa5a9
 drivers/cdrom/cdrom.c          |    3 +--
94aa5a9
 drivers/ide/ide-floppy_ioctl.c |    3 +--
94aa5a9
 drivers/scsi/sd.c              |    2 +-
94aa5a9
 include/linux/blkdev.h         |    2 ++
94aa5a9
 8 files changed, 18 insertions(+), 12 deletions(-)
94aa5a9
94aa5a9
diff --git a/block/scsi_ioctl.c b/block/scsi_ioctl.c
94aa5a9
index 4f4230b..57ac937 100644
94aa5a9
--- a/block/scsi_ioctl.c
94aa5a9
+++ b/block/scsi_ioctl.c
94aa5a9
@@ -691,6 +691,13 @@ int scsi_cmd_ioctl(struct request_queue *q, struct gendisk *bd_disk, fmode_t mod
94aa5a9
 }
94aa5a9
 EXPORT_SYMBOL(scsi_cmd_ioctl);
94aa5a9
 
94aa5a9
+int scsi_cmd_blk_ioctl(struct block_device *bd, fmode_t mode,
94aa5a9
+		       unsigned int cmd, void __user *arg)
94aa5a9
+{
94aa5a9
+	return scsi_cmd_ioctl(bd->bd_disk->queue, bd->bd_disk, mode, cmd, arg);
94aa5a9
+}
94aa5a9
+EXPORT_SYMBOL(scsi_cmd_blk_ioctl);
94aa5a9
+
94aa5a9
 static int __init blk_scsi_ioctl_init(void)
94aa5a9
 {
94aa5a9
 	blk_set_cmd_filter_defaults(&blk_default_cmd_filter);
94aa5a9
diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c
94aa5a9
index c2f9b3e..1dab802 100644
94aa5a9
--- a/drivers/block/cciss.c
94aa5a9
+++ b/drivers/block/cciss.c
94aa5a9
@@ -1716,7 +1716,7 @@ static int cciss_ioctl(struct block_device *bdev, fmode_t mode,
94aa5a9
 	case CCISS_BIG_PASSTHRU:
94aa5a9
 		return cciss_bigpassthru(h, argp);
94aa5a9
 
94aa5a9
-	/* scsi_cmd_ioctl handles these, below, though some are not */
94aa5a9
+	/* scsi_cmd_blk_ioctl handles these, below, though some are not */
94aa5a9
 	/* very meaningful for cciss.  SG_IO is the main one people want. */
94aa5a9
 
94aa5a9
 	case SG_GET_VERSION_NUM:
94aa5a9
@@ -1727,9 +1727,9 @@ static int cciss_ioctl(struct block_device *bdev, fmode_t mode,
94aa5a9
 	case SG_EMULATED_HOST:
94aa5a9
 	case SG_IO:
94aa5a9
 	case SCSI_IOCTL_SEND_COMMAND:
94aa5a9
-		return scsi_cmd_ioctl(disk->queue, disk, mode, cmd, argp);
94aa5a9
+		return scsi_cmd_blk_ioctl(bdev, mode, cmd, argp);
94aa5a9
 
94aa5a9
-	/* scsi_cmd_ioctl would normally handle these, below, but */
94aa5a9
+	/* scsi_cmd_blk_ioctl would normally handle these, below, but */
94aa5a9
 	/* they aren't a good fit for cciss, as CD-ROMs are */
94aa5a9
 	/* not supported, and we don't have any bus/target/lun */
94aa5a9
 	/* which we present to the kernel. */
94aa5a9
diff --git a/drivers/block/ub.c b/drivers/block/ub.c
94aa5a9
index 0e376d4..7333b9e 100644
94aa5a9
--- a/drivers/block/ub.c
94aa5a9
+++ b/drivers/block/ub.c
94aa5a9
@@ -1744,12 +1744,11 @@ static int ub_bd_release(struct gendisk *disk, fmode_t mode)
94aa5a9
 static int ub_bd_ioctl(struct block_device *bdev, fmode_t mode,
94aa5a9
     unsigned int cmd, unsigned long arg)
94aa5a9
 {
94aa5a9
-	struct gendisk *disk = bdev->bd_disk;
94aa5a9
 	void __user *usermem = (void __user *) arg;
94aa5a9
 	int ret;
94aa5a9
 
94aa5a9
 	mutex_lock(&ub_mutex);
94aa5a9
-	ret = scsi_cmd_ioctl(disk->queue, disk, mode, cmd, usermem);
94aa5a9
+	ret = scsi_cmd_blk_ioctl(bdev, mode, cmd, usermem);
94aa5a9
 	mutex_unlock(&ub_mutex);
94aa5a9
 
94aa5a9
 	return ret;
94aa5a9
diff --git a/drivers/block/virtio_blk.c b/drivers/block/virtio_blk.c
94aa5a9
index 079c088..5d7a934 100644
94aa5a9
--- a/drivers/block/virtio_blk.c
94aa5a9
+++ b/drivers/block/virtio_blk.c
94aa5a9
@@ -236,8 +236,8 @@ static int virtblk_ioctl(struct block_device *bdev, fmode_t mode,
94aa5a9
 	if (!virtio_has_feature(vblk->vdev, VIRTIO_BLK_F_SCSI))
94aa5a9
 		return -ENOTTY;
94aa5a9
 
94aa5a9
-	return scsi_cmd_ioctl(disk->queue, disk, mode, cmd,
94aa5a9
-			      (void __user *)data);
94aa5a9
+	return scsi_cmd_blk_ioctl(bdev, mode, cmd,
94aa5a9
+				  (void __user *)data);
94aa5a9
 }
94aa5a9
 
94aa5a9
 /* We provide getgeo only to please some old bootloader/partitioning tools */
94aa5a9
diff --git a/drivers/cdrom/cdrom.c b/drivers/cdrom/cdrom.c
94aa5a9
index f997c27..cedb231 100644
94aa5a9
--- a/drivers/cdrom/cdrom.c
94aa5a9
+++ b/drivers/cdrom/cdrom.c
94aa5a9
@@ -2747,12 +2747,11 @@ int cdrom_ioctl(struct cdrom_device_info *cdi, struct block_device *bdev,
94aa5a9
 {
94aa5a9
 	void __user *argp = (void __user *)arg;
94aa5a9
 	int ret;
94aa5a9
-	struct gendisk *disk = bdev->bd_disk;
94aa5a9
 
94aa5a9
 	/*
94aa5a9
 	 * Try the generic SCSI command ioctl's first.
94aa5a9
 	 */
94aa5a9
-	ret = scsi_cmd_ioctl(disk->queue, disk, mode, cmd, argp);
94aa5a9
+	ret = scsi_cmd_blk_ioctl(bdev, mode, cmd, argp);
94aa5a9
 	if (ret != -ENOTTY)
94aa5a9
 		return ret;
94aa5a9
 
94aa5a9
diff --git a/drivers/ide/ide-floppy_ioctl.c b/drivers/ide/ide-floppy_ioctl.c
94aa5a9
index d267b7a..a22ca84 100644
94aa5a9
--- a/drivers/ide/ide-floppy_ioctl.c
94aa5a9
+++ b/drivers/ide/ide-floppy_ioctl.c
94aa5a9
@@ -292,8 +292,7 @@ int ide_floppy_ioctl(ide_drive_t *drive, struct block_device *bdev,
94aa5a9
 	 * and CDROM_SEND_PACKET (legacy) ioctls
94aa5a9
 	 */
94aa5a9
 	if (cmd != CDROM_SEND_PACKET && cmd != SCSI_IOCTL_SEND_COMMAND)
94aa5a9
-		err = scsi_cmd_ioctl(bdev->bd_disk->queue, bdev->bd_disk,
94aa5a9
-				mode, cmd, argp);
94aa5a9
+		err = scsi_cmd_blk_ioctl(bdev, mode, cmd, argp);
94aa5a9
 
94aa5a9
 	if (err == -ENOTTY)
94aa5a9
 		err = generic_ide_ioctl(drive, bdev, cmd, arg);
94aa5a9
diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
94aa5a9
index 953773c..c88885d 100644
94aa5a9
--- a/drivers/scsi/sd.c
94aa5a9
+++ b/drivers/scsi/sd.c
94aa5a9
@@ -1095,7 +1095,7 @@ static int sd_ioctl(struct block_device *bdev, fmode_t mode,
94aa5a9
 			error = scsi_ioctl(sdp, cmd, p);
94aa5a9
 			break;
94aa5a9
 		default:
94aa5a9
-			error = scsi_cmd_ioctl(disk->queue, disk, mode, cmd, p);
94aa5a9
+			error = scsi_cmd_blk_ioctl(bdev, mode, cmd, p);
94aa5a9
 			if (error != -ENOTTY)
94aa5a9
 				break;
94aa5a9
 			error = scsi_ioctl(sdp, cmd, p);
94aa5a9
diff --git a/include/linux/blkdev.h b/include/linux/blkdev.h
94aa5a9
index 5e30b45..aa829a4 100644
94aa5a9
--- a/include/linux/blkdev.h
94aa5a9
+++ b/include/linux/blkdev.h
94aa5a9
@@ -675,6 +675,8 @@ extern int blk_insert_cloned_request(struct request_queue *q,
94aa5a9
 				     struct request *rq);
94aa5a9
 extern void blk_delay_queue(struct request_queue *, unsigned long);
94aa5a9
 extern void blk_recount_segments(struct request_queue *, struct bio *);
94aa5a9
+extern int scsi_cmd_blk_ioctl(struct block_device *, fmode_t,
94aa5a9
+			      unsigned int, void __user *);
94aa5a9
 extern int scsi_cmd_ioctl(struct request_queue *, struct gendisk *, fmode_t,
94aa5a9
 			  unsigned int, void __user *);
94aa5a9
 extern int sg_scsi_ioctl(struct request_queue *, struct gendisk *, fmode_t,
94aa5a9
-- 
94aa5a9
1.7.7.5
94aa5a9