1 // SPDX-License-Identifier: GPL-2.0+
3 * Watchdog driver for DA9063 PMICs.
5 * Copyright(c) 2012 Dialog Semiconductor Ltd.
7 * Author: Mariusz Wojtasik <mariusz.wojtasik@diasemi.com>
11 #include <linux/kernel.h>
12 #include <linux/module.h>
13 #include <linux/watchdog.h>
14 #include <linux/platform_device.h>
15 #include <linux/uaccess.h>
16 #include <linux/slab.h>
17 #include <linux/i2c.h>
18 #include <linux/delay.h>
19 #include <linux/mfd/da9063/registers.h>
20 #include <linux/mfd/da9063/core.h>
21 #include <linux/property.h>
22 #include <linux/regmap.h>
25 * Watchdog selector to timeout in seconds.
27 * others: timeout = 2048 ms * 2^(TWDSCALE-1).
29 static const unsigned int wdt_timeout
[] = { 0, 2, 4, 8, 16, 32, 65, 131 };
30 static bool use_sw_pm
;
32 #define DA9063_TWDSCALE_DISABLE 0
33 #define DA9063_TWDSCALE_MIN 1
34 #define DA9063_TWDSCALE_MAX (ARRAY_SIZE(wdt_timeout) - 1)
35 #define DA9063_WDT_MIN_TIMEOUT wdt_timeout[DA9063_TWDSCALE_MIN]
36 #define DA9063_WDT_MAX_TIMEOUT wdt_timeout[DA9063_TWDSCALE_MAX]
37 #define DA9063_WDG_TIMEOUT wdt_timeout[3]
38 #define DA9063_RESET_PROTECTION_MS 256
40 static unsigned int da9063_wdt_timeout_to_sel(unsigned int secs
)
44 for (i
= DA9063_TWDSCALE_MIN
; i
<= DA9063_TWDSCALE_MAX
; i
++) {
45 if (wdt_timeout
[i
] >= secs
)
49 return DA9063_TWDSCALE_MAX
;
53 * Read the currently active timeout.
54 * Zero means the watchdog is disabled.
56 static unsigned int da9063_wdt_read_timeout(struct da9063
*da9063
)
60 regmap_read(da9063
->regmap
, DA9063_REG_CONTROL_D
, &val
);
62 return wdt_timeout
[val
& DA9063_TWDSCALE_MASK
];
65 static int da9063_wdt_disable_timer(struct da9063
*da9063
)
67 return regmap_update_bits(da9063
->regmap
, DA9063_REG_CONTROL_D
,
69 DA9063_TWDSCALE_DISABLE
);
73 da9063_wdt_update_timeout(struct da9063
*da9063
, unsigned int timeout
)
79 * The watchdog triggers a reboot if a timeout value is already
80 * programmed because the timeout value combines two functions
81 * in one: indicating the counter limit and starting the watchdog.
82 * The watchdog must be disabled to be able to change the timeout
83 * value if the watchdog is already running. Then we can set the
84 * new timeout value which enables the watchdog again.
86 ret
= da9063_wdt_disable_timer(da9063
);
90 usleep_range(150, 300);
91 regval
= da9063_wdt_timeout_to_sel(timeout
);
93 return regmap_update_bits(da9063
->regmap
, DA9063_REG_CONTROL_D
,
94 DA9063_TWDSCALE_MASK
, regval
);
97 static int da9063_wdt_start(struct watchdog_device
*wdd
)
99 struct da9063
*da9063
= watchdog_get_drvdata(wdd
);
102 ret
= da9063_wdt_update_timeout(da9063
, wdd
->timeout
);
104 dev_err(da9063
->dev
, "Watchdog failed to start (err = %d)\n",
110 static int da9063_wdt_stop(struct watchdog_device
*wdd
)
112 struct da9063
*da9063
= watchdog_get_drvdata(wdd
);
115 ret
= da9063_wdt_disable_timer(da9063
);
117 dev_alert(da9063
->dev
, "Watchdog failed to stop (err = %d)\n",
123 static int da9063_wdt_ping(struct watchdog_device
*wdd
)
125 struct da9063
*da9063
= watchdog_get_drvdata(wdd
);
129 * Prevent pings from occurring late in system poweroff/reboot sequence
130 * and possibly locking out restart handler from accessing i2c bus.
132 if (system_state
> SYSTEM_RUNNING
)
135 ret
= regmap_write(da9063
->regmap
, DA9063_REG_CONTROL_F
,
138 dev_alert(da9063
->dev
, "Failed to ping the watchdog (err = %d)\n",
144 static int da9063_wdt_set_timeout(struct watchdog_device
*wdd
,
145 unsigned int timeout
)
147 struct da9063
*da9063
= watchdog_get_drvdata(wdd
);
151 * There are two cases when a set_timeout() will be called:
152 * 1. The watchdog is off and someone wants to set the timeout for the
154 * 2. The watchdog is already running and a new timeout value should be
157 * The watchdog can't store a timeout value not equal zero without
158 * enabling the watchdog, so the timeout must be buffered by the driver.
160 if (watchdog_active(wdd
))
161 ret
= da9063_wdt_update_timeout(da9063
, timeout
);
164 dev_err(da9063
->dev
, "Failed to set watchdog timeout (err = %d)\n",
167 wdd
->timeout
= wdt_timeout
[da9063_wdt_timeout_to_sel(timeout
)];
172 static int da9063_wdt_restart(struct watchdog_device
*wdd
, unsigned long action
,
175 struct da9063
*da9063
= watchdog_get_drvdata(wdd
);
176 struct i2c_client
*client
= to_i2c_client(da9063
->dev
);
177 union i2c_smbus_data msg
;
181 * Don't use regmap because it is not atomic safe. Additionally, use
182 * unlocked flavor of i2c_smbus_xfer to avoid scenario where i2c bus
183 * might previously be locked by some process unable to release the
184 * lock due to interrupts already being disabled at this late stage.
186 msg
.byte
= DA9063_SHUTDOWN
;
187 ret
= __i2c_smbus_xfer(client
->adapter
, client
->addr
, client
->flags
,
188 I2C_SMBUS_WRITE
, DA9063_REG_CONTROL_F
,
189 I2C_SMBUS_BYTE_DATA
, &msg
);
192 dev_alert(da9063
->dev
, "Failed to shutdown (err = %d)\n",
195 /* wait for reset to assert... */
201 static const struct watchdog_info da9063_watchdog_info
= {
202 .options
= WDIOF_SETTIMEOUT
| WDIOF_KEEPALIVEPING
,
203 .identity
= "DA9063 Watchdog",
206 static const struct watchdog_ops da9063_watchdog_ops
= {
207 .owner
= THIS_MODULE
,
208 .start
= da9063_wdt_start
,
209 .stop
= da9063_wdt_stop
,
210 .ping
= da9063_wdt_ping
,
211 .set_timeout
= da9063_wdt_set_timeout
,
212 .restart
= da9063_wdt_restart
,
215 static int da9063_wdt_probe(struct platform_device
*pdev
)
217 struct device
*dev
= &pdev
->dev
;
218 struct da9063
*da9063
;
219 struct watchdog_device
*wdd
;
220 unsigned int timeout
;
225 da9063
= dev_get_drvdata(dev
->parent
);
229 wdd
= devm_kzalloc(dev
, sizeof(*wdd
), GFP_KERNEL
);
233 use_sw_pm
= device_property_present(dev
, "dlg,use-sw-pm");
235 wdd
->info
= &da9063_watchdog_info
;
236 wdd
->ops
= &da9063_watchdog_ops
;
237 wdd
->min_timeout
= DA9063_WDT_MIN_TIMEOUT
;
238 wdd
->max_timeout
= DA9063_WDT_MAX_TIMEOUT
;
239 wdd
->min_hw_heartbeat_ms
= DA9063_RESET_PROTECTION_MS
;
241 wdd
->status
= WATCHDOG_NOWAYOUT_INIT_STATUS
;
243 watchdog_set_restart_priority(wdd
, 128);
244 watchdog_set_drvdata(wdd
, da9063
);
245 dev_set_drvdata(dev
, wdd
);
247 wdd
->timeout
= DA9063_WDG_TIMEOUT
;
249 /* Use pre-configured timeout if watchdog is already running. */
250 timeout
= da9063_wdt_read_timeout(da9063
);
252 wdd
->timeout
= timeout
;
254 /* Set timeout, maybe override it with DT value, scale it */
255 watchdog_init_timeout(wdd
, 0, dev
);
256 da9063_wdt_set_timeout(wdd
, wdd
->timeout
);
258 /* Update timeout if the watchdog is already running. */
260 da9063_wdt_update_timeout(da9063
, wdd
->timeout
);
261 set_bit(WDOG_HW_RUNNING
, &wdd
->status
);
264 return devm_watchdog_register_device(dev
, wdd
);
267 static int __maybe_unused
da9063_wdt_suspend(struct device
*dev
)
269 struct watchdog_device
*wdd
= dev_get_drvdata(dev
);
274 if (watchdog_active(wdd
))
275 return da9063_wdt_stop(wdd
);
280 static int __maybe_unused
da9063_wdt_resume(struct device
*dev
)
282 struct watchdog_device
*wdd
= dev_get_drvdata(dev
);
287 if (watchdog_active(wdd
))
288 return da9063_wdt_start(wdd
);
293 static SIMPLE_DEV_PM_OPS(da9063_wdt_pm_ops
,
294 da9063_wdt_suspend
, da9063_wdt_resume
);
296 static struct platform_driver da9063_wdt_driver
= {
297 .probe
= da9063_wdt_probe
,
299 .name
= DA9063_DRVNAME_WATCHDOG
,
300 .pm
= &da9063_wdt_pm_ops
,
303 module_platform_driver(da9063_wdt_driver
);
305 MODULE_AUTHOR("Mariusz Wojtasik <mariusz.wojtasik@diasemi.com>");
306 MODULE_DESCRIPTION("Watchdog driver for Dialog DA9063");
307 MODULE_LICENSE("GPL");
308 MODULE_ALIAS("platform:" DA9063_DRVNAME_WATCHDOG
);