1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Driver for watchdog device controlled through GPIO-line
5 * Author: 2013, Alexander Shiyan <shc_work@mail.ru>
8 #include <linux/delay.h>
10 #include <linux/gpio/consumer.h>
11 #include <linux/mod_devicetable.h>
12 #include <linux/module.h>
13 #include <linux/platform_device.h>
14 #include <linux/property.h>
15 #include <linux/watchdog.h>
17 static bool nowayout
= WATCHDOG_NOWAYOUT
;
18 module_param(nowayout
, bool, 0);
19 MODULE_PARM_DESC(nowayout
,
20 "Watchdog cannot be stopped once started (default="
21 __MODULE_STRING(WATCHDOG_NOWAYOUT
) ")");
23 #define SOFT_TIMEOUT_MIN 1
24 #define SOFT_TIMEOUT_DEF 60
31 struct gpio_wdt_priv
{
32 struct gpio_desc
*gpiod
;
36 struct watchdog_device wdd
;
39 static void gpio_wdt_disable(struct gpio_wdt_priv
*priv
)
42 gpiod_set_value_cansleep(priv
->gpiod
, 1);
44 /* Put GPIO back to tristate */
45 if (priv
->hw_algo
== HW_ALGO_TOGGLE
)
46 gpiod_direction_input(priv
->gpiod
);
49 static int gpio_wdt_ping(struct watchdog_device
*wdd
)
51 struct gpio_wdt_priv
*priv
= watchdog_get_drvdata(wdd
);
53 switch (priv
->hw_algo
) {
55 /* Toggle output pin */
56 priv
->state
= !priv
->state
;
57 gpiod_set_value_cansleep(priv
->gpiod
, priv
->state
);
61 gpiod_set_value_cansleep(priv
->gpiod
, 1);
63 gpiod_set_value_cansleep(priv
->gpiod
, 0);
69 static int gpio_wdt_start(struct watchdog_device
*wdd
)
71 struct gpio_wdt_priv
*priv
= watchdog_get_drvdata(wdd
);
74 gpiod_direction_output(priv
->gpiod
, priv
->state
);
76 set_bit(WDOG_HW_RUNNING
, &wdd
->status
);
78 return gpio_wdt_ping(wdd
);
81 static int gpio_wdt_stop(struct watchdog_device
*wdd
)
83 struct gpio_wdt_priv
*priv
= watchdog_get_drvdata(wdd
);
85 if (!priv
->always_running
) {
86 gpio_wdt_disable(priv
);
88 set_bit(WDOG_HW_RUNNING
, &wdd
->status
);
94 static const struct watchdog_info gpio_wdt_ident
= {
95 .options
= WDIOF_MAGICCLOSE
| WDIOF_KEEPALIVEPING
|
97 .identity
= "GPIO Watchdog",
100 static const struct watchdog_ops gpio_wdt_ops
= {
101 .owner
= THIS_MODULE
,
102 .start
= gpio_wdt_start
,
103 .stop
= gpio_wdt_stop
,
104 .ping
= gpio_wdt_ping
,
107 static int gpio_wdt_probe(struct platform_device
*pdev
)
109 struct device
*dev
= &pdev
->dev
;
110 struct gpio_wdt_priv
*priv
;
111 enum gpiod_flags gflags
;
112 unsigned int hw_margin
;
116 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
120 platform_set_drvdata(pdev
, priv
);
122 ret
= device_property_read_string(dev
, "hw_algo", &algo
);
125 if (!strcmp(algo
, "toggle")) {
126 priv
->hw_algo
= HW_ALGO_TOGGLE
;
128 } else if (!strcmp(algo
, "level")) {
129 priv
->hw_algo
= HW_ALGO_LEVEL
;
130 gflags
= GPIOD_OUT_LOW
;
135 priv
->gpiod
= devm_gpiod_get(dev
, NULL
, gflags
);
136 if (IS_ERR(priv
->gpiod
))
137 return PTR_ERR(priv
->gpiod
);
139 ret
= device_property_read_u32(dev
, "hw_margin_ms", &hw_margin
);
142 /* Disallow values lower than 2 and higher than 65535 ms */
143 if (hw_margin
< 2 || hw_margin
> 65535)
146 priv
->always_running
= device_property_read_bool(dev
, "always-running");
148 watchdog_set_drvdata(&priv
->wdd
, priv
);
150 priv
->wdd
.info
= &gpio_wdt_ident
;
151 priv
->wdd
.ops
= &gpio_wdt_ops
;
152 priv
->wdd
.min_timeout
= SOFT_TIMEOUT_MIN
;
153 priv
->wdd
.max_hw_heartbeat_ms
= hw_margin
;
154 priv
->wdd
.parent
= dev
;
155 priv
->wdd
.timeout
= SOFT_TIMEOUT_DEF
;
157 watchdog_init_timeout(&priv
->wdd
, 0, dev
);
158 watchdog_set_nowayout(&priv
->wdd
, nowayout
);
160 watchdog_stop_on_reboot(&priv
->wdd
);
162 if (priv
->always_running
)
163 gpio_wdt_start(&priv
->wdd
);
165 return devm_watchdog_register_device(dev
, &priv
->wdd
);
168 static const struct of_device_id gpio_wdt_dt_ids
[] = {
169 { .compatible
= "linux,wdt-gpio", },
172 MODULE_DEVICE_TABLE(of
, gpio_wdt_dt_ids
);
174 static struct platform_driver gpio_wdt_driver
= {
177 .of_match_table
= gpio_wdt_dt_ids
,
179 .probe
= gpio_wdt_probe
,
182 #ifdef CONFIG_GPIO_WATCHDOG_ARCH_INITCALL
183 static int __init
gpio_wdt_init(void)
185 return platform_driver_register(&gpio_wdt_driver
);
187 arch_initcall(gpio_wdt_init
);
189 module_platform_driver(gpio_wdt_driver
);
192 MODULE_AUTHOR("Alexander Shiyan <shc_work@mail.ru>");
193 MODULE_DESCRIPTION("GPIO Watchdog");
194 MODULE_LICENSE("GPL");