Blame SOURCES/0057-Handle-sys-devices-virtual-nvme-fabrics-nvme-subsyst.patch

4e0e09
From ad1d30fc5e20b933b6ad59d35c13e0193cd68a2d Mon Sep 17 00:00:00 2001
4e0e09
From: Peter Jones <pjones@redhat.com>
4e0e09
Date: Wed, 2 Oct 2019 17:04:12 -0400
4e0e09
Subject: [PATCH 57/63] Handle
4e0e09
 /sys/devices/virtual/{nvme-fabrics,nvme-subsystem} devices
4e0e09
4e0e09
Signed-off-by: Peter Jones <pjones@redhat.com>
4e0e09
---
4e0e09
 src/linux-nvme.c         | 59 ++++++++++++++++++++++-----
4e0e09
 src/linux-virtual-root.c | 88 ++++++++++++++++++++++++++++++++++++++++
4e0e09
 src/linux.c              | 43 +++++++++++++++++---
4e0e09
 src/linux.h              |  4 +-
4e0e09
 4 files changed, 176 insertions(+), 18 deletions(-)
4e0e09
 create mode 100644 src/linux-virtual-root.c
4e0e09
4e0e09
diff --git a/src/linux-nvme.c b/src/linux-nvme.c
4e0e09
index 455c4c7ba9b..7ca2fa3c283 100644
4e0e09
--- a/src/linux-nvme.c
4e0e09
+++ b/src/linux-nvme.c
4e0e09
@@ -1,6 +1,6 @@
4e0e09
 /*
4e0e09
  * libefiboot - library for the manipulation of EFI boot variables
4e0e09
- * Copyright 2012-2018 Red Hat, Inc.
4e0e09
+ * Copyright 2012-2019 Red Hat, Inc.
4e0e09
  *
4e0e09
  * This library is free software; you can redistribute it and/or
4e0e09
  * modify it under the terms of the GNU Lesser General Public License as
4e0e09
@@ -15,7 +15,6 @@
4e0e09
  * You should have received a copy of the GNU Lesser General Public
4e0e09
  * License along with this library; if not, see
4e0e09
  * <http://www.gnu.org/licenses/>.
4e0e09
- *
4e0e09
  */
4e0e09
 
4e0e09
 #include "fix_coverity.h"
4e0e09
@@ -24,6 +23,7 @@
4e0e09
 #include <fcntl.h>
4e0e09
 #include <inttypes.h>
4e0e09
 #include <stdint.h>
4e0e09
+#include <sys/param.h>
4e0e09
 #include <unistd.h>
4e0e09
 
4e0e09
 #include "efiboot.h"
4e0e09
@@ -34,6 +34,12 @@
4e0e09
  * /sys/dev/block/$major:$minor looks like:
4e0e09
  * 259:0 -> ../../devices/pci0000:00/0000:00:1d.0/0000:05:00.0/nvme/nvme0/nvme0n1
4e0e09
  * 259:1 -> ../../devices/pci0000:00/0000:00:1d.0/0000:05:00.0/nvme/nvme0/nvme0n1/nvme0n1p1
4e0e09
+ * or:
4e0e09
+ * 259:0 ->../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1
4e0e09
+ * 259:1 ->../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1/nvme0n1p1
4e0e09
+ * or:
4e0e09
+ * 259:5 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1
4e0e09
+ * 259:6 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1/nvme0n1p1
4e0e09
  *
4e0e09
  * /sys/dev/block/259:0/device looks like:
4e0e09
  * device -> ../../nvme0
4e0e09
@@ -53,17 +59,44 @@ parse_nvme(struct device *dev, const char *current, const char *root UNUSED)
4e0e09
 	int rc;
4e0e09
 	int32_t tosser0, tosser1, tosser2, ctrl_id, ns_id, partition;
4e0e09
 	uint8_t *filebuf = NULL;
4e0e09
+	int pos0 = -1, pos1 = -1, pos2 = -1;
4e0e09
 	ssize_t sz = 0;
