Commit 7963e521 authored by Keith Busch's avatar Keith Busch Committed by Jens Axboe

NVMe: Passthrough IOCTL for IO commands

The NVME_IOCTL_SUBMIT_IO only works for IO commands with block data
transfers and isn't usable for other NVMe commands like flush,
data set management, or any sort of vendor unique command. The
NVME_IOCTL_ADMIN_CMD, however, can easily be modified to accept arbitrary
IO commands in addition to arbitrary admin commands without breaking
backward compatibility. This patch just adds a new IOCTL to distinguish
if the driver should submit the command on an IO or Admin queue.
Signed-off-by: default avatarKeith Busch <keith.busch@intel.com>
Signed-off-by: default avatarMatthew Wilcox <matthew.r.wilcox@intel.com>
Signed-off-by: default avatarJens Axboe <axboe@fb.com>
parent 1b9dbf7f
...@@ -1732,10 +1732,10 @@ static int nvme_submit_io(struct nvme_ns *ns, struct nvme_user_io __user *uio) ...@@ -1732,10 +1732,10 @@ static int nvme_submit_io(struct nvme_ns *ns, struct nvme_user_io __user *uio)
return status; return status;
} }
static int nvme_user_admin_cmd(struct nvme_dev *dev, static int nvme_user_cmd(struct nvme_dev *dev,
struct nvme_admin_cmd __user *ucmd) struct nvme_passthru_cmd __user *ucmd, bool ioq)
{ {
struct nvme_admin_cmd cmd; struct nvme_passthru_cmd cmd;
struct nvme_command c; struct nvme_command c;
int status, length; int status, length;
struct nvme_iod *uninitialized_var(iod); struct nvme_iod *uninitialized_var(iod);
...@@ -1774,6 +1774,9 @@ static int nvme_user_admin_cmd(struct nvme_dev *dev, ...@@ -1774,6 +1774,9 @@ static int nvme_user_admin_cmd(struct nvme_dev *dev,
ADMIN_TIMEOUT; ADMIN_TIMEOUT;
if (length != cmd.data_len) if (length != cmd.data_len)
status = -ENOMEM; status = -ENOMEM;
else if (ioq)
status = nvme_submit_sync_cmd(dev, this_cpu_read(*dev->io_queue), &c,
&cmd.result, timeout);
else else
status = nvme_submit_sync_cmd(dev, 0, &c, &cmd.result, timeout); status = nvme_submit_sync_cmd(dev, 0, &c, &cmd.result, timeout);
...@@ -1799,7 +1802,9 @@ static int nvme_ioctl(struct block_device *bdev, fmode_t mode, unsigned int cmd, ...@@ -1799,7 +1802,9 @@ static int nvme_ioctl(struct block_device *bdev, fmode_t mode, unsigned int cmd,
force_successful_syscall_return(); force_successful_syscall_return();
return ns->ns_id; return ns->ns_id;
case NVME_IOCTL_ADMIN_CMD: case NVME_IOCTL_ADMIN_CMD:
return nvme_user_admin_cmd(ns->dev, (void __user *)arg); return nvme_user_cmd(ns->dev, (void __user *)arg, false);
case NVME_IOCTL_IO_CMD:
return nvme_user_cmd(ns->dev, (void __user *)arg, true);
case NVME_IOCTL_SUBMIT_IO: case NVME_IOCTL_SUBMIT_IO:
return nvme_submit_io(ns, (void __user *)arg); return nvme_submit_io(ns, (void __user *)arg);
case SG_GET_VERSION_NUM: case SG_GET_VERSION_NUM:
...@@ -2743,7 +2748,9 @@ static long nvme_dev_ioctl(struct file *f, unsigned int cmd, unsigned long arg) ...@@ -2743,7 +2748,9 @@ static long nvme_dev_ioctl(struct file *f, unsigned int cmd, unsigned long arg)
struct nvme_dev *dev = f->private_data; struct nvme_dev *dev = f->private_data;
switch (cmd) { switch (cmd) {
case NVME_IOCTL_ADMIN_CMD: case NVME_IOCTL_ADMIN_CMD:
return nvme_user_admin_cmd(dev, (void __user *)arg); return nvme_user_cmd(dev, (void __user *)arg, false);
case NVME_IOCTL_IO_CMD:
return nvme_user_cmd(dev, (void __user *)arg, true);
default: default:
return -ENOTTY; return -ENOTTY;
} }
......
...@@ -503,7 +503,7 @@ struct nvme_user_io { ...@@ -503,7 +503,7 @@ struct nvme_user_io {
__u16 appmask; __u16 appmask;
}; };
struct nvme_admin_cmd { struct nvme_passthru_cmd {
__u8 opcode; __u8 opcode;
__u8 flags; __u8 flags;
__u16 rsvd1; __u16 rsvd1;
...@@ -524,8 +524,11 @@ struct nvme_admin_cmd { ...@@ -524,8 +524,11 @@ struct nvme_admin_cmd {
__u32 result; __u32 result;
}; };
#define nvme_admin_cmd nvme_passthru_cmd
#define NVME_IOCTL_ID _IO('N', 0x40) #define NVME_IOCTL_ID _IO('N', 0x40)
#define NVME_IOCTL_ADMIN_CMD _IOWR('N', 0x41, struct nvme_admin_cmd) #define NVME_IOCTL_ADMIN_CMD _IOWR('N', 0x41, struct nvme_admin_cmd)
#define NVME_IOCTL_SUBMIT_IO _IOW('N', 0x42, struct nvme_user_io) #define NVME_IOCTL_SUBMIT_IO _IOW('N', 0x42, struct nvme_user_io)
#define NVME_IOCTL_IO_CMD _IOWR('N', 0x43, struct nvme_passthru_cmd)
#endif /* _UAPI_LINUX_NVME_H */ #endif /* _UAPI_LINUX_NVME_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment