Blame nvme-lightnvm.c

Packit Service b7b338
/*
Packit Service b7b338
 * lightnvm.c -- LightNVM NVMe integration.
Packit Service b7b338
 *
Packit Service b7b338
 * Copyright (c) 2016, CNEX Labs.
Packit Service b7b338
 *
Packit Service b7b338
 * Written by Matias Bjoerling <matias@cnexlabs.com>
Packit Service b7b338
 *
Packit Service b7b338
 * This program is free software; you can redistribute it and/or
Packit Service b7b338
 * modify it under the terms of the GNU General Public License
Packit Service b7b338
 * as published by the Free Software Foundation; either version 2
Packit Service b7b338
 * of the License, or (at your option) any later version.
Packit Service b7b338
 *
Packit Service b7b338
 * This program is distributed in the hope that it will be useful,
Packit Service b7b338
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service b7b338
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service b7b338
 * GNU General Public License for more details.
Packit Service b7b338
 *
Packit Service b7b338
 * You should have received a copy of the GNU General Public License
Packit Service b7b338
 * along with this program; if not, write to the Free Software
Packit Service b7b338
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
Packit Service b7b338
 */
Packit Service b7b338
Packit Service b7b338
#include <stdlib.h>
Packit Service b7b338
#include <stdio.h>
Packit Service b7b338
#include <unistd.h>
Packit Service b7b338
#include <fcntl.h>
Packit Service b7b338
#include <sys/stat.h>
Packit Service b7b338
#include <sys/ioctl.h>
Packit Service b7b338
#include <string.h>
Packit Service b7b338
#include <errno.h>
Packit Service b7b338
#include <inttypes.h>
Packit Service b7b338
Packit Service b7b338
#include "nvme-lightnvm.h"
Packit Service b7b338
#include "nvme-print.h"
Packit Service b7b338
#include "nvme-ioctl.h"
Packit Service b7b338
Packit Service b7b338
static int lnvm_open(void)
Packit Service b7b338
{
Packit Service b7b338
	char dev[FILENAME_MAX] = NVM_CTRL_FILE;
Packit Service b7b338
	int fd;
Packit Service b7b338
Packit Service b7b338
	fd = open(dev, O_WRONLY);
Packit Service b7b338
	if (fd < 0) {
Packit Service b7b338
		printf("Failed to open LightNVM mgmt interface\n");
Packit Service b7b338
		perror(dev);
Packit Service b7b338
		return fd;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	return fd;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void lnvm_close(int fd)
Packit Service b7b338
{
Packit Service b7b338
	close(fd);
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_init(char *dev, char *mmtype)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_dev_init init;
Packit Service b7b338
	int fd, ret;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	memset(&init, 0, sizeof(struct nvm_ioctl_dev_init));
Packit Service b7b338
	strncpy(init.dev, dev, DISK_NAME_LEN - 1);
Packit Service b7b338
	strncpy(init.mmtype, mmtype, NVM_MMTYPE_LEN - 1);
Packit Service b7b338
Packit Service b7b338
	ret = ioctl(fd, NVM_DEV_INIT, &init);
Packit Service b7b338
	switch (errno) {
Packit Service b7b338
	case EINVAL:
Packit Service b7b338
		printf("Initialization failed.\n");
Packit Service b7b338
		break;
Packit Service b7b338
	case EEXIST:
Packit Service b7b338
		printf("Device has already been initialized.\n");
Packit Service b7b338
		break;
Packit Service b7b338
	case 0:
Packit Service b7b338
		break;
Packit Service b7b338
	default:
Packit Service b7b338
		printf("Unknown error occurred (%d)\n", errno);
Packit Service b7b338
		break;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
Packit Service b7b338
	return ret;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_list_devices(void)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_get_devices devs;
Packit Service b7b338
	int fd, ret, i;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	ret = ioctl(fd, NVM_GET_DEVICES, &devs);
Packit Service b7b338
	if (ret) {
Packit Service b7b338
		lnvm_close(fd);
Packit Service b7b338
		return ret;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	printf("Number of devices: %u\n", devs.nr_devices);
Packit Service b7b338
	printf("%-12s\t%-12s\tVersion\n", "Device", "Block manager");
Packit Service b7b338
Packit Service b7b338
	for (i = 0; i < devs.nr_devices && i < 31; i++) {
Packit Service b7b338
		struct nvm_ioctl_device_info *info = &devs.info[i];
Packit Service b7b338
Packit Service b7b338
		printf("%-12s\t%-12s\t(%u,%u,%u)\n", info->devname, info->bmname,
Packit Service b7b338
				info->bmversion[0], info->bmversion[1],
Packit Service b7b338
				info->bmversion[2]);
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
Packit Service b7b338
	return 0;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_info(void)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_info c;
Packit Service b7b338
	int fd, ret, i;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	memset(&c, 0, sizeof(struct nvm_ioctl_info));
Packit Service b7b338
	ret = ioctl(fd, NVM_INFO, &c);
Packit Service b7b338
	if (ret) {
Packit Service b7b338
		lnvm_close(fd);
Packit Service b7b338
		return ret;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	printf("LightNVM (%u,%u,%u). %u target type(s) registered.\n",
Packit Service b7b338
			c.version[0], c.version[1], c.version[2], c.tgtsize);
Packit Service b7b338
	printf("Type\tVersion\n");
Packit Service b7b338
Packit Service b7b338
	for (i = 0; i < c.tgtsize; i++) {
Packit Service b7b338
		struct nvm_ioctl_info_tgt *tgt = &c.tgts[i];
Packit Service b7b338
Packit Service b7b338
		printf("%s\t(%u,%u,%u)\n",
Packit Service b7b338
				tgt->tgtname, tgt->version[0], tgt->version[1],
Packit Service b7b338
				tgt->version[2]);
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
	return 0;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_create_tgt(char *devname, char *tgtname, char *tgttype,
Packit Service b7b338
					int lun_begin, int lun_end,
Packit Service b7b338
					int over_prov, int flags)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_create c;
Packit Service b7b338
	int fd, ret;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	strncpy(c.dev, devname, DISK_NAME_LEN - 1);
Packit Service b7b338
	strncpy(c.tgtname, tgtname, DISK_NAME_LEN - 1);
Packit Service b7b338
	strncpy(c.tgttype, tgttype, NVM_TTYPE_NAME_MAX - 1);
Packit Service b7b338
	c.flags = flags;
Packit Service b7b338
Packit Service b7b338
	/* Fall back into simple IOCTL version if no extended attributes used */
Packit Service b7b338
	if (over_prov != -1) {
Packit Service b7b338
		c.conf.type = NVM_CONFIG_TYPE_EXTENDED;
Packit Service b7b338
		c.conf.e.lun_begin = lun_begin;
Packit Service b7b338
		c.conf.e.lun_end = lun_end;
Packit Service b7b338
		c.conf.e.over_prov = over_prov;
Packit Service b7b338
	} else {
Packit Service b7b338
		c.conf.type = NVM_CONFIG_TYPE_SIMPLE;
Packit Service b7b338
		c.conf.s.lun_begin = lun_begin;
Packit Service b7b338
		c.conf.s.lun_end = lun_end;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	ret = ioctl(fd, NVM_DEV_CREATE, &c);
Packit Service b7b338
	if (ret)
Packit Service b7b338
		fprintf(stderr, "Creation of target failed. Please see dmesg.\n");
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
	return ret;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_remove_tgt(char *tgtname)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_remove c;
Packit Service b7b338
	int fd, ret;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	strncpy(c.tgtname, tgtname, DISK_NAME_LEN - 1);
Packit Service b7b338
	c.flags = 0;
Packit Service b7b338
Packit Service b7b338
	ret = ioctl(fd, NVM_DEV_REMOVE, &c);
Packit Service b7b338
	if (ret)
Packit Service b7b338
		fprintf(stderr, "Remove of target failed. Please see dmesg.\n");
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
	return ret;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_factory_init(char *devname, int erase_only_marked,
Packit Service b7b338
						int clear_host_marks,
Packit Service b7b338
						int clear_bb_marks)
Packit Service b7b338
{
Packit Service b7b338
	struct nvm_ioctl_dev_factory fact;
Packit Service b7b338
	int fd, ret;
Packit Service b7b338
Packit Service b7b338
	fd = lnvm_open();
Packit Service b7b338
	if (fd < 0)
Packit Service b7b338
		return fd;
Packit Service b7b338
Packit Service b7b338
	memset(&fact, 0, sizeof(struct nvm_ioctl_dev_factory));
Packit Service b7b338
Packit Service b7b338
	strncpy(fact.dev, devname, DISK_NAME_LEN - 1);
Packit Service b7b338
	if (erase_only_marked)
Packit Service b7b338
		fact.flags |= NVM_FACTORY_ERASE_ONLY_USER;
Packit Service b7b338
	if (clear_host_marks)
Packit Service b7b338
		fact.flags |= NVM_FACTORY_RESET_HOST_BLKS;
Packit Service b7b338
	if (clear_bb_marks)
Packit Service b7b338
		fact.flags |= NVM_FACTORY_RESET_GRWN_BBLKS;
Packit Service b7b338
Packit Service b7b338
	ret = ioctl(fd, NVM_DEV_FACTORY, &fact);
Packit Service b7b338
	switch (errno) {
Packit Service b7b338
	case EINVAL:
Packit Service b7b338
		fprintf(stderr, "Factory reset failed.\n");
Packit Service b7b338
		break;
Packit Service b7b338
	case 0:
Packit Service b7b338
		break;
Packit Service b7b338
	default:
Packit Service b7b338
		fprintf(stderr, "Unknown error occurred (%d)\n", errno);
Packit Service b7b338
		break;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	lnvm_close(fd);
Packit Service b7b338
	return ret;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_id_grp(void *t, int human)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_nvm_id12_group *grp = t;
Packit Service b7b338
	uint32_t mpos = (uint32_t)le32_to_cpu(grp->mpos);
Packit Service b7b338
	uint32_t mccap = (uint32_t)le32_to_cpu(grp->mccap);
Packit Service b7b338
Packit Service b7b338
	printf(" mtype   : %d\n", grp->mtype);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (grp->mtype == LNVM_IDFY_GRP_MTYPE_NAND)
Packit Service b7b338
			printf("           NAND Flash Memory\n");
Packit Service b7b338
		else
Packit Service b7b338
			printf("           Reserved\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf(" fmtype  : %d\n", grp->fmtype);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (grp->fmtype == LNVM_IDFY_GRP_FMTYPE_SLC)
Packit Service b7b338
			printf("           Single bit Level Cell flash (SLC)\n");
Packit Service b7b338
		else if (grp->fmtype == LNVM_IDFY_GRP_FMTYPE_MLC)
Packit Service b7b338
			printf("           Two bit Level Cell flash (MLC)\n");
Packit Service b7b338
		else if (grp->fmtype == LNVM_IDFY_GRP_FMTYPE_TLC)
Packit Service b7b338
			printf("           Three bit Level Cell flash (TLC)\n");
Packit Service b7b338
		else
Packit Service b7b338
			printf("           Reserved\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf(" chnls   : %d\n", grp->num_ch);
Packit Service b7b338
	printf(" luns    : %d\n", grp->num_lun);
Packit Service b7b338
	printf(" plns    : %d\n", grp->num_pln);
Packit Service b7b338
	printf(" blks    : %d\n", (uint16_t)le16_to_cpu(grp->num_blk));
Packit Service b7b338
	printf(" pgs     : %d\n", (uint16_t)le16_to_cpu(grp->num_pg));
Packit Service b7b338
	printf(" fpg_sz  : %d\n", (uint16_t)le16_to_cpu(grp->fpg_sz));
Packit Service b7b338
	printf(" csecs   : %d\n", (uint16_t)le16_to_cpu(grp->csecs));
Packit Service b7b338
	printf(" sos     : %d\n", (uint16_t)le16_to_cpu(grp->sos));
Packit Service b7b338
	printf(" trdt    : %d\n", (uint32_t)le32_to_cpu(grp->trdt));
Packit Service b7b338
	printf(" trdm    : %d\n", (uint32_t)le32_to_cpu(grp->trdm));
Packit Service b7b338
	printf(" tprt    : %d\n", (uint32_t)le32_to_cpu(grp->tprt));
Packit Service b7b338
	printf(" tprm    : %d\n", (uint32_t)le32_to_cpu(grp->tprm));
Packit Service b7b338
	printf(" tbet    : %d\n", (uint32_t)le32_to_cpu(grp->tbet));
Packit Service b7b338
	printf(" tbem    : %d\n", (uint32_t)le32_to_cpu(grp->tbem));
Packit Service b7b338
	printf(" mpos    : %#x\n", mpos);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_SNGL_PLN_RD))
Packit Service b7b338
			printf("           [0]: Single plane read\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_DUAL_PLN_RD))
Packit Service b7b338
			printf("           [1]: Dual plane read\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_QUAD_PLN_RD))
Packit Service b7b338
			printf("           [2]: Quad plane read\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_SNGL_PLN_PRG))
Packit Service b7b338
			printf("           [8]: Single plane program\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_DUAL_PLN_PRG))
Packit Service b7b338
			printf("           [9]: Dual plane program\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_QUAD_PLN_PRG))
Packit Service b7b338
			printf("           [10]: Quad plane program\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_SNGL_PLN_ERS))
Packit Service b7b338
			printf("           [16]: Single plane erase\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_DUAL_PLN_ERS))
Packit Service b7b338
			printf("           [17]: Dual plane erase\n");
Packit Service b7b338
		if (mpos & (1 << LNVM_IDFY_GRP_MPOS_QUAD_PLN_ERS))
Packit Service b7b338
			printf("           [18]: Quad plane erase\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf(" mccap   : %#x\n", mccap);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_GRP_MCCAP_SLC))
Packit Service b7b338
			printf("           [0]: SLC mode\n");
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_GRP_MCCAP_CMD_SUSP))
Packit Service b7b338
			printf("           [1]: Command suspension\n");
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_GRP_MCCAP_SCRAMBLE))
Packit Service b7b338
			printf("           [2]: Scramble\n");
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_GRP_MCCAP_ENCRYPT))
Packit Service b7b338
			printf("           [3]: Encryption\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf(" cpar    : %#x\n", (uint16_t)le16_to_cpu(grp->cpar));
Packit Service b7b338
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_ppaf(struct nvme_nvm_addr_format *ppaf)
Packit Service b7b338
{
Packit Service b7b338
	printf("ppaf     :\n");
Packit Service b7b338
	printf(" ch offs : %d ch bits  : %d\n",
Packit Service b7b338
					ppaf->ch_offset, ppaf->ch_len);
Packit Service b7b338
	printf(" lun offs: %d lun bits : %d\n",
Packit Service b7b338
					ppaf->lun_offset, ppaf->lun_len);
Packit Service b7b338
	printf(" pl offs : %d pl bits  : %d\n",
Packit Service b7b338
					ppaf->pln_offset, ppaf->pln_len);
Packit Service b7b338
	printf(" blk offs: %d blk bits : %d\n",
Packit Service b7b338
					ppaf->blk_offset, ppaf->blk_len);
Packit Service b7b338
	printf(" pg offs : %d pg bits  : %d\n",
Packit Service b7b338
					ppaf->pg_offset, ppaf->pg_len);
Packit Service b7b338
	printf(" sec offs: %d sec bits : %d\n",
Packit Service b7b338
					ppaf->sect_offset, ppaf->sect_len);
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_id12_ns(void *t, unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	int i;
Packit Service b7b338
	int human = flags & VERBOSE;
Packit Service b7b338
	struct nvme_nvm_id12 *id = t;
Packit Service b7b338
Packit Service b7b338
	uint32_t cap = (uint32_t) le32_to_cpu(id->cap);
Packit Service b7b338
	uint32_t dom = (uint32_t) le32_to_cpu(id->dom);
Packit Service b7b338
	uint32_t cgrps = id->cgrps;
Packit Service b7b338
Packit Service b7b338
	if (id->cgrps > 4) {
Packit Service b7b338
		fprintf(stderr, "invalid identify geometry returned\n");
Packit Service b7b338
		return;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	printf("verid    : %#x\n", id->ver_id);
Packit Service b7b338
	printf("vmnt     : %#x\n", id->vmnt);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (!id->vmnt)
Packit Service b7b338
			printf("           Generic/Enable opcodes as found in this spec.");
Packit Service b7b338
		else
Packit Service b7b338
			printf("           Reserved/Reserved for future opcode configurations");
Packit Service b7b338
	}
Packit Service b7b338
	printf("\n");
Packit Service b7b338
	printf("cgrps    : %d\n", id->cgrps);
Packit Service b7b338
	printf("cap      : %#x\n", cap);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (cap & (1 << LNVM_IDFY_CAP_BAD_BLK_TBL_MGMT))
Packit Service b7b338
			printf("           [0]: Bad block table management\n");
Packit Service b7b338
		if (cap & (1 << LNVM_IDFY_CAP_HYBRID_CMD_SUPP))
Packit Service b7b338
			printf("           [1]: Hybrid command support\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf("dom      : %#x\n", dom);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (dom & (1 << LNVM_IDFY_DOM_HYBRID_MODE))
Packit Service b7b338
			printf("           [0]: Hybrid mode (L2P MAP is in device)\n");
Packit Service b7b338
		if (dom & (1 << LNVM_IDFY_DOM_ECC_MODE))
Packit Service b7b338
			printf("           [1]: Error Code Correction(ECC) mode\n");
Packit Service b7b338
	}
Packit Service b7b338
	show_lnvm_ppaf(&id->ppaf);
Packit Service b7b338
Packit Service b7b338
	for (i = 0; i < cgrps; i++) {
Packit Service b7b338
		printf("grp      : %d\n", i);
Packit Service b7b338
		show_lnvm_id_grp((void *)&id->groups[i], human);
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_id20_ns(struct nvme_nvm_id20 *id, unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	int human = flags & VERBOSE;
Packit Service b7b338
	uint32_t mccap = (uint32_t) le32_to_cpu(id->mccap);
Packit Service b7b338
Packit Service b7b338
	printf("ver_major     : %#x\n", id->mjr);
Packit Service b7b338
	printf("ver_minor     : %#x\n", id->mnr);
Packit Service b7b338
Packit Service b7b338
	printf("mccap         : %#x\n", mccap);
Packit Service b7b338
	if (human) {
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_CAP_VCOPY))
Packit Service b7b338
			printf("           [0]: Vector copy support\n");
Packit Service b7b338
		if (mccap & (1 << LNVM_IDFY_CAP_MRESETS))
Packit Service b7b338
			printf("           [1]: Multiple resets support\n");
Packit Service b7b338
	}
Packit Service b7b338
	printf("wit           : %d\n", id->wit);
Packit Service b7b338
Packit Service b7b338
	printf("lba format\n");
Packit Service b7b338
	printf(" grp len      : %d\n", id->lbaf.grp_len);
Packit Service b7b338
	printf(" pu len       : %d\n", id->lbaf.pu_len);
Packit Service b7b338
	printf(" chk len      : %d\n", id->lbaf.chk_len);
Packit Service b7b338
	printf(" clba len     : %d\n", id->lbaf.lba_len);
Packit Service b7b338
Packit Service b7b338
	printf("geometry\n");
Packit Service b7b338
	printf(" num_grp      : %d\n", le16_to_cpu(id->num_grp));
Packit Service b7b338
	printf(" num_pu       : %d\n", le16_to_cpu(id->num_pu));
Packit Service b7b338
	printf(" num_chk      : %d\n", le32_to_cpu(id->num_chk));
Packit Service b7b338
	printf(" clba         : %d\n", le32_to_cpu(id->clba));
Packit Service b7b338
	printf("write req\n");
Packit Service b7b338
	printf(" ws_min       : %d\n", le32_to_cpu(id->ws_min));
Packit Service b7b338
	printf(" ws_opt       : %d\n", le32_to_cpu(id->ws_opt));
Packit Service b7b338
	printf(" mw_cunits    : %d\n", le32_to_cpu(id->mw_cunits));
Packit Service b7b338
	printf(" maxoc        : %d\n", le32_to_cpu(id->maxoc));
Packit Service b7b338
	printf(" maxocpu      : %d\n", le32_to_cpu(id->maxocpu));
Packit Service b7b338
	printf("perf metrics\n");
Packit Service b7b338
	printf(" trdt (ns)    : %d\n", le32_to_cpu(id->trdt));
Packit Service b7b338
	printf(" trdm (ns)    : %d\n", le32_to_cpu(id->trdm));
Packit Service b7b338
	printf(" twrt (ns)    : %d\n", le32_to_cpu(id->twrt));
Packit Service b7b338
	printf(" twrm (ns)    : %d\n", le32_to_cpu(id->twrm));
Packit Service b7b338
	printf(" tcrst (ns)   : %d\n", le32_to_cpu(id->tcrst));
Packit Service b7b338
	printf(" tcrsm (ns)   : %d\n", le32_to_cpu(id->tcrsm));
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_id_ns(struct nvme_nvm_id *id, unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	switch (id->ver_id) {
Packit Service b7b338
		case 1:
Packit Service b7b338
			show_lnvm_id12_ns((void *) id, flags);
Packit Service b7b338
		break;
Packit Service b7b338
		case 2:
Packit Service b7b338
			show_lnvm_id20_ns((void *) id, flags);
Packit Service b7b338
		break;
Packit Service b7b338
		default:
Packit Service b7b338
			fprintf(stderr, "Version %d not supported.\n",
Packit Service b7b338
					id->ver_id);
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_get_identity(int fd, int nsid, struct nvme_nvm_id *nvm_id)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_admin_cmd cmd = {
Packit Service b7b338
		.opcode		= nvme_nvm_admin_identity,
Packit Service b7b338
		.nsid		= nsid,
Packit Service b7b338
		.addr		= (__u64)(uintptr_t)nvm_id,
Packit Service b7b338
		.data_len	= sizeof(struct nvme_nvm_id),
Packit Service b7b338
	};
Packit Service b7b338
Packit Service b7b338
	return nvme_submit_passthru(fd, NVME_IOCTL_ADMIN_CMD, &cmd);
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_id_ns(int fd, int nsid, unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_nvm_id nvm_id;
Packit Service b7b338
	int err;
Packit Service b7b338
Packit Service b7b338
	err = lnvm_get_identity(fd, nsid, &nvm_id);
Packit Service b7b338
	if (!err) {
Packit Service b7b338
		if (flags & BINARY)
Packit Service b7b338
			d_raw((unsigned char *)&nvm_id, sizeof(nvm_id));
Packit Service b7b338
		else
Packit Service b7b338
			show_lnvm_id_ns(&nvm_id, flags);
Packit Service b7b338
	} else if (err > 0)
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x) NSID:%d\n",
Packit Service b7b338
			nvme_status_to_string(err), err, nsid);
Packit Service b7b338
	return err;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static inline const char *print_chunk_state(__u8 cs)
Packit Service b7b338
{
Packit Service b7b338
	switch (cs) {
Packit Service b7b338
	case 1 << 0:	return "FREE";
Packit Service b7b338
	case 1 << 1:	return "CLOSED";
Packit Service b7b338
	case 1 << 2:	return "OPEN";
Packit Service b7b338
	case 1 << 3:	return "OFFLINE";
Packit Service b7b338
	default:	return "UNKNOWN";
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static inline const char *print_chunk_type(__u8 ct)
Packit Service b7b338
{
Packit Service b7b338
	switch (ct & 0xF) {
Packit Service b7b338
	case 1 << 0:	return "SEQWRITE_REQ";
Packit Service b7b338
	case 1 << 1:	return "RANDWRITE_ALLOWED";
Packit Service b7b338
	default:	return "UNKNOWN";
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static inline const char *print_chunk_attr(__u8 ct)
Packit Service b7b338
{
Packit Service b7b338
	switch (ct & 0xF0) {
Packit Service b7b338
	case 1 << 4:	return "DEVIATED";
Packit Service b7b338
	default:	return "NONE";
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_chunk_log(struct nvme_nvm_chunk_desc *chunk_log,
Packit Service b7b338
				__u32 data_len)
Packit Service b7b338
{
Packit Service b7b338
	int nr_entry = data_len / sizeof(struct nvme_nvm_chunk_desc);
Packit Service b7b338
	int idx;
Packit Service b7b338
Packit Service b7b338
	printf("Total chunks in namespace: %d\n", nr_entry);
Packit Service b7b338
	for (idx = 0; idx < nr_entry; idx++) {
Packit Service b7b338
		struct nvme_nvm_chunk_desc *desc = &chunk_log[idx];
Packit Service b7b338
Packit Service b7b338
		printf(" [%5d] { ", idx);
Packit Service b7b338
		printf("SLBA: 0x%016"PRIx64, le64_to_cpu(desc->slba));
Packit Service b7b338
		printf(", WP: 0x%016"PRIx64, le64_to_cpu(desc->wp));
Packit Service b7b338
		printf(", CNLB: 0x%016"PRIx64, le64_to_cpu(desc->cnlb));
Packit Service b7b338
		printf(", State: %-8s", print_chunk_state(desc->cs));
Packit Service b7b338
		printf(", Type: %-20s", print_chunk_type(desc->ct));
Packit Service b7b338
		printf(", Attr: %-8s", print_chunk_attr(desc->ct));
Packit Service b7b338
		printf(", WLI: %4d }\n", desc->wli);
Packit Service b7b338
	}
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_chunk_log(int fd, __u32 nsid, __u32 data_len, void *data,
Packit Service b7b338
			unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	int err;
Packit Service b7b338
Packit Service b7b338
	err = nvme_get_log13(fd, nsid, NVM_LID_CHUNK_INFO, 0, 0, 0,
Packit Service b7b338
			false, data_len, data);
Packit Service b7b338
	if (err > 0) {
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x) NSID:%d\n",
Packit Service b7b338
			nvme_status_to_string(err), err, nsid);
Packit Service b7b338
Packit Service b7b338
		goto out;
Packit Service b7b338
	} else if (err < 0) {
Packit Service b7b338
		err = -errno;
Packit Service b7b338
		perror("nvme_get_log13");
Packit Service b7b338
Packit Service b7b338
		goto out;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (flags & BINARY)
Packit Service b7b338
		d_raw(data, data_len);
Packit Service b7b338
	else
Packit Service b7b338
		show_lnvm_chunk_log(data, data_len);
Packit Service b7b338
Packit Service b7b338
out:
Packit Service b7b338
	return err;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static void show_lnvm_bbtbl(struct nvme_nvm_bb_tbl *tbl)
Packit Service b7b338
{
Packit Service b7b338
	printf("verid    : %#x\n", (uint16_t)le16_to_cpu(tbl->verid));
Packit Service b7b338
	printf("tblks    : %d\n", (uint32_t)le32_to_cpu(tbl->tblks));
Packit Service b7b338
	printf("tfact    : %d\n", (uint32_t)le32_to_cpu(tbl->tfact));
Packit Service b7b338
	printf("tgrown   : %d\n", (uint32_t)le32_to_cpu(tbl->tgrown));
Packit Service b7b338
	printf("tdresv   : %d\n", (uint32_t)le32_to_cpu(tbl->tdresv));
Packit Service b7b338
	printf("thresv   : %d\n", (uint32_t)le32_to_cpu(tbl->thresv));
Packit Service b7b338
	printf("Use raw output to retrieve table.\n");
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static int __lnvm_do_get_bbtbl(int fd, struct nvme_nvm_id12 *id,
Packit Service b7b338
						struct ppa_addr ppa,
Packit Service b7b338
						unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_nvm_id12_group *grp = &id->groups[0];
Packit Service b7b338
	int bbtblsz = ((uint16_t)le16_to_cpu(grp->num_blk) * grp->num_pln);
Packit Service b7b338
	int bufsz = bbtblsz + sizeof(struct nvme_nvm_bb_tbl);
Packit Service b7b338
	struct nvme_nvm_bb_tbl *bbtbl;
Packit Service b7b338
	int err;
Packit Service b7b338
Packit Service b7b338
	bbtbl = calloc(1, bufsz);
Packit Service b7b338
	if (!bbtbl)
Packit Service b7b338
		return -ENOMEM;
Packit Service b7b338
Packit Service b7b338
	struct nvme_nvm_getbbtbl cmd = {
Packit Service b7b338
		.opcode		= nvme_nvm_admin_get_bb_tbl,
Packit Service b7b338
		.nsid		= cpu_to_le32(1),
Packit Service b7b338
		.addr		= (__u64)(uintptr_t)bbtbl,
Packit Service b7b338
		.data_len	= bufsz,
Packit Service b7b338
		.ppa		= cpu_to_le64(ppa.ppa),
Packit Service b7b338
	};
Packit Service b7b338
	void *tmp = &cm;;
Packit Service b7b338
	struct nvme_passthru_cmd *nvme_cmd = tmp;
Packit Service b7b338
Packit Service b7b338
	err = nvme_submit_passthru(fd, NVME_IOCTL_ADMIN_CMD, nvme_cmd);
Packit Service b7b338
	if (err > 0) {
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x)\n",
Packit Service b7b338
			nvme_status_to_string(err), err);
Packit Service b7b338
		free(bbtbl);
Packit Service b7b338
		return err;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (flags & BINARY)
Packit Service b7b338
		d_raw((unsigned char *)&bbtbl->blk, bbtblsz);
Packit Service b7b338
	else {
Packit Service b7b338
		printf("LightNVM Bad Block Stats:\n");
Packit Service b7b338
		show_lnvm_bbtbl(bbtbl);
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	free(bbtbl);
Packit Service b7b338
	return 0;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_get_bbtbl(int fd, int nsid, int lunid, int chid, unsigned int flags)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_nvm_id12 nvm_id;
Packit Service b7b338
	struct ppa_addr ppa;
Packit Service b7b338
	int err;
Packit Service b7b338
	void *tmp = &nvm_id;
Packit Service b7b338
Packit Service b7b338
	err = lnvm_get_identity(fd, nsid, (struct nvme_nvm_id *)tmp);
Packit Service b7b338
	if (err) {
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x)\n",
Packit Service b7b338
			nvme_status_to_string(err), err);
Packit Service b7b338
		return err;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (nvm_id.ver_id != 1) {
Packit Service b7b338
		fprintf(stderr, "Get bad block table not supported on version %d\n",
Packit Service b7b338
				nvm_id.ver_id);
Packit Service b7b338
		return -EINVAL;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (chid >= nvm_id.groups[0].num_ch ||
Packit Service b7b338
					lunid >= nvm_id.groups[0].num_lun) {
Packit Service b7b338
		fprintf(stderr, "Out of bound channel id or LUN id\n");
Packit Service b7b338
		return -EINVAL;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	ppa.ppa = 0;
Packit Service b7b338
	ppa.g.lun = lunid;
Packit Service b7b338
	ppa.g.ch = chid;
Packit Service b7b338
Packit Service b7b338
	ppa = generic_to_dev_addr(&nvm_id.ppaf, ppa);
Packit Service b7b338
Packit Service b7b338
	return __lnvm_do_get_bbtbl(fd, &nvm_id, ppa, flags);
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
static int __lnvm_do_set_bbtbl(int fd, struct ppa_addr ppa, __u8 value)
Packit Service b7b338
{
Packit Service b7b338
	int err;
Packit Service b7b338
Packit Service b7b338
	struct nvme_nvm_setbbtbl cmd = {
Packit Service b7b338
		.opcode		= nvme_nvm_admin_set_bb_tbl,
Packit Service b7b338
		.nsid		= cpu_to_le32(1),
Packit Service b7b338
		.ppa		= cpu_to_le64(ppa.ppa),
Packit Service b7b338
		.nlb		= cpu_to_le16(0),
Packit Service b7b338
		.value		= value,
Packit Service b7b338
	};
Packit Service b7b338
	void *tmp = &cm;;
Packit Service b7b338
	struct nvme_passthru_cmd *nvme_cmd = tmp;
Packit Service b7b338
Packit Service b7b338
	err = nvme_submit_passthru(fd, NVME_IOCTL_ADMIN_CMD, nvme_cmd);
Packit Service b7b338
	if (err > 0) {
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x)\n",
Packit Service b7b338
			nvme_status_to_string(err), err);
Packit Service b7b338
		return err;
Packit Service b7b338
	}
Packit Service b7b338
	return 0;
Packit Service b7b338
}
Packit Service b7b338
Packit Service b7b338
int lnvm_do_set_bbtbl(int fd, int nsid,
Packit Service b7b338
				int chid, int lunid, int plnid, int blkid,
Packit Service b7b338
				__u8 value)
Packit Service b7b338
{
Packit Service b7b338
	struct nvme_nvm_id12 nvm_id;
Packit Service b7b338
	struct ppa_addr ppa;
Packit Service b7b338
	int err;
Packit Service b7b338
	void *tmp = &nvm_id;
Packit Service b7b338
Packit Service b7b338
	err = lnvm_get_identity(fd, nsid, (struct nvme_nvm_id *)tmp);
Packit Service b7b338
	if (err) {
Packit Service b7b338
		fprintf(stderr, "NVMe Status:%s(%x)\n",
Packit Service b7b338
			nvme_status_to_string(err), err);
Packit Service b7b338
		return err;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (nvm_id.ver_id != 1) {
Packit Service b7b338
		fprintf(stderr, "Set bad block table not supported on version %d\n",
Packit Service b7b338
				nvm_id.ver_id);
Packit Service b7b338
		return -EINVAL;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	if (chid >= nvm_id.groups[0].num_ch ||
Packit Service b7b338
					lunid >= nvm_id.groups[0].num_lun ||
Packit Service b7b338
					plnid >= nvm_id.groups[0].num_pln ||
Packit Service b7b338
					blkid >= le16_to_cpu(nvm_id.groups[0].num_blk)) {
Packit Service b7b338
		fprintf(stderr, "Out of bound channel id, LUN id, plane id, or"\
Packit Service b7b338
				"block id\n");
Packit Service b7b338
		return -EINVAL;
Packit Service b7b338
	}
Packit Service b7b338
Packit Service b7b338
	ppa.ppa = 0;
Packit Service b7b338
	ppa.g.lun = lunid;
Packit Service b7b338
	ppa.g.ch = chid;
Packit Service b7b338
	ppa.g.pl = plnid;
Packit Service b7b338
	ppa.g.blk = blkid;
Packit Service b7b338
Packit Service b7b338
	ppa = generic_to_dev_addr(&nvm_id.ppaf, ppa);
Packit Service b7b338
Packit Service b7b338
	return __lnvm_do_set_bbtbl(fd, ppa, value);
Packit Service b7b338
}