[PATCH] watchdog: Add software pretimeout support

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



This adds the option to use a software timer to generate a watchdog
pretimeout event for hardware watchdogs that do not support the
pretimeout feature.

With this enabled, all watchdogs will appear to have pretimeout support
in userspace. If no pretimeout value is set, there will be no change in
the watchdog's behavior. If a pretimeout value is set for a specific
watchdog that does not have built-in pretimeout support, a timer will be
started that should fire at the specified time before the watchdog
timeout would occur. When the watchdog is successfully pinged, the timer
will be restarted. If the timer is allowed to fire it will generate a
pretimeout event.

If the watchdog does support a pretimeout natively, that functionality
will be used instead of the software timer.

Signed-off-by: Curtis Klein <curtis.klein@xxxxxxx>
---
 drivers/watchdog/Kconfig               |  8 +++++
 drivers/watchdog/watchdog_dev.c        | 62 +++++++++++++++++++++++++++++++---
 drivers/watchdog/watchdog_pretimeout.c |  7 ++--
 3 files changed, 71 insertions(+), 6 deletions(-)

diff --git a/drivers/watchdog/Kconfig b/drivers/watchdog/Kconfig
index f22e373..6a4554b 100644
--- a/drivers/watchdog/Kconfig
+++ b/drivers/watchdog/Kconfig
@@ -73,6 +73,14 @@ config WATCHDOG_SYSFS
 	  Say Y here if you want to enable watchdog device status read through
 	  sysfs attributes.
 
+config WATCHDOG_SOFTWARE_PRETIMEOUT
+	bool "Enable software timer based pretimeout support"
+	help
+	  Enable this if you want to use a software timer based pretimeout for
+	  watchdogs that do not have pretimeout support. Be aware that because
+	  this pretimeout is purely software based, it may not be able to fire
+	  before the actual watchdog fires in some situations.
+
 comment "Watchdog Pretimeout Governors"
 
 config WATCHDOG_PRETIMEOUT_GOV
diff --git a/drivers/watchdog/watchdog_dev.c b/drivers/watchdog/watchdog_dev.c
index 2946f3a..ae4b5b4 100644
--- a/drivers/watchdog/watchdog_dev.c
+++ b/drivers/watchdog/watchdog_dev.c
@@ -7,6 +7,8 @@
  *
  *	(c) Copyright 2008-2011 Wim Van Sebroeck <wim@xxxxxxxxx>.
  *
+ *	(c) Copyright 2020 Hewlett Packard Enterprise Development LP.
+ *
  *
  *	This source code is part of the generic code that can be used
  *	by all the watchdog timer drivers.
@@ -64,6 +66,9 @@ struct watchdog_core_data {
 	ktime_t open_deadline;
 	struct hrtimer timer;
 	struct kthread_work work;
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+	struct hrtimer pretimeout_timer;
+#endif
 	unsigned long status;		/* Internal status bits */
 #define _WDOG_DEV_OPEN		0	/* Opened ? */
 #define _WDOG_ALLOW_RELEASE	1	/* Did we receive the magic char ? */
@@ -185,6 +190,17 @@ static int __watchdog_ping(struct watchdog_device *wdd)
 	else
 		err = wdd->ops->start(wdd); /* restart watchdog */
 
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+	if (!(wdd->info->options & WDIOF_PRETIMEOUT) && wdd->pretimeout) {
+		if (err == 0)
+			hrtimer_start(&wd_data->pretimeout_timer,
+				      ktime_set(wdd->timeout - wdd->pretimeout, 0),
+				      HRTIMER_MODE_REL);
+	} else {
+		hrtimer_cancel(&wd_data->pretimeout_timer);
+	}
+#endif
+
 	watchdog_update_worker(wdd);
 
 	return err;
@@ -250,6 +266,19 @@ static enum hrtimer_restart watchdog_timer_expired(struct hrtimer *timer)
 	return HRTIMER_NORESTART;
 }
 
