|
@@ -171,11 +171,43 @@ static void do_virtblk_request(struct request_queue *q)
|
|
vblk->vq->vq_ops->kick(vblk->vq);
|
|
vblk->vq->vq_ops->kick(vblk->vq);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+/* return ATA identify data
|
|
|
|
+ */
|
|
|
|
+static int virtblk_identify(struct gendisk *disk, void *argp)
|
|
|
|
+{
|
|
|
|
+ struct virtio_blk *vblk = disk->private_data;
|
|
|
|
+ void *opaque;
|
|
|
|
+ int err = -ENOMEM;
|
|
|
|
+
|
|
|
|
+ opaque = kmalloc(VIRTIO_BLK_ID_BYTES, GFP_KERNEL);
|
|
|
|
+ if (!opaque)
|
|
|
|
+ goto out;
|
|
|
|
+
|
|
|
|
+ err = virtio_config_buf(vblk->vdev, VIRTIO_BLK_F_IDENTIFY,
|
|
|
|
+ offsetof(struct virtio_blk_config, identify), opaque,
|
|
|
|
+ VIRTIO_BLK_ID_BYTES);
|
|
|
|
+
|
|
|
|
+ if (err)
|
|
|
|
+ goto out_kfree;
|
|
|
|
+
|
|
|
|
+ if (copy_to_user(argp, opaque, VIRTIO_BLK_ID_BYTES))
|
|
|
|
+ err = -EFAULT;
|
|
|
|
+
|
|
|
|
+out_kfree:
|
|
|
|
+ kfree(opaque);
|
|
|
|
+out:
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
static int virtblk_ioctl(struct block_device *bdev, fmode_t mode,
|
|
static int virtblk_ioctl(struct block_device *bdev, fmode_t mode,
|
|
unsigned cmd, unsigned long data)
|
|
unsigned cmd, unsigned long data)
|
|
{
|
|
{
|
|
struct gendisk *disk = bdev->bd_disk;
|
|
struct gendisk *disk = bdev->bd_disk;
|
|
struct virtio_blk *vblk = disk->private_data;
|
|
struct virtio_blk *vblk = disk->private_data;
|
|
|
|
+ void __user *argp = (void __user *)data;
|
|
|
|
+
|
|
|
|
+ if (cmd == HDIO_GET_IDENTITY)
|
|
|
|
+ return virtblk_identify(disk, argp);
|
|
|
|
|
|
/*
|
|
/*
|
|
* Only allow the generic SCSI ioctls if the host can support it.
|
|
* Only allow the generic SCSI ioctls if the host can support it.
|
|
@@ -183,8 +215,7 @@ static int virtblk_ioctl(struct block_device *bdev, fmode_t mode,
|
|
if (!virtio_has_feature(vblk->vdev, VIRTIO_BLK_F_SCSI))
|
|
if (!virtio_has_feature(vblk->vdev, VIRTIO_BLK_F_SCSI))
|
|
return -ENOIOCTLCMD;
|
|
return -ENOIOCTLCMD;
|
|
|
|
|
|
- return scsi_cmd_ioctl(disk->queue, disk, mode, cmd,
|
|
|
|
- (void __user *)data);
|
|
|
|
|
|
+ return scsi_cmd_ioctl(disk->queue, disk, mode, cmd, argp);
|
|
}
|
|
}
|
|
|
|
|
|
/* We provide getgeo only to please some old bootloader/partitioning tools */
|
|
/* We provide getgeo only to please some old bootloader/partitioning tools */
|
|
@@ -390,7 +421,7 @@ static struct virtio_device_id id_table[] = {
|
|
static unsigned int features[] = {
|
|
static unsigned int features[] = {
|
|
VIRTIO_BLK_F_BARRIER, VIRTIO_BLK_F_SEG_MAX, VIRTIO_BLK_F_SIZE_MAX,
|
|
VIRTIO_BLK_F_BARRIER, VIRTIO_BLK_F_SEG_MAX, VIRTIO_BLK_F_SIZE_MAX,
|
|
VIRTIO_BLK_F_GEOMETRY, VIRTIO_BLK_F_RO, VIRTIO_BLK_F_BLK_SIZE,
|
|
VIRTIO_BLK_F_GEOMETRY, VIRTIO_BLK_F_RO, VIRTIO_BLK_F_BLK_SIZE,
|
|
- VIRTIO_BLK_F_SCSI,
|
|
|
|
|
|
+ VIRTIO_BLK_F_SCSI, VIRTIO_BLK_F_IDENTIFY
|
|
};
|
|
};
|
|
|
|
|
|
static struct virtio_driver virtio_blk = {
|
|
static struct virtio_driver virtio_blk = {
|