Blame SOURCES/libvirt-RHEL-virscsi-Check-device-type-before-getting-it-s-dev-node-name.patch

fbe740
From f66beef45382be2aed6d021a409e90f8114c8671 Mon Sep 17 00:00:00 2001
fbe740
Message-Id: <f66beef45382be2aed6d021a409e90f8114c8671@dist-git>
fbe740
From: Michal Privoznik <mprivozn@redhat.com>
fbe740
Date: Fri, 6 Mar 2020 15:52:21 +0100
fbe740
Subject: [PATCH] RHEL: virscsi: Check device type before getting it's /dev
fbe740
 node name
fbe740
fbe740
Not all SCSI devices are block devices, therefore
fbe740
/sys/bus/scsi/devices/X:X:X:X/block/ directory does not always
fbe740
exist. Check if the SCSI device is a block device beforehand.
fbe740
fbe740
https://bugzilla.redhat.com/show_bug.cgi?id=1808390
fbe740
fbe740
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
fbe740
Signed-off-by: Andrea Bolognani <abologna@redhat.com>
fbe740
Message-Id: <20200306145226.1610708-2-abologna@redhat.com>
fbe740
Reviewed-by: Jiri Denemark <jdenemar@redhat.com>
fbe740
---
fbe740
 src/util/virscsi.c             | 146 ++++++++++++++++++++++++++++++---
fbe740
 tests/virscsidata/0-0-0-0/type |   1 +
fbe740
 tests/virscsidata/1-0-0-0/type |   1 +
fbe740
 3 files changed, 137 insertions(+), 11 deletions(-)
fbe740
 create mode 100644 tests/virscsidata/0-0-0-0/type
fbe740
 create mode 100644 tests/virscsidata/1-0-0-0/type
