]> granicus.if.org Git - zfs/commitdiff
zed crashes when devid not present
authorMatthew Ahrens <mahrens@delphix.com>
Fri, 26 Jul 2019 19:07:48 +0000 (12:07 -0700)
committerBrian Behlendorf <behlendorf1@llnl.gov>
Fri, 26 Jul 2019 19:07:48 +0000 (12:07 -0700)
zed core dumps due to a NULL pointer in zfs_agent_iter_vdev(). The
gs_devid is NULL, but the nvl has a "devid" entry.

zfs_agent_post_event() checks that ZFS_EV_VDEV_GUID or DEV_IDENTIFIER is
present in nvl, but then later it and zfs_agent_iter_vdev() assume that
DEV_IDENTIFIER is present and thus gs_devid is set.

Typically this is not a problem because usually either all vdevs have
devid's, or none of them do. Since zfs_agent_iter_vdev() first checks if
the vdev has devid before dereferencing gs_devid, the problem isn't
typically encountered. However, if some vdevs have devid's and some do
not, then the problem is easily reproduced.  This can happen if the pool
has been moved from a system that has devid's to one that does not.

The fix is for zfs_agent_iter_vdev() to only try to match the devid's if
both nvl and gsp have devid's present.

Reviewed-by: Prashanth Sreenivasa <pks@delphix.com>
Reviewed-by: Don Brady <don.brady@delphix.com>
Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Reviewed-by: loli10K <ezomori.nozomu@gmail.com>
Signed-off-by: Matthew Ahrens <mahrens@delphix.com>
External-issue: DLPX-65090
Closes #9054
Closes #9060

cmd/zed/agents/zfs_agents.c

index 6d392604bcebe4b4d5f649400c0b1fd19eb6f125..006e0ab99f47d08418b41417a6e28da1da88bbe1 100644 (file)
@@ -116,7 +116,8 @@ zfs_agent_iter_vdev(zpool_handle_t *zhp, nvlist_t *nvl, void *arg)
        /*
         * On a devid match, grab the vdev guid and expansion time, if any.
         */
-       if ((nvlist_lookup_string(nvl, ZPOOL_CONFIG_DEVID, &path) == 0) &&
+       if (gsp->gs_devid != NULL &&
+           (nvlist_lookup_string(nvl, ZPOOL_CONFIG_DEVID, &path) == 0) &&
            (strcmp(gsp->gs_devid, path) == 0)) {
                (void) nvlist_lookup_uint64(nvl, ZPOOL_CONFIG_GUID,
                    &gsp->gs_vdev_guid);