[PATCH] Added 'advantech_ec_wdt' watchdog driver

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

 



This patch adds the 'advantech_ec_wdt' kernel module which provides
WDT support for Advantech platforms with ITE based Embedded Controller.

Signed-off-by: Thomas Kastner <thomas.kastner@xxxxxxxxxxxxx>
---
 drivers/watchdog/Kconfig            |   7 +
 drivers/watchdog/Makefile           |   1 +
 drivers/watchdog/advantech_ec_wdt.c | 247 ++++++++++++++++++++++++++++
 3 files changed, 255 insertions(+)
 create mode 100644 drivers/watchdog/advantech_ec_wdt.c

diff --git a/drivers/watchdog/Kconfig b/drivers/watchdog/Kconfig
index 688922fc4edb..afe14f530291 100644
--- a/drivers/watchdog/Kconfig
+++ b/drivers/watchdog/Kconfig
@@ -1055,6 +1055,13 @@ config ADVANTECH_WDT
 	  feature. More information can be found at
 	  <https://www.advantech.com.tw/products/>
 
+config ADVANTECH_EC_WDT
+	tristate "Advantech Embedded Controller Watchdog Timer"
+	depends on X86
+	help
+	   	This driver supports Advantech products with ITE based Embedded Controller.
+ 		It does not support Advantech products with other ECs or without EC.
+
 config ALIM1535_WDT
 	tristate "ALi M1535 PMU Watchdog Timer"
 	depends on X86 && PCI
diff --git a/drivers/watchdog/Makefile b/drivers/watchdog/Makefile
index cdeb119e6e61..2768dc2348af 100644
--- a/drivers/watchdog/Makefile
+++ b/drivers/watchdog/Makefile
@@ -102,6 +102,7 @@ obj-$(CONFIG_SUNPLUS_WATCHDOG) += sunplus_wdt.o
 # X86 (i386 + ia64 + x86_64) Architecture
 obj-$(CONFIG_ACQUIRE_WDT) += acquirewdt.o
 obj-$(CONFIG_ADVANTECH_WDT) += advantechwdt.o
+obj-$(CONFIG_ADVANTECH_EC_WDT) += advantech_ec_wdt.o
 obj-$(CONFIG_ALIM1535_WDT) += alim1535_wdt.o
 obj-$(CONFIG_ALIM7101_WDT) += alim7101_wdt.o
 obj-$(CONFIG_EBC_C384_WDT) += ebc-c384_wdt.o