+
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+static enum hrtimer_restart watchdog_software_pretimeout(struct hrtimer *timer)
+{
+	struct watchdog_core_data *wd_data;
+
+	wd_data = container_of(timer, struct watchdog_core_data, pretimeout_timer);
+
+	watchdog_notify_pretimeout(wd_data->wdd);
+	return HRTIMER_NORESTART;
+}
+#endif
+
 /*
  *	watchdog_start: wrapper to start the watchdog.
  *	@wdd: the watchdog device to start
@@ -287,6 +316,12 @@ static int watchdog_start(struct watchdog_device *wdd)
 		}
 	}
 
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+		if (!(wdd->info->options & WDIOF_PRETIMEOUT) && wdd->pretimeout)
+			hrtimer_start(&wd_data->pretimeout_timer,
+				      ktime_set(wdd->timeout - wdd->pretimeout, 0),
+				      HRTIMER_MODE_REL);
+#endif
 	return err;
 }
 
@@ -325,6 +360,10 @@ static int watchdog_stop(struct watchdog_device *wdd)
 	if (err == 0) {
 		clear_bit(WDOG_ACTIVE, &wdd->status);
 		watchdog_update_worker(wdd);
+
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+		hrtimer_cancel(&wdd->wd_data->pretimeout_timer);
+#endif
 	}
 
 	return err;
@@ -361,6 +400,9 @@ static unsigned int watchdog_get_status(struct watchdog_device *wdd)
 	if (test_and_clear_bit(_WDOG_KEEPALIVE, &wd_data->status))
 		status |= WDIOF_KEEPALIVEPING;
 
+	if (IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT))
+		status |= WDIOF_PRETIMEOUT;
+
 	return status;
 }
 
@@ -408,7 +450,8 @@ static int watchdog_set_pretimeout(struct watchdog_device *wdd,
 {
 	int err = 0;
 
-	if (!(wdd->info->options & WDIOF_PRETIMEOUT))
+	if (!(wdd->info->options & WDIOF_PRETIMEOUT) &&
+	    !IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT))
 		return -EOPNOTSUPP;
 
 	if (watchdog_pretimeout_invalid(wdd, timeout))
@@ -595,12 +638,14 @@ static umode_t wdt_is_visible(struct kobject *kobj, struct attribute *attr,
 	if (attr == &dev_attr_timeleft.attr && !wdd->ops->get_timeleft)
 		mode = 0;
 	else if (attr == &dev_attr_pretimeout.attr &&
-		 !(wdd->info->options & WDIOF_PRETIMEOUT))
+		 !(wdd->info->options & WDIOF_PRETIMEOUT) &&
+		 !IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT))
 		mode = 0;
 	else if ((attr == &dev_attr_pretimeout_governor.attr ||
 		  attr == &dev_attr_pretimeout_available_governors.attr) &&
-		 (!(wdd->info->options & WDIOF_PRETIMEOUT) ||
-		  !IS_ENABLED(CONFIG_WATCHDOG_PRETIMEOUT_GOV)))
+		 ((!(wdd->info->options & WDIOF_PRETIMEOUT) &&
+		   !IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT)) ||
+		   !IS_ENABLED(CONFIG_WATCHDOG_PRETIMEOUT_GOV)))
 		mode = 0;
 
 	return mode;
@@ -1010,6 +1055,11 @@ static int watchdog_cdev_register(struct watchdog_device *wdd)
 	hrtimer_init(&wd_data->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL_HARD);
 	wd_data->timer.function = watchdog_timer_expired;
 
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+	hrtimer_init(&wd_data->pretimeout_timer,  CLOCK_MONOTONIC, HRTIMER_MODE_REL);
+	wd_data->pretimeout_timer.function = watchdog_software_pretimeout;
+#endif
+
 	if (wdd->id == 0) {
 		old_wd_data = wd_data;
 		watchdog_miscdev.parent = wdd->parent;
@@ -1097,6 +1147,10 @@ static void watchdog_cdev_unregister(struct watchdog_device *wdd)
 	hrtimer_cancel(&wd_data->timer);
 	kthread_cancel_work_sync(&wd_data->work);
 
+#ifdef CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT
+	hrtimer_cancel(&wd_data->pretimeout_timer);
+#endif
+
 	put_device(&wd_data->dev);
 }
 
diff --git a/drivers/watchdog/watchdog_pretimeout.c b/drivers/watchdog/watchdog_pretimeout.c
index 01ca84b..cb3f8f4 100644
--- a/drivers/watchdog/watchdog_pretimeout.c
+++ b/drivers/watchdog/watchdog_pretimeout.c
@@ -1,6 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-or-later
 /*
  * Copyright (C) 2015-2016 Mentor Graphics
+ * Copyright (C) 2020 Hewlett Packard Enterprise Development LP.
  */
 
 #include <linux/list.h>
@@ -177,7 +178,8 @@ int watchdog_register_pretimeout(struct watchdog_device *wdd)
 {
 	struct watchdog_pretimeout *p;
 
-	if (!(wdd->info->options & WDIOF_PRETIMEOUT))
+	if (!(wdd->info->options & WDIOF_PRETIMEOUT) &&
+	    !IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT))
 		return 0;
 
 	p = kzalloc(sizeof(*p), GFP_KERNEL);
@@ -197,7 +199,8 @@ void watchdog_unregister_pretimeout(struct watchdog_device *wdd)
 {
 	struct watchdog_pretimeout *p, *t;
 
-	if (!(wdd->info->options & WDIOF_PRETIMEOUT))
+	if (!(wdd->info->options & WDIOF_PRETIMEOUT) &&
+	    !IS_ENABLED(CONFIG_WATCHDOG_SOFTWARE_PRETIMEOUT))
 		return;
 
 	spin_lock_irq(&pretimeout_lock);
-- 
2.7.4




[Index of Archives]     [Linux ARM Kernel]     [Linux ARM]     [Linux Omap]     [Fedora ARM]     [IETF Annouce]     [Security]     [Bugtraq]     [Linux]     [Linux OMAP]     [Linux MIPS]     [eCos]     [Asterisk Internet PBX]     [Linux API]

  Powered by Linux