Re: [PATCH v5] Add sun4v_wdt watchdog driver

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

 



On 01/27/2016 05:54 PM, Wim Coekaerts wrote:
This driver adds sparc hypervisor watchdog support. The default
timeout is 60 seconds and the range is between 1 and
31536000 seconds. Both watchdog-resolution and
watchdog-max-timeout MD properties settings are supported.

Signed-off-by: Wim Coekaerts <wim.coekaerts@xxxxxxxxxx>
---
Change in v5:
   - After proper review of the hv watchdog api, the hv takes care of
	rounding to multiple of resolution internally so just check
	if it's <= 1 second.
   - Clean up _ping() and simplify _probe() value checking

Change in v4:
   - timeout in seconds
   - call the hv with support for microsecond resolution

Changes in v3:
   - Modify sun4v_mach_set_watchdog to allow for NULL and remove
	time_remaining
   - Cleanup includes
   - Consolidate _start and _ping into _ping since they were the same
   - Fix checkpatch warnings
   - Remove pr_info()s
   - Clean up return codes to be standard kernel return values
   - Consolidate _init and _probe into _init
	Don't use platform_device anymore because this is really
	just a driver that depends on a hv call. This now looks more
	like softdog.
   - Cleanly check for sun4v architecture and remove extra tests
   - Convert to ms timer support and honor resolution
	since most drivers use seconds I added _ms to make it clear
   - Clean up validation against the properties
   - Use sane values for MIN/MAX
---
  Documentation/watchdog/watchdog-parameters.txt |    4 +
  arch/sparc/kernel/hvcalls.S                    |    3 +-
  arch/sparc/kernel/sparc_ksyms_64.c             |    1 +
  drivers/watchdog/Kconfig                       |   11 ++
  drivers/watchdog/Makefile                      |    1 +
  drivers/watchdog/sun4v_wdt.c                   |  211 ++++++++++++++++++++++++
  6 files changed, 230 insertions(+), 1 deletions(-)
  create mode 100644 drivers/watchdog/sun4v_wdt.c

diff --git a/Documentation/watchdog/watchdog-parameters.txt b/Documentation/watchdog/watchdog-parameters.txt
index 9f9ec9f..4e4b6f1 100644
--- a/Documentation/watchdog/watchdog-parameters.txt
+++ b/Documentation/watchdog/watchdog-parameters.txt
@@ -400,3 +400,7 @@ wm8350_wdt:
  nowayout: Watchdog cannot be stopped once started
  	(default=kernel config parameter)
  -------------------------------------------------
+sun4v_wdt:
+timeout_ms: Watchdog timeout in milliseconds 1..180000, default=60000)
+nowayout: Watchdog cannot be stopped once started
+-------------------------------------------------
diff --git a/arch/sparc/kernel/hvcalls.S b/arch/sparc/kernel/hvcalls.S
index afbaba5..d127130 100644
--- a/arch/sparc/kernel/hvcalls.S
+++ b/arch/sparc/kernel/hvcalls.S
@@ -338,8 +338,9 @@ ENTRY(sun4v_mach_set_watchdog)
  	mov	%o1, %o4
  	mov	HV_FAST_MACH_SET_WATCHDOG, %o5
  	ta	HV_FAST_TRAP
+	brnz,a,pn %o4, 0f
  	stx	%o1, [%o4]
-	retl
+0:	retl
  	 nop
  ENDPROC(sun4v_mach_set_watchdog)

diff --git a/arch/sparc/kernel/sparc_ksyms_64.c b/arch/sparc/kernel/sparc_ksyms_64.c
index a92d5d2..9e034f2 100644
--- a/arch/sparc/kernel/sparc_ksyms_64.c
+++ b/arch/sparc/kernel/sparc_ksyms_64.c
@@ -37,6 +37,7 @@ EXPORT_SYMBOL(sun4v_niagara_getperf);
  EXPORT_SYMBOL(sun4v_niagara_setperf);
  EXPORT_SYMBOL(sun4v_niagara2_getperf);
  EXPORT_SYMBOL(sun4v_niagara2_setperf);
+EXPORT_SYMBOL(sun4v_mach_set_watchdog);

  /* from hweight.S */
  EXPORT_SYMBOL(__arch_hweight8);
diff --git a/drivers/watchdog/Kconfig b/drivers/watchdog/Kconfig
index 4f0e7be..30d38ae 100644
--- a/drivers/watchdog/Kconfig
+++ b/drivers/watchdog/Kconfig
@@ -1565,6 +1565,17 @@ config WATCHDOG_RIO
  	  machines.  The watchdog timeout period is normally one minute but
  	  can be changed with a boot-time parameter.