diff --git a/drivers/watchdog/advantech_ec_wdt.c b/drivers/watchdog/advantech_ec_wdt.c
new file mode 100644
index 000000000000..f3babaa918f7
--- /dev/null
+++ b/drivers/watchdog/advantech_ec_wdt.c
@@ -0,0 +1,247 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ *	Advantech Embedded Controller Watchdog Driver
+ *
+ *	This driver supports Advantech products with ITE based Embedded Controller.
+ *	It does not support Advantech products with other ECs or without EC.
+ *
+ *	This software is provided "as is" without warranty of any kind.
+ *
+ *	Copyright (C) 2022 Advantech Europe B.V.
+ *	<thomas.kastner@xxxxxxxxxxxxx>
+ */
+
+#include <linux/delay.h>
+#include <linux/module.h>
+#include <linux/moduleparam.h>
+#include <linux/io.h>
+#include <linux/ioport.h>
+#include <linux/isa.h>
+#include <linux/kernel.h>
+#include <linux/types.h>
+#include <linux/watchdog.h>
+#include <linux/seq_file.h>
+#include <linux/uaccess.h>
+#include <linux/spinlock.h>
+
+#define DRIVER_NAME		"advantech_ec_wdt"
+
+/* EC IO region */
+#define BASE_ADDR		0x299
+#define ADDR_EXTENT		2
+
+/* EC interface definitions */
+#define EC_ADDR_CMD		0x29A
+#define EC_ADDR_DATA		0x299
+#define EC_CMD_EC_PROBE		0x30
+#define EC_CMD_COMM		0x89
+#define EC_CMD_WDT_START	0x28
+#define EC_CMD_WDT_STOP		0x29
+#define EC_CMD_WDT_RESET	0x2A
+#define EC_DAT_EN_DLY_H		0x58
+#define EC_DAT_EN_DLY_L		0x59
+#define EC_DAT_RST_DLY_H	0x5E
+#define EC_DAT_RST_DLY_L	0x5F
+#define EC_MAGIC		0x95
+
+/* module parameters */
+#define MIN_TIME		1
+#define MAX_TIME		6000 /* 100 minutes */
+#define DEFAULT_TIME		60
+
+static unsigned int timeout = DEFAULT_TIME;
+module_param(timeout, uint, 0);
+MODULE_PARM_DESC(timeout,
+	"Default Watchdog timer setting ("
+	__MODULE_STRING(DEFAULT_TIME) "s)."
+	"The range is from " __MODULE_STRING(MIN_TIME)
+	" to " __MODULE_STRING(MAX_TIME) ".");
+
+static int adv_ec_wdt_ping(struct watchdog_device *wdd)
+{
+	pr_debug("%s: Strobing watchdog\n", __func__);
+
+	/* reset watchdog */
+	outb(EC_CMD_WDT_RESET, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	return 0;
+}
+
+static int adv_ec_wdt_set_timeout(struct watchdog_device *wdd, unsigned int t)
+{
+	unsigned int val;
+
+	/* scale time to EC 100 ms base */
+	val = t*10;
+
+	pr_debug("%s: Setting timeout value of %i\n", __func__, val);
+	/* reset enable delay, just in case it was set by BIOS etc. */
+	outb(EC_CMD_COMM, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	outb(EC_DAT_EN_DLY_H, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+	outb(0, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+
+	outb(EC_CMD_COMM, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	outb(EC_DAT_EN_DLY_L, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+	outb(0, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+
+	/* set reset delay */
+	outb(EC_CMD_COMM, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	outb(EC_DAT_RST_DLY_H, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+	outb((val >> 8), EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+
+	outb(EC_CMD_COMM, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	outb(EC_DAT_RST_DLY_L, EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+	outb((val & 0xFF), EC_ADDR_DATA);
+	usleep_range(10000, 11000);
+
+	wdd->timeout = t;
+	return 0;
+}
+
+
+static int adv_ec_probe(void)
+{
+	unsigned int val;
+
+	/* Check for magic byte */
+	outb(EC_CMD_EC_PROBE, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+	val = inb(EC_ADDR_DATA);
+
+	pr_debug("%s: probe result: %02X\n", __func__, val);
+
+	if (val == EC_MAGIC)
+		return 0;
+	else
+		return -ENODEV;
+}
+
+
+static int adv_ec_wdt_start(struct watchdog_device *wdd)
+{
+	adv_ec_wdt_set_timeout(wdd, wdd->timeout);
+
+	pr_debug("%s: Enabling watchdog timer\n", __func__);
+
+	/* Enable the watchdog timer */
+	outb(EC_CMD_WDT_START, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+
+	return 0;
+}
+
+static int adv_ec_wdt_stop(struct watchdog_device *wdd)
+{
+	pr_debug("%s: Disabling watchdog timer\n", __func__);
+
+	/* Disable the watchdog timer */
+	outb(EC_CMD_WDT_STOP, EC_ADDR_CMD);
+	usleep_range(10000, 11000);
+
+	return 0;
+}
+
+static const struct watchdog_info adv_ec_wdt_info = {
+	.identity =	DRIVER_NAME,
+	.options =	WDIOF_SETTIMEOUT |
+			WDIOF_MAGICCLOSE |
+			WDIOF_KEEPALIVEPING,
+};
+
+static const struct watchdog_ops adv_ec_wdt_ops = {
+	.owner =	THIS_MODULE,
+	.start =	adv_ec_wdt_start,
+	.stop =		adv_ec_wdt_stop,
+	.ping =		adv_ec_wdt_ping,
+	.set_timeout =	adv_ec_wdt_set_timeout,
+};
+
+static struct watchdog_device adv_ec_wdt_dev = {
+	.info =		&adv_ec_wdt_info,
+	.ops =		&adv_ec_wdt_ops,
+	.min_timeout =	MIN_TIME,
+	.max_timeout =	MAX_TIME,
+};
+
+static int adv_ec_wdt_probe(struct device *dev, unsigned int id)
+{
+	int ret;
+
+	if (!devm_request_region(dev, BASE_ADDR, ADDR_EXTENT, dev_name(dev))) {
+		dev_err(dev, "Unable to lock port addresses (0x%X-0x%X)\n",
+			BASE_ADDR, BASE_ADDR + ADDR_EXTENT);
+		return -EBUSY;
+	}
+
+	/* probe for EC */
+	ret = adv_ec_probe();
+	if (ret) {
+		dev_err(dev, "Error: cannot detect Advantech EC\n");
+		return -ENODEV;
+	}
+
+	adv_ec_wdt_dev.timeout = timeout;
+
+	ret = devm_watchdog_register_device(dev, &adv_ec_wdt_dev);
+
+	return ret;
+}
+
+static void adv_ec_wdt_remove(struct device *dev, unsigned int id)
+{
+	/* stop the watchdog */
+	adv_ec_wdt_stop(NULL);
+
+	/* watchdog_unregister() and release_region() are called automatically */
+
+	return;
+}
+
+static struct isa_driver adv_ec_wdt_driver = {
+	.probe		= adv_ec_wdt_probe,
+	.remove		= adv_ec_wdt_remove,
+	.driver		= {
+	.name		= DRIVER_NAME,
+	},
+};
+
+
+static int __init adv_ec_wdt_init(void)
+{
+	/* Check boot parameters to verify that their initial values */
+	/* are in range. */
+	if ((timeout < MIN_TIME) ||
+	    (timeout > MAX_TIME)) {
+		pr_err("Watchdog timer: value of timeout %d (dec) "
+		  "is out of range from %d to %d (dec)\n",
+		  timeout, MIN_TIME, MAX_TIME);
+		return -EINVAL;
+	}
+
+	return isa_register_driver(&adv_ec_wdt_driver, 1);
+}
+
+static void __exit adv_ec_wdt_exit(void)
+{
+	isa_unregister_driver(&adv_ec_wdt_driver);
+}
+
+module_init(adv_ec_wdt_init);
+module_exit(adv_ec_wdt_exit);
+
+MODULE_AUTHOR("Thomas Kastner <thomas.kastne@xxxxxxxxxxxxx>");
+MODULE_DESCRIPTION("Advantech Embedded Controller Watchdog Device Driver");
+MODULE_LICENSE("GPL");
+MODULE_VERSION("20221012");
+MODULE_ALIAS("isa:" DRIVER_NAME);
-- 
2.34.1




[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