return status;
}
-static int nvme_user_admin_cmd(struct nvme_dev *dev,
- struct nvme_admin_cmd __user *ucmd)
+static int nvme_user_cmd(struct nvme_dev *dev,
+ struct nvme_passthru_cmd __user *ucmd, bool ioq)
{
- struct nvme_admin_cmd cmd;
+ struct nvme_passthru_cmd cmd;
struct nvme_command c;
int status, length;
struct nvme_iod *uninitialized_var(iod);
ADMIN_TIMEOUT;
if (length != cmd.data_len)
status = -ENOMEM;
+ else if (ioq)
+ status = nvme_submit_sync_cmd(dev, this_cpu_read(*dev->io_queue), &c,
+ &cmd.result, timeout);
else
status = nvme_submit_sync_cmd(dev, 0, &c, &cmd.result, timeout);
force_successful_syscall_return();
return ns->ns_id;
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:
return nvme_submit_io(ns, (void __user *)arg);
case SG_GET_VERSION_NUM:
struct nvme_dev *dev = f->private_data;
switch (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:
return -ENOTTY;
}
__u16 appmask;
};
-struct nvme_admin_cmd {
+struct nvme_passthru_cmd {
__u8 opcode;
__u8 flags;
__u16 rsvd1;
__u32 result;
};
+#define nvme_admin_cmd nvme_passthru_cmd
+
#define NVME_IOCTL_ID _IO('N', 0x40)
#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_IO_CMD _IOWR('N', 0x43, struct nvme_passthru_cmd)
#endif /* _UAPI_LINUX_NVME_H */