4e0e09
-	int pos0 = 0, pos1 = 0;
4e0e09
+	struct subdir {
4e0e09
+		const char * const name;
4e0e09
+		const char * const fmt;
4e0e09
+		int *pos0, *pos1;
4e0e09
+	} subdirs[] = {
4e0e09
+		{"nvme-subsysN/", "%nnvme-subsys%d/%n", &pos0, &pos2},
4e0e09
+		{"ctl/", "%nctl/%n%n", &pos0, &pos1},
4e0e09
+		{"nvme/", "%nnvme/%n%n", &pos0, &pos1},
4e0e09
+		{NULL, }
4e0e09
+	};
4e0e09
 
4e0e09
 	debug("entry");
4e0e09
 
4e0e09
-	debug("searching for nvme/nvme0/nvme0n1 or nvme/nvme0/nvme0n1/nvme0n1p1");
4e0e09
-	rc = sscanf(current, "nvme/nvme%d/nvme%dn%d%n/nvme%dn%dp%d%n",
4e0e09
-	            &tosser0, &ctrl_id, &ns_id, &pos0,
4e0e09
-	            &tosser1, &tosser2, &partition, &pos1);
4e0e09
-	debug("current:'%s' rc:%d pos0:%d pos1:%d\n", current, rc, pos0, pos1);
4e0e09
-	dbgmk("         ", pos0, pos1);
4e0e09
+	/*
4e0e09
+	 * in this case, *any* of these is okay.
4e0e09
+	 */
4e0e09
+	for (int i = 0; subdirs[i].name; i++) {
4e0e09
+		debug("searching for %s", subdirs[i].name);
4e0e09
+		pos0 = tosser0 = pos1 = -1;
4e0e09
+		rc = sscanf(current, subdirs[i].fmt, &pos0, &pos1, &pos2);
4e0e09
+		debug("current:'%s' rc:%d pos0:%d pos1:%d\n", current, rc,
4e0e09
+		      *subdirs[i].pos0, *subdirs[i].pos1);
4e0e09
+		dbgmk("         ", *subdirs[i].pos0, *subdirs[i].pos1);
4e0e09
+		if (*subdirs[i].pos0 >= 0 && *subdirs[i].pos1 >= *subdirs[i].pos0) {
4e0e09
+			sz += *subdirs[i].pos1;
4e0e09
+			current += *subdirs[i].pos1;
4e0e09
+			break;
4e0e09
+		}
4e0e09
+	}
4e0e09
+
4e0e09
+	debug("searching for nvme0/nvme0n1 or nvme0/nvme0n1/nvme0n1p1");
4e0e09
+	rc = sscanf(current, "%nnvme%d/nvme%dn%d%n/nvme%dn%dp%d%n",
4e0e09
+	            &pos0, &tosser0, &ctrl_id, &ns_id, &pos1,
4e0e09
+	            &tosser1, &tosser2, &partition, &pos2);
4e0e09
+	debug("current:'%s' rc:%d pos0:%d pos1:%d pos2:%d\n", current, rc, pos0, pos1, pos2);
4e0e09
+	dbgmk("         ", pos0, MAX(pos1,pos2));
4e0e09
 	/*
4e0e09
 	 * If it isn't of that form, it's not one of our nvme devices.
4e0e09
 	 */
4e0e09
@@ -79,14 +112,15 @@ parse_nvme(struct device *dev, const char *current, const char *root UNUSED)
4e0e09
 	        if (dev->part == -1)
4e0e09
 	                dev->part = partition;
4e0e09
 
4e0e09
-	        pos0 = pos1;
4e0e09
+		pos1 = pos2;
4e0e09
 	}
4e0e09
 
4e0e09
-	sz += pos0;
4e0e09
+	sz += pos1;
4e0e09
 
4e0e09
 	/*
4e0e09
 	 * now fish the eui out of sysfs is there is one...
4e0e09
 	 */