+config WATCHDOG_SUN4V
+	tristate "Sun4v Watchdog support"
+	select WATCHDOG_CORE
+	depends on SPARC64
+	help
+	  Say Y here to support the hypervisor watchdog capability embedded
+	  in the SPARC sun4v architecture.
+
+	  To compile this driver as a module, choose M here. The module will
+	  be called sun4v_wdt.
+
  # XTENSA Architecture

  # Xen Architecture
diff --git a/drivers/watchdog/Makefile b/drivers/watchdog/Makefile
index f566753..f6a6a38 100644
--- a/drivers/watchdog/Makefile
+++ b/drivers/watchdog/Makefile
@@ -179,6 +179,7 @@ obj-$(CONFIG_SH_WDT) += shwdt.o

  obj-$(CONFIG_WATCHDOG_RIO)		+= riowd.o
  obj-$(CONFIG_WATCHDOG_CP1XXX)		+= cpwd.o
+obj-$(CONFIG_WATCHDOG_SUN4V)		+= sun4v_wdt.o

  # XTENSA Architecture

diff --git a/drivers/watchdog/sun4v_wdt.c b/drivers/watchdog/sun4v_wdt.c
new file mode 100644
index 0000000..42da024
--- /dev/null
+++ b/drivers/watchdog/sun4v_wdt.c
@@ -0,0 +1,211 @@
+/*
+ *	sun4v watchdog timer
+ *	(c) Copyright 2016 Oracle Corporation
+ *
+ *	Implement a simple watchdog driver using the built-in sun4v hypervisor
+ *	watchdog support. If time expires, the hypervisor stops or bounces
+ *	the guest domain.
+ *
+ *	This program is free software; you can redistribute it and/or
+ *	modify it under the terms of the GNU General Public License
+ *	as published by the Free Software Foundation; either version
+ *	2 of the License, or (at your option) any later version.
+ */
+
+#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
+
+#include <linux/errno.h>
+#include <linux/init.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/moduleparam.h>
+#include <linux/watchdog.h>
+#include <asm/hypervisor.h>
+#include <asm/mdesc.h>
+
+#define WDT_TIMEOUT			60
+#define WDT_MAX_TIMEOUT			31536000
+#define WDT_MIN_TIMEOUT			1
+#define WDT_DEFAULT_RESOLUTION_MS	1000	/* 1 second */
+
+static unsigned int wdt_max_timeout = WDT_MAX_TIMEOUT;
+
+static unsigned int timeout = WDT_TIMEOUT;
+module_param(timeout, uint, S_IRUGO);
+MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds (default="
+	__MODULE_STRING(WDT_TIMEOUT) ")");
+
+static bool nowayout = WATCHDOG_NOWAYOUT;
+module_param(nowayout, bool, S_IRUGO);
+MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default="
+	__MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
+
+static int sun4v_wdt_stop(struct watchdog_device *wdd)
+{
+	sun4v_mach_set_watchdog(0, NULL);
+
+	return 0;
+}
+
+static int sun4v_wdt_ping(struct watchdog_device *wdd)
+{
+	int hverr;
+
+	/*
+	 * HV watchdog timer will round up the timeout
+	 * passed in to the nearest multiple of the
+	 * watchdog resolution in milliseconds.
+	 */
+	hverr = sun4v_mach_set_watchdog(wdd->timeout * 1000, NULL);
+	if (hverr == HV_EINVAL)
+		return -EINVAL;
+
+	return 0;
+}
+
+static int sun4v_wdt_set_timeout(struct watchdog_device *wdd,
+				 unsigned int timeout)
+{
+	wdd->timeout = timeout;
+
+	return 0;
+}
+
+static const struct watchdog_info sun4v_wdt_ident = {
+	.options =	WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE,
+	.identity =	"sun4v hypervisor watchdog",
+	.firmware_version = 0,
+};
+
+static struct watchdog_ops sun4v_wdt_ops = {
+	.owner =	THIS_MODULE,
+	.start =	sun4v_wdt_ping,
+	.stop =		sun4v_wdt_stop,
+	.ping =		sun4v_wdt_ping,
+	.set_timeout =	sun4v_wdt_set_timeout,
+};
+
+static struct watchdog_device wdd = {
+	.info = &sun4v_wdt_ident,
+	.ops = &sun4v_wdt_ops,
+	.min_timeout = WDT_MIN_TIMEOUT,
+};
+
+static int hvapi_registered;
+
+static int __init sun4v_wdt_init(void)
+{
+	struct mdesc_handle *handle;
+	u64 node;
+	const u64 *value;
+	u64 wdt_max_property;
+	u64 wdt_resolution_property;
+	int ret = 0;
+	unsigned long major = 1, minor = 1;
+
+	/*
+	 * This is a safe way to validate if we are on the right
+	 * platform.
+	 */
+	if (sun4v_hvapi_register(HV_GRP_CORE, major, &minor) != 0)
+		return -ENODEV;
+	if (minor < 1) {
+		sun4v_hvapi_unregister(HV_GRP_CORE);
+		return -ENODEV;
+	}
+	hvapi_registered = 1;
+

I don't think you need this variable. See below.

+	/*
+	 * There are 2 properties that can be set from the control
+	 * domain for the watchdog.
+	 * watchdog-resolution
+	 * watchdog-max-timeout
+	 *
+	 * We can expect a handle to be returned otherwise something
+	 * serious is wrong. Correct to return -ENODEV here.
+	 */
+
+	handle = mdesc_grab();
+	if (!handle)

Should probably call sun4v_hvapi_unregister(HV_GRP_CORE) here.

+		return -ENODEV;
+
+	node = mdesc_node_by_name(handle, MDESC_NODE_NULL, "platform");
+	if (node == MDESC_NODE_NULL) {
+		mdesc_release(handle);
+		sun4v_hvapi_unregister(HV_GRP_CORE);
+		return -ENODEV;
+	}
+
+	/* Allow for resolution up to 1s (default) */
+	value = mdesc_get_property(handle, node, "watchdog-resolution", NULL);
+	if (value) {
+		wdt_resolution_property = *value;
+		if (wdt_resolution_property == 0 ||
+		    wdt_resolution_property > WDT_DEFAULT_RESOLUTION_MS) {
+			mdesc_release(handle);
+			sun4v_hvapi_unregister(HV_GRP_CORE);

Please implement proper error handling using goto, per CodingStyle.

+			return -EINVAL;
+		}
+	}
+
+	value = mdesc_get_property(handle, node, "watchdog-max-timeout", NULL);
+	if (value) {
+		wdt_max_property = *value;
+		/*
+		 * If the property (in ms) is set to a value smaller than
+		 * WDT_MIN_TIMEOUT, return -EINVAL.
+		 */
+		if (wdt_max_property < (WDT_MIN_TIMEOUT * 1000)) {

Unecessary ( ) in expression.

+			mdesc_release(handle);
+			sun4v_hvapi_unregister(HV_GRP_CORE);
+			return -EINVAL;
+		}
+
+		/*
+		 * If the property is set to a value smaller than
+		 * WDT_MAX_TIMEOUT then set wdt_max_timeout to
+		 * the value of the property in seconds.
+		 */
+		if (wdt_max_property < (wdt_max_timeout * 1000))

Unnecessary ( ) in expression.

+			wdt_max_timeout = wdt_max_property  / 1000;
+	}
+
+	mdesc_release(handle);
+
+	if (timeout < WDT_MIN_TIMEOUT)
+		timeout = WDT_MIN_TIMEOUT;
+	if (timeout > wdt_max_timeout)
+		timeout = wdt_max_timeout;

watchdog_init_timeout() would normally take care of this, if you set the
module parameter to 0 and initialize wdt.timeout with the default timeout.

If you don't want to use watchdog_init_timeout() to get a range instead
of the default if the provided timeout is invalid, please use clamp_val().

+
+	wdd.max_timeout = wdt_max_timeout;
+	wdd.timeout = timeout;
+
+	watchdog_set_nowayout(&wdd, nowayout);
+
+	ret = watchdog_register_device(&wdd);
+	if (ret) {
+		sun4v_hvapi_unregister(HV_GRP_CORE);
+		return ret;
+	}
+
+	pr_info("initialized (timeout=%dms, nowayout=%d)\n",
+		 wdd.timeout, nowayout);
+
+	return 0;
+}
+
+
+static void __exit sun4v_wdt_exit(void)
+{
+	if (hvapi_registered)
+		sun4v_hvapi_unregister(HV_GRP_CORE);

hvapi_registered is not needed. This function is only called
if the init function was successful.

+	sun4v_wdt_stop(&wdd);

This should not be needed either. The module won't be released if nowayout is set
if the driver is open. If nowayout is not set, it will only be released after the
watchdog device has been closed. If it was not closed properly, the watchdog may
still be running, but I don't think you want to stop the watchdog in that case
when unloading the module, as it would defeat magicclose.

+	watchdog_unregister_device(&wdd);
+}
+
+module_init(sun4v_wdt_init);
+module_exit(sun4v_wdt_exit);
+
+MODULE_AUTHOR("Wim Coekaerts <wim.coekaerts@xxxxxxxxxx>");
+MODULE_DESCRIPTION("sun4v watchdog driver");
+MODULE_LICENSE("GPL");


--
To unsubscribe from this list: send the line "unsubscribe sparclinux" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Kernel Development]     [DCCP]     [Linux ARM Development]     [Linux]     [Photo]     [Yosemite Help]     [Linux ARM Kernel]     [Linux SCSI]     [Linux x86_64]     [Linux Hams]

  Powered by Linux