Blame nvme-ioctl.c

Packit dd4ba5
#include <assert.h>
Packit dd4ba5
#include <sys/ioctl.h>
Packit dd4ba5
#include <sys/stat.h>
Packit dd4ba5
#include <errno.h>
Packit dd4ba5
Packit dd4ba5
#include <getopt.h>
Packit dd4ba5
#include <fcntl.h>
Packit dd4ba5
#include <inttypes.h>
Packit dd4ba5
#include <locale.h>
Packit dd4ba5
#include <stdio.h>
Packit dd4ba5
#include <stdlib.h>
Packit dd4ba5
#include <string.h>
Packit dd4ba5
#include <unistd.h>
Packit dd4ba5
#include <math.h>
Packit dd4ba5
Packit dd4ba5
#include "nvme-ioctl.h"
Packit dd4ba5
Packit dd4ba5
static int nvme_verify_chr(int fd)
Packit dd4ba5
{
Packit dd4ba5
	static struct stat nvme_stat;
Packit dd4ba5
	int err = fstat(fd, &nvme_stat);
Packit dd4ba5
Packit dd4ba5
	if (err < 0) {
Packit dd4ba5
		perror("fstat");
Packit dd4ba5
		return errno;
Packit dd4ba5
	}
Packit dd4ba5
	if (!S_ISCHR(nvme_stat.st_mode)) {
Packit dd4ba5
		fprintf(stderr,
Packit dd4ba5
			"Error: requesting reset on non-controller handle\n");
Packit dd4ba5
		return ENOTBLK;
Packit dd4ba5
	}
Packit dd4ba5
	return 0;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_subsystem_reset(int fd)
Packit dd4ba5
{
Packit dd4ba5
	int ret;
Packit dd4ba5
Packit dd4ba5
	ret = nvme_verify_chr(fd);
Packit dd4ba5
	if (ret)
Packit dd4ba5
		return ret;
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_SUBSYS_RESET);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_reset_controller(int fd)
Packit dd4ba5
{
Packit dd4ba5
	int ret;
Packit dd4ba5
Packit dd4ba5
	ret = nvme_verify_chr(fd);
Packit dd4ba5
	if (ret)
Packit dd4ba5
		return ret;
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_RESET);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_ns_rescan(int fd)
Packit dd4ba5
{
Packit dd4ba5
	int ret;
Packit dd4ba5
Packit dd4ba5
	ret = nvme_verify_chr(fd);
Packit dd4ba5
	if (ret)
Packit dd4ba5
		return ret;
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_RESCAN);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_nsid(int fd)
Packit dd4ba5
{
Packit dd4ba5
	static struct stat nvme_stat;
Packit dd4ba5
	int err = fstat(fd, &nvme_stat);
Packit dd4ba5
Packit dd4ba5
	if (err < 0)
Packit dd4ba5
		return -errno;
Packit dd4ba5
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_ID);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_submit_passthru(int fd, unsigned long ioctl_cmd,
Packit dd4ba5
			 struct nvme_passthru_cmd *cmd)
Packit dd4ba5
{
Packit dd4ba5
	return ioctl(fd, ioctl_cmd, cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_submit_admin_passthru(int fd, struct nvme_passthru_cmd *cmd)
Packit dd4ba5
{
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_ADMIN_CMD, cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_submit_io_passthru(int fd, struct nvme_passthru_cmd *cmd)
Packit dd4ba5
{
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_IO_CMD, cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_passthru(int fd, unsigned long ioctl_cmd, __u8 opcode,
Packit dd4ba5
		  __u8 flags, __u16 rsvd,
Packit dd4ba5
		  __u32 nsid, __u32 cdw2, __u32 cdw3, __u32 cdw10, __u32 cdw11,
Packit dd4ba5
		  __u32 cdw12, __u32 cdw13, __u32 cdw14, __u32 cdw15,
Packit dd4ba5
		  __u32 data_len, void *data, __u32 metadata_len,
Packit dd4ba5
		  void *metadata, __u32 timeout_ms, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= opcode,
Packit dd4ba5
		.flags		= flags,
Packit dd4ba5
		.rsvd1		= rsvd,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw2		= cdw2,
Packit dd4ba5
		.cdw3		= cdw3,
Packit dd4ba5
		.metadata	= (__u64)(uintptr_t) metadata,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.metadata_len	= metadata_len,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.cdw11		= cdw11,
Packit dd4ba5
		.cdw12		= cdw12,
Packit dd4ba5
		.cdw13		= cdw13,
Packit dd4ba5
		.cdw14		= cdw14,
Packit dd4ba5
		.cdw15		= cdw15,
Packit dd4ba5
		.timeout_ms	= timeout_ms,
Packit dd4ba5
		.result		= 0,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_passthru(fd, ioctl_cmd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_io(int fd, __u8 opcode, __u64 slba, __u16 nblocks, __u16 control,
Packit dd4ba5
	    __u32 dsmgmt, __u32 reftag, __u16 apptag, __u16 appmask, void *data,
Packit dd4ba5
	    void *metadata)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_user_io io = {
Packit dd4ba5
		.opcode		= opcode,
Packit dd4ba5
		.flags		= 0,
Packit dd4ba5
		.control	= control,
Packit dd4ba5
		.nblocks	= nblocks,
Packit dd4ba5
		.rsvd		= 0,
Packit dd4ba5
		.metadata	= (__u64)(uintptr_t) metadata,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.slba		= slba,
Packit dd4ba5
		.dsmgmt		= dsmgmt,
Packit dd4ba5
		.reftag		= reftag,
Packit dd4ba5
		.appmask	= appmask,
Packit dd4ba5
		.apptag		= apptag,
Packit dd4ba5
	};
Packit dd4ba5
	return ioctl(fd, NVME_IOCTL_SUBMIT_IO, &io);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_read(int fd, __u64 slba, __u16 nblocks, __u16 control, __u32 dsmgmt,
Packit dd4ba5
	      __u32 reftag, __u16 apptag, __u16 appmask, void *data,
Packit dd4ba5
	      void *metadata)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_io(fd, nvme_cmd_read, slba, nblocks, control, dsmgmt,
Packit dd4ba5
		       reftag, apptag, appmask, data, metadata);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_write(int fd, __u64 slba, __u16 nblocks, __u16 control, __u32 dsmgmt,
Packit dd4ba5
	       __u32 reftag, __u16 apptag, __u16 appmask, void *data,
Packit dd4ba5
	       void *metadata)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_io(fd, nvme_cmd_write, slba, nblocks, control, dsmgmt,
Packit dd4ba5
		       reftag, apptag, appmask, data, metadata);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_compare(int fd, __u64 slba, __u16 nblocks, __u16 control, __u32 dsmgmt,
Packit dd4ba5
		 __u32 reftag, __u16 apptag, __u16 appmask, void *data,
Packit dd4ba5
		 void *metadata)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_io(fd, nvme_cmd_compare, slba, nblocks, control, dsmgmt,
Packit dd4ba5
		       reftag, apptag, appmask, data, metadata);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_verify(int fd, __u32 nsid, __u64 slba, __u16 nblocks,
Packit dd4ba5
		__u16 control, __u32 reftag, __u16 apptag, __u16 appmask)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_verify,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= slba & 0xffffffff,
Packit dd4ba5
		.cdw11		= slba >> 32,
Packit dd4ba5
		.cdw12		= nblocks | (control << 16),
Packit dd4ba5
		.cdw14		= reftag,
Packit dd4ba5
		.cdw15		= apptag | (appmask << 16),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_passthru_io(int fd, __u8 opcode, __u8 flags, __u16 rsvd,
Packit dd4ba5
		     __u32 nsid, __u32 cdw2, __u32 cdw3, __u32 cdw10,
Packit dd4ba5
		     __u32 cdw11, __u32 cdw12, __u32 cdw13, __u32 cdw14,
Packit dd4ba5
		     __u32 cdw15, __u32 data_len, void *data,
Packit dd4ba5
		     __u32 metadata_len, void *metadata, __u32 timeout_ms)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_passthru(fd, NVME_IOCTL_IO_CMD, opcode, flags, rsvd, nsid,
Packit dd4ba5
			     cdw2, cdw3, cdw10, cdw11, cdw12, cdw13, cdw14,
Packit dd4ba5
			     cdw15, data_len, data, metadata_len, metadata,
Packit dd4ba5
			     timeout_ms, NULL);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_write_zeros(int fd, __u32 nsid, __u64 slba, __u16 nlb,
Packit dd4ba5
		     __u16 control, __u32 reftag, __u16 apptag, __u16 appmask)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_write_zeroes,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= slba & 0xffffffff,
Packit dd4ba5
		.cdw11		= slba >> 32,
Packit dd4ba5
		.cdw12		= nlb | (control << 16),
Packit dd4ba5
		.cdw14		= reftag,
Packit dd4ba5
		.cdw15		= apptag | (appmask << 16),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_write_uncorrectable(int fd, __u32 nsid, __u64 slba, __u16 nlb)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_write_uncor,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= slba & 0xffffffff,
Packit dd4ba5
		.cdw11		= slba >> 32,
Packit dd4ba5
		.cdw12		= nlb,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_flush(int fd, __u32 nsid)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_flush,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_dsm(int fd, __u32 nsid, __u32 cdw11, struct nvme_dsm_range *dsm,
Packit dd4ba5
	     __u16 nr_ranges)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_dsm,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) dsm,
Packit dd4ba5
		.data_len	= nr_ranges * sizeof(*dsm),
Packit dd4ba5
		.cdw10		= nr_ranges - 1,
Packit dd4ba5
		.cdw11		= cdw11,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
struct nvme_dsm_range *nvme_setup_dsm_range(__u32 *ctx_attrs, __u32 *llbas,
Packit dd4ba5
					    __u64 *slbas, __u16 nr_ranges)
Packit dd4ba5
{
Packit dd4ba5
	int i;
Packit dd4ba5
	struct nvme_dsm_range *dsm = malloc(nr_ranges * sizeof(*dsm));
Packit dd4ba5
Packit dd4ba5
	if (!dsm) {
Packit dd4ba5
		fprintf(stderr, "malloc: %s\n", strerror(errno));
Packit dd4ba5
		return NULL;
Packit dd4ba5
	}
Packit dd4ba5
	for (i = 0; i < nr_ranges; i++) {
Packit dd4ba5
		dsm[i].cattr = cpu_to_le32(ctx_attrs[i]);
Packit dd4ba5
		dsm[i].nlb = cpu_to_le32(llbas[i]);
Packit dd4ba5
		dsm[i].slba = cpu_to_le64(slbas[i]);
Packit dd4ba5
	}
Packit dd4ba5
	return dsm;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_resv_acquire(int fd, __u32 nsid, __u8 rtype, __u8 racqa,
Packit dd4ba5
		      bool iekey, __u64 crkey, __u64 nrkey)
Packit dd4ba5
{
Packit dd4ba5
	__le64 payload[2] = { cpu_to_le64(crkey), cpu_to_le64(nrkey) };
Packit dd4ba5
	__u32 cdw10 = (racqa & 0x7) | (iekey ? 1 << 3 : 0) | rtype << 8;
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_resv_acquire,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) (payload),
Packit dd4ba5
		.data_len	= sizeof(payload),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_resv_register(int fd, __u32 nsid, __u8 rrega, __u8 cptpl,
Packit dd4ba5
		       bool iekey, __u64 crkey, __u64 nrkey)
Packit dd4ba5
{
Packit dd4ba5
	__le64 payload[2] = { cpu_to_le64(crkey), cpu_to_le64(nrkey) };
Packit dd4ba5
	__u32 cdw10 = (rrega & 0x7) | (iekey ? 1 << 3 : 0) | cptpl << 30;
Packit dd4ba5
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_resv_register,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) (payload),
Packit dd4ba5
		.data_len	= sizeof(payload),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_resv_release(int fd, __u32 nsid, __u8 rtype, __u8 rrela,
Packit dd4ba5
		      bool iekey, __u64 crkey)
Packit dd4ba5
{
Packit dd4ba5
	__le64 payload[1] = { cpu_to_le64(crkey) };
Packit dd4ba5
	__u32 cdw10 = (rrela & 0x7) | (iekey ? 1 << 3 : 0) | rtype << 8;
Packit dd4ba5
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_resv_release,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) (payload),
Packit dd4ba5
		.data_len	= sizeof(payload),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_resv_report(int fd, __u32 nsid, __u32 numd, __u32 cdw11, void *data)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_cmd_resv_report,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= numd,
Packit dd4ba5
		.cdw11		= cdw11,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= (numd + 1) << 2,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_io_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify13(int fd, __u32 nsid, __u32 cdw10, __u32 cdw11, void *data)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_identify,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= NVME_IDENTIFY_DATA_SIZE,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.cdw11		= cdw11,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify(int fd, __u32 nsid, __u32 cdw10, void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_identify13(fd, nsid, cdw10, 0, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ctrl(int fd, void *data)
Packit dd4ba5
{
Packit dd4ba5
	memset(data, 0, sizeof(struct nvme_id_ctrl));
Packit dd4ba5
	return nvme_identify(fd, 0, 1, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ns(int fd, __u32 nsid, bool present, void *data)
Packit dd4ba5
{
Packit dd4ba5
	int cns = present ? NVME_ID_CNS_NS_PRESENT : NVME_ID_CNS_NS;
Packit dd4ba5
Packit dd4ba5
	return nvme_identify(fd, nsid, cns, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ns_list(int fd, __u32 nsid, bool all, void *data)
Packit dd4ba5
{
Packit dd4ba5
	int cns = all ? NVME_ID_CNS_NS_PRESENT_LIST : NVME_ID_CNS_NS_ACTIVE_LIST;
Packit dd4ba5
Packit dd4ba5
	return nvme_identify(fd, nsid, cns, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ctrl_list(int fd, __u32 nsid, __u16 cntid, void *data)
Packit dd4ba5
{
Packit dd4ba5
	int cns = nsid ? NVME_ID_CNS_CTRL_NS_LIST : NVME_ID_CNS_CTRL_LIST;
Packit dd4ba5
Packit dd4ba5
	return nvme_identify(fd, nsid, (cntid << 16) | cns, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_secondary_ctrl_list(int fd, __u32 nsid, __u16 cntid, void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_identify(fd, nsid, (cntid << 16) | NVME_ID_CNS_SCNDRY_CTRL_LIST, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ns_descs(int fd, __u32 nsid, void *data)
Packit dd4ba5
{
Packit dd4ba5
Packit dd4ba5
	return nvme_identify(fd, nsid, NVME_ID_CNS_NS_DESC_LIST, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_nvmset(int fd, __u16 nvmset_id, void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_identify13(fd, 0, NVME_ID_CNS_NVMSET_LIST, nvmset_id, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_ns_granularity(int fd, void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_identify13(fd, 0, NVME_ID_CNS_NS_GRANULARITY, 0, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_identify_uuid(int fd, void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_identify(fd, 0, NVME_ID_CNS_UUID_LIST, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_log14(int fd, __u32 nsid, __u8 log_id, __u8 lsp, __u64 lpo,
Packit dd4ba5
                 __u16 lsi, bool rae, __u8 uuid_ix, __u32 data_len, void *data)
Packit dd4ba5
{
Packit dd4ba5
	__u32 numd = (data_len >> 2) - 1;
Packit dd4ba5
	__u16 numdu = numd >> 16, numdl = numd & 0xffff;
Packit dd4ba5
	__u32 cdw10 = log_id | (numdl << 16) | (rae ? 1 << 15 : 0) | lsp << 8;
Packit dd4ba5
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_get_log_page,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.cdw11		= numdu | (lsi << 16),
Packit dd4ba5
		.cdw12		= lpo & 0xffffffff,
Packit dd4ba5
		.cdw13		= lpo >> 32,
Packit dd4ba5
		.cdw14		= uuid_ix,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_log13(int fd, __u32 nsid, __u8 log_id, __u8 lsp,
Packit dd4ba5
		 __u64 lpo, __u16 lsi, bool rae, __u32 data_len,
Packit dd4ba5
		 void *data)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log14(fd, nsid, log_id, lsp, lpo, lsi, rae, 0,
Packit dd4ba5
			      data_len, data);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_log(int fd, __u32 nsid, __u8 log_id, bool rae,
Packit dd4ba5
		 __u32 data_len, void *data)
Packit dd4ba5
{
Packit dd4ba5
	__u32 offset = 0, xfer_len = data_len;
Packit dd4ba5
	void *ptr = data;
Packit dd4ba5
	int ret;
Packit dd4ba5
Packit dd4ba5
	/*
Packit dd4ba5
	 * 4k is the smallest possible transfer unit, so by
Packit dd4ba5
	 * restricting ourselves for 4k transfers we avoid having
Packit dd4ba5
	 * to check the MDTS value of the controller.
Packit dd4ba5
	 */
Packit dd4ba5
	do {
Packit dd4ba5
		xfer_len = data_len - offset;
Packit dd4ba5
		if (xfer_len > 4096)
Packit dd4ba5
			xfer_len = 4096;
Packit dd4ba5
Packit dd4ba5
		ret = nvme_get_log13(fd, nsid, log_id, NVME_NO_LOG_LSP,
Packit dd4ba5
				     offset, 0, rae, xfer_len, ptr);
Packit dd4ba5
		if (ret)
Packit dd4ba5
			return ret;
Packit dd4ba5
Packit dd4ba5
		offset += xfer_len;
Packit dd4ba5
		ptr += xfer_len;
Packit dd4ba5
	} while (offset < data_len);
Packit dd4ba5
Packit dd4ba5
	return 0;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_telemetry_log(int fd, void *lp, int generate_report,
Packit dd4ba5
			   int ctrl_init, size_t log_page_size, __u64 offset)
Packit dd4ba5
{
Packit dd4ba5
	if (ctrl_init)
Packit dd4ba5
		return nvme_get_log13(fd, NVME_NSID_ALL, NVME_LOG_TELEMETRY_CTRL,
Packit dd4ba5
				      NVME_NO_LOG_LSP, offset,
Packit dd4ba5
				      0, 1, log_page_size, lp);
Packit dd4ba5
	if (generate_report)
Packit dd4ba5
		return nvme_get_log13(fd, NVME_NSID_ALL, NVME_LOG_TELEMETRY_HOST,
Packit dd4ba5
				      NVME_TELEM_LSP_CREATE, offset,
Packit dd4ba5
				      0, 1, log_page_size, lp);
Packit dd4ba5
	else
Packit dd4ba5
		return nvme_get_log13(fd, NVME_NSID_ALL, NVME_LOG_TELEMETRY_HOST,
Packit dd4ba5
				      NVME_NO_LOG_LSP, offset,
Packit dd4ba5
				      0, 1, log_page_size, lp);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_fw_log(int fd, struct nvme_firmware_log_page *fw_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, NVME_NSID_ALL, NVME_LOG_FW_SLOT, true,
Packit dd4ba5
			sizeof(*fw_log), fw_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_changed_ns_list_log(int fd, struct nvme_changed_ns_list_log *changed_ns_list_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, 0, NVME_LOG_CHANGED_NS, true,
Packit dd4ba5
			sizeof(changed_ns_list_log->log),
Packit dd4ba5
			changed_ns_list_log->log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_error_log(int fd, int entries, struct nvme_error_log_page *err_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, NVME_NSID_ALL, NVME_LOG_ERROR, false,
Packit dd4ba5
			entries * sizeof(*err_log), err_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_endurance_log(int fd, __u16 group_id, struct nvme_endurance_group_log *endurance_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log13(fd, 0, NVME_LOG_ENDURANCE_GROUP, 0, 0, group_id, 0,
Packit dd4ba5
			sizeof(*endurance_log), endurance_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_smart_log(int fd, __u32 nsid, struct nvme_smart_log *smart_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, nsid, NVME_LOG_SMART, false,
Packit dd4ba5
			sizeof(*smart_log), smart_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_ana_log(int fd, void *ana_log, size_t ana_log_len, int rgo)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log13(fd, NVME_NSID_ALL, NVME_LOG_ANA, rgo, 0, 0,
Packit dd4ba5
			true, ana_log_len, ana_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_self_test_log(int fd, __u32 nsid, struct nvme_self_test_log *self_test_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, nsid, NVME_LOG_DEVICE_SELF_TEST, false,
Packit dd4ba5
		sizeof(*self_test_log), self_test_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_effects_log(int fd, struct nvme_effects_log_page *effects_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, NVME_NSID_ALL, NVME_LOG_CMD_EFFECTS, false,
Packit dd4ba5
			sizeof(*effects_log), effects_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_discovery_log(int fd, struct nvmf_disc_rsp_page_hdr *log, __u32 size)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, 0, NVME_LOG_DISC, false, size, log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_sanitize_log(int fd, struct nvme_sanitize_log_page *sanitize_log)
Packit dd4ba5
{
Packit dd4ba5
	return nvme_get_log(fd, 0, NVME_LOG_SANITIZE, false,
Packit dd4ba5
			sizeof(*sanitize_log), sanitize_log);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_feature(int fd, __u8 opcode, __u32 nsid, __u32 cdw10, __u32 cdw11,
Packit dd4ba5
		 __u32 cdw12, __u32 data_len, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= opcode,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.cdw11		= cdw11,
Packit dd4ba5
		.cdw12		= cdw12,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_set_feature(int fd, __u32 nsid, __u8 fid, __u32 value, __u32 cdw12,
Packit dd4ba5
		     bool save, __u32 data_len, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	__u32 cdw10 = fid | (save ? 1 << 31 : 0);
Packit dd4ba5
Packit dd4ba5
	return nvme_feature(fd, nvme_admin_set_features, nsid, cdw10, value,
Packit dd4ba5
			    cdw12, data_len, data, result);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
Packit dd4ba5
int nvme_get_property(int fd, int offset, uint64_t *value)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_fabrics_command,
Packit dd4ba5
		.nsid		= nvme_fabrics_type_property_get,
Packit dd4ba5
		.cdw10		= is_64bit_reg(offset),
Packit dd4ba5
		.cdw11		= offset,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && value)
Packit dd4ba5
		*value = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_properties(int fd, void **pbar)
Packit dd4ba5
{
Packit dd4ba5
	int offset;
Packit dd4ba5
	uint64_t value;
Packit dd4ba5
	int err, size = getpagesize();
Packit dd4ba5
Packit dd4ba5
	*pbar = malloc(size);
Packit dd4ba5
	if (!*pbar) {
Packit dd4ba5
		fprintf(stderr, "malloc: %s\n", strerror(errno));
Packit dd4ba5
		return -ENOMEM;
Packit dd4ba5
	}
Packit dd4ba5
Packit dd4ba5
	memset(*pbar, 0xff, size);
Packit dd4ba5
	for (offset = NVME_REG_CAP; offset <= NVME_REG_CMBSZ;) {
Packit dd4ba5
		err = nvme_get_property(fd, offset, &value);
Packit dd4ba5
		if (err > 0 && (err & 0xff) == NVME_SC_INVALID_FIELD) {
Packit dd4ba5
			err = 0;
Packit dd4ba5
			value = -1;
Packit dd4ba5
		} else if (err) {
Packit dd4ba5
			free(*pbar);
Packit dd4ba5
			break;
Packit dd4ba5
		}
Packit dd4ba5
		if (is_64bit_reg(offset)) {
Packit dd4ba5
			*(uint64_t *)(*pbar + offset) = value;
Packit dd4ba5
			offset += 8;
Packit dd4ba5
		} else {
Packit dd4ba5
			*(uint32_t *)(*pbar + offset) = value;
Packit dd4ba5
			offset += 4;
Packit dd4ba5
		}
Packit dd4ba5
	}
Packit dd4ba5
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_set_property(int fd, int offset, uint64_t value)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_passthru_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_fabrics_command,
Packit dd4ba5
		.nsid		= nvme_fabrics_type_property_set,
Packit dd4ba5
		.cdw10		= is_64bit_reg(offset),
Packit dd4ba5
		.cdw11		= offset,
Packit dd4ba5
		.cdw12		= value & 0xffffffff,
Packit dd4ba5
		.cdw13		= value >> 32,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_feature(int fd, __u32 nsid, __u8 fid, __u8 sel, __u32 cdw11,
Packit dd4ba5
		     __u32 data_len, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	__u32 cdw10 = fid | sel << 8;
Packit dd4ba5
Packit dd4ba5
	return nvme_feature(fd, nvme_admin_get_features, nsid, cdw10, cdw11,
Packit dd4ba5
			    0, data_len, data, result);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_format(int fd, __u32 nsid, __u8 lbaf, __u8 ses, __u8 pi,
Packit dd4ba5
		__u8 pil, __u8 ms, __u32 timeout)
Packit dd4ba5
{
Packit dd4ba5
	__u32 cdw10 = lbaf | ms << 4 | pi << 5 | pil << 8 | ses << 9;
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_format_nvm,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= cdw10,
Packit dd4ba5
		.timeout_ms	= timeout,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_ns_create(int fd, __u64 nsze, __u64 ncap, __u8 flbas, __u8 dps,
Packit dd4ba5
		__u8 nmic, __u32 anagrpid, __u16 nvmsetid,  __u32 timeout,
Packit dd4ba5
		__u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_id_ns ns = {
Packit dd4ba5
		.nsze		= cpu_to_le64(nsze),
Packit dd4ba5
		.ncap		= cpu_to_le64(ncap),
Packit dd4ba5
		.flbas		= flbas,
Packit dd4ba5
		.dps		= dps,
Packit dd4ba5
		.nmic		= nmic,
Packit dd4ba5
		.anagrpid	= anagrpid,
Packit dd4ba5
		.nvmsetid	= nvmsetid,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_ns_mgmt,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) ((void *)&ns),
Packit dd4ba5
		.cdw10		= 0,
Packit dd4ba5
		.data_len	= 0x1000,
Packit dd4ba5
		.timeout_ms	= timeout,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_ns_delete(int fd, __u32 nsid, __u32 timeout)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_ns_mgmt,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= 1,
Packit dd4ba5
		.timeout_ms	= timeout,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_ns_attachment(int fd, __u32 nsid, __u16 num_ctrls, __u16 *ctrlist,
Packit dd4ba5
		       bool attach)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_controller_list cntlist = {
Packit dd4ba5
		.num = cpu_to_le16(num_ctrls),
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_ns_attach,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t)&cntlist,
Packit dd4ba5
		.cdw10		= attach ? 0 : 1,
Packit dd4ba5
		.data_len	= 0x1000,
Packit dd4ba5
	};
Packit dd4ba5
	int i;
Packit dd4ba5
Packit dd4ba5
	for (i = 0; i < num_ctrls; i++)
Packit dd4ba5
		cntlist.identifier[i] = cpu_to_le16(ctrlist[i]);
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_fw_download(int fd, __u32 offset, __u32 data_len, void *data)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_download_fw,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
		.cdw10		= (data_len >> 2) - 1,
Packit dd4ba5
		.cdw11		= offset >> 2,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_fw_commit(int fd, __u8 slot, __u8 action, __u8 bpid)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_activate_fw,
Packit dd4ba5
		.cdw10		= (bpid << 31) | (action << 3) | slot,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_sec_send(int fd, __u32 nsid, __u8 nssf, __u16 spsp,
Packit dd4ba5
		  __u8 secp, __u32 tl, __u32 data_len, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_security_send,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= secp << 24 | spsp << 8 | nssf,
Packit dd4ba5
		.cdw11		= tl,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_sec_recv(int fd, __u32 nsid, __u8 nssf, __u16 spsp,
Packit dd4ba5
		  __u8 secp, __u32 al, __u32 data_len, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_security_recv,
Packit dd4ba5
		.nsid		= nsid,
Packit dd4ba5
		.cdw10		= secp << 24 | spsp << 8 | nssf,
Packit dd4ba5
		.cdw11		= al,
Packit dd4ba5
		.addr		= (__u64)(uintptr_t) data,
Packit dd4ba5
		.data_len	= data_len,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
	return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_get_lba_status(int fd, __u64 slba, __u32 mndw, __u8 atype, __u16 rl,
Packit dd4ba5
		void *data)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode =  nvme_admin_get_lba_status,
Packit dd4ba5
		.addr = (__u64)(uintptr_t) data,
Packit dd4ba5
		.cdw10 = slba & 0xffffffff,
Packit dd4ba5
		.cdw11 = slba >> 32,
Packit dd4ba5
		.cdw12 = mndw,
Packit dd4ba5
		.cdw13 = (atype << 24) | rl,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_dir_send(int fd, __u32 nsid, __u16 dspec, __u8 dtype, __u8 doper,
Packit dd4ba5
                  __u32 data_len, __u32 dw12, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
        struct nvme_admin_cmd cmd = {
Packit dd4ba5
                .opcode         = nvme_admin_directive_send,
Packit dd4ba5
                .addr           = (__u64)(uintptr_t) data,
Packit dd4ba5
                .data_len       = data_len,
Packit dd4ba5
                .nsid           = nsid,
Packit dd4ba5
                .cdw10          = data_len? (data_len >> 2) - 1 : 0,
Packit dd4ba5
                .cdw11          = dspec << 16 | dtype << 8 | doper,
Packit dd4ba5
                .cdw12          = dw12,
Packit dd4ba5
        };
Packit dd4ba5
        int err;
Packit dd4ba5
Packit dd4ba5
        err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
        if (!err && result)
Packit dd4ba5
                *result = cmd.result;
Packit dd4ba5
        return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_dir_recv(int fd, __u32 nsid, __u16 dspec, __u8 dtype, __u8 doper,
Packit dd4ba5
                  __u32 data_len, __u32 dw12, void *data, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
        struct nvme_admin_cmd cmd = {
Packit dd4ba5
                .opcode         = nvme_admin_directive_recv,
Packit dd4ba5
                .addr           = (__u64)(uintptr_t) data,
Packit dd4ba5
                .data_len       = data_len,
Packit dd4ba5
                .nsid           = nsid,
Packit dd4ba5
                .cdw10          = data_len? (data_len >> 2) - 1 : 0,
Packit dd4ba5
                .cdw11          = dspec << 16 | dtype << 8 | doper,
Packit dd4ba5
                .cdw12          = dw12,
Packit dd4ba5
        };
Packit dd4ba5
        int err;
Packit dd4ba5
Packit dd4ba5
        err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
        if (!err && result)
Packit dd4ba5
                *result = cmd.result;
Packit dd4ba5
        return err;
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_sanitize(int fd, __u8 sanact, __u8 ause, __u8 owpass, __u8 oipbp,
Packit dd4ba5
		  __u8 no_dealloc, __u32 ovrpat)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode		= nvme_admin_sanitize_nvm,
Packit dd4ba5
		.cdw10		= no_dealloc << 9 | oipbp << 8 |
Packit dd4ba5
				  owpass << NVME_SANITIZE_OWPASS_SHIFT |
Packit dd4ba5
				  ause << 3 | sanact,
Packit dd4ba5
		.cdw11		= ovrpat,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_self_test_start(int fd, __u32 nsid, __u32 cdw10)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode = nvme_admin_dev_self_test,
Packit dd4ba5
		.nsid = nsid,
Packit dd4ba5
		.cdw10 = cdw10,
Packit dd4ba5
	};
Packit dd4ba5
Packit dd4ba5
	return nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
}
Packit dd4ba5
Packit dd4ba5
int nvme_virtual_mgmt(int fd, __u32 cdw10, __u32 cdw11, __u32 *result)
Packit dd4ba5
{
Packit dd4ba5
	struct nvme_admin_cmd cmd = {
Packit dd4ba5
		.opcode = nvme_admin_virtual_mgmt,
Packit dd4ba5
		.cdw10  = cdw10,
Packit dd4ba5
		.cdw11  = cdw11,
Packit dd4ba5
	};
Packit dd4ba5
	int err;
Packit dd4ba5
Packit dd4ba5
	err = nvme_submit_admin_passthru(fd, &cmd);
Packit dd4ba5
	if (!err && result)
Packit dd4ba5
		*result = cmd.result;
Packit dd4ba5
Packit dd4ba5
	return err;
Packit dd4ba5
}