4e0e09
+	debug("looking for the eui");
4e0e09
 	char *euipath = NULL;
4e0e09
 	rc = read_sysfs_file(&filebuf, "class/block/nvme%dn%d/eui", ctrl_id, ns_id);
4e0e09
 	if (rc < 0 && (errno == ENOENT || errno == ENOTDIR)) {
4e0e09
@@ -109,6 +143,9 @@ parse_nvme(struct device *dev, const char *current, const char *root UNUSED)
4e0e09
 	                errno = EINVAL;
4e0e09
 	                return -1;
4e0e09
 	        }
4e0e09
+		debug("eui is %02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx",
4e0e09
+		      eui[0], eui[1], eui[2], eui[3],
4e0e09
+		      eui[4], eui[5], eui[6], eui[7]);
4e0e09
 	        dev->nvme_info.has_eui = 1;
4e0e09
 	        memcpy(dev->nvme_info.eui, eui, sizeof(eui));
4e0e09
 	}
4e0e09
diff --git a/src/linux-virtual-root.c b/src/linux-virtual-root.c
4e0e09
new file mode 100644
4e0e09
index 00000000000..b2d36b4095f
4e0e09
--- /dev/null
4e0e09
+++ b/src/linux-virtual-root.c
4e0e09
@@ -0,0 +1,88 @@
4e0e09
+/*
4e0e09
+ * libefiboot - library for the manipulation of EFI boot variables
4e0e09
+ * Copyright 2012-2019 Red Hat, Inc.
4e0e09
+ *
4e0e09
+ * This library is free software; you can redistribute it and/or
4e0e09
+ * modify it under the terms of the GNU Lesser General Public License as
4e0e09
+ * published by the Free Software Foundation; either version 2.1 of the
4e0e09
+ * License, or (at your option) any later version.
4e0e09
+ *
4e0e09
+ * This library is distributed in the hope that it will be useful,
4e0e09
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
4e0e09
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
4e0e09
+ * Lesser General Public License for more details.
4e0e09
+ *
4e0e09
+ * You should have received a copy of the GNU Lesser General Public
4e0e09
+ * License along with this library; if not, see
4e0e09
+ * <http://www.gnu.org/licenses/>.
4e0e09
+ */
4e0e09
+
4e0e09
+#include "fix_coverity.h"
4e0e09
+
4e0e09
+#include <errno.h>
4e0e09
+#include <fcntl.h>
4e0e09
+#include <inttypes.h>
4e0e09
+#include <stdint.h>
4e0e09
+#include <unistd.h>
4e0e09
+
4e0e09
+#include "efiboot.h"
4e0e09
+
4e0e09
+/*
4e0e09
+ * Support virtually rooted devices (fibre+nvme, etc.)
4e0e09
+ *
4e0e09
+ * /sys/dev/block/$major:$minor looks like:
4e0e09
+ * 259:0 ->../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1
4e0e09
+ * 259:1 ->../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1/nvme0n1p1
4e0e09
+ * or:
4e0e09
+ * 259:5 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1
4e0e09
+ * 259:6 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1/nvme0n1p1
4e0e09
+ */
4e0e09
+
4e0e09
+static ssize_t
4e0e09
+parse_virtual_root(struct device *dev UNUSED, const char *current, const char *root UNUSED)
4e0e09
+{
4e0e09
+	int rc;
4e0e09
+	ssize_t sz;
4e0e09
+	int pos0 = 0, pos1 = 0;
4e0e09
+	struct subdir {
4e0e09
+		const char * const name;
4e0e09
+		const char * const fmt;
4e0e09
+	} subdirs[] = {
4e0e09
+		{"../../devices/virtual", "%n../../devices/virtual/%n"},
4e0e09
+		{"nvme-subsystem/", "%nnvme-subsystem/%n"},
4e0e09
+		{"nvme-fabrics/ctl/", "%nnvme-fabrics/ctl/%n"},
4e0e09
+		{NULL, NULL}
4e0e09
+	};
4e0e09
+
4e0e09
+	debug("entry");
4e0e09
+
4e0e09
+	for (int i = 0; subdirs[i].name; i++) {
4e0e09
+		debug("searching for %s", subdirs[i].name);
4e0e09
+		pos0 = pos1 = -1;
4e0e09
+		rc = sscanf(current, subdirs[i].fmt, &pos0, &pos1);
4e0e09
+		debug("current:'%s' rc:%d pos0:%d pos1:%d\n", current, rc, pos0, pos1);
4e0e09
+		dbgmk("        ", pos0, pos1);
4e0e09
+		if (rc == 1) {
4e0e09
+			sz += pos1;
4e0e09
+			current += pos1;
4e0e09
+			if (i > 0)
4e0e09
+				goto found;
4e0e09
+		}
4e0e09
+	}
4e0e09
+
4e0e09
+	sz = 0;
4e0e09
+found:
4e0e09
+	debug("current:'%s' sz:%zd\n", current, sz);
4e0e09
+	return sz;
4e0e09
+}
4e0e09
+
4e0e09
+static enum interface_type virtual_root_iftypes[] = { virtual_root, unknown };
4e0e09
+
4e0e09
+struct dev_probe HIDDEN virtual_root_parser = {
4e0e09
+	.name = "virtual_root",
4e0e09
+	.iftypes = virtual_root_iftypes,
4e0e09
+	.flags = DEV_ABBREV_ONLY|DEV_PROVIDES_ROOT,
4e0e09
+	.parse = parse_virtual_root,
4e0e09
+};
4e0e09
+
4e0e09
+// vim:fenc=utf-8:tw=75:noet
4e0e09
diff --git a/src/linux.c b/src/linux.c
4e0e09
index 30db22d95dd..7dd8d4cd858 100644
4e0e09
--- a/src/linux.c
4e0e09
+++ b/src/linux.c
4e0e09
@@ -170,16 +170,17 @@ int HIDDEN
4e0e09
 set_disk_and_part_name(struct device *dev)
