mrc0mmand / rpms / lvm2

Forked from rpms/lvm2 2 years ago
Clone

Blame SOURCES/lvm2-2_02_181-dmeventd-base-vdo-plugin.patch

0d8a0a
 WHATS_NEW_DM                                   |   1 +
0d8a0a
 configure                                      |   3 +-
0d8a0a
 configure.ac                                   |   1 +
0d8a0a
 daemons/dmeventd/libdevmapper-event.c          |   1 +
0d8a0a
 daemons/dmeventd/plugins/Makefile.in           |   9 +-
0d8a0a
 daemons/dmeventd/plugins/vdo/.exported_symbols |   3 +
0d8a0a
 daemons/dmeventd/plugins/vdo/Makefile.in       |  36 +++
0d8a0a
 daemons/dmeventd/plugins/vdo/dmeventd_vdo.c    | 406 +++++++++++++++++++++++++
0d8a0a
 8 files changed, 453 insertions(+), 7 deletions(-)
0d8a0a
 create mode 100644 daemons/dmeventd/plugins/vdo/.exported_symbols
0d8a0a
 create mode 100644 daemons/dmeventd/plugins/vdo/Makefile.in
0d8a0a
 create mode 100644 daemons/dmeventd/plugins/vdo/dmeventd_vdo.c
0d8a0a
0d8a0a
diff --git a/WHATS_NEW_DM b/WHATS_NEW_DM
0d8a0a
index e77352a..c42ee17 100644
0d8a0a
--- a/WHATS_NEW_DM
0d8a0a
+++ b/WHATS_NEW_DM
0d8a0a
@@ -1,5 +1,6 @@
0d8a0a
 Version 1.02.150 - 
0d8a0a
 =================================
0d8a0a
+  Add vdo plugin for monitoring VDO devices.
0d8a0a
 
0d8a0a
 Version 1.02.149 - 19th July 2018
0d8a0a
 =================================
0d8a0a
diff --git a/configure b/configure
0d8a0a
index 10b49c6..eb7d70b 100755
0d8a0a
--- a/configure
0d8a0a
+++ b/configure
0d8a0a
@@ -15648,7 +15648,7 @@ _ACEOF
0d8a0a
 
0d8a0a
 
0d8a0a
 ################################################################################
