Blame SOURCES/0255-RHBZ-1638651-marginal-path.patch

4728c8
---
4728c8
 libmultipath/Makefile      |    7 
4728c8
 libmultipath/config.h      |   12 
4728c8
 libmultipath/configure.c   |   18 -
4728c8
 libmultipath/configure.h   |    3 
4728c8
 libmultipath/defaults.h    |    1 
4728c8
 libmultipath/dict.c        |  410 ++++++++++++++++++++++++
4728c8
 libmultipath/io_err_stat.c |  763 +++++++++++++++++++++++++++++++++++++++++++++
4728c8
 libmultipath/io_err_stat.h |   15 
4728c8
 libmultipath/propsel.c     |   98 +++++
4728c8
 libmultipath/propsel.h     |    4 
4728c8
 libmultipath/structs.h     |   14 
4728c8
 libmultipath/time-util.c   |   42 ++
4728c8
 libmultipath/time-util.h   |   13 
4728c8
 libmultipath/uevent.c      |   38 ++
4728c8
 libmultipath/uevent.h      |    2 
4728c8
 multipath/multipath.conf.5 |  108 ++++++
4728c8
 multipathd/cli_handlers.c  |    2 
4728c8
 multipathd/main.c          |   64 +++
4728c8
 18 files changed, 1599 insertions(+), 15 deletions(-)
4728c8
4728c8
Index: multipath-tools-130222/libmultipath/Makefile
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/Makefile
4728c8
+++ multipath-tools-130222/libmultipath/Makefile
4728c8
@@ -7,16 +7,17 @@ include ../Makefile.inc
4728c8
 SONAME=0
4728c8
 DEVLIB = libmultipath.so
4728c8
 LIBS = $(DEVLIB).$(SONAME)
4728c8
-LIBDEPS = -lpthread -ldl -ldevmapper -ludev -L$(mpathcmddir) -lmpathcmd
4728c8
+LIBDEPS = -lpthread -ldl -ldevmapper -ludev -L$(mpathcmddir) -lmpathcmd -laio
4728c8
 CFLAGS += -fPIC -I$(mpathcmddir) -I$(mpathpersistdir)
4728c8
 
4728c8
 OBJS = memory.o parser.o vector.o devmapper.o \
4728c8
        hwtable.o blacklist.o util.o dmparser.o config.o \
4728c8
        structs.o discovery.o propsel.o dict.o \
4728c8
-       pgpolicies.o debug.o regex.o defaults.o uevent.o \
4728c8
+       pgpolicies.o debug.o regex.o defaults.o uevent.o time-util.o \
4728c8
        switchgroup.o uxsock.o print.o alias.o log_pthread.o \
4728c8
        log.o configure.o structs_vec.o sysfs.o prio.o checkers.o \
4728c8
-       lock.o waiter.o file.o wwids.o prioritizers/alua_rtpg.o prkey.o
4728c8
+       lock.o waiter.o file.o wwids.o prioritizers/alua_rtpg.o prkey.o \
4728c8
+       io_err_stat.o
4728c8
 
4728c8
 LIBDM_API_FLUSH = $(shell grep -Ecs '^[a-z]*[[:space:]]+dm_task_no_flush' /usr/include/libdevmapper.h)
4728c8
 