4e0e09
 {
4e0e09
 	int rc = -1;
4e0e09
-
4e0e09
-	/*
4e0e09
-	 * results are like such:
4e0e09
-	 * maj:min -> ../../devices/pci$PCI_STUFF/$BLOCKDEV_STUFF/block/$DISK/$PART
4e0e09
-	 */
4e0e09
-
4e0e09
 	char *ultimate = pathseg(dev->link, -1);
4e0e09
 	char *penultimate = pathseg(dev->link, -2);
4e0e09
 	char *approximate = pathseg(dev->link, -3);
4e0e09
 	char *proximate = pathseg(dev->link, -4);
4e0e09
+	char *psl5 = pathseg(dev->link, -5);
4e0e09
+
4e0e09
+
4e0e09
+	/*
4e0e09
+	 * devlinks look something like:
4e0e09
+	 * maj:min -> ../../devices/pci$PCI_STUFF/$BLOCKDEV_STUFF/block/$DISK/$PART
4e0e09
+	 */
4e0e09
 
4e0e09
 	errno = 0;
4e0e09
 	debug("dev->disk_name:%p dev->part_name:%p", dev->disk_name, dev->part_name);
4e0e09
@@ -188,6 +189,7 @@ set_disk_and_part_name(struct device *dev)
4e0e09
 	debug("penultimate:'%s'", penultimate ? : "");
4e0e09
 	debug("approximate:'%s'", approximate ? : "");
4e0e09
 	debug("proximate:'%s'", proximate ? : "");
4e0e09
+	debug("psl5:'%s'", psl5 ? : "");
4e0e09
 
4e0e09
 	if (ultimate && penultimate &&
4e0e09
 	    ((proximate && !strcmp(proximate, "nvme")) ||
4e0e09
@@ -232,6 +234,34 @@ set_disk_and_part_name(struct device *dev)
4e0e09
 	        set_disk_name(dev, "%s", ultimate);
4e0e09
 	        debug("disk:%s", ultimate);
4e0e09
 		rc = 0;
4e0e09
+	} else if ((proximate && ultimate && !strcmp(proximate, "nvme-fabrics")) ||
4e0e09
+		    (approximate && ultimate && !strcmp(approximate, "nvme-subsystem"))) {
4e0e09
+		/*
4e0e09
+		 * 259:0 ->../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1
4e0e09
+		 *				 ^ proximate            ^ ultimate
4e0e09
+		 * or
4e0e09
+		 * 259:5 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1
4e0e09
+		 *                                ^ approximate  ^ penultimate
4e0e09
+		 *                                                   ultimate ^
4e0e09
+		 */
4e0e09
+		set_disk_name(dev, "%s", ultimate);
4e0e09
+		debug("disk:%s", ultimate);
4e0e09
+		rc = 0;
4e0e09
+	} else if ((psl5 && penultimate && ultimate && !strcmp(psl5, "nvme-fabrics")) ||
4e0e09
+		   (proximate && penultimate && ultimate && !strcmp(proximate, "nvme-subsystem"))) {
4e0e09
+		/*
4e0e09
+		 * 259:1 -> ../../devices/virtual/nvme-fabrics/ctl/nvme0/nvme0n1/nvme0n1p1
4e0e09
+		 *                                ^psl5                  ^ penultimate
4e0e09
+		 *                                                      ultimate ^
4e0e09
+		 * or
4e0e09
+		 * 259:6 -> ../../devices/virtual/nvme-subsystem/nvme-subsys0/nvme0n1/nvme0n1p1
4e0e09
+		 *                                ^ proximate                 ^ penultimate
4e0e09
+		 *                                                           ultimate ^
4e0e09
+		 */
4e0e09
+		set_disk_name(dev, "%s", penultimate);
4e0e09
+		set_part_name(dev, "%s", ultimate);
4e0e09
+		debug("disk:%s part:%s", penultimate, ultimate);
4e0e09
+		rc = 0;
4e0e09
 	}
4e0e09
 
4e0e09
 	if (rc < 0)
4e0e09
@@ -248,6 +278,7 @@ static struct dev_probe *dev_probes[] = {
4e0e09
 	&acpi_root_parser,
4e0e09
 	&pci_root_parser,
4e0e09
 	&soc_root_parser,
4e0e09
+	&virtual_root_parser,
4e0e09
 	&pci_parser,
4e0e09
 	&virtblk_parser,
4e0e09
 	&sas_parser,
4e0e09
diff --git a/src/linux.h b/src/linux.h
4e0e09
index ae9835ef7ce..6bfc5869254 100644
4e0e09
--- a/src/linux.h
4e0e09
+++ b/src/linux.h
4e0e09
@@ -99,7 +99,8 @@ struct emmc_info {
4e0e09
 
4e0e09
 enum interface_type {
4e0e09
 	unknown,
4e0e09
-	isa, acpi_root, pci_root, soc_root, pci, network,
4e0e09
+	isa, acpi_root, pci_root, soc_root, virtual_root,
4e0e09
+	pci, network,
4e0e09
 	ata, atapi, scsi, sata, sas,
4e0e09
 	usb, i1394, fibre, i2o,
4e0e09
 	md, virtblk,
4e0e09
@@ -346,6 +347,7 @@ extern struct dev_probe pmem_parser;
4e0e09
 extern struct dev_probe pci_root_parser;
4e0e09
 extern struct dev_probe acpi_root_parser;
4e0e09
 extern struct dev_probe soc_root_parser;
4e0e09
+extern struct dev_probe virtual_root_parser;
4e0e09
 extern struct dev_probe pci_parser;
4e0e09
 extern struct dev_probe sas_parser;
4e0e09
 extern struct dev_probe sata_parser;
4e0e09
-- 
4e0e09
2.26.2
4e0e09