0d8a0a
-ac_config_files="$ac_config_files Makefile make.tmpl daemons/Makefile daemons/clvmd/Makefile daemons/cmirrord/Makefile daemons/dmeventd/Makefile daemons/dmeventd/libdevmapper-event.pc daemons/dmeventd/plugins/Makefile daemons/dmeventd/plugins/lvm2/Makefile daemons/dmeventd/plugins/raid/Makefile daemons/dmeventd/plugins/mirror/Makefile daemons/dmeventd/plugins/snapshot/Makefile daemons/dmeventd/plugins/thin/Makefile daemons/dmfilemapd/Makefile daemons/lvmdbusd/Makefile daemons/lvmdbusd/lvmdbusd daemons/lvmdbusd/lvmdb.py daemons/lvmdbusd/lvm_shell_proxy.py daemons/lvmdbusd/path.py daemons/lvmetad/Makefile daemons/lvmpolld/Makefile daemons/lvmlockd/Makefile device_mapper/Makefile conf/Makefile conf/example.conf conf/lvmlocal.conf conf/command_profile_template.profile conf/metadata_profile_template.profile include/.symlinks include/Makefile lib/Makefile lib/format1/Makefile lib/format_pool/Makefile lib/locking/Makefile include/lvm-version.h libdaemon/Makefile libdaemon/client/Makefile libdaemon/server/Makefile libdm/Makefile libdm/libdevmapper.pc liblvm/Makefile liblvm/liblvm2app.pc man/Makefile po/Makefile python/Makefile python/setup.py scripts/blkdeactivate.sh scripts/blk_availability_init_red_hat scripts/blk_availability_systemd_red_hat.service scripts/clvmd_init_red_hat scripts/cmirrord_init_red_hat scripts/com.redhat.lvmdbus1.service scripts/dm_event_systemd_red_hat.service scripts/dm_event_systemd_red_hat.socket scripts/lvm2_cluster_activation_red_hat.sh scripts/lvm2_cluster_activation_systemd_red_hat.service scripts/lvm2_clvmd_systemd_red_hat.service scripts/lvm2_cmirrord_systemd_red_hat.service scripts/lvm2_lvmdbusd_systemd_red_hat.service scripts/lvm2_lvmetad_init_red_hat scripts/lvm2_lvmetad_systemd_red_hat.service scripts/lvm2_lvmetad_systemd_red_hat.socket scripts/lvm2_lvmpolld_init_red_hat scripts/lvm2_lvmpolld_systemd_red_hat.service scripts/lvm2_lvmpolld_systemd_red_hat.socket scripts/lvm2_lvmlockd_systemd_red_hat.service scripts/lvm2_lvmlocking_systemd_red_hat.service scripts/lvm2_monitoring_init_red_hat scripts/lvm2_monitoring_systemd_red_hat.service scripts/lvm2_pvscan_systemd_red_hat@.service scripts/lvm2_tmpfiles_red_hat.conf scripts/lvmdump.sh scripts/Makefile test/Makefile test/api/Makefile test/api/python_lvm_unit.py test/unit/Makefile tools/Makefile udev/Makefile"
0d8a0a
+ac_config_files="$ac_config_files Makefile make.tmpl daemons/Makefile daemons/clvmd/Makefile daemons/cmirrord/Makefile daemons/dmeventd/Makefile daemons/dmeventd/libdevmapper-event.pc daemons/dmeventd/plugins/Makefile daemons/dmeventd/plugins/lvm2/Makefile daemons/dmeventd/plugins/raid/Makefile daemons/dmeventd/plugins/mirror/Makefile daemons/dmeventd/plugins/snapshot/Makefile daemons/dmeventd/plugins/thin/Makefile daemons/dmeventd/plugins/vdo/Makefile daemons/dmfilemapd/Makefile daemons/lvmdbusd/Makefile daemons/lvmdbusd/lvmdbusd daemons/lvmdbusd/lvmdb.py daemons/lvmdbusd/lvm_shell_proxy.py daemons/lvmdbusd/path.py daemons/lvmetad/Makefile daemons/lvmpolld/Makefile daemons/lvmlockd/Makefile device_mapper/Makefile conf/Makefile conf/example.conf conf/lvmlocal.conf conf/command_profile_template.profile conf/metadata_profile_template.profile include/.symlinks include/Makefile lib/Makefile lib/format1/Makefile lib/format_pool/Makefile lib/locking/Makefile include/lvm-version.h libdaemon/Makefile libdaemon/client/Makefile libdaemon/server/Makefile libdm/Makefile libdm/libdevmapper.pc liblvm/Makefile liblvm/liblvm2app.pc man/Makefile po/Makefile python/Makefile python/setup.py scripts/blkdeactivate.sh scripts/blk_availability_init_red_hat scripts/blk_availability_systemd_red_hat.service scripts/clvmd_init_red_hat scripts/cmirrord_init_red_hat scripts/com.redhat.lvmdbus1.service scripts/dm_event_systemd_red_hat.service scripts/dm_event_systemd_red_hat.socket scripts/lvm2_cluster_activation_red_hat.sh scripts/lvm2_cluster_activation_systemd_red_hat.service scripts/lvm2_clvmd_systemd_red_hat.service scripts/lvm2_cmirrord_systemd_red_hat.service scripts/lvm2_lvmdbusd_systemd_red_hat.service scripts/lvm2_lvmetad_init_red_hat scripts/lvm2_lvmetad_systemd_red_hat.service scripts/lvm2_lvmetad_systemd_red_hat.socket scripts/lvm2_lvmpolld_init_red_hat scripts/lvm2_lvmpolld_systemd_red_hat.service scripts/lvm2_lvmpolld_systemd_red_hat.socket scripts/lvm2_lvmlockd_systemd_red_hat.service scripts/lvm2_lvmlocking_systemd_red_hat.service scripts/lvm2_monitoring_init_red_hat scripts/lvm2_monitoring_systemd_red_hat.service scripts/lvm2_pvscan_systemd_red_hat@.service scripts/lvm2_tmpfiles_red_hat.conf scripts/lvmdump.sh scripts/Makefile test/Makefile test/api/Makefile test/api/python_lvm_unit.py test/unit/Makefile tools/Makefile udev/Makefile"
0d8a0a
 
0d8a0a
 cat >confcache <<\_ACEOF
0d8a0a
 # This file is a shell script that caches the results of configure
0d8a0a
@@ -16356,6 +16356,7 @@ do
0d8a0a
     "daemons/dmeventd/plugins/mirror/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/dmeventd/plugins/mirror/Makefile" ;;
0d8a0a
     "daemons/dmeventd/plugins/snapshot/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/dmeventd/plugins/snapshot/Makefile" ;;
0d8a0a
     "daemons/dmeventd/plugins/thin/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/dmeventd/plugins/thin/Makefile" ;;
0d8a0a
+    "daemons/dmeventd/plugins/vdo/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/dmeventd/plugins/vdo/Makefile" ;;
0d8a0a
     "daemons/dmfilemapd/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/dmfilemapd/Makefile" ;;
0d8a0a
     "daemons/lvmdbusd/Makefile") CONFIG_FILES="$CONFIG_FILES daemons/lvmdbusd/Makefile" ;;