fbe740
fbe740
diff --git a/src/util/virscsi.c b/src/util/virscsi.c
fbe740
index 06659c45c7..c40857977f 100644
fbe740
--- a/src/util/virscsi.c
fbe740
+++ b/src/util/virscsi.c
fbe740
@@ -50,6 +50,32 @@ struct _virUsedByInfo {
fbe740
 typedef struct _virUsedByInfo virUsedByInfo;
fbe740
 typedef virUsedByInfo *virUsedByInfoPtr;
fbe740
 
fbe740
+
fbe740
+/* Keep in sync with scsi/scsi_proto.h */
fbe740
+typedef enum {
fbe740
+    VIR_SCSI_DEVICE_TYPE_NONE = -1,
fbe740
+    VIR_SCSI_DEVICE_TYPE_DISK = 0x00,
fbe740
+    VIR_SCSI_DEVICE_TYPE_TAPE = 0x01,
fbe740
+    VIR_SCSI_DEVICE_TYPE_PRINTER = 0x02,
fbe740
+    VIR_SCSI_DEVICE_TYPE_PROCESSOR = 0x03,
fbe740
+    VIR_SCSI_DEVICE_TYPE_WORM = 0x04,
fbe740
+    VIR_SCSI_DEVICE_TYPE_ROM = 0x05,
fbe740
+    VIR_SCSI_DEVICE_TYPE_SCANNER = 0x06,
fbe740
+    VIR_SCSI_DEVICE_TYPE_MOD = 0x07,
fbe740
+    VIR_SCSI_DEVICE_TYPE_MEDIUM_CHANGER = 0x08,
fbe740
+    VIR_SCSI_DEVICE_TYPE_COMM = 0x09,
fbe740
+    VIR_SCSI_DEVICE_TYPE_RAID = 0x0c,
fbe740
+    VIR_SCSI_DEVICE_TYPE_ENCLOSURE = 0x0d,
fbe740
+    VIR_SCSI_DEVICE_TYPE_RBC = 0x0e,
fbe740
+    VIR_SCSI_DEVICE_TYPE_OSD = 0x11,
fbe740
+    VIR_SCSI_DEVICE_TYPE_ZBC = 0x14,
fbe740
+    VIR_SCSI_DEVICE_TYPE_WLUN = 0x1e,
fbe740
+    VIR_SCSI_DEVICE_TYPE_NO_LUN = 0x7f,
fbe740
+
fbe740
+    VIR_SCSI_DEVICE_TYPE_LAST,
fbe740
+} virSCSIDeviceType;
fbe740
+
fbe740
+
fbe740
 struct _virSCSIDevice {
fbe740
     unsigned int adapter;
fbe740
     unsigned int bus;
fbe740
@@ -134,6 +160,84 @@ virSCSIDeviceGetSgName(const char *sysfs_prefix,
fbe740
     return sg;
fbe740
 }
fbe740
 
fbe740
+
fbe740
+static int
fbe740
+virSCSIDeviceGetType(const char *prefix,
fbe740
+                     unsigned int adapter,
fbe740
+                     unsigned int bus,
fbe740
+                     unsigned int target,
fbe740
+                     unsigned long long unit,
fbe740
+                     virSCSIDeviceType *type)
fbe740
+{
fbe740
+    int intType;
fbe740
+
fbe740
+    if (virFileReadValueInt(&intType,
fbe740
+                            "%s/%d:%u:%u:%llu/type",
fbe740
+                            prefix, adapter, bus, target, unit) < 0)
fbe740
+        return -1;
fbe740
+
fbe740
+    switch (intType) {
fbe740
+    case VIR_SCSI_DEVICE_TYPE_DISK:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_TAPE:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_PRINTER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_PROCESSOR:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_WORM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ROM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_SCANNER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_MOD:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_MEDIUM_CHANGER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_COMM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_RAID:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ENCLOSURE:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_RBC:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_OSD:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ZBC:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_WLUN:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_NO_LUN:
fbe740
+        *type = intType;
fbe740
+        break;
fbe740
+
fbe740
+    default:
fbe740
+        virReportError(VIR_ERR_INTERNAL_ERROR,
fbe740
+                       _("unknown SCSI device type: %x"),
fbe740
+                       intType);
fbe740
+        return -1;
fbe740
+    }
fbe740
+
fbe740
+    return 0;
fbe740
+}
fbe740
+
fbe740
+
fbe740
+static char *
fbe740
+virSCSIDeviceGetDevNameBlock(const char *prefix,
fbe740
+                             unsigned int adapter,
fbe740
+                             unsigned int bus,
fbe740
+                             unsigned int target,
fbe740
+                             unsigned long long unit)
fbe740
+{
fbe740
+    DIR *dir = NULL;
fbe740
+    struct dirent *entry;
fbe740
+    g_autofree char *path = NULL;
fbe740
+    char *name = NULL;
fbe740
+
fbe740
+    path = g_strdup_printf("%s/%d:%u:%u:%llu/block",
fbe740
+                           prefix, adapter, bus, target, unit);
fbe740
+
fbe740
+    if (virDirOpen(&dir, path) < 0)
fbe740
+        goto cleanup;
fbe740
+
fbe740
+    while (virDirRead(dir, &entry, path) > 0) {
fbe740
+        name = g_strdup(entry->d_name);
fbe740
+        break;
fbe740
+    }
fbe740
+
fbe740
+ cleanup:
fbe740
+    VIR_DIR_CLOSE(dir);
fbe740
+
fbe740
+    return name;
fbe740
+}
fbe740
+
fbe740
+
fbe740
 /* Returns device name (e.g. "sdc") on success, or NULL
fbe740
  * on failure.
fbe740
  */
fbe740
@@ -144,32 +248,52 @@ virSCSIDeviceGetDevName(const char *sysfs_prefix,
fbe740
                         unsigned int target,
fbe740
                         unsigned long long unit)
fbe740
 {
fbe740
-    DIR *dir = NULL;
fbe740
-    struct dirent *entry;
fbe740
-    g_autofree char *path = NULL;
fbe740
     char *name = NULL;
fbe740
     unsigned int adapter_id;
fbe740
+    virSCSIDeviceType type;
fbe740
     const char *prefix = sysfs_prefix ? sysfs_prefix : SYSFS_SCSI_DEVICES;
fbe740
 
fbe740
     if (virSCSIDeviceGetAdapterId(adapter, &adapter_id) < 0)
fbe740
         return NULL;
fbe740
 
fbe740
-    path = g_strdup_printf("%s/%d:%u:%u:%llu/block", prefix, adapter_id, bus,
fbe740
-                           target, unit);
fbe740
+    if (virSCSIDeviceGetType(prefix, adapter_id,
fbe740
+                             bus, target, unit, &type) < 0)
fbe740
+        return NULL;
fbe740
 
fbe740
-    if (virDirOpen(&dir, path) < 0)
fbe740
-        goto cleanup;
fbe740
+    switch (type) {
fbe740
+    case VIR_SCSI_DEVICE_TYPE_DISK:
fbe740
+        name = virSCSIDeviceGetDevNameBlock(prefix, adapter_id, bus, target, unit);
fbe740
+        break;
fbe740
 
fbe740
-    while (virDirRead(dir, &entry, path) > 0) {
fbe740
-        name = g_strdup(entry->d_name);
fbe740
+    case VIR_SCSI_DEVICE_TYPE_TAPE:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_PRINTER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_PROCESSOR:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_WORM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ROM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_SCANNER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_MOD:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_MEDIUM_CHANGER:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_COMM:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_RAID:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ENCLOSURE:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_RBC:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_OSD:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_ZBC:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_WLUN:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_NO_LUN:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_NONE:
fbe740
+    case VIR_SCSI_DEVICE_TYPE_LAST:
fbe740
+    default:
fbe740
+        virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
fbe740
+                       _("unsupported SCSI device type: %x"),
fbe740
+                       type);
fbe740
         break;
fbe740
     }
fbe740
 
fbe740
- cleanup:
fbe740
-    VIR_DIR_CLOSE(dir);
fbe740
     return name;
fbe740
 }
fbe740
 
fbe740
+
fbe740
 virSCSIDevicePtr
fbe740
 virSCSIDeviceNew(const char *sysfs_prefix,
fbe740
                  const char *adapter,
fbe740
diff --git a/tests/virscsidata/0-0-0-0/type b/tests/virscsidata/0-0-0-0/type
fbe740
new file mode 100644
fbe740
index 0000000000..573541ac97
fbe740
--- /dev/null
fbe740
+++ b/tests/virscsidata/0-0-0-0/type
fbe740
@@ -0,0 +1 @@
fbe740
+0
fbe740
diff --git a/tests/virscsidata/1-0-0-0/type b/tests/virscsidata/1-0-0-0/type
fbe740
new file mode 100644
fbe740
index 0000000000..573541ac97
fbe740
--- /dev/null
fbe740
+++ b/tests/virscsidata/1-0-0-0/type
fbe740
@@ -0,0 +1 @@
fbe740
+0
fbe740
-- 
fbe740
2.25.1
fbe740