multipathd: ignore failed wwid recheck
authorBenjamin Marzinski <bmarzins@redhat.com>
Sat, 30 Mar 2019 06:05:56 +0000 (01:05 -0500)
committerChristophe Varoqui <christophe.varoqui@opensvc.com>
Thu, 18 Apr 2019 11:03:34 +0000 (13:03 +0200)
If disable_changed_wwids is set, when multipathd gets a change event on
a path, it verifies that the wwid hasn't changed in uev_update_path().
If get_uid() failed, uev_update_path treated this as a wwid change to 0.
This could cause paths to suddenly be dropped due to an issue with
getting the wwid.  Even if get_uid() failed because the path was down,
it no change uevent happend when it later became active, multipathd
would continue to ignore the path. Also, scsi_uid_fallback() clears the
failure return if it doesn't attempt to fallback, causing get_uid()
to return success, when it actually failed.

Multipathd should neither set nor clear wwid_changed if get_uid()
returned failure. Also, scsi_uid_fallback() should retain the old return
value if it doesn't attempt to fallback.

Signed-off-by: Benjamin Marzinski <bmarzins@redhat.com>
libmultipath/discovery.c
multipathd/main.c

index 729bcb9..b08cb2d 100644 (file)
@@ -1755,9 +1755,9 @@ get_vpd_uid(struct path * pp)
 }
 
 static ssize_t scsi_uid_fallback(struct path *pp, int path_state,
-                            const char **origin)
+                            const char **origin, ssize_t old_len)
 {
-       ssize_t len = 0;
+       ssize_t len = old_len;
        int retrigger;
        struct config *conf;
 
@@ -1828,7 +1828,7 @@ get_uid (struct path * pp, int path_state, struct udev_device *udev)
                        origin = "sysfs";
                }
                if (len <= 0 && pp->bus == SYSFS_BUS_SCSI)
-                       len = scsi_uid_fallback(pp, path_state, &origin);
+                       len = scsi_uid_fallback(pp, path_state, &origin, len);
        }
        if ( len < 0 ) {
                condlog(1, "%s: failed to get %s uid: %s",
index 678ecf8..fd83a6a 100644 (file)
@@ -1234,9 +1234,11 @@ uev_update_path (struct uevent *uev, struct vectors * vecs)
                        goto out;
 
                strcpy(wwid, pp->wwid);
-               get_uid(pp, pp->state, uev->udev);
+               rc = get_uid(pp, pp->state, uev->udev);
 
-               if (strncmp(wwid, pp->wwid, WWID_SIZE) != 0) {
+               if (rc != 0)
+                       strcpy(pp->wwid, wwid);
+               else if (strncmp(wwid, pp->wwid, WWID_SIZE) != 0) {
                        condlog(0, "%s: path wwid changed from '%s' to '%s'. %s",
                                uev->kernel, wwid, pp->wwid,
                                (disable_changed_wwids ? "disallowing" :