0d8a0a
     "daemons/lvmdbusd/lvmdbusd") CONFIG_FILES="$CONFIG_FILES daemons/lvmdbusd/lvmdbusd" ;;
0d8a0a
diff --git a/configure.ac b/configure.ac
0d8a0a
index 9fa0c76..24bae39 100644
0d8a0a
--- a/configure.ac
0d8a0a
+++ b/configure.ac
0d8a0a
@@ -2155,6 +2155,7 @@ daemons/dmeventd/plugins/raid/Makefile
0d8a0a
 daemons/dmeventd/plugins/mirror/Makefile
0d8a0a
 daemons/dmeventd/plugins/snapshot/Makefile
0d8a0a
 daemons/dmeventd/plugins/thin/Makefile
0d8a0a
+daemons/dmeventd/plugins/vdo/Makefile
0d8a0a
 daemons/dmfilemapd/Makefile
0d8a0a
 daemons/lvmdbusd/Makefile
0d8a0a
 daemons/lvmdbusd/lvmdbusd
0d8a0a
diff --git a/daemons/dmeventd/libdevmapper-event.c b/daemons/dmeventd/libdevmapper-event.c
0d8a0a
index 9aeb4c5..f9a8a2b 100644
0d8a0a
--- a/daemons/dmeventd/libdevmapper-event.c
0d8a0a
+++ b/daemons/dmeventd/libdevmapper-event.c
0d8a0a
@@ -645,6 +645,7 @@ int dm_event_register_handler(const struct dm_event_handler *dmevh)
0d8a0a
 	uuid = dm_task_get_uuid(dmt);
0d8a0a
 
0d8a0a
 	if (!strstr(dmevh->dso, "libdevmapper-event-lvm2thin.so") &&
0d8a0a
+	    !strstr(dmevh->dso, "libdevmapper-event-lvm2vdo.so") &&
0d8a0a
 	    !strstr(dmevh->dso, "libdevmapper-event-lvm2snapshot.so") &&
0d8a0a
 	    !strstr(dmevh->dso, "libdevmapper-event-lvm2mirror.so") &&
0d8a0a
 	    !strstr(dmevh->dso, "libdevmapper-event-lvm2raid.so"))
0d8a0a
diff --git a/daemons/dmeventd/plugins/Makefile.in b/daemons/dmeventd/plugins/Makefile.in
0d8a0a
index 27edf55..951dd2b 100644
0d8a0a
--- a/daemons/dmeventd/plugins/Makefile.in
0d8a0a
+++ b/daemons/dmeventd/plugins/Makefile.in
0d8a0a
@@ -1,6 +1,6 @@
0d8a0a
 #
0d8a0a
 # Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
0d8a0a
-# Copyright (C) 2004-2005, 2011 Red Hat, Inc. All rights reserved.
0d8a0a
+# Copyright (C) 2004-2018 Red Hat, Inc. All rights reserved.
0d8a0a
 #
0d8a0a
 # This file is part of LVM2.
0d8a0a
 #
0d8a0a
@@ -16,11 +16,7 @@ srcdir = @srcdir@
0d8a0a
 top_srcdir = @top_srcdir@
0d8a0a
 top_builddir = @top_builddir@
0d8a0a
 
0d8a0a
-SUBDIRS += lvm2 snapshot raid thin mirror
0d8a0a
-
0d8a0a
-ifeq ($(MAKECMDGOALS),distclean)
0d8a0a
-  SUBDIRS = lvm2 mirror snapshot raid thin
0d8a0a
-endif
0d8a0a
+SUBDIRS += lvm2 snapshot raid thin mirror vdo
0d8a0a
 
0d8a0a
 include $(top_builddir)/make.tmpl
0d8a0a
 
0d8a0a
@@ -28,3 +24,4 @@ snapshot: lvm2
0d8a0a
 mirror: lvm2
0d8a0a
 raid: lvm2
0d8a0a
 thin: lvm2
