Right now, multipathd only ever calls pathinfo on a path outside of the vecs lock in one circumstance, when it's adding a new path from a uevent. Doing this can cause weirdness or crash multipathd. First off, the path checker code is written assuming that only one checker instance will be active at a time. If this isn't the case, two processes can modify the checker's refcounts at the same time, potentially causing a checker to get removed while still in use (if two threads try to increment the refcount at the same time, causing it to only increment once). Another issue is that since the vecs lock isn't grabbed until after all of the pathinfo is gathered, and the vecs lock is the only thing keeping the uevent servicing thread from running at the same time as a reconfigure is happening, it is possible to have multipathd add a path using the old config after reconfigure has already added that path using the new config. Leading to two versions of the path on the pathvec. There are possibly other issues with running pathinfo outside of the vecs lock that I haven't noticed, as well. The easiest solution is to call pathinfo on the path while holding the vecs lock, to keep other threads from modifying the checker structure or the config while setting up the path. This is what happens every other time multipathd calls pathinfo, including when a path is added through the multipathd command interface. This does mean that any time spent calling pathinfo on new paths will block other threads that need the vecs lock, but since the checker is now always called in async mode, and the prio function will use a short timeout if called on a failed path, this shouldn't be any more noticeable than the calls to check_path() for already existing paths. Signed-off-by: Benjamin Marzinski <bmarzins@xxxxxxxxxx> --- multipathd/main.c | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/multipathd/main.c b/multipathd/main.c index 7b364cfe..9e01cfaa 100644 --- a/multipathd/main.c +++ b/multipathd/main.c @@ -906,9 +906,8 @@ uev_add_path (struct uevent *uev, struct vectors * vecs, int need_do_map) } } } - lock_cleanup_pop(vecs->lock); if (pp) - return ret; + goto out; /* * get path vital state @@ -920,13 +919,13 @@ uev_add_path (struct uevent *uev, struct vectors * vecs, int need_do_map) pthread_cleanup_pop(1); if (!pp) { if (ret == PATHINFO_SKIPPED) - return 0; - condlog(3, "%s: failed to get path info", uev->kernel); - return 1; + ret = 0; + else { + condlog(3, "%s: failed to get path info", uev->kernel); + ret = 1; + } + goto out; } - pthread_cleanup_push(cleanup_lock, &vecs->lock); - lock(&vecs->lock); - pthread_testcancel(); ret = store_path(vecs->pathvec, pp); if (!ret) { conf = get_multipath_config(); @@ -940,6 +939,7 @@ uev_add_path (struct uevent *uev, struct vectors * vecs, int need_do_map) free_path(pp); ret = 1; } +out: lock_cleanup_pop(vecs->lock); return ret; } -- 2.17.2 -- dm-devel mailing list dm-devel@xxxxxxxxxx https://www.redhat.com/mailman/listinfo/dm-devel