1 // SPDX-License-Identifier: GPL-2.0+
5 * Copyright (C) 2004, 2005 Nokia Corporation
7 * Based on code written by Amit Kucheria and Michael Buesch.
8 * Rewritten by Aaro Koskinen.
11 #include <linux/devm-helpers.h>
12 #include <linux/slab.h>
13 #include <linux/errno.h>
14 #include <linux/device.h>
15 #include <linux/kernel.h>
16 #include <linux/module.h>
17 #include <linux/mfd/retu.h>
18 #include <linux/watchdog.h>
19 #include <linux/platform_device.h>
21 /* Watchdog timer values in seconds */
22 #define RETU_WDT_MAX_TIMER 63
25 struct retu_dev
*rdev
;
27 struct delayed_work ping_work
;
31 * Since Retu watchdog cannot be disabled in hardware, we must kick it
32 * with a timer until userspace watchdog software takes over. If
33 * CONFIG_WATCHDOG_NOWAYOUT is set, we never start the feeding.
35 static void retu_wdt_ping_enable(struct retu_wdt_dev
*wdev
)
37 retu_write(wdev
->rdev
, RETU_REG_WATCHDOG
, RETU_WDT_MAX_TIMER
);
38 schedule_delayed_work(&wdev
->ping_work
,
39 round_jiffies_relative(RETU_WDT_MAX_TIMER
* HZ
/ 2));
42 static void retu_wdt_ping_disable(struct retu_wdt_dev
*wdev
)
44 retu_write(wdev
->rdev
, RETU_REG_WATCHDOG
, RETU_WDT_MAX_TIMER
);
45 cancel_delayed_work_sync(&wdev
->ping_work
);
48 static void retu_wdt_ping_work(struct work_struct
*work
)
50 struct retu_wdt_dev
*wdev
= container_of(to_delayed_work(work
),
51 struct retu_wdt_dev
, ping_work
);
52 retu_wdt_ping_enable(wdev
);
55 static int retu_wdt_start(struct watchdog_device
*wdog
)
57 struct retu_wdt_dev
*wdev
= watchdog_get_drvdata(wdog
);
59 retu_wdt_ping_disable(wdev
);
61 return retu_write(wdev
->rdev
, RETU_REG_WATCHDOG
, wdog
->timeout
);
64 static int retu_wdt_stop(struct watchdog_device
*wdog
)
66 struct retu_wdt_dev
*wdev
= watchdog_get_drvdata(wdog
);
68 retu_wdt_ping_enable(wdev
);
73 static int retu_wdt_ping(struct watchdog_device
*wdog
)
75 struct retu_wdt_dev
*wdev
= watchdog_get_drvdata(wdog
);
77 return retu_write(wdev
->rdev
, RETU_REG_WATCHDOG
, wdog
->timeout
);
80 static int retu_wdt_set_timeout(struct watchdog_device
*wdog
,
83 struct retu_wdt_dev
*wdev
= watchdog_get_drvdata(wdog
);
85 wdog
->timeout
= timeout
;
86 return retu_write(wdev
->rdev
, RETU_REG_WATCHDOG
, wdog
->timeout
);
89 static const struct watchdog_info retu_wdt_info
= {
90 .options
= WDIOF_SETTIMEOUT
| WDIOF_MAGICCLOSE
| WDIOF_KEEPALIVEPING
,
91 .identity
= "Retu watchdog",
94 static const struct watchdog_ops retu_wdt_ops
= {
96 .start
= retu_wdt_start
,
97 .stop
= retu_wdt_stop
,
98 .ping
= retu_wdt_ping
,
99 .set_timeout
= retu_wdt_set_timeout
,
102 static int retu_wdt_probe(struct platform_device
*pdev
)
104 struct retu_dev
*rdev
= dev_get_drvdata(pdev
->dev
.parent
);
105 bool nowayout
= WATCHDOG_NOWAYOUT
;
106 struct watchdog_device
*retu_wdt
;
107 struct retu_wdt_dev
*wdev
;
110 retu_wdt
= devm_kzalloc(&pdev
->dev
, sizeof(*retu_wdt
), GFP_KERNEL
);
114 wdev
= devm_kzalloc(&pdev
->dev
, sizeof(*wdev
), GFP_KERNEL
);
118 retu_wdt
->info
= &retu_wdt_info
;
119 retu_wdt
->ops
= &retu_wdt_ops
;
120 retu_wdt
->timeout
= RETU_WDT_MAX_TIMER
;
121 retu_wdt
->min_timeout
= 0;
122 retu_wdt
->max_timeout
= RETU_WDT_MAX_TIMER
;
123 retu_wdt
->parent
= &pdev
->dev
;
125 watchdog_set_drvdata(retu_wdt
, wdev
);
126 watchdog_set_nowayout(retu_wdt
, nowayout
);
129 wdev
->dev
= &pdev
->dev
;
131 ret
= devm_delayed_work_autocancel(&pdev
->dev
, &wdev
->ping_work
,
136 ret
= devm_watchdog_register_device(&pdev
->dev
, retu_wdt
);
141 retu_wdt_ping(retu_wdt
);
143 retu_wdt_ping_enable(wdev
);
148 static struct platform_driver retu_wdt_driver
= {
149 .probe
= retu_wdt_probe
,
154 module_platform_driver(retu_wdt_driver
);
156 MODULE_ALIAS("platform:retu-wdt");
157 MODULE_DESCRIPTION("Retu watchdog");
158 MODULE_AUTHOR("Amit Kucheria");
159 MODULE_AUTHOR("Aaro Koskinen <aaro.koskinen@iki.fi>");
160 MODULE_LICENSE("GPL");