0d8a0a
+vdo: lvm2
0d8a0a
diff --git a/daemons/dmeventd/plugins/vdo/.exported_symbols b/daemons/dmeventd/plugins/vdo/.exported_symbols
0d8a0a
new file mode 100644
0d8a0a
index 0000000..b88c705
0d8a0a
--- /dev/null
0d8a0a
+++ b/daemons/dmeventd/plugins/vdo/.exported_symbols
0d8a0a
@@ -0,0 +1,3 @@
0d8a0a
+process_event
0d8a0a
+register_device
0d8a0a
+unregister_device
0d8a0a
diff --git a/daemons/dmeventd/plugins/vdo/Makefile.in b/daemons/dmeventd/plugins/vdo/Makefile.in
0d8a0a
new file mode 100644
0d8a0a
index 0000000..bda738a
0d8a0a
--- /dev/null
0d8a0a
+++ b/daemons/dmeventd/plugins/vdo/Makefile.in
0d8a0a
@@ -0,0 +1,36 @@
0d8a0a
+#
0d8a0a
+# Copyright (C) 2018 Red Hat, Inc. All rights reserved.
0d8a0a
+#
0d8a0a
+# This file is part of LVM2.
0d8a0a
+#
0d8a0a
+# This copyrighted material is made available to anyone wishing to use,
0d8a0a
+# modify, copy, or redistribute it subject to the terms and conditions
0d8a0a
+# of the GNU General Public License v.2.
0d8a0a
+#
0d8a0a
+# You should have received a copy of the GNU General Public License
0d8a0a
+# along with this program; if not, write to the Free Software Foundation,
0d8a0a
+# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
0d8a0a
+
0d8a0a
+srcdir = @srcdir@
0d8a0a
+top_srcdir = @top_srcdir@
0d8a0a
+top_builddir = @top_builddir@
0d8a0a
+
0d8a0a
+INCLUDES += -I$(top_srcdir)/daemons/dmeventd/plugins/lvm2
0d8a0a
+CLDFLAGS += -L$(top_builddir)/daemons/dmeventd/plugins/lvm2
0d8a0a
+
0d8a0a
+SOURCES = dmeventd_vdo.c
0d8a0a
+
0d8a0a
+LIB_NAME = libdevmapper-event-lvm2vdo
0d8a0a
+LIB_SHARED = $(LIB_NAME).$(LIB_SUFFIX)
0d8a0a
+LIB_VERSION = $(LIB_VERSION_LVM)
0d8a0a
+
0d8a0a
+CFLOW_LIST = $(SOURCES)
0d8a0a
+CFLOW_LIST_TARGET = $(LIB_NAME).cflow
0d8a0a
+
0d8a0a
+include $(top_builddir)/make.tmpl
0d8a0a
+
0d8a0a
+LIBS += -ldevmapper-event-lvm2 $(INTERNAL_LIBS)
0d8a0a
+
0d8a0a
+install_lvm2: install_dm_plugin
0d8a0a
+
0d8a0a
+install: install_lvm2
0d8a0a
diff --git a/daemons/dmeventd/plugins/vdo/dmeventd_vdo.c b/daemons/dmeventd/plugins/vdo/dmeventd_vdo.c
0d8a0a
new file mode 100644
0d8a0a
index 0000000..d77ca79
0d8a0a
--- /dev/null
0d8a0a
+++ b/daemons/dmeventd/plugins/vdo/dmeventd_vdo.c
0d8a0a
@@ -0,0 +1,406 @@
0d8a0a
+/*
0d8a0a
+ * Copyright (C) 2018 Red Hat, Inc. All rights reserved.
0d8a0a
+ *
0d8a0a
+ * This file is part of LVM2.
0d8a0a
+ *
0d8a0a
+ * This copyrighted material is made available to anyone wishing to use,
0d8a0a
+ * modify, copy, or redistribute it subject to the terms and conditions
0d8a0a
+ * of the GNU Lesser General Public License v.2.1.
0d8a0a
+ *
0d8a0a
+ * You should have received a copy of the GNU Lesser General Public License
0d8a0a
+ * along with this program; if not, write to the Free Software Foundation,
0d8a0a
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
0d8a0a
+ */
0d8a0a
+
0d8a0a
+#include "lib/misc/lib.h"
0d8a0a
+#include "dmeventd_lvm.h"
0d8a0a
+#include "daemons/dmeventd/libdevmapper-event.h"
0d8a0a
+#include "device_mapper/vdo/target.h"
0d8a0a
+
0d8a0a
+#include <sys/wait.h>
0d8a0a
+#include <stdarg.h>
0d8a0a
+
0d8a0a
+/* First warning when VDO pool is 80% full. */
0d8a0a
+#define WARNING_THRESH	(DM_PERCENT_1 * 80)
0d8a0a
+/* Run a check every 5%. */
0d8a0a
+#define CHECK_STEP	(DM_PERCENT_1 *  5)
0d8a0a
+/* Do not bother checking VDO pool is less than 50% full. */
0d8a0a
+#define CHECK_MINIMUM	(DM_PERCENT_1 * 50)
0d8a0a
+
0d8a0a
+#define MAX_FAILS	(256)  /* ~42 mins between cmd call retry with 10s delay */
0d8a0a
+
0d8a0a
+#define VDO_DEBUG 0
0d8a0a
+
0d8a0a
+struct dso_state {
0d8a0a
+	struct dm_pool *mem;
0d8a0a
+	int percent_check;
0d8a0a
+	int percent;
0d8a0a
+	uint64_t known_data_size;
0d8a0a
+	unsigned fails;
0d8a0a
+	unsigned max_fails;
0d8a0a
+	int restore_sigset;
0d8a0a
+	sigset_t old_sigset;
0d8a0a
+	pid_t pid;
0d8a0a
+	char *argv[3];
0d8a0a
+	const char *cmd_str;
0d8a0a
+	const char *name;
0d8a0a
+};
0d8a0a
+
0d8a0a
+DM_EVENT_LOG_FN("vdo")
0d8a0a
+
0d8a0a
+static int _run_command(struct dso_state *state)
0d8a0a
+{
0d8a0a
+	char val[16];
0d8a0a
+	int i;
0d8a0a
+
0d8a0a
+	/* Mark for possible lvm2 command we are running from dmeventd
0d8a0a
+	 * lvm2 will not try to talk back to dmeventd while processing it */
0d8a0a
+	(void) setenv("LVM_RUN_BY_DMEVENTD", "1", 1);
0d8a0a
+
0d8a0a
+	if (state->percent) {
0d8a0a
+		/* Prepare some known data to env vars for easy use */
0d8a0a
+		if (dm_snprintf(val, sizeof(val), "%d",
0d8a0a
+				state->percent / DM_PERCENT_1) != -1)
0d8a0a
+			(void) setenv("DMEVENTD_VDO_POOL", val, 1);
0d8a0a
+	} else {
0d8a0a
+		/* For an error event it's for a user to check status and decide */
0d8a0a
+		log_debug("Error event processing.");
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	log_verbose("Executing command: %s", state->cmd_str);
0d8a0a
+
0d8a0a
+	/* TODO:
0d8a0a
+	 *   Support parallel run of 'task' and it's waitpid maintainence
0d8a0a
+	 *   ATM we can't handle signaling of  SIGALRM
0d8a0a
+	 *   as signalling is not allowed while 'process_event()' is running
0d8a0a
+	 */
0d8a0a
+	if (!(state->pid = fork())) {
0d8a0a
+		/* child */
0d8a0a
+		(void) close(0);
0d8a0a
+		for (i = 3; i < 255; ++i) (void) close(i);
0d8a0a
+		execvp(state->argv[0], state->argv);
0d8a0a
+		_exit(errno);
0d8a0a
+	} else if (state->pid == -1) {
0d8a0a
+		log_error("Can't fork command %s.", state->cmd_str);
0d8a0a
+		state->fails = 1;
0d8a0a
+		return 0;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	return 1;
0d8a0a
+}
0d8a0a
+
0d8a0a
+static int _use_policy(struct dm_task *dmt, struct dso_state *state)
0d8a0a
+{
0d8a0a
+#if VDO_DEBUG
0d8a0a
+	log_debug("dmeventd executes: %s.", state->cmd_str);
0d8a0a
+#endif
0d8a0a
+	if (state->argv[0])
0d8a0a
+		return _run_command(state);
0d8a0a
+
0d8a0a
+	if (!dmeventd_lvm2_run_with_lock(state->cmd_str)) {
0d8a0a
+		log_error("Failed command for %s.", dm_task_get_name(dmt));
0d8a0a
+		state->fails = 1;
0d8a0a
+		return 0;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	state->fails = 0;
0d8a0a
+
0d8a0a
+	return 1;
0d8a0a
+}
0d8a0a
+
0d8a0a
+/* Check if executed command has finished
0d8a0a
+ * Only 1 command may run */
0d8a0a
+static int _wait_for_pid(struct dso_state *state)
0d8a0a
+{
0d8a0a
+	int status = 0;
0d8a0a
+
0d8a0a
+	if (state->pid == -1)
0d8a0a
+		return 1;
0d8a0a
+
0d8a0a
+	if (!waitpid(state->pid, &status, WNOHANG))
0d8a0a
+		return 0;
0d8a0a
+
0d8a0a
+	/* Wait for finish */
0d8a0a
+	if (WIFEXITED(status)) {
0d8a0a
+		log_verbose("Child %d exited with status %d.",
0d8a0a
+			    state->pid, WEXITSTATUS(status));
0d8a0a
+		state->fails = WEXITSTATUS(status) ? 1 : 0;
0d8a0a
+	} else {
0d8a0a
+		if (WIFSIGNALED(status))
0d8a0a
+			log_verbose("Child %d was terminated with status %d.",
0d8a0a
+				    state->pid, WTERMSIG(status));
0d8a0a
+		state->fails = 1;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	state->pid = -1;
0d8a0a
+
0d8a0a
+	return 1;
0d8a0a
+}
0d8a0a
+
0d8a0a
+void process_event(struct dm_task *dmt,
0d8a0a
+		   enum dm_event_mask event __attribute__((unused)),
0d8a0a
+		   void **user)
0d8a0a
+{
0d8a0a
+	const char *device = dm_task_get_name(dmt);
0d8a0a
+	struct dso_state *state = *user;
0d8a0a
+	void *next = NULL;
0d8a0a
+	uint64_t start, length;
0d8a0a
+	char *target_type = NULL;
0d8a0a
+	char *params;
0d8a0a
+	int needs_policy = 0;
0d8a0a
+	struct dm_task *new_dmt = NULL;
0d8a0a
+	struct dm_vdo_status_parse_result vdop = { .status = NULL };
0d8a0a
+
0d8a0a
+#if VDO_DEBUG
0d8a0a
+	log_debug("Watch for VDO %s:%.2f%%.", state->name,
0d8a0a
+		  dm_percent_to_round_float(state->percent_check, 2));
0d8a0a
+#endif
0d8a0a
+	if (!_wait_for_pid(state)) {
0d8a0a
+		log_warn("WARNING: Skipping event, child %d is still running (%s).",
0d8a0a
+			 state->pid, state->cmd_str);
0d8a0a
+		return;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	if (event & DM_EVENT_DEVICE_ERROR) {
0d8a0a
+#if VDO_DEBUG
0d8a0a
+		log_debug("VDO event error.");
0d8a0a
+#endif
0d8a0a
+		/* Error -> no need to check and do instant resize */
0d8a0a
+		state->percent = 0;
0d8a0a
+		if (_use_policy(dmt, state))
0d8a0a
+			goto out;
0d8a0a
+
0d8a0a
+		stack;
0d8a0a
+
0d8a0a
+		if (!(new_dmt = dm_task_create(DM_DEVICE_STATUS)))
0d8a0a
+			goto_out;
0d8a0a
+
0d8a0a
+		if (!dm_task_set_uuid(new_dmt, dm_task_get_uuid(dmt)))
0d8a0a
+			goto_out;
0d8a0a
+
0d8a0a
+		/* Non-blocking status read */
0d8a0a
+		if (!dm_task_no_flush(new_dmt))
0d8a0a
+			log_warn("WARNING: Can't set no_flush for dm status.");
0d8a0a
+
0d8a0a
+		if (!dm_task_run(new_dmt))
0d8a0a
+			goto_out;
0d8a0a
+
0d8a0a
+		dmt = new_dmt;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	dm_get_next_target(dmt, next, &start, &length, &target_type, &params);
0d8a0a
+
0d8a0a
+	if (!target_type || (strcmp(target_type, "vdo") != 0)) {
0d8a0a
+		log_error("Invalid target type.");
0d8a0a
+		goto out;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	if (!dm_vdo_status_parse(state->mem, params, &vdop)) {
0d8a0a
+		log_error("Failed to parse status.");
0d8a0a
+		goto out;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	state->percent = dm_make_percent(vdop.status->used_blocks,
0d8a0a
+					 vdop.status->total_blocks);
0d8a0a
+
0d8a0a
+#if VDO_DEBUG
0d8a0a
+	log_debug("VDO %s status  %.2f%% " FMTu64 "/" FMTu64 ".",
0d8a0a
+		  state->name, dm_percent_to_round_float(state->percent, 2),
0d8a0a
+		  vdop.status->used_blocks, vdop.status->total_blocks);
0d8a0a
+#endif
0d8a0a
+
0d8a0a
+	/* VDO pool size had changed. Clear the threshold. */
0d8a0a
+	if (state->known_data_size != vdop.status->total_blocks) {
0d8a0a
+		state->percent_check = CHECK_MINIMUM;
0d8a0a
+		state->known_data_size = vdop.status->total_blocks;
0d8a0a
+		state->fails = 0;
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	/*
0d8a0a
+	 * Trigger action when threshold boundary is exceeded.
0d8a0a
+	 * Report 80% threshold warning when it's used above 80%.
0d8a0a
+	 * Only 100% is exception as it cannot be surpased so policy
0d8a0a
+	 * action is called for:  >50%, >55% ... >95%, 100%
0d8a0a
+	 */
0d8a0a
+	if ((state->percent > WARNING_THRESH) &&
0d8a0a
+	    (state->percent > state->percent_check))
0d8a0a
+		log_warn("WARNING: VDO %s %s is now %.2f%% full.",
0d8a0a
+			 state->name, device,
0d8a0a
+			 dm_percent_to_round_float(state->percent, 2));
0d8a0a
+	if (state->percent > CHECK_MINIMUM) {
0d8a0a
+		/* Run action when usage raised more than CHECK_STEP since the last time */
0d8a0a
+		if (state->percent > state->percent_check)
0d8a0a
+			needs_policy = 1;
0d8a0a
+		state->percent_check = (state->percent / CHECK_STEP + 1) * CHECK_STEP;
0d8a0a
+		if (state->percent_check == DM_PERCENT_100)
0d8a0a
+			state->percent_check--; /* Can't get bigger then 100% */
0d8a0a
+	} else
0d8a0a
+		state->percent_check = CHECK_MINIMUM;
0d8a0a
+
0d8a0a
+	/* Reduce number of _use_policy() calls by power-of-2 factor till frequency of MAX_FAILS is reached.
0d8a0a
+	 * Avoids too high number of error retries, yet shows some status messages in log regularly.
0d8a0a
+	 * i.e. PV could have been pvmoved and VG/LV was locked for a while...
0d8a0a
+	 */
0d8a0a
+	if (state->fails) {
0d8a0a
+		if (state->fails++ <= state->max_fails) {
0d8a0a
+			log_debug("Postponing frequently failing policy (%u <= %u).",
0d8a0a
+				  state->fails - 1, state->max_fails);
0d8a0a
+			return;
0d8a0a
+		}
0d8a0a
+		if (state->max_fails < MAX_FAILS)
0d8a0a
+			state->max_fails <<= 1;
0d8a0a
+		state->fails = needs_policy = 1; /* Retry failing command */
0d8a0a
+	} else
0d8a0a
+		state->max_fails = 1; /* Reset on success */
0d8a0a
+
0d8a0a
+	/* FIXME: ATM nothing can be done, drop 0, once it becomes useful */
0d8a0a
+	if (0 && needs_policy)
0d8a0a
+		_use_policy(dmt, state);
0d8a0a
+out:
0d8a0a
+	if (vdop.status)
0d8a0a
+		dm_pool_free(state->mem, vdop.status);
0d8a0a
+
0d8a0a
+	if (new_dmt)
0d8a0a
+		dm_task_destroy(new_dmt);
0d8a0a
+}
0d8a0a
+
0d8a0a
+/* Handle SIGCHLD for a thread */
0d8a0a
+static void _sig_child(int signum __attribute__((unused)))
0d8a0a
+{
0d8a0a
+	/* empty SIG_IGN */;
0d8a0a
+}
0d8a0a
+
0d8a0a
+/* Setup handler for SIGCHLD when executing external command
0d8a0a
+ * to get quick 'waitpid()' reaction
0d8a0a
+ * It will interrupt syscall just like SIGALRM and
0d8a0a
+ * invoke process_event().
0d8a0a
+ */
0d8a0a
+static void _init_thread_signals(struct dso_state *state)
0d8a0a
+{
0d8a0a
+	struct sigaction act = { .sa_handler = _sig_child };
0d8a0a
+	sigset_t my_sigset;
0d8a0a
+
0d8a0a
+	sigemptyset(&my_sigset);
0d8a0a
+
0d8a0a
+	if (sigaction(SIGCHLD, &act, NULL))
0d8a0a
+		log_warn("WARNING: Failed to set SIGCHLD action.");
0d8a0a
+	else if (sigaddset(&my_sigset, SIGCHLD))
0d8a0a
+		log_warn("WARNING: Failed to add SIGCHLD to set.");
0d8a0a
+	else if (pthread_sigmask(SIG_UNBLOCK, &my_sigset, &state->old_sigset))
0d8a0a
+		log_warn("WARNING: Failed to unblock SIGCHLD.");
0d8a0a
+	else
0d8a0a
+		state->restore_sigset = 1;
0d8a0a
+}
0d8a0a
+
0d8a0a
+static void _restore_thread_signals(struct dso_state *state)
0d8a0a
+{
0d8a0a
+	if (state->restore_sigset &&
0d8a0a
+	    pthread_sigmask(SIG_SETMASK, &state->old_sigset, NULL))
0d8a0a
+		log_warn("WARNING: Failed to block SIGCHLD.");
0d8a0a
+}
0d8a0a
+
0d8a0a
+int register_device(const char *device,
0d8a0a
+		    const char *uuid,
0d8a0a
+		    int major __attribute__((unused)),
0d8a0a
+		    int minor __attribute__((unused)),
0d8a0a
+		    void **user)
0d8a0a
+{
0d8a0a
+	struct dso_state *state;
0d8a0a
+	const char *cmd;
0d8a0a
+	char *str;
0d8a0a
+	char cmd_str[PATH_MAX + 128 + 2]; /* cmd ' ' vg/lv \0 */
0d8a0a
+        const char *name = "pool";
0d8a0a
+
0d8a0a
+	if (!dmeventd_lvm2_init_with_pool("vdo_pool_state", state))
0d8a0a
+		goto_bad;
0d8a0a
+
0d8a0a
+	state->cmd_str = "";
0d8a0a
+
0d8a0a
+	/* Search for command for LVM- prefixed devices only */
0d8a0a
+	cmd = (strncmp(uuid, "LVM-", 4) == 0) ? "_dmeventd_vdo_command" : "";
0d8a0a
+
0d8a0a
+	if (!dmeventd_lvm2_command(state->mem, cmd_str, sizeof(cmd_str), cmd, device))
0d8a0a
+		goto_bad;
0d8a0a
+
0d8a0a
+	if (strncmp(cmd_str, "lvm ", 4) == 0) {
0d8a0a
+		if (!(state->cmd_str = dm_pool_strdup(state->mem, cmd_str + 4))) {
0d8a0a
+			log_error("Failed to copy lvm VDO command.");
0d8a0a
+				goto bad;
0d8a0a
+		}
0d8a0a
+	} else if (cmd_str[0] == '/') {
0d8a0a
+		if (!(state->cmd_str = dm_pool_strdup(state->mem, cmd_str))) {
0d8a0a
+			log_error("Failed to copy VDO command.");
0d8a0a
+			goto bad;
0d8a0a
+		}
0d8a0a
+
0d8a0a
+		/* Find last space before 'vg/lv' */
0d8a0a
+		if (!(str = strrchr(state->cmd_str, ' ')))
0d8a0a
+			goto inval;
0d8a0a
+
0d8a0a
+		if (!(state->argv[0] = dm_pool_strndup(state->mem, state->cmd_str,
0d8a0a
+						       str - state->cmd_str))) {
0d8a0a
+			log_error("Failed to copy command.");
0d8a0a
+			goto bad;
0d8a0a
+		}
0d8a0a
+
0d8a0a
+		state->argv[1] = str + 1;  /* 1 argument - vg/lv */
0d8a0a
+		_init_thread_signals(state);
0d8a0a
+	} else if (cmd[0] == 0) {
0d8a0a
+		state->name = "volume"; /* What to use with 'others?' */
0d8a0a
+	} else/* Unuspported command format */
0d8a0a
+		goto inval;
0d8a0a
+
0d8a0a
+	state->pid = -1;
0d8a0a
+	state->name = name;
0d8a0a
+	*user = state;
0d8a0a
+
0d8a0a
+	log_info("Monitoring VDO %s %s.", name, device);
0d8a0a
+
0d8a0a
+	return 1;
0d8a0a
+inval:
0d8a0a
+	log_error("Invalid command for monitoring: %s.", cmd_str);
0d8a0a
+bad:
0d8a0a
+	log_error("Failed to monitor VDO %s %s.", name, device);
0d8a0a
+
0d8a0a
+	if (state)
0d8a0a
+		dmeventd_lvm2_exit_with_pool(state);
0d8a0a
+
0d8a0a
+	return 0;
0d8a0a
+}
0d8a0a
+
0d8a0a
+int unregister_device(const char *device,
0d8a0a
+		      const char *uuid __attribute__((unused)),
0d8a0a
+		      int major __attribute__((unused)),
0d8a0a
+		      int minor __attribute__((unused)),
0d8a0a
+		      void **user)
0d8a0a
+{
0d8a0a
+	struct dso_state *state = *user;
0d8a0a
+	const char *name = state->name;
0d8a0a
+	int i;
0d8a0a
+
0d8a0a
+	for (i = 0; !_wait_for_pid(state) && (i < 6); ++i) {
0d8a0a
+		if (i == 0)
0d8a0a
+			/* Give it 2 seconds, then try to terminate & kill it */
0d8a0a
+			log_verbose("Child %d still not finished (%s) waiting.",
0d8a0a
+				    state->pid, state->cmd_str);
0d8a0a
+		else if (i == 3) {
0d8a0a
+			log_warn("WARNING: Terminating child %d.", state->pid);
0d8a0a
+			kill(state->pid, SIGINT);
0d8a0a
+			kill(state->pid, SIGTERM);
0d8a0a
+		} else if (i == 5) {
0d8a0a
+			log_warn("WARNING: Killing child %d.", state->pid);
0d8a0a
+			kill(state->pid, SIGKILL);
0d8a0a
+		}
0d8a0a
+		sleep(1);
0d8a0a
+	}
0d8a0a
+
0d8a0a
+	if (state->pid != -1)
0d8a0a
+		log_warn("WARNING: Cannot kill child %d!", state->pid);
0d8a0a
+
0d8a0a
+	_restore_thread_signals(state);
0d8a0a
+
0d8a0a
+	dmeventd_lvm2_exit_with_pool(state);
0d8a0a
+	log_info("No longer monitoring VDO %s %s.", name, device);
0d8a0a
+
0d8a0a
+	return 1;
0d8a0a
+}