1 // SPDX-License-Identifier: GPL-2.0
3 * Renesas RZ/A Series WDT Driver
5 * Copyright (C) 2017 Renesas Electronics America, Inc.
6 * Copyright (C) 2017 Chris Brandt
9 #include <linux/bitops.h>
10 #include <linux/clk.h>
11 #include <linux/delay.h>
13 #include <linux/module.h>
15 #include <linux/platform_device.h>
16 #include <linux/watchdog.h>
18 #define DEFAULT_TIMEOUT 30
20 /* Watchdog Timer Registers */
22 #define WTCSR_MAGIC 0xA500
23 #define WTSCR_WT BIT(6)
24 #define WTSCR_TME BIT(5)
25 #define WTSCR_CKS(i) (i)
28 #define WTCNT_MAGIC 0x5A00
31 #define WRCSR_MAGIC 0x5A00
32 #define WRCSR_RSTE BIT(6)
33 #define WRCSR_CLEAR_WOVF 0xA500 /* special value */
35 /* The maximum CKS register setting value to get the longest timeout */
39 #define DIVIDER_3BIT 16384 /* Clock divider when CKS = 0x7 */
40 #define DIVIDER_4BIT 4194304 /* Clock divider when CKS = 0xF */
43 struct watchdog_device wdev
;
50 static void rza_wdt_calc_timeout(struct rza_wdt
*priv
, int timeout
)
52 unsigned long rate
= clk_get_rate(priv
->clk
);
55 if (priv
->cks
== CKS_4BIT
) {
56 ticks
= DIV_ROUND_UP(timeout
* rate
, DIVIDER_4BIT
);
59 * Since max_timeout was set in probe, we know that the timeout
60 * value passed will never calculate to a tick value greater
63 priv
->count
= 256 - ticks
;
66 /* Start timer with longest timeout */
70 pr_debug("%s: timeout set to %u (WTCNT=%d)\n", __func__
,
71 timeout
, priv
->count
);
74 static int rza_wdt_start(struct watchdog_device
*wdev
)
76 struct rza_wdt
*priv
= watchdog_get_drvdata(wdev
);
79 writew(WTCSR_MAGIC
| 0, priv
->base
+ WTCSR
);
81 /* Must dummy read WRCSR:WOVF at least once before clearing */
82 readb(priv
->base
+ WRCSR
);
83 writew(WRCSR_CLEAR_WOVF
, priv
->base
+ WRCSR
);
85 rza_wdt_calc_timeout(priv
, wdev
->timeout
);
87 writew(WRCSR_MAGIC
| WRCSR_RSTE
, priv
->base
+ WRCSR
);
88 writew(WTCNT_MAGIC
| priv
->count
, priv
->base
+ WTCNT
);
89 writew(WTCSR_MAGIC
| WTSCR_WT
| WTSCR_TME
|
90 WTSCR_CKS(priv
->cks
), priv
->base
+ WTCSR
);
95 static int rza_wdt_stop(struct watchdog_device
*wdev
)
97 struct rza_wdt
*priv
= watchdog_get_drvdata(wdev
);
99 writew(WTCSR_MAGIC
| 0, priv
->base
+ WTCSR
);
104 static int rza_wdt_ping(struct watchdog_device
*wdev
)
106 struct rza_wdt
*priv
= watchdog_get_drvdata(wdev
);
108 writew(WTCNT_MAGIC
| priv
->count
, priv
->base
+ WTCNT
);
110 pr_debug("%s: timeout = %u\n", __func__
, wdev
->timeout
);
115 static int rza_set_timeout(struct watchdog_device
*wdev
, unsigned int timeout
)
117 wdev
->timeout
= timeout
;
122 static int rza_wdt_restart(struct watchdog_device
*wdev
, unsigned long action
,
125 struct rza_wdt
*priv
= watchdog_get_drvdata(wdev
);
128 writew(WTCSR_MAGIC
| 0, priv
->base
+ WTCSR
);
130 /* Must dummy read WRCSR:WOVF at least once before clearing */
131 readb(priv
->base
+ WRCSR
);
132 writew(WRCSR_CLEAR_WOVF
, priv
->base
+ WRCSR
);
135 * Start timer with fastest clock source and only 1 clock left before
136 * overflow with reset option enabled.
138 writew(WRCSR_MAGIC
| WRCSR_RSTE
, priv
->base
+ WRCSR
);
139 writew(WTCNT_MAGIC
| 255, priv
->base
+ WTCNT
);
140 writew(WTCSR_MAGIC
| WTSCR_WT
| WTSCR_TME
, priv
->base
+ WTCSR
);
143 * Actually make sure the above sequence hits hardware before sleeping.
147 /* Wait for WDT overflow (reset) */
153 static const struct watchdog_info rza_wdt_ident
= {
154 .options
= WDIOF_MAGICCLOSE
| WDIOF_KEEPALIVEPING
| WDIOF_SETTIMEOUT
,
155 .identity
= "Renesas RZ/A WDT Watchdog",
158 static const struct watchdog_ops rza_wdt_ops
= {
159 .owner
= THIS_MODULE
,
160 .start
= rza_wdt_start
,
161 .stop
= rza_wdt_stop
,
162 .ping
= rza_wdt_ping
,
163 .set_timeout
= rza_set_timeout
,
164 .restart
= rza_wdt_restart
,
167 static int rza_wdt_probe(struct platform_device
*pdev
)
169 struct device
*dev
= &pdev
->dev
;
170 struct rza_wdt
*priv
;
173 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
177 priv
->base
= devm_platform_ioremap_resource(pdev
, 0);
178 if (IS_ERR(priv
->base
))
179 return PTR_ERR(priv
->base
);
181 priv
->clk
= devm_clk_get(dev
, NULL
);
182 if (IS_ERR(priv
->clk
))
183 return PTR_ERR(priv
->clk
);
185 rate
= clk_get_rate(priv
->clk
);
187 dev_err(dev
, "invalid clock rate (%ld)\n", rate
);
191 priv
->wdev
.info
= &rza_wdt_ident
;
192 priv
->wdev
.ops
= &rza_wdt_ops
;
193 priv
->wdev
.parent
= dev
;
195 priv
->cks
= (u8
)(uintptr_t) of_device_get_match_data(dev
);
196 if (priv
->cks
== CKS_4BIT
) {
197 /* Assume slowest clock rate possible (CKS=0xF) */
198 priv
->wdev
.max_timeout
= (DIVIDER_4BIT
* U8_MAX
) / rate
;
200 } else if (priv
->cks
== CKS_3BIT
) {
201 /* Assume slowest clock rate possible (CKS=7) */
202 rate
/= DIVIDER_3BIT
;
205 * Since the max possible timeout of our 8-bit count
206 * register is less than a second, we must use
207 * max_hw_heartbeat_ms.
209 priv
->wdev
.max_hw_heartbeat_ms
= (1000 * U8_MAX
) / rate
;
210 dev_dbg(dev
, "max hw timeout of %dms\n",
211 priv
->wdev
.max_hw_heartbeat_ms
);
214 priv
->wdev
.min_timeout
= 1;
215 priv
->wdev
.timeout
= DEFAULT_TIMEOUT
;
217 watchdog_init_timeout(&priv
->wdev
, 0, dev
);
218 watchdog_set_drvdata(&priv
->wdev
, priv
);
220 return devm_watchdog_register_device(dev
, &priv
->wdev
);
223 static const struct of_device_id rza_wdt_of_match
[] = {
224 { .compatible
= "renesas,r7s9210-wdt", .data
= (void *)CKS_4BIT
, },
225 { .compatible
= "renesas,rza-wdt", .data
= (void *)CKS_3BIT
, },
228 MODULE_DEVICE_TABLE(of
, rza_wdt_of_match
);
230 static struct platform_driver rza_wdt_driver
= {
231 .probe
= rza_wdt_probe
,
234 .of_match_table
= rza_wdt_of_match
,
238 module_platform_driver(rza_wdt_driver
);
240 MODULE_DESCRIPTION("Renesas RZ/A WDT Driver");
241 MODULE_AUTHOR("Chris Brandt <chris.brandt@renesas.com>");
242 MODULE_LICENSE("GPL v2");