4728c8
Index: multipath-tools-130222/libmultipath/config.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/config.h
4728c8
+++ multipath-tools-130222/libmultipath/config.h
4728c8
@@ -67,6 +67,10 @@ struct hwentry {
4728c8
 	int deferred_remove;
4728c8
 	int delay_watch_checks;
4728c8
 	int delay_wait_checks;
4728c8
+	int marginal_path_err_sample_time;
4728c8
+	int marginal_path_err_rate_threshold;
4728c8
+	int marginal_path_err_recheck_gap_time;
4728c8
+	int marginal_path_double_failed_time;
4728c8
 	int skip_kpartx;
4728c8
 	int max_sectors_kb;
4728c8
 	int unpriv_sgio;
4728c8
@@ -100,6 +104,10 @@ struct mpentry {
4728c8
 	int deferred_remove;
4728c8
 	int delay_watch_checks;
4728c8
 	int delay_wait_checks;
4728c8
+	int marginal_path_err_sample_time;
4728c8
+	int marginal_path_err_rate_threshold;
4728c8
+	int marginal_path_err_recheck_gap_time;
4728c8
+	int marginal_path_double_failed_time;
4728c8
 	int skip_kpartx;
4728c8
 	int max_sectors_kb;
4728c8
 	int unpriv_sgio;
4728c8
@@ -153,6 +161,10 @@ struct config {
4728c8
 	int processed_main_config;
4728c8
 	int delay_watch_checks;
4728c8
 	int delay_wait_checks;
4728c8
+	int marginal_path_err_sample_time;
4728c8
+	int marginal_path_err_rate_threshold;
4728c8
+	int marginal_path_err_recheck_gap_time;
4728c8
+	int marginal_path_double_failed_time;
4728c8
 	int retrigger_tries;
4728c8
 	int retrigger_delay;
4728c8
 	int new_bindings_in_boot;
4728c8
Index: multipath-tools-130222/libmultipath/configure.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/configure.c
4728c8
+++ multipath-tools-130222/libmultipath/configure.c
4728c8
@@ -42,6 +42,7 @@
4728c8
 #include "uxsock.h"
4728c8
 #include "wwids.h"
4728c8
 #include "sysfs.h"
4728c8
+#include "io_err_stat.h"
4728c8
 
4728c8
 /* group paths in pg by host adapter
4728c8
  */
4728c8
@@ -257,7 +258,8 @@ int rr_optimize_path_order(struct pathgr
4728c8
 }
4728c8
 
4728c8
 extern int
4728c8
-setup_map (struct multipath * mpp, char * params, int params_size)
4728c8
+setup_map (struct multipath * mpp, char * params, int params_size,
4728c8
+	   struct vectors *vecs)
4728c8
 {
4728c8
 	struct pathgroup * pgp;
4728c8
 	int i, old_nr_active;
4728c8
@@ -297,11 +299,21 @@ setup_map (struct multipath * mpp, char
4728c8
 	select_deferred_remove(mpp);
4728c8
 	select_delay_watch_checks(mpp);
4728c8
 	select_delay_wait_checks(mpp);
4728c8
+	select_marginal_path_err_sample_time(mpp);
4728c8
+	select_marginal_path_err_rate_threshold(mpp);
4728c8
+	select_marginal_path_err_recheck_gap_time(mpp);
4728c8
+	select_marginal_path_double_failed_time(mpp);
4728c8
 	select_skip_kpartx(mpp);
4728c8
 	select_max_sectors_kb(mpp);
4728c8
 	select_unpriv_sgio(mpp);
4728c8
 
4728c8
 	sysfs_set_scsi_tmo(mpp);
4728c8
+
4728c8
+	if (mpp->marginal_path_double_failed_time > 0 &&
4728c8
+	    mpp->marginal_path_err_sample_time > 0 &&
4728c8
+	    mpp->marginal_path_err_recheck_gap_time > 0 &&
4728c8
+	    mpp->marginal_path_err_rate_threshold >= 0)
4728c8
+		start_io_err_stat_thread(vecs);
4728c8
 	/*
4728c8
 	 * assign paths to path groups -- start with no groups and all paths
4728c8
 	 * in mpp->paths
4728c8
@@ -867,7 +879,7 @@ coalesce_paths (struct vectors * vecs, v
4728c8
 		verify_paths(mpp, vecs, NULL);
4728c8
 
4728c8
 		params[0] = '\0';
4728c8
-		if (setup_map(mpp, params, PARAMS_SIZE)) {
4728c8
+		if (setup_map(mpp, params, PARAMS_SIZE, vecs)) {
4728c8
 			remove_map(mpp, vecs, 0);
4728c8
 			continue;
4728c8
 		}
4728c8
@@ -1118,7 +1130,7 @@ extern int reload_map(struct vectors *ve
4728c8
 		vector_foreach_slot (mpp->paths, pp, i)
4728c8
 			pathinfo(pp, conf->hwtable, DI_PRIO);
4728c8
 	}
4728c8
-	if (setup_map(mpp, params, PARAMS_SIZE)) {
4728c8
+	if (setup_map(mpp, params, PARAMS_SIZE, vecs)) {
4728c8
 		condlog(0, "%s: failed to setup map", mpp->alias);
4728c8
 		return 1;
4728c8
 	}
4728c8
Index: multipath-tools-130222/libmultipath/configure.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/configure.h
4728c8
+++ multipath-tools-130222/libmultipath/configure.h
4728c8
@@ -24,7 +24,8 @@ enum actions {
4728c8
 #define FLUSH_ONE 1
4728c8
 #define FLUSH_ALL 2
4728c8
 
4728c8
-int setup_map (struct multipath * mpp, char * params, int params_size );
4728c8
+int setup_map (struct multipath * mpp, char * params, int params_size,
4728c8
+	       struct vectors *vecs);
4728c8
 int domap (struct multipath * mpp, char * params);
4728c8
 int reinstate_paths (struct multipath *mpp);
4728c8
 int check_daemon(void);
4728c8
Index: multipath-tools-130222/libmultipath/defaults.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/defaults.h
4728c8
+++ multipath-tools-130222/libmultipath/defaults.h
4728c8
@@ -22,6 +22,7 @@
4728c8
 #define DEFAULT_DETECT_CHECKER DETECT_CHECKER_OFF
4728c8
 #define DEFAULT_DEFERRED_REMOVE DEFERRED_REMOVE_OFF
4728c8
 #define DEFAULT_DELAY_CHECKS DELAY_CHECKS_OFF
4728c8
+#define DEFAULT_MARGINAL_PATH MARGINAL_PATH_OFF
4728c8
 #define DEFAULT_RETRIGGER_DELAY 10
4728c8
 #define DEFAULT_RETRIGGER_TRIES 3
4728c8
 #define DEFAULT_UEV_WAIT_TIMEOUT 30
4728c8
Index: multipath-tools-130222/libmultipath/dict.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/dict.c
4728c8
+++ multipath-tools-130222/libmultipath/dict.c
4728c8
@@ -1077,6 +1077,81 @@ def_all_tg_pt_handler(vector strvec)
4728c8
         return 0;
4728c8
 }
4728c8
 
4728c8
+static int
4728c8
+def_marginal_path_err_sample_time_handler(vector strvec)
4728c8
+{
4728c8
+	char * buff;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		conf->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((conf->marginal_path_err_sample_time = atoi(buff)) < 1)
4728c8
+		conf->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+def_marginal_path_err_rate_threshold_handler(vector strvec)
4728c8
+{
4728c8
+	char * buff;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		conf->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+	else if ((conf->marginal_path_err_rate_threshold = atoi(buff)) < 1)
4728c8
+		conf->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+def_marginal_path_err_recheck_gap_time_handler(vector strvec)
4728c8
+{
4728c8
+	char * buff;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		conf->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((conf->marginal_path_err_recheck_gap_time = atoi(buff)) < 1)
4728c8
+		conf->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+def_marginal_path_double_failed_time_handler(vector strvec)
4728c8
+{
4728c8
+	char * buff;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		conf->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((conf->marginal_path_double_failed_time = atoi(buff)) < 1)
4728c8
+		conf->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
 
4728c8
 /*
4728c8
  * blacklist block handlers
4728c8
@@ -2055,6 +2130,98 @@ hw_all_tg_pt_handler(vector strvec)
4728c8
 	return 0;
4728c8
 }
4728c8
 
4728c8
+static int
4728c8
+hw_marginal_path_err_sample_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct hwentry *hwe = VECTOR_LAST_SLOT(conf->hwtable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!hwe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		hwe->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((hwe->marginal_path_err_sample_time = atoi(buff)) < 1)
4728c8
+		hwe->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+hw_marginal_path_err_rate_threshold_handler(vector strvec)
4728c8
+{
4728c8
+	struct hwentry *hwe = VECTOR_LAST_SLOT(conf->hwtable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!hwe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		hwe->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+	else if ((hwe->marginal_path_err_rate_threshold = atoi(buff)) < 1)
4728c8
+		hwe->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+hw_marginal_path_err_recheck_gap_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct hwentry *hwe = VECTOR_LAST_SLOT(conf->hwtable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!hwe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		hwe->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((hwe->marginal_path_err_recheck_gap_time = atoi(buff)) < 1)
4728c8
+		hwe->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+hw_marginal_path_double_failed_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct hwentry *hwe = VECTOR_LAST_SLOT(conf->hwtable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!hwe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		hwe->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((hwe->marginal_path_double_failed_time = atoi(buff)) < 1)
4728c8
+		hwe->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
 /*
4728c8
  * multipaths block handlers
4728c8
  */
4728c8
@@ -2659,6 +2826,98 @@ mp_ghost_delay_handler(vector strvec)
4728c8
 	return 0;
4728c8
 }
4728c8
 
4728c8
+static int
4728c8
+mp_marginal_path_err_sample_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct mpentry *mpe = VECTOR_LAST_SLOT(conf->mptable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!mpe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		mpe->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((mpe->marginal_path_err_sample_time = atoi(buff)) < 1)
4728c8
+		mpe->marginal_path_err_sample_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+mp_marginal_path_err_rate_threshold_handler(vector strvec)
4728c8
+{
4728c8
+	struct mpentry *mpe = VECTOR_LAST_SLOT(conf->mptable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!mpe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		mpe->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+	else if ((mpe->marginal_path_err_rate_threshold = atoi(buff)) < 1)
4728c8
+		mpe->marginal_path_err_rate_threshold = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+mp_marginal_path_err_recheck_gap_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct mpentry *mpe = VECTOR_LAST_SLOT(conf->mptable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!mpe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		mpe->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((mpe->marginal_path_err_recheck_gap_time = atoi(buff)) < 1)
4728c8
+		mpe->marginal_path_err_recheck_gap_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+mp_marginal_path_double_failed_time_handler(vector strvec)
4728c8
+{
4728c8
+	struct mpentry *mpe = VECTOR_LAST_SLOT(conf->mptable);
4728c8
+	char * buff;
4728c8
+
4728c8
+	if (!mpe)
4728c8
+		return 1;
4728c8
+
4728c8
+	buff = set_value(strvec);
4728c8
+	if (!buff)
4728c8
+		return 1;
4728c8
+
4728c8
+	if ((strlen(buff) == 2 && !strcmp(buff, "no")) ||
4728c8
+	    (strlen(buff) == 1 && !strcmp(buff, "0")))
4728c8
+		mpe->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+	else if ((mpe->marginal_path_double_failed_time = atoi(buff)) < 1)
4728c8
+		mpe->marginal_path_double_failed_time = MARGINAL_PATH_OFF;
4728c8
+
4728c8
+	FREE(buff);
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
 /*
4728c8
  * config file keywords printing
4728c8
  */
4728c8
@@ -2989,6 +3248,56 @@ snprint_mp_ghost_delay (char * buff, int
4728c8
 }
4728c8
 
4728c8
 static int
4728c8
+snprint_mp_marginal_path_err_sample_time (char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct mpentry * mpe = (struct mpentry *)data;
4728c8
+
4728c8
+	if (mpe->marginal_path_err_sample_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (mpe->marginal_path_err_sample_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", mpe->marginal_path_err_sample_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_mp_marginal_path_err_rate_threshold (char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct mpentry * mpe = (struct mpentry *)data;
4728c8
+
4728c8
+	if (mpe->marginal_path_err_rate_threshold == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (mpe->marginal_path_err_rate_threshold == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", mpe->marginal_path_err_rate_threshold);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_mp_marginal_path_err_recheck_gap_time (char * buff, int len,
4728c8
+					       void * data)
4728c8
+{
4728c8
+	struct mpentry * mpe = (struct mpentry *)data;
4728c8
+
4728c8
+	if (mpe->marginal_path_err_recheck_gap_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (mpe->marginal_path_err_recheck_gap_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d",
4728c8
+			mpe->marginal_path_err_recheck_gap_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_mp_marginal_path_double_failed_time (char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct mpentry * mpe = (struct mpentry *)data;
4728c8
+
4728c8
+	if (mpe->marginal_path_double_failed_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (mpe->marginal_path_double_failed_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", mpe->marginal_path_double_failed_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
 snprint_hw_fast_io_fail(char * buff, int len, void * data)
4728c8
 {
4728c8
 	struct hwentry * hwe = (struct hwentry *)data;
4728c8
@@ -3429,6 +3738,55 @@ snprint_hw_all_tg_pt(char * buff, int le
4728c8
 }
4728c8
 
4728c8
 static int
4728c8
+snprint_hw_marginal_path_err_sample_time(char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct hwentry * hwe = (struct hwentry *)data;
4728c8
+
4728c8
+	if (hwe->marginal_path_err_sample_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (hwe->marginal_path_err_sample_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", hwe->marginal_path_err_sample_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_hw_marginal_path_err_rate_threshold(char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct hwentry * hwe = (struct hwentry *)data;
4728c8
+
4728c8
+	if (hwe->marginal_path_err_rate_threshold == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (hwe->marginal_path_err_rate_threshold == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", hwe->marginal_path_err_rate_threshold);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_hw_marginal_path_err_recheck_gap_time(char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct hwentry * hwe = (struct hwentry *)data;
4728c8
+
4728c8
+	if (hwe->marginal_path_err_recheck_gap_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (hwe->marginal_path_err_recheck_gap_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d",
4728c8
+			hwe->marginal_path_err_recheck_gap_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_hw_marginal_path_double_failed_time(char * buff, int len, void * data)
4728c8
+{
4728c8
+	struct hwentry * hwe = (struct hwentry *)data;
4728c8
+
4728c8
+	if (hwe->marginal_path_double_failed_time == MARGINAL_PATH_UNDEF)
4728c8
+		return 0;
4728c8
+	if (hwe->marginal_path_double_failed_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", hwe->marginal_path_double_failed_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
 snprint_def_polling_interval (char * buff, int len, void * data)
4728c8
 {
4728c8
 	return snprintf(buff, len, "%i", conf->checkint);
4728c8
@@ -3945,6 +4303,46 @@ snprint_def_all_tg_pt(char * buff, int l
4728c8
 }
4728c8
 
4728c8
 static int
4728c8
+snprint_def_marginal_path_err_sample_time(char * buff, int len, void * data)
4728c8
+{
4728c8
+	if (conf->marginal_path_err_sample_time == MARGINAL_PATH_UNDEF ||
4728c8
+	    conf->marginal_path_err_sample_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d", conf->marginal_path_err_sample_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_def_marginal_path_err_rate_threshold(char * buff, int len, void * data)
4728c8
+{
4728c8
+	if (conf->marginal_path_err_rate_threshold == MARGINAL_PATH_UNDEF ||
4728c8
+	    conf->marginal_path_err_rate_threshold == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d",
4728c8
+			conf->marginal_path_err_rate_threshold);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_def_marginal_path_err_recheck_gap_time(char * buff, int len,
4728c8
+					       void * data)
4728c8
+{
4728c8
+	if (conf->marginal_path_err_recheck_gap_time == MARGINAL_PATH_UNDEF ||
4728c8
+	    conf->marginal_path_err_recheck_gap_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d",
4728c8
+			conf->marginal_path_err_recheck_gap_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
+snprint_def_marginal_path_double_failed_time(char * buff, int len, void * data)
4728c8
+{
4728c8
+	if (conf->marginal_path_double_failed_time == MARGINAL_PATH_UNDEF ||
4728c8
+	    conf->marginal_path_double_failed_time == MARGINAL_PATH_OFF)
4728c8
+		return snprintf(buff, len, "no");
4728c8
+	return snprintf(buff, len, "%d",
4728c8
+			conf->marginal_path_double_failed_time);
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
 snprint_ble_simple (char * buff, int len, void * data)
4728c8
 {
4728c8
 	struct blentry * ble = (struct blentry *)data;
4728c8
@@ -4043,6 +4441,10 @@ init_keywords(void)
4728c8
 	install_keyword("unpriv_sgio", &def_unpriv_sgio_handler, &snprint_def_unpriv_sgio);
4728c8
 	install_keyword("ghost_delay", &def_ghost_delay_handler, &snprint_def_ghost_delay);
4728c8
 	install_keyword("all_tg_pt", &def_all_tg_pt_handler, &snprint_def_all_tg_pt);
4728c8
+	install_keyword("marginal_path_err_sample_time", &def_marginal_path_err_sample_time_handler, &snprint_def_marginal_path_err_sample_time);
4728c8
+	install_keyword("marginal_path_err_rate_threshold", &def_marginal_path_err_rate_threshold_handler, &snprint_def_marginal_path_err_rate_threshold);
4728c8
+	install_keyword("marginal_path_err_recheck_gap_time", &def_marginal_path_err_recheck_gap_time_handler, &snprint_def_marginal_path_err_recheck_gap_time);
4728c8
+	install_keyword("marginal_path_double_failed_time", &def_marginal_path_double_failed_time_handler, &snprint_def_marginal_path_double_failed_time);
4728c8
 	__deprecated install_keyword("default_selector", &def_selector_handler, NULL);
4728c8
 	__deprecated install_keyword("default_path_grouping_policy", &def_pgpolicy_handler, NULL);
4728c8
 	__deprecated install_keyword("default_uid_attribute", &def_uid_attribute_handler, NULL);
4728c8
@@ -4120,6 +4522,10 @@ init_keywords(void)
4728c8
 	install_keyword("unpriv_sgio", &hw_unpriv_sgio_handler, &snprint_hw_unpriv_sgio);
4728c8
 	install_keyword("ghost_delay", &hw_ghost_delay_handler, &snprint_hw_ghost_delay);
4728c8
 	install_keyword("all_tg_pt", &hw_all_tg_pt_handler, &snprint_hw_all_tg_pt);
4728c8
+	install_keyword("marginal_path_err_sample_time", &hw_marginal_path_err_sample_time_handler, &snprint_hw_marginal_path_err_sample_time);
4728c8
+	install_keyword("marginal_path_err_rate_threshold", &hw_marginal_path_err_rate_threshold_handler, &snprint_hw_marginal_path_err_rate_threshold);
4728c8
+	install_keyword("marginal_path_err_recheck_gap_time", &hw_marginal_path_err_recheck_gap_time_handler, &snprint_hw_marginal_path_err_recheck_gap_time);
4728c8
+	install_keyword("marginal_path_double_failed_time", &hw_marginal_path_double_failed_time_handler, &snprint_hw_marginal_path_double_failed_time);
4728c8
 	install_sublevel_end();
4728c8
 
4728c8
 	install_keyword_root("overrides", &nop_handler);
4728c8
@@ -4184,5 +4590,9 @@ init_keywords(void)
4728c8
 	install_keyword("max_sectors_kb", &mp_max_sectors_kb_handler, &snprint_mp_max_sectors_kb);
4728c8
 	install_keyword("unpriv_sgio", &mp_unpriv_sgio_handler, &snprint_mp_unpriv_sgio);
4728c8
 	install_keyword("ghost_delay", &mp_ghost_delay_handler, &snprint_mp_ghost_delay);
4728c8
+	install_keyword("marginal_path_err_sample_time", &mp_marginal_path_err_sample_time_handler, &snprint_mp_marginal_path_err_sample_time);
4728c8
+	install_keyword("marginal_path_err_rate_threshold", &mp_marginal_path_err_rate_threshold_handler, &snprint_mp_marginal_path_err_rate_threshold);
4728c8
+	install_keyword("marginal_path_err_recheck_gap_time", &mp_marginal_path_err_recheck_gap_time_handler, &snprint_mp_marginal_path_err_recheck_gap_time);
4728c8
+	install_keyword("marginal_path_double_failed_time", &mp_marginal_path_double_failed_time_handler, &snprint_mp_marginal_path_double_failed_time);
4728c8
 	install_sublevel_end();
4728c8
 }
4728c8
Index: multipath-tools-130222/libmultipath/io_err_stat.c
4728c8
===================================================================
4728c8
--- /dev/null
4728c8
+++ multipath-tools-130222/libmultipath/io_err_stat.c
4728c8
@@ -0,0 +1,763 @@
4728c8
+/*
4728c8
+ * (C) Copyright HUAWEI Technology Corp. 2017, All Rights Reserved.
4728c8
+ *
4728c8
+ * io_err_stat.c
4728c8
+ * version 1.0
4728c8
+ *
4728c8
+ * IO error stream statistic process for path failure event from kernel
4728c8
+ *
4728c8
+ * Author(s): Guan Junxiong 2017 <guanjunxiong@huawei.com>
4728c8
+ *
4728c8
+ * This file is released under the GPL version 2, or any later version.
4728c8
+ */
4728c8
+
4728c8
+#include <unistd.h>
4728c8
+#include <pthread.h>
4728c8
+#include <signal.h>
4728c8
+#include <fcntl.h>
4728c8
+#include <sys/stat.h>
4728c8
+#include <sys/ioctl.h>
4728c8
+#include <linux/fs.h>
4728c8
+#include <libaio.h>
4728c8
+#include <errno.h>
4728c8
+#include <sys/mman.h>
4728c8
+
4728c8
+#include "vector.h"
4728c8
+#include "memory.h"
4728c8
+#include "checkers.h"
4728c8
+#include "config.h"
4728c8
+#include "structs.h"
4728c8
+#include "structs_vec.h"
4728c8
+#include "devmapper.h"
4728c8
+#include "debug.h"
4728c8
+#include "lock.h"
4728c8
+#include "time-util.h"
4728c8
+#include "io_err_stat.h"
4728c8
+
4728c8
+#define IOTIMEOUT_SEC			60
4728c8
+#define TIMEOUT_NO_IO_NSEC		10000000 /*10ms = 10000000ns*/
4728c8
+#define FLAKY_PATHFAIL_THRESHOLD	2
4728c8
+#define CONCUR_NR_EVENT			32
4728c8
+
4728c8
+#define PATH_IO_ERR_IN_CHECKING		-1
4728c8
+#define PATH_IO_ERR_WAITING_TO_CHECK	-2
4728c8
+
4728c8
+#define io_err_stat_log(prio, fmt, args...) \
4728c8
+	condlog(prio, "io error statistic: " fmt, ##args)
4728c8
+
4728c8
+
4728c8
+struct io_err_stat_pathvec {
4728c8
+	pthread_mutex_t mutex;
4728c8
+	vector		pathvec;
4728c8
+};
4728c8
+
4728c8
+struct dio_ctx {
4728c8
+	struct timespec	io_starttime;
4728c8
+	int		blksize;
4728c8
+	void		*buf;
4728c8
+	struct iocb	io;
4728c8
+};
4728c8
+
4728c8
+struct io_err_stat_path {
4728c8
+	char		devname[FILE_NAME_SIZE];
4728c8
+	int		fd;
4728c8
+	struct dio_ctx	*dio_ctx_array;
4728c8
+	int		io_err_nr;
4728c8
+	int		io_nr;
4728c8
+	struct timespec	start_time;
4728c8
+
4728c8
+	int		total_time;
4728c8
+	int		err_rate_threshold;
4728c8
+};
4728c8
+
4728c8
+pthread_t		io_err_stat_thr;
4728c8
+pthread_attr_t		io_err_stat_attr;
4728c8
+
4728c8
+static pthread_mutex_t io_err_thread_lock = PTHREAD_MUTEX_INITIALIZER;
4728c8
+static pthread_cond_t io_err_thread_cond = PTHREAD_COND_INITIALIZER;
4728c8
+static int io_err_thread_running = 0;
4728c8
+
4728c8
+#define uatomic_read(ptr) __atomic_load_n((ptr), __ATOMIC_SEQ_CST)
4728c8
+#define uatomic_set(ptr, val) __atomic_store_n((ptr), (val), __ATOMIC_SEQ_CST)
4728c8
+
4728c8
+static struct io_err_stat_pathvec *paths;
4728c8
+struct vectors *vecs;
4728c8
+io_context_t	ioctx;
4728c8
+
4728c8
+static void cancel_inflight_io(struct io_err_stat_path *pp);
4728c8
+
4728c8
+struct io_err_stat_path *find_err_path_by_dev(vector pathvec, char *dev)
4728c8
+{
4728c8
+	int i;
4728c8
+	struct io_err_stat_path *pp;
4728c8
+
4728c8
+	if (!pathvec)
4728c8
+		return NULL;
4728c8
+	vector_foreach_slot(pathvec, pp, i)
4728c8
+		if (!strcmp(pp->devname, dev))
4728c8
+			return pp;
4728c8
+
4728c8
+	io_err_stat_log(4, "%s: not found in check queue", dev);
4728c8
+
4728c8
+	return NULL;
4728c8
+}
4728c8
+
4728c8
+static int init_each_dio_ctx(struct dio_ctx *ct, int blksize,
4728c8
+		unsigned long pgsize)
4728c8
+{
4728c8
+	ct->blksize = blksize;
4728c8
+	if (posix_memalign(&ct->buf, pgsize, blksize))
4728c8
+		return 1;
4728c8
+	memset(ct->buf, 0, blksize);
4728c8
+	ct->io_starttime.tv_sec = 0;
4728c8
+	ct->io_starttime.tv_nsec = 0;
4728c8
+
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static void deinit_each_dio_ctx(struct dio_ctx *ct)
4728c8
+{
4728c8
+	if (ct->buf)
4728c8
+		free(ct->buf);
4728c8
+}
4728c8
+
4728c8
+static int setup_directio_ctx(struct io_err_stat_path *p)
4728c8
+{
4728c8
+	unsigned long pgsize = getpagesize();
4728c8
+	char fpath[PATH_MAX];
4728c8
+	int blksize = 0;
4728c8
+	int i;
4728c8
+
4728c8
+	if (snprintf(fpath, PATH_MAX, "/dev/%s", p->devname) >= PATH_MAX)
4728c8
+		return 1;
4728c8
+	if (p->fd < 0)
4728c8
+		p->fd = open(fpath, O_RDONLY | O_DIRECT);
4728c8
+	if (p->fd < 0)
4728c8
+		return 1;
4728c8
+
4728c8
+	p->dio_ctx_array = MALLOC(sizeof(struct dio_ctx) * CONCUR_NR_EVENT);
4728c8
+	if (!p->dio_ctx_array)
4728c8
+		goto fail_close;
4728c8
+
4728c8
+	if (ioctl(p->fd, BLKBSZGET, &blksize) < 0) {
4728c8
+		io_err_stat_log(4, "%s:cannot get blocksize, set default 512",
4728c8
+				p->devname);
4728c8
+		blksize = 512;
4728c8
+	}
4728c8
+	if (!blksize)
4728c8
+		goto free_pdctx;
4728c8
+
4728c8
+	for (i = 0; i < CONCUR_NR_EVENT; i++) {
4728c8
+		if (init_each_dio_ctx(p->dio_ctx_array + i, blksize, pgsize))
4728c8
+			goto deinit;
4728c8
+	}
4728c8
+	return 0;
4728c8
+
4728c8
+deinit:
4728c8
+	for (i = 0; i < CONCUR_NR_EVENT; i++)
4728c8
+		deinit_each_dio_ctx(p->dio_ctx_array + i);
4728c8
+free_pdctx:
4728c8
+	FREE(p->dio_ctx_array);
4728c8
+fail_close:
4728c8
+	close(p->fd);
4728c8
+
4728c8
+	return 1;
4728c8
+}
4728c8
+
4728c8
+static void destroy_directio_ctx(struct io_err_stat_path *p)
4728c8
+{
4728c8
+	int i;
4728c8
+
4728c8
+	if (!p || !p->dio_ctx_array)
4728c8
+		return;
4728c8
+	cancel_inflight_io(p);
4728c8
+
4728c8
+	for (i = 0; i < CONCUR_NR_EVENT; i++)
4728c8
+		deinit_each_dio_ctx(p->dio_ctx_array + i);
4728c8
+	FREE(p->dio_ctx_array);
4728c8
+
4728c8
+	if (p->fd > 0)
4728c8
+		close(p->fd);
4728c8
+}
4728c8
+
4728c8
+static struct io_err_stat_path *alloc_io_err_stat_path(void)
4728c8
+{
4728c8
+	struct io_err_stat_path *p;
4728c8
+
4728c8
+	p = (struct io_err_stat_path *)MALLOC(sizeof(*p));
4728c8
+	if (!p)
4728c8
+		return NULL;
4728c8
+
4728c8
+	memset(p->devname, 0, sizeof(p->devname));
4728c8
+	p->io_err_nr = 0;
4728c8
+	p->io_nr = 0;
4728c8
+	p->total_time = 0;
4728c8
+	p->start_time.tv_sec = 0;
4728c8
+	p->start_time.tv_nsec = 0;
4728c8
+	p->err_rate_threshold = 0;
4728c8
+	p->fd = -1;
4728c8
+
4728c8
+	return p;
4728c8
+}
4728c8
+
4728c8
+static void free_io_err_stat_path(struct io_err_stat_path *p)
4728c8
+{
4728c8
+	FREE(p);
4728c8
+}
4728c8
+
4728c8
+static struct io_err_stat_pathvec *alloc_pathvec(void)
4728c8
+{
4728c8
+	struct io_err_stat_pathvec *p;
4728c8
+	int r;
4728c8
+
4728c8
+	p = (struct io_err_stat_pathvec *)MALLOC(sizeof(*p));
4728c8
+	if (!p)
4728c8
+		return NULL;
4728c8
+	p->pathvec = vector_alloc();
4728c8
+	if (!p->pathvec)
4728c8
+		goto out_free_struct_pathvec;
4728c8
+	r = pthread_mutex_init(&p->mutex, NULL);
4728c8
+	if (r)
4728c8
+		goto out_free_member_pathvec;
4728c8
+
4728c8
+	return p;
4728c8
+
4728c8
+out_free_member_pathvec:
4728c8
+	vector_free(p->pathvec);
4728c8
+out_free_struct_pathvec:
4728c8
+	FREE(p);
4728c8
+	return NULL;
4728c8
+}
4728c8
+
4728c8
+static void free_io_err_pathvec(struct io_err_stat_pathvec *p)
4728c8
+{
4728c8
+	struct io_err_stat_path *path;
4728c8
+	int i;
4728c8
+
4728c8
+	if (!p)
4728c8
+		return;
4728c8
+	pthread_mutex_destroy(&p->mutex);
4728c8
+	if (!p->pathvec) {
4728c8
+		vector_foreach_slot(p->pathvec, path, i) {
4728c8
+			destroy_directio_ctx(path);
4728c8
+			free_io_err_stat_path(path);
4728c8
+		}
4728c8
+		vector_free(p->pathvec);
4728c8
+	}
4728c8
+	FREE(p);
4728c8
+}
4728c8
+
4728c8
+/*
4728c8
+ * return value
4728c8
+ * 0: enqueue OK
4728c8
+ * 1: fails because of internal error
4728c8
+ */
4728c8
+static int enqueue_io_err_stat_by_path(struct path *path)
4728c8
+{
4728c8
+	struct io_err_stat_path *p;
4728c8
+
4728c8
+	pthread_mutex_lock(&paths->mutex);
4728c8
+	p = find_err_path_by_dev(paths->pathvec, path->dev);
4728c8
+	if (p) {
4728c8
+		pthread_mutex_unlock(&paths->mutex);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	pthread_mutex_unlock(&paths->mutex);
4728c8
+
4728c8
+	p = alloc_io_err_stat_path();
4728c8
+	if (!p)
4728c8
+		return 1;
4728c8
+
4728c8
+	memcpy(p->devname, path->dev, sizeof(p->devname));
4728c8
+	p->total_time = path->mpp->marginal_path_err_sample_time;
4728c8
+	p->err_rate_threshold = path->mpp->marginal_path_err_rate_threshold;
4728c8
+
4728c8
+	if (setup_directio_ctx(p))
4728c8
+		goto free_ioerr_path;
4728c8
+	pthread_mutex_lock(&paths->mutex);
4728c8
+	if (!vector_alloc_slot(paths->pathvec))
4728c8
+		goto unlock_destroy;
4728c8
+	vector_set_slot(paths->pathvec, p);
4728c8
+	pthread_mutex_unlock(&paths->mutex);
4728c8
+
4728c8
+	io_err_stat_log(2, "%s: enqueue path %s to check",
4728c8
+			path->mpp->alias, path->dev);
4728c8
+	return 0;
4728c8
+
4728c8
+unlock_destroy:
4728c8
+	pthread_mutex_unlock(&paths->mutex);
4728c8
+	destroy_directio_ctx(p);
4728c8
+free_ioerr_path:
4728c8
+	free_io_err_stat_path(p);
4728c8
+
4728c8
+	return 1;
4728c8
+}
4728c8
+
4728c8
+int io_err_stat_handle_pathfail(struct path *path)
4728c8
+{
4728c8
+	struct timespec curr_time;
4728c8
+
4728c8
+	if (uatomic_read(&io_err_thread_running) == 0)
4728c8
+		return 1;
4728c8
+
4728c8
+	if (path->io_err_disable_reinstate) {
4728c8
+		io_err_stat_log(3, "%s: reinstate is already disabled",
4728c8
+				path->dev);
4728c8
+		return 1;
4728c8
+	}
4728c8
+	if (path->io_err_pathfail_cnt < 0)
4728c8
+		return 1;
4728c8
+
4728c8
+	if (!path->mpp)
4728c8
+		return 1;
4728c8
+	if (path->mpp->marginal_path_double_failed_time <= 0 ||
4728c8
+		path->mpp->marginal_path_err_sample_time <= 0 ||
4728c8
+		path->mpp->marginal_path_err_recheck_gap_time <= 0 ||
4728c8
+		path->mpp->marginal_path_err_rate_threshold < 0) {
4728c8
+		io_err_stat_log(4, "%s: parameter not set", path->mpp->alias);
4728c8
+		return 1;
4728c8
+	}
4728c8
+	if (path->mpp->marginal_path_err_sample_time < (2 * IOTIMEOUT_SEC)) {
4728c8
+		io_err_stat_log(2, "%s: marginal_path_err_sample_time should not less than %d",
4728c8
+				path->mpp->alias, 2 * IOTIMEOUT_SEC);
4728c8
+		return 1;
4728c8
+	}
4728c8
+	/*
4728c8
+	 * The test should only be started for paths that have failed
4728c8
+	 * repeatedly in a certain time frame, so that we have reason
4728c8
+	 * to assume they're flaky. Without bother the admin to configure
4728c8
+	 * the repeated count threshold and time frame, we assume a path
4728c8
+	 * which fails at least twice within 60 seconds is flaky.
4728c8
+	 */
4728c8
+	if (clock_gettime(CLOCK_MONOTONIC, &curr_time) != 0)
4728c8
+		return 1;
4728c8
+	if (path->io_err_pathfail_cnt == 0) {
4728c8
+		path->io_err_pathfail_cnt++;
4728c8
+		path->io_err_pathfail_starttime = curr_time.tv_sec;
4728c8
+		io_err_stat_log(5, "%s: start path flakiness pre-checking",
4728c8
+				path->dev);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if ((curr_time.tv_sec - path->io_err_pathfail_starttime) >
4728c8
+			path->mpp->marginal_path_double_failed_time) {
4728c8
+		path->io_err_pathfail_cnt = 0;
4728c8
+		path->io_err_pathfail_starttime = curr_time.tv_sec;
4728c8
+		io_err_stat_log(5, "%s: restart path flakiness pre-checking",
4728c8
+				path->dev);
4728c8
+	}
4728c8
+	path->io_err_pathfail_cnt++;
4728c8
+	if (path->io_err_pathfail_cnt >= FLAKY_PATHFAIL_THRESHOLD) {
4728c8
+		path->io_err_disable_reinstate = 1;
4728c8
+		path->io_err_pathfail_cnt = PATH_IO_ERR_WAITING_TO_CHECK;
4728c8
+		/* enqueue path as soon as it comes up */
4728c8
+		path->io_err_dis_reinstate_time = 0;
4728c8
+		if (path->state != PATH_DOWN) {
4728c8
+			int oldstate = path->state;
4728c8
+			io_err_stat_log(2, "%s: mark as failed", path->dev);
4728c8
+			path->mpp->stat_path_failures++;
4728c8
+			path->state = PATH_DOWN;
4728c8
+			path->dmstate = PSTATE_FAILED;
4728c8
+			if (oldstate == PATH_UP || oldstate == PATH_GHOST)
4728c8
+				update_queue_mode_del_path(path->mpp);
4728c8
+			if (path->tick > conf->checkint)
4728c8
+				path->tick = conf->checkint;
4728c8
+		}
4728c8
+	}
4728c8
+
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+int need_io_err_check(struct path *pp)
4728c8
+{
4728c8
+	struct timespec curr_time;
4728c8
+	int r;
4728c8
+
4728c8
+	if (uatomic_read(&io_err_thread_running) == 0)
4728c8
+		return 0;
4728c8
+	if (pp->mpp->nr_active <= 0) {
4728c8
+		io_err_stat_log(2, "%s: recover path early", pp->dev);
4728c8
+		goto recover;
4728c8
+	}
4728c8
+	if (pp->io_err_pathfail_cnt != PATH_IO_ERR_WAITING_TO_CHECK)
4728c8
+		return 1;
4728c8
+	if (clock_gettime(CLOCK_MONOTONIC, &curr_time) != 0 ||
4728c8
+	    (curr_time.tv_sec - pp->io_err_dis_reinstate_time) >
4728c8
+			pp->mpp->marginal_path_err_recheck_gap_time) {
4728c8
+		io_err_stat_log(4, "%s: reschedule checking after %d seconds",
4728c8
+				pp->dev,
4728c8
+				pp->mpp->marginal_path_err_recheck_gap_time);
4728c8
+		r = enqueue_io_err_stat_by_path(pp);
4728c8
+		/*
4728c8
+		 * Enqueue fails because of internal error.
4728c8
+		 * In this case , we recover this path
4728c8
+		 * Or else,  return 1 to set path state to PATH_SHAKY
4728c8
+		 */
4728c8
+		if (r == 1) {
4728c8
+			io_err_stat_log(3, "%s: enqueue fails, recovering",
4728c8
+					pp->dev);
4728c8
+			goto recover;
4728c8
+		} else
4728c8
+			pp->io_err_pathfail_cnt = PATH_IO_ERR_IN_CHECKING;
4728c8
+	}
4728c8
+
4728c8
+	return 1;
4728c8
+
4728c8
+recover:
4728c8
+	pp->io_err_pathfail_cnt = 0;
4728c8
+	pp->io_err_disable_reinstate = 0;
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int delete_io_err_stat_by_addr(struct io_err_stat_path *p)
4728c8
+{
4728c8
+	int i;
4728c8
+
4728c8
+	i = find_slot(paths->pathvec, p);
4728c8
+	if (i != -1)
4728c8
+		vector_del_slot(paths->pathvec, i);
4728c8
+
4728c8
+	destroy_directio_ctx(p);
4728c8
+	free_io_err_stat_path(p);
4728c8
+
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static void account_async_io_state(struct io_err_stat_path *pp, int rc)
4728c8
+{
4728c8
+	switch (rc) {
4728c8
+	case PATH_DOWN:
4728c8
+		pp->io_err_nr++;
4728c8
+		break;
4728c8
+	case PATH_UNCHECKED:
4728c8
+	case PATH_UP:
4728c8
+	case PATH_PENDING:
4728c8
+		break;
4728c8
+	default:
4728c8
+		break;
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static int poll_io_err_stat(struct vectors *vecs, struct io_err_stat_path *pp)
4728c8
+{
4728c8
+	struct timespec currtime, difftime;
4728c8
+	struct path *path;
4728c8
+	double err_rate;
4728c8
+
4728c8
+	if (clock_gettime(CLOCK_MONOTONIC, &currtime) != 0)
4728c8
+		return 1;
4728c8
+	timespecsub(&currtime, &pp->start_time, &difftime);
4728c8
+	if (difftime.tv_sec < pp->total_time)
4728c8
+		return 0;
4728c8
+
4728c8
+	io_err_stat_log(4, "%s: check end", pp->devname);
4728c8
+
4728c8
+	err_rate = pp->io_nr == 0 ? 0 : (pp->io_err_nr * 1000.0f) / pp->io_nr;
4728c8
+	io_err_stat_log(3, "%s: IO error rate (%.1f/1000)",
4728c8
+			pp->devname, err_rate);
4728c8
+	pthread_cleanup_push(cleanup_lock, &vecs->lock);
4728c8
+	lock(vecs->lock);
4728c8
+	pthread_testcancel();
4728c8
+	path = find_path_by_dev(vecs->pathvec, pp->devname);
4728c8
+	if (!path) {
4728c8
+		io_err_stat_log(4, "path %s not found'", pp->devname);
4728c8
+	} else if (err_rate <= pp->err_rate_threshold) {
4728c8
+		path->io_err_pathfail_cnt = 0;
4728c8
+		path->io_err_disable_reinstate = 0;
4728c8
+		io_err_stat_log(3, "%s: (%d/%d) good to enable reinstating",
4728c8
+				pp->devname, pp->io_err_nr, pp->io_nr);
4728c8
+		/*
4728c8
+		 * schedule path check as soon as possible to
4728c8
+		 * update path state. Do NOT reinstate dm path here
4728c8
+		 */
4728c8
+		path->tick = 1;
4728c8
+
4728c8
+	} else if (path->mpp && path->mpp->nr_active > 0) {
4728c8
+		io_err_stat_log(3, "%s: keep failing the dm path %s",
4728c8
+				path->mpp->alias, path->dev);
4728c8
+		path->io_err_pathfail_cnt = PATH_IO_ERR_WAITING_TO_CHECK;
4728c8
+		path->io_err_disable_reinstate = 1;
4728c8
+		path->io_err_dis_reinstate_time = currtime.tv_sec;
4728c8
+		io_err_stat_log(3, "%s: disable reinstating of %s",
4728c8
+				path->mpp->alias, path->dev);
4728c8
+	} else {
4728c8
+		path->io_err_pathfail_cnt = 0;
4728c8
+		path->io_err_disable_reinstate = 0;
4728c8
+		io_err_stat_log(3, "%s: there is orphan path, enable reinstating",
4728c8
+				pp->devname);
4728c8
+	}
4728c8
+	lock_cleanup_pop(vecs->lock);
4728c8
+
4728c8
+	delete_io_err_stat_by_addr(pp);
4728c8
+
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+static int send_each_async_io(struct dio_ctx *ct, int fd, char *dev)
4728c8
+{
4728c8
+	int rc = -1;
4728c8
+
4728c8
+	if (ct->io_starttime.tv_nsec == 0 &&
4728c8
+			ct->io_starttime.tv_sec == 0) {
4728c8
+		struct iocb *ios[1] = { &ct->io };
4728c8
+
4728c8
+		if (clock_gettime(CLOCK_MONOTONIC, &ct->io_starttime) != 0) {
4728c8
+			ct->io_starttime.tv_sec = 0;
4728c8
+			ct->io_starttime.tv_nsec = 0;
4728c8
+			return rc;
4728c8
+		}
4728c8
+		io_prep_pread(&ct->io, fd, ct->buf, ct->blksize, 0);
4728c8
+		if (io_submit(ioctx, 1, ios) != 1) {
4728c8
+			io_err_stat_log(5, "%s: io_submit error %i",
4728c8
+					dev, errno);
4728c8
+			return rc;
4728c8
+		}
4728c8
+		rc = 0;
4728c8
+	}
4728c8
+
4728c8
+	return rc;
4728c8
+}
4728c8
+
4728c8
+static void send_batch_async_ios(struct io_err_stat_path *pp)
4728c8
+{
4728c8
+	int i;
4728c8
+	struct dio_ctx *ct;
4728c8
+	struct timespec currtime, difftime;
4728c8
+
4728c8
+	if (clock_gettime(CLOCK_MONOTONIC, &currtime) != 0)
4728c8
+		return;
4728c8
+	/*
4728c8
+	 * Give a free time for all IO to complete or timeout
4728c8
+	 */
4728c8
+	if (pp->start_time.tv_sec != 0) {
4728c8
+		timespecsub(&currtime, &pp->start_time, &difftime);
4728c8
+		if (difftime.tv_sec + IOTIMEOUT_SEC >= pp->total_time)
4728c8
+			return;
4728c8
+	}
4728c8
+
4728c8
+	for (i = 0; i < CONCUR_NR_EVENT; i++) {
4728c8
+		ct = pp->dio_ctx_array + i;
4728c8
+		if (!send_each_async_io(ct, pp->fd, pp->devname))
4728c8
+			pp->io_nr++;
4728c8
+	}
4728c8
+	if (pp->start_time.tv_sec == 0 && pp->start_time.tv_nsec == 0 &&
4728c8
+		clock_gettime(CLOCK_MONOTONIC, &pp->start_time)) {
4728c8
+		pp->start_time.tv_sec = 0;
4728c8
+		pp->start_time.tv_nsec = 0;
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static int try_to_cancel_timeout_io(struct dio_ctx *ct, struct timespec *t,
4728c8
+		char *dev)
4728c8
+{
4728c8
+	struct timespec	difftime;
4728c8
+	struct io_event	event;
4728c8
+	int		rc = PATH_UNCHECKED;
4728c8
+	int		r;
4728c8
+
4728c8
+	if (ct->io_starttime.tv_sec == 0)
4728c8
+		return rc;
4728c8
+	timespecsub(t, &ct->io_starttime, &difftime);
4728c8
+	if (difftime.tv_sec > IOTIMEOUT_SEC) {
4728c8
+		struct iocb *ios[1] = { &ct->io };
4728c8
+
4728c8
+		io_err_stat_log(5, "%s: abort check on timeout", dev);
4728c8
+		r = io_cancel(ioctx, ios[0], &event);
4728c8
+		if (r)
4728c8
+			io_err_stat_log(5, "%s: io_cancel error %i",
4728c8
+					dev, errno);
4728c8
+		ct->io_starttime.tv_sec = 0;
4728c8
+		ct->io_starttime.tv_nsec = 0;
4728c8
+		rc = PATH_DOWN;
4728c8
+	} else {
4728c8
+		rc = PATH_PENDING;
4728c8
+	}
4728c8
+
4728c8
+	return rc;
4728c8
+}
4728c8
+
4728c8
+static void poll_async_io_timeout(void)
4728c8
+{
4728c8
+	struct io_err_stat_path *pp;
4728c8
+	struct timespec curr_time;
4728c8
+	int		rc = PATH_UNCHECKED;
4728c8
+	int		i, j;
4728c8
+
4728c8
+	if (clock_gettime(CLOCK_MONOTONIC, &curr_time) != 0)
4728c8
+		return;
4728c8
+	vector_foreach_slot(paths->pathvec, pp, i) {
4728c8
+		for (j = 0; j < CONCUR_NR_EVENT; j++) {
4728c8
+			rc = try_to_cancel_timeout_io(pp->dio_ctx_array + j,
4728c8
+					&curr_time, pp->devname);
4728c8
+			account_async_io_state(pp, rc);
4728c8
+		}
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static void cancel_inflight_io(struct io_err_stat_path *pp)
4728c8
+{
4728c8
+	struct io_event event;
4728c8
+	int i, r;
4728c8
+
4728c8
+	for (i = 0; i < CONCUR_NR_EVENT; i++) {
4728c8
+		struct dio_ctx *ct = pp->dio_ctx_array + i;
4728c8
+		struct iocb *ios[1] = { &ct->io };
4728c8
+
4728c8
+		if (ct->io_starttime.tv_sec == 0
4728c8
+				&& ct->io_starttime.tv_nsec == 0)
4728c8
+			continue;
4728c8
+		io_err_stat_log(5, "%s: abort infligh io",
4728c8
+				pp->devname);
4728c8
+		r = io_cancel(ioctx, ios[0], &event);
4728c8
+		if (r)
4728c8
+			io_err_stat_log(5, "%s: io_cancel error %d, %i",
4728c8
+					pp->devname, r, errno);
4728c8
+		ct->io_starttime.tv_sec = 0;
4728c8
+		ct->io_starttime.tv_nsec = 0;
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static inline int handle_done_dio_ctx(struct dio_ctx *ct, struct io_event *ev)
4728c8
+{
4728c8
+	ct->io_starttime.tv_sec = 0;
4728c8
+	ct->io_starttime.tv_nsec = 0;
4728c8
+	return (ev->res == ct->blksize) ? PATH_UP : PATH_DOWN;
4728c8
+}
4728c8
+
4728c8
+static void handle_async_io_done_event(struct io_event *io_evt)
4728c8
+{
4728c8
+	struct io_err_stat_path *pp;
4728c8
+	struct dio_ctx *ct;
4728c8
+	int rc = PATH_UNCHECKED;
4728c8
+	int i, j;
4728c8
+
4728c8
+	vector_foreach_slot(paths->pathvec, pp, i) {
4728c8
+		for (j = 0; j < CONCUR_NR_EVENT; j++) {
4728c8
+			ct = pp->dio_ctx_array + j;
4728c8
+			if (&ct->io == io_evt->obj) {
4728c8
+				rc = handle_done_dio_ctx(ct, io_evt);
4728c8
+				account_async_io_state(pp, rc);
4728c8
+				return;
4728c8
+			}
4728c8
+		}
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static void process_async_ios_event(int timeout_nsecs, char *dev)
4728c8
+{
4728c8
+	struct io_event events[CONCUR_NR_EVENT];
4728c8
+	int		i, n;
4728c8
+	struct timespec	timeout = { .tv_nsec = timeout_nsecs };
4728c8
+
4728c8
+	errno = 0;
4728c8
+	n = io_getevents(ioctx, 1L, CONCUR_NR_EVENT, events, &timeout);
4728c8
+	if (n < 0) {
4728c8
+		io_err_stat_log(3, "%s: async io events returned %d (errno=%s)",
4728c8
+				dev, n, strerror(errno));
4728c8
+	} else {
4728c8
+		for (i = 0; i < n; i++)
4728c8
+			handle_async_io_done_event(&events[i]);
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+static void service_paths(void)
4728c8
+{
4728c8
+	struct io_err_stat_path *pp;
4728c8
+	int i;
4728c8
+
4728c8
+	pthread_mutex_lock(&paths->mutex);
4728c8
+	vector_foreach_slot(paths->pathvec, pp, i) {
4728c8
+		send_batch_async_ios(pp);
4728c8
+		process_async_ios_event(TIMEOUT_NO_IO_NSEC, pp->devname);
4728c8
+		poll_async_io_timeout();
4728c8
+		poll_io_err_stat(vecs, pp);
4728c8
+	}
4728c8
+	pthread_mutex_unlock(&paths->mutex);
4728c8
+}
4728c8
+
4728c8
+static void cleanup_unlock(void *arg)
4728c8
+{
4728c8
+	pthread_mutex_unlock((pthread_mutex_t*) arg);
4728c8
+}
4728c8
+
4728c8
+static void cleanup_exited(void *arg)
4728c8
+{
4728c8
+	uatomic_set(&io_err_thread_running, 0);
4728c8
+}
4728c8
+
4728c8
+static void *io_err_stat_loop(void *data)
4728c8
+{
4728c8
+	vecs = (struct vectors *)data;
4728c8
+
4728c8
+	pthread_cleanup_push(cleanup_exited, NULL);
4728c8
+
4728c8
+	mlockall(MCL_CURRENT | MCL_FUTURE);
4728c8
+
4728c8
+	pthread_mutex_lock(&io_err_thread_lock);
4728c8
+	uatomic_set(&io_err_thread_running, 1);
4728c8
+	pthread_cond_broadcast(&io_err_thread_cond);
4728c8
+	pthread_mutex_unlock(&io_err_thread_lock);
4728c8
+
4728c8
+	while (1) {
4728c8
+		service_paths();
4728c8
+		usleep(100000);
4728c8
+	}
4728c8
+
4728c8
+	pthread_cleanup_pop(1);
4728c8
+	return NULL;
4728c8
+}
4728c8
+
4728c8
+int start_io_err_stat_thread(void *data)
4728c8
+{
4728c8
+	int ret;
4728c8
+
4728c8
+	if (uatomic_read(&io_err_thread_running) == 1)
4728c8
+		return 0;
4728c8
+
4728c8
+	if (io_setup(CONCUR_NR_EVENT, &ioctx) != 0) {
4728c8
+		io_err_stat_log(4, "io_setup failed");
4728c8
+		return 1;
4728c8
+	}
4728c8
+	paths = alloc_pathvec();
4728c8
+	if (!paths)
4728c8
+		goto destroy_ctx;
4728c8
+
4728c8
+	pthread_mutex_lock(&io_err_thread_lock);
4728c8
+	pthread_cleanup_push(cleanup_unlock, &io_err_thread_lock);
4728c8
+
4728c8
+	ret = pthread_create(&io_err_stat_thr, &io_err_stat_attr,
4728c8
+			     io_err_stat_loop, data);
4728c8
+
4728c8
+	while (!ret && !uatomic_read(&io_err_thread_running) &&
4728c8
+	       pthread_cond_wait(&io_err_thread_cond,
4728c8
+				 &io_err_thread_lock) == 0);
4728c8
+
4728c8
+	pthread_cleanup_pop(1);
4728c8
+
4728c8
+	if (ret) {
4728c8
+		io_err_stat_log(0, "cannot create io_error statistic thread");
4728c8
+		goto out_free;
4728c8
+	}
4728c8
+
4728c8
+	io_err_stat_log(2, "io_error statistic thread started");
4728c8
+	return 0;
4728c8
+
4728c8
+out_free:
4728c8
+	free_io_err_pathvec(paths);
4728c8
+destroy_ctx:
4728c8
+	io_destroy(ioctx);
4728c8
+	io_err_stat_log(0, "failed to start io_error statistic thread");
4728c8
+	return 1;
4728c8
+}
4728c8
+
4728c8
+void stop_io_err_stat_thread(void)
4728c8
+{
4728c8
+	if (io_err_stat_thr == (pthread_t)0)
4728c8
+		return;
4728c8
+
4728c8
+	if (uatomic_read(&io_err_thread_running) == 1)
4728c8
+		pthread_cancel(io_err_stat_thr);
4728c8
+
4728c8
+	pthread_join(io_err_stat_thr, NULL);
4728c8
+	free_io_err_pathvec(paths);
4728c8
+	io_destroy(ioctx);
4728c8
+}
4728c8
Index: multipath-tools-130222/libmultipath/io_err_stat.h
4728c8
===================================================================
4728c8
--- /dev/null
4728c8
+++ multipath-tools-130222/libmultipath/io_err_stat.h
4728c8
@@ -0,0 +1,15 @@
4728c8
+#ifndef _IO_ERR_STAT_H
4728c8
+#define _IO_ERR_STAT_H
4728c8
+
4728c8
+#include "vector.h"
4728c8
+#include "lock.h"
4728c8
+
4728c8
+
4728c8
+extern pthread_attr_t io_err_stat_attr;
4728c8
+
4728c8
+int start_io_err_stat_thread(void *data);
4728c8
+void stop_io_err_stat_thread(void);
4728c8
+int io_err_stat_handle_pathfail(struct path *path);
4728c8
+int need_io_err_check(struct path *pp);
4728c8
+
4728c8
+#endif /* _IO_ERR_STAT_H */
4728c8
Index: multipath-tools-130222/libmultipath/propsel.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/propsel.c
4728c8
+++ multipath-tools-130222/libmultipath/propsel.c
4728c8
@@ -956,6 +956,104 @@ select_delay_wait_checks (struct multipa
4728c8
 }
4728c8
 
4728c8
 extern int
4728c8
+select_marginal_path_err_sample_time(struct multipath * mp)
4728c8
+{
4728c8
+	if (mp->mpe &&
4728c8
+	    mp->mpe->marginal_path_err_sample_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_sample_time = mp->mpe->marginal_path_err_sample_time;
4728c8
+		condlog(3, "marginal_path_err_sample_time = %i (multipath setting)", mp->marginal_path_err_sample_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (mp->hwe &&
4728c8
+	    mp->hwe->marginal_path_err_sample_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_sample_time = mp->hwe->marginal_path_err_sample_time;
4728c8
+		condlog(3, "marginal_path_err_sample_time = %i (controler setting)", mp->marginal_path_err_sample_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (conf->marginal_path_err_sample_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_sample_time = conf->marginal_path_err_sample_time;
4728c8
+		condlog(3, "marginal_path_err_sample_time = %i (config file default)", mp->marginal_path_err_sample_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	mp->marginal_path_err_sample_time = DEFAULT_DELAY_CHECKS;
4728c8
+	condlog(3, "marginal_path_err_sample_time = DISABLED (internal default)");
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+extern int
4728c8
+select_marginal_path_err_rate_threshold(struct multipath * mp)
4728c8
+{
4728c8
+	if (mp->mpe &&
4728c8
+	    mp->mpe->marginal_path_err_rate_threshold != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_rate_threshold = mp->mpe->marginal_path_err_rate_threshold;
4728c8
+		condlog(3, "marginal_path_err_rate_threshold = %i (multipath setting)", mp->marginal_path_err_rate_threshold);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (mp->hwe &&
4728c8
+	    mp->hwe->marginal_path_err_rate_threshold != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_rate_threshold = mp->hwe->marginal_path_err_rate_threshold;
4728c8
+		condlog(3, "marginal_path_err_rate_threshold = %i (controler setting)", mp->marginal_path_err_rate_threshold);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (conf->marginal_path_err_rate_threshold != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_rate_threshold = conf->marginal_path_err_rate_threshold;
4728c8
+		condlog(3, "marginal_path_err_rate_threshold = %i (config file default)", mp->marginal_path_err_rate_threshold);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	mp->marginal_path_err_rate_threshold = DEFAULT_DELAY_CHECKS;
4728c8
+	condlog(3, "marginal_path_err_rate_threshold = DISABLED (internal default)");
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+extern int
4728c8
+select_marginal_path_err_recheck_gap_time(struct multipath * mp)
4728c8
+{
4728c8
+	if (mp->mpe && mp->mpe->marginal_path_err_recheck_gap_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_recheck_gap_time = mp->mpe->marginal_path_err_recheck_gap_time;
4728c8
+		condlog(3, "marginal_path_err_recheck_gap_time = %i (multipath setting)", mp->marginal_path_err_recheck_gap_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (mp->hwe && mp->hwe->marginal_path_err_recheck_gap_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_recheck_gap_time = mp->hwe->marginal_path_err_recheck_gap_time;
4728c8
+		condlog(3, "marginal_path_err_recheck_gap_time = %i (controler setting)", mp->marginal_path_err_recheck_gap_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (conf->marginal_path_err_recheck_gap_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_err_recheck_gap_time = conf->marginal_path_err_recheck_gap_time;
4728c8
+		condlog(3, "marginal_path_err_recheck_gap_time = %i (config file default)", mp->marginal_path_err_recheck_gap_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	mp->marginal_path_err_recheck_gap_time = DEFAULT_DELAY_CHECKS;
4728c8
+	condlog(3, "marginal_path_err_recheck_gap_time = DISABLED (internal default)");
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+extern int
4728c8
+select_marginal_path_double_failed_time(struct multipath * mp)
4728c8
+{
4728c8
+	if (mp->mpe &&
4728c8
+	    mp->mpe->marginal_path_double_failed_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_double_failed_time = mp->mpe->marginal_path_double_failed_time;
4728c8
+		condlog(3, "marginal_path_double_failed_time = %i (multipath setting)", mp->marginal_path_double_failed_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (mp->hwe &&
4728c8
+	    mp->hwe->marginal_path_double_failed_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_double_failed_time = mp->hwe->marginal_path_double_failed_time;
4728c8
+		condlog(3, "marginal_path_double_failed_time = %i (controler setting)", mp->marginal_path_double_failed_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	if (conf->marginal_path_double_failed_time != MARGINAL_PATH_UNDEF) {
4728c8
+		mp->marginal_path_double_failed_time = conf->marginal_path_double_failed_time;
4728c8
+		condlog(3, "marginal_path_double_failed_time = %i (config file default)", mp->marginal_path_double_failed_time);
4728c8
+		return 0;
4728c8
+	}
4728c8
+	mp->marginal_path_double_failed_time = DEFAULT_DELAY_CHECKS;
4728c8
+	condlog(3, "marginal_path_double_failed_time = DISABLED (internal default)");
4728c8
+	return 0;
4728c8
+}
4728c8
+
4728c8
+extern int
4728c8
 select_skip_kpartx (struct multipath * mp)
4728c8
 {
4728c8
 	if (mp->mpe && mp->mpe->skip_kpartx != SKIP_KPARTX_UNDEF) {
4728c8
Index: multipath-tools-130222/libmultipath/propsel.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/propsel.h
4728c8
+++ multipath-tools-130222/libmultipath/propsel.h
4728c8
@@ -24,6 +24,10 @@ int select_detect_checker(struct path *
4728c8
 int select_deferred_remove(struct multipath *mp);
4728c8
 int select_delay_watch_checks (struct multipath * mp);
4728c8
 int select_delay_wait_checks (struct multipath * mp);
4728c8
+int select_marginal_path_err_sample_time(struct multipath *mp);
4728c8
+int select_marginal_path_err_rate_threshold(struct multipath *mp);
4728c8
+int select_marginal_path_err_recheck_gap_time(struct multipath *mp);
4728c8
+int select_marginal_path_double_failed_time(struct multipath *mp);
4728c8
 int select_skip_kpartx (struct multipath * mp);
4728c8
 int select_max_sectors_kb (struct multipath * mp);
4728c8
 int select_unpriv_sgio (struct multipath * mp);
4728c8
Index: multipath-tools-130222/libmultipath/structs.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/structs.h
4728c8
+++ multipath-tools-130222/libmultipath/structs.h
4728c8
@@ -3,6 +3,7 @@
4728c8
 
4728c8
 #include <sys/types.h>
4728c8
 #include <inttypes.h>
4728c8
+#include <time.h>
4728c8
 
4728c8
 #include "prio.h"
4728c8
 #include "byteorder.h"
4728c8
@@ -176,6 +177,11 @@ enum delay_checks_states {
4728c8
 	DELAY_CHECKS_UNDEF = 0,
4728c8
 };
4728c8
 
4728c8
+enum marginal_path_states {
4728c8
+	MARGINAL_PATH_OFF = -1,
4728c8
+	MARGINAL_PATH_UNDEF = 0,
4728c8
+};
4728c8
+
4728c8
 enum missing_udev_info_states {
4728c8
 	INFO_OK,
4728c8
 	INFO_MISSING,
4728c8
@@ -252,6 +258,10 @@ struct path {
4728c8
 	int missing_udev_info;
4728c8
 	int retriggers;
4728c8
 	int wwid_changed;
4728c8
+	time_t io_err_dis_reinstate_time;
4728c8
+	int io_err_disable_reinstate;
4728c8
+	int io_err_pathfail_cnt;
4728c8
+	int io_err_pathfail_starttime;
4728c8
 
4728c8
 	/* configlet pointers */
4728c8
 	struct hwentry * hwe;
4728c8
@@ -285,6 +295,10 @@ struct multipath {
4728c8
 	int deferred_remove;
4728c8
 	int delay_watch_checks;
4728c8
 	int delay_wait_checks;
4728c8
+	int marginal_path_err_sample_time;
4728c8
+	int marginal_path_err_rate_threshold;
4728c8
+	int marginal_path_err_recheck_gap_time;
4728c8
+	int marginal_path_double_failed_time;
4728c8
 	int force_udev_reload;
4728c8
 	int skip_kpartx;
4728c8
 	int max_sectors_kb;
4728c8
Index: multipath-tools-130222/libmultipath/time-util.c
4728c8
===================================================================
4728c8
--- /dev/null
4728c8
+++ multipath-tools-130222/libmultipath/time-util.c
4728c8
@@ -0,0 +1,42 @@
4728c8
+#include <assert.h>
4728c8
+#include <pthread.h>
4728c8
+#include <time.h>
4728c8
+#include "time-util.h"
4728c8
+
4728c8
+/* Initialize @cond as a condition variable that uses the monotonic clock */
4728c8
+void pthread_cond_init_mono(pthread_cond_t *cond)
4728c8
+{
4728c8
+	pthread_condattr_t attr;
4728c8
+	int res;
4728c8
+
4728c8
+	res = pthread_condattr_init(&attr);
4728c8
+	assert(res == 0);
4728c8
+	res = pthread_condattr_setclock(&attr, CLOCK_MONOTONIC);
4728c8
+	assert(res == 0);
4728c8
+	res = pthread_cond_init(cond, &attr);
4728c8
+	assert(res == 0);
4728c8
+	res = pthread_condattr_destroy(&attr);
4728c8
+	assert(res == 0);
4728c8
+}
4728c8
+
4728c8
+/* Ensure that 0 <= ts->tv_nsec && ts->tv_nsec < 1000 * 1000 * 1000. */
4728c8
+void normalize_timespec(struct timespec *ts)
4728c8
+{
4728c8
+	while (ts->tv_nsec < 0) {
4728c8
+		ts->tv_nsec += 1000UL * 1000 * 1000;
4728c8
+		ts->tv_sec--;
4728c8
+	}
4728c8
+	while (ts->tv_nsec >= 1000UL * 1000 * 1000) {
4728c8
+		ts->tv_nsec -= 1000UL * 1000 * 1000;
4728c8
+		ts->tv_sec++;
4728c8
+	}
4728c8
+}
4728c8
+
4728c8
+/* Compute *res = *a - *b */
4728c8
+void timespecsub(const struct timespec *a, const struct timespec *b,
4728c8
+		 struct timespec *res)
4728c8
+{
4728c8
+	res->tv_sec = a->tv_sec - b->tv_sec;
4728c8
+	res->tv_nsec = a->tv_nsec - b->tv_nsec;
4728c8
+	normalize_timespec(res);
4728c8
+}
4728c8
Index: multipath-tools-130222/libmultipath/time-util.h
4728c8
===================================================================
4728c8
--- /dev/null
4728c8
+++ multipath-tools-130222/libmultipath/time-util.h
4728c8
@@ -0,0 +1,13 @@
4728c8
+#ifndef _TIME_UTIL_H_
4728c8
+#define _TIME_UTIL_H_
4728c8
+
4728c8
+#include <pthread.h>
4728c8
+
4728c8
+struct timespec;
4728c8
+
4728c8
+void pthread_cond_init_mono(pthread_cond_t *cond);
4728c8
+void normalize_timespec(struct timespec *ts);
4728c8
+void timespecsub(const struct timespec *a, const struct timespec *b,
4728c8
+		 struct timespec *res);
4728c8
+
4728c8
+#endif /* _TIME_UTIL_H_ */
4728c8
Index: multipath-tools-130222/libmultipath/uevent.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/uevent.c
4728c8
+++ multipath-tools-130222/libmultipath/uevent.c
4728c8
@@ -616,12 +616,46 @@ uevent_get_dm_name(struct uevent *uev)
4728c8
 	int i;
4728c8
 
4728c8
 	for (i = 0; uev->envp[i] != NULL; i++) {
4728c8
-		if (!strncmp(uev->envp[i], "DM_NAME", 6) &&
4728c8
-		    strlen(uev->envp[i]) > 7) {
4728c8
+		if (!strncmp(uev->envp[i], "DM_NAME", 7) &&
4728c8
+		    strlen(uev->envp[i]) > 8) {
4728c8
 			p = MALLOC(strlen(uev->envp[i] + 8) + 1);
4728c8
 			strcpy(p, uev->envp[i] + 8);
4728c8
 			break;
4728c8
 		}
4728c8
 	}
4728c8
+	return p;
4728c8
+}
4728c8
+
4728c8
+extern char *
4728c8
+uevent_get_dm_path(struct uevent *uev)
4728c8
+{
4728c8
+	char *p = NULL;
4728c8
+	int i;
4728c8
+
4728c8
+	for (i = 0; uev->envp[i] != NULL; i++) {
4728c8
+		if (!strncmp(uev->envp[i], "DM_PATH", 7) &&
4728c8
+		    strlen(uev->envp[i]) > 8) {
4728c8
+			p = MALLOC(strlen(uev->envp[i] + 8) + 1);
4728c8
+			strcpy(p, uev->envp[i] + 8);
4728c8
+			break;
4728c8
+		}
4728c8
+	}
4728c8
+	return p;
4728c8
+}
4728c8
+
4728c8
+extern char *
4728c8
+uevent_get_dm_action(struct uevent *uev)
4728c8
+{
4728c8
+	char *p = NULL;
4728c8
+	int i;
4728c8
+
4728c8
+	for (i = 0; uev->envp[i] != NULL; i++) {
4728c8
+		if (!strncmp(uev->envp[i], "DM_ACTION", 9) &&
4728c8
+		    strlen(uev->envp[i]) > 10) {
4728c8
+			p = MALLOC(strlen(uev->envp[i] + 10) + 1);
4728c8
+			strcpy(p, uev->envp[i] + 10);
4728c8
+			break;
4728c8
+		}
4728c8
+	}
4728c8
 	return p;
4728c8
 }
4728c8
Index: multipath-tools-130222/libmultipath/uevent.h
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/libmultipath/uevent.h
4728c8
+++ multipath-tools-130222/libmultipath/uevent.h
4728c8
@@ -36,5 +36,7 @@ int uevent_get_major(struct uevent *uev)
4728c8
 int uevent_get_minor(struct uevent *uev);
4728c8
 int uevent_get_disk_ro(struct uevent *uev);
4728c8
 char *uevent_get_dm_name(struct uevent *uev);
4728c8
+char *uevent_get_dm_path(struct uevent *uev);
4728c8
+char *uevent_get_dm_action(struct uevent *uev);
4728c8
 
4728c8
 #endif /* _UEVENT_H */
4728c8
Index: multipath-tools-130222/multipath/multipath.conf.5
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/multipath/multipath.conf.5
4728c8
+++ multipath-tools-130222/multipath/multipath.conf.5
4728c8
@@ -527,7 +527,7 @@ recently become valid for this many chec
4728c8
 being watched, when they next become valid, they will not be used until they
4728c8
 have stayed up for
4728c8
 .I delay_wait_checks
4728c8
-checks. Default is
4728c8
+checks. See "Shaky paths detection" below. Default is
4728c8
 .I no
4728c8
 .TP
4728c8
 .B delay_wait_checks
4728c8
@@ -537,9 +537,56 @@ online fails again within
4728c8
 checks, the next time it comes back online, it will marked and delayed, and not
4728c8
 used until it has passed
4728c8
 .I delay_wait_checks
4728c8
-checks. Default is
4728c8
+checks. See "Shaky paths detection" below. Default is
4728c8
 .I no
4728c8
 .TP
4728c8
+.B marginal_path_double_failed_time
4728c8
+One of the four parameters of supporting path check based on accounting IO
4728c8
+error such as intermittent error. When a path failed event occurs twice in
4728c8
+\fImarginal_path_double_failed_time\fR seconds due to an IO error and all the
4728c8
+other three parameters are set, multipathd will fail the path and enqueue
4728c8
+this path into a queue of which members are sent a couple of continuous
4728c8
+direct reading asynchronous IOs at a fixed sample rate of 10HZ to start IO
4728c8
+error accounting process. See "Shaky paths detection" below. Default is
4728c8
+\fIno\fR
4728c8
+.TP
4728c8
+.B marginal_path_err_sample_time
4728c8
+One of the four parameters of supporting path check based on accounting IO
4728c8
+error such as intermittent error. If it is set to a value no less than 120,
4728c8
+when a path fail event occurs twice in \fImarginal_path_double_failed_time\fR
4728c8
+second due to an IO error, multipathd will fail the path and enqueue this
4728c8
+path into a queue of which members are sent a couple of continuous direct
4728c8
+reading asynchronous IOs at a fixed sample rate of 10HZ to start the IO
4728c8
+accounting process for the path will last for
4728c8
+\fImarginal_path_err_sample_time\fR.
4728c8
+If the rate of IO error on a particular path is greater than the
4728c8
+\fImarginal_path_err_rate_threshold\fR, then the path will not reinstate for
4728c8
+\fImarginal_path_err_recheck_gap_time\fR seconds unless there is only one
4728c8
+active path. After \fImarginal_path_err_recheck_gap_time\fR expires, the path
4728c8
+will be requeueed for rechecking. If checking result is good enough, the
4728c8
+path will be reinstated. See "Shaky paths detection" below. Default is
4728c8
+\fIno\fR
4728c8
+.TP
4728c8
+.B marginal_path_err_rate_threshold
4728c8
+The error rate threshold as a permillage (1/1000). One of the four parameters
4728c8
+of supporting path check based on accounting IO error such as intermittent
4728c8
+error. Refer to \fImarginal_path_err_sample_time\fR. If the rate of IO errors
4728c8
+on a particular path is greater than this parameter, then the path will not
4728c8
+reinstate for \fImarginal_path_err_recheck_gap_time\fR seconds unless there is
4728c8
+only one active path. See "Shaky paths detection" below. Default is \fIno\fR
4728c8
+.TP
4728c8
+.B marginal_path_err_recheck_gap_time
4728c8
+One of the four parameters of supporting path check based on accounting IO
4728c8
+error such as intermittent error. Refer to
4728c8
+\fImarginal_path_err_sample_time\fR. If this parameter is set to a positive
4728c8
+value, the failed path of  which the IO error rate is larger than
4728c8
+\fImarginal_path_err_rate_threshold\fR will be kept in failed state for
4728c8
+\fImarginal_path_err_recheck_gap_time\fR seconds. When
4728c8
+\fImarginal_path_err_recheck_gap_time\fR seconds expires, the path will be
4728c8
+requeueed for checking. If checking result is good enough, the path will be
4728c8
+reinstated, or else it will keep failed. See "Shaky paths detection" below.
4728c8
+Default is \fIno\fR
4728c8
+.TP
4728c8
 .B missing_uev_wait_timeout
4728c8
 Controls how many seconds multipathd will wait, after a new multipath device
4728c8
 is created, to receive a change event from udev for the device, before
4728c8
@@ -771,6 +818,14 @@ section:
4728c8
 .TP
4728c8
 .B delay_wait_checks
4728c8
 .TP
4728c8
+.B marginal_path_err_sample_time
4728c8
+.TP
4728c8
+.B marginal_path_err_rate_threshold
4728c8
+.TP
4728c8
+.B marginal_path_err_recheck_gap_time
4728c8
+.TP
4728c8
+.B marginal_path_double_failed_time
4728c8
+.TP
4728c8
 .B skip_kpartx
4728c8
 .TP
4728c8
 .B max_sectors_kb
4728c8
@@ -877,6 +932,14 @@ section:
4728c8
 .TP
4728c8
 .B delay_wait_checks
4728c8
 .TP
4728c8
+.B marginal_path_err_sample_time
4728c8
+.TP
4728c8
+.B marginal_path_err_rate_threshold
4728c8
+.TP
4728c8
+.B marginal_path_err_recheck_gap_time
4728c8
+.TP
4728c8
+.B marginal_path_double_failed_time
4728c8
+.TP
4728c8
 .B skip_kpartx
4728c8
 .TP
4728c8
 .B max_sectors_kb
4728c8
@@ -887,6 +950,47 @@ section:
4728c8
 .RE
4728c8
 .PD
4728c8
 .LP
4728c8
+.SH "Shaky paths detection"
4728c8
+A common problem in SAN setups is the occurence of intermittent errors: a
4728c8
+path is unreachable, then reachable again for a short time, disappears again,
4728c8
+and so forth. This happens typically on unstable interconnects. It is
4728c8
+undesirable to switch pathgroups unnecessarily on such frequent, unreliable
4728c8
+events. \fImultipathd\fR supports two different methods for detecting this
4728c8
+situation and dealing with it. Both methods share the same basic mode of
4728c8
+operation: If a path is found to be \(dqshaky\(dq or \(dqflipping\(dq,
4728c8
+and appears to be in healthy status, it is not reinstated (put back to use)
4728c8
+immediately. Instead, it is watched for some time, and only reinstated
4728c8
+if the healthy state appears to be stable. The logic of determining
4728c8
+\(dqshaky\(dq condition, as well as the logic when to reinstate,
4728c8
+differs between the two methods.
4728c8
+.TP 8
4728c8
+.B \(dqdelay_checks\(dq failure tracking
4728c8
+If a path fails again within a
4728c8
+\fIdelay_watch_checks\fR interval after a failure, don't
4728c8
+reinstate it until it passes a \fIdelay_wait_checks\fR interval
4728c8
+in always good status.
4728c8
+The intervals are measured in \(dqticks\(dq, i.e. the
4728c8
+time between path checks by multipathd, which is variable and controlled by the
4728c8
+\fIpolling_interval\fR and \fImax_polling_interval\fR parameters.
4728c8
+.TP
4728c8
+.B \(dqmarginal_path\(dq failure tracking
4728c8
+If a second failure event (good->bad transition) occurs within
4728c8
+\fImarginal_path_double_failed_time\fR seconds after a failure, high-frequency
4728c8
+monitoring is started for the affected path: I/O is sent at a rate of 10 per
4728c8
+second. This is done for \fImarginal_path_err_sample_time\fR seconds. During
4728c8
+this period, the path is not reinstated. If the
4728c8
+rate of errors remains below \fImarginal_path_err_rate_threshold\fR during the
4728c8
+monitoring period, the path is reinstated. Otherwise, it
4728c8
+is kept in failed state for \fImarginal_path_err_recheck_gap_time\fR, and
4728c8
+after that, it is monitored again. For this method, time intervals are measured
4728c8
+in seconds.
4728c8
+.RE
4728c8
+.LP
4728c8
+See the documentation
4728c8
+of the individual options above for details.
4728c8
+It is \fBstrongly discouraged\fR to use more than one of these methods for any
4728c8
+given multipath map, because the two concurrent methods may interact in
4728c8
+unpredictable ways.
4728c8
 .SH "KNOWN ISSUES"
4728c8
 The usage of
4728c8
 .B queue_if_no_path
4728c8
Index: multipath-tools-130222/multipathd/cli_handlers.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/multipathd/cli_handlers.c
4728c8
+++ multipath-tools-130222/multipathd/cli_handlers.c
4728c8
@@ -721,7 +721,7 @@ int resize_map(struct multipath *mpp, un
4728c8
 
4728c8
 	mpp->size = size;
4728c8
 	update_mpp_paths(mpp, vecs->pathvec);
4728c8
-	setup_map(mpp, params, PARAMS_SIZE);
4728c8
+	setup_map(mpp, params, PARAMS_SIZE, vecs);
4728c8
 	mpp->action = ACT_RESIZE;
4728c8
 	if (domap(mpp, params) <= 0) {
4728c8
 		condlog(0, "%s: failed to resize map : %s", mpp->alias,
4728c8
Index: multipath-tools-130222/multipathd/main.c
4728c8
===================================================================
4728c8
--- multipath-tools-130222.orig/multipathd/main.c
4728c8
+++ multipath-tools-130222/multipathd/main.c
4728c8
@@ -56,6 +56,7 @@
4728c8
 #include <log.h>
4728c8
 #include <file.h>
4728c8
 #include <prkey.h>
4728c8
+#include <io_err_stat.h>
4728c8
 
4728c8
 #include "main.h"
4728c8
 #include "pidfile.h"
4728c8
@@ -274,7 +275,7 @@ retry:
4728c8
 	mpp->action = ACT_RELOAD;
4728c8
 
4728c8
 	extract_hwe_from_path(mpp);
4728c8
-	if (setup_map(mpp, params, PARAMS_SIZE)) {
4728c8
+	if (setup_map(mpp, params, PARAMS_SIZE, vecs)) {
4728c8
 		condlog(0, "%s: failed to setup new map in update", mpp->alias);
4728c8
 		retries = -1;
4728c8
 		goto fail;
4728c8
@@ -638,7 +639,7 @@ rescan:
4728c8
 	/*
4728c8
 	 * push the map to the device-mapper
4728c8
 	 */
4728c8
-	if (setup_map(mpp, params, PARAMS_SIZE)) {
4728c8
+	if (setup_map(mpp, params, PARAMS_SIZE, vecs)) {
4728c8
 		condlog(0, "%s: failed to setup map for addition of new "
4728c8
 			"path %s", mpp->alias, pp->dev);
4728c8
 		goto fail_map;
4728c8
@@ -771,7 +772,7 @@ ev_remove_path (struct path *pp, struct
4728c8
 			 */
4728c8
 		}
4728c8
 
4728c8
-		if (setup_map(mpp, params, PARAMS_SIZE)) {
4728c8
+		if (setup_map(mpp, params, PARAMS_SIZE, vecs)) {
4728c8
 			condlog(0, "%s: failed to setup map for"
4728c8
 				" removal of path %s", mpp->alias, pp->dev);
4728c8
 			goto fail;
4728c8
@@ -891,6 +892,41 @@ uev_update_path (struct uevent *uev, str
4728c8
 }
4728c8
 
4728c8
 static int
4728c8
+uev_pathfail_check(struct uevent *uev, struct vectors *vecs)
4728c8
+{
4728c8
+	char *action = NULL, *devt = NULL;
4728c8
+	struct path *pp;
4728c8
+	int r = 1;
4728c8
+
4728c8
+	action = uevent_get_dm_action(uev);
4728c8
+	if (!action)
4728c8
+		return 1;
4728c8
+	if (strncmp(action, "PATH_FAILED", 11))
4728c8
+		goto out;
4728c8
+	devt = uevent_get_dm_path(uev);
4728c8
+	if (!devt) {
4728c8
+		condlog(3, "%s: No DM_PATH in uevent", uev->kernel);
4728c8
+		goto out;
4728c8
+	}
4728c8
+
4728c8
+	pp = find_path_by_devt(vecs->pathvec, devt);
4728c8
+	if (!pp)
4728c8
+		goto out_devt;
4728c8
+	r = io_err_stat_handle_pathfail(pp);
4728c8
+
4728c8
+	if (r)
4728c8
+		condlog(3, "io_err_stat: %s: cannot handle pathfail uevent",
4728c8
+			pp->dev);
4728c8
+out_devt:
4728c8
+	FREE(devt);
4728c8
+	FREE(action);
4728c8
+	return r;
4728c8
+out:
4728c8
+	FREE(action);
4728c8
+	return 1;
4728c8
+}
4728c8
+
4728c8
+static int
4728c8
 map_discovery (struct vectors * vecs)
4728c8
 {
4728c8
 	struct multipath * mpp;
4728c8
@@ -974,6 +1010,14 @@ uev_trigger (struct uevent * uev, void *
4728c8
 	if (!strncmp(uev->kernel, "dm-", 3)) {
4728c8
 		if (!strncmp(uev->action, "change", 6)) {
4728c8
 			r = uev_add_map(uev, vecs);
4728c8
+
4728c8
+			/*
4728c8
+			 * the kernel-side dm-mpath issues a PATH_FAILED event
4728c8
+			 * when it encounters a path IO error. It is reason-
4728c8
+			 * able be the entry of path IO error accounting pro-
4728c8
+			 * cess.
4728c8
+			 */
4728c8
+			uev_pathfail_check(uev, vecs);
4728c8
 			goto out;
4728c8
 		}
4728c8
 		if (!strncmp(uev->action, "remove", 6)) {
4728c8
@@ -1405,6 +1449,17 @@ check_path (struct vectors * vecs, struc
4728c8
 		return;
4728c8
 
4728c8
 	if ((newstate == PATH_UP || newstate == PATH_GHOST) &&
4728c8
+	    pp->io_err_disable_reinstate && need_io_err_check(pp)) {
4728c8
+		pp->state = PATH_SHAKY;
4728c8
+		/*
4728c8
+		 * to reschedule as soon as possible,so that this path can
4728c8
+		 * be recoverd in time
4728c8
+		 */
4728c8
+		pp->tick = 1;
4728c8
+		return;
4728c8
+	}
4728c8
+
4728c8
+	if ((newstate == PATH_UP || newstate == PATH_GHOST) &&
4728c8
 	     pp->wait_checks > 0) {
4728c8
 		if (pp->mpp && pp->mpp->nr_active > 0) {
4728c8
 			pp->state = PATH_DELAYED;
4728c8
@@ -1955,6 +2010,7 @@ child (void * param)
4728c8
 	setup_thread_attr(&misc_attr, 64 * 1024, 1);
4728c8
 	setup_thread_attr(&uevent_attr, 128 * 1024, 1);
4728c8
 	setup_thread_attr(&waiter_attr, 32 * 1024, 1);
4728c8
+	setup_thread_attr(&io_err_stat_attr, 32 * 1024, 0);
4728c8
 
4728c8
 	if (logsink) {
4728c8
 		setup_thread_attr(&log_attr, 64 * 1024, 0);
4728c8
@@ -2097,6 +2153,8 @@ child (void * param)
4728c8
 	*/
4728c8
 	cleanup_checkers();
4728c8
 	cleanup_prio();
4728c8
+	stop_io_err_stat_thread();
4728c8
+	pthread_attr_destroy(&io_err_stat_attr);
4728c8
 
4728c8
 	dm_lib_release();
4728c8
 	dm_lib_exit();