1
2
3
4
5
6#include <clk.h>
7#include <common.h>
8#include <dm.h>
9#include <reset.h>
10#include <wdt.h>
11#include <asm/io.h>
12#include <linux/bitops.h>
13
14#define DW_WDT_CR 0x00
15#define DW_WDT_TORR 0x04
16#define DW_WDT_CRR 0x0C
17
18#define DW_WDT_CR_EN_OFFSET 0x00
19#define DW_WDT_CR_RMOD_OFFSET 0x01
20#define DW_WDT_CRR_RESTART_VAL 0x76
21
22struct designware_wdt_priv {
23 void __iomem *base;
24 unsigned int clk_khz;
25 struct reset_ctl_bulk resets;
26};
27
28
29
30
31
32static int designware_wdt_settimeout(void __iomem *base, unsigned int clk_khz,
33 unsigned int timeout)
34{
35 signed int i;
36
37
38 i = fls(timeout * clk_khz - 1) - 16;
39 i = clamp(i, 0, 15);
40
41 writel(i | (i << 4), base + DW_WDT_TORR);
42
43 return 0;
44}
45
46static void designware_wdt_enable(void __iomem *base)
47{
48 writel(BIT(DW_WDT_CR_EN_OFFSET), base + DW_WDT_CR);
49}
50
51static unsigned int designware_wdt_is_enabled(void __iomem *base)
52{
53 return readl(base + DW_WDT_CR) & BIT(0);
54}
55
56static void designware_wdt_reset_common(void __iomem *base)
57{
58 if (designware_wdt_is_enabled(base))
59
60 writel(DW_WDT_CRR_RESTART_VAL, base + DW_WDT_CRR);
61}
62
63#if !CONFIG_IS_ENABLED(WDT)
64void hw_watchdog_reset(void)
65{
66 designware_wdt_reset_common((void __iomem *)CONFIG_DW_WDT_BASE);
67}
68
69void hw_watchdog_init(void)
70{
71
72 hw_watchdog_reset();
73
74 designware_wdt_settimeout((void __iomem *)CONFIG_DW_WDT_BASE,
75 CONFIG_DW_WDT_CLOCK_KHZ,
76 CONFIG_WATCHDOG_TIMEOUT_MSECS);
77
78 designware_wdt_enable((void __iomem *)CONFIG_DW_WDT_BASE);
79
80 hw_watchdog_reset();
81}
82#else
83static int designware_wdt_reset(struct udevice *dev)
84{
85 struct designware_wdt_priv *priv = dev_get_priv(dev);
86
87 designware_wdt_reset_common(priv->base);
88
89 return 0;
90}
91
92static int designware_wdt_stop(struct udevice *dev)
93{
94 struct designware_wdt_priv *priv = dev_get_priv(dev);
95
96 designware_wdt_reset(dev);
97 writel(0, priv->base + DW_WDT_CR);
98
99 if (CONFIG_IS_ENABLED(DM_RESET)) {
100 int ret;
101
102 ret = reset_assert_bulk(&priv->resets);
103 if (ret)
104 return ret;
105
106 ret = reset_deassert_bulk(&priv->resets);
107 if (ret)
108 return ret;
109 }
110
111 return 0;
112}
113
114static int designware_wdt_start(struct udevice *dev, u64 timeout, ulong flags)
115{
116 struct designware_wdt_priv *priv = dev_get_priv(dev);
117
118 designware_wdt_stop(dev);
119
120
121 designware_wdt_settimeout(priv->base, priv->clk_khz, timeout);
122
123 designware_wdt_enable(priv->base);
124
125
126 return designware_wdt_reset(dev);
127}
128
129static int designware_wdt_probe(struct udevice *dev)
130{
131 struct designware_wdt_priv *priv = dev_get_priv(dev);
132 __maybe_unused int ret;
133
134 priv->base = dev_remap_addr(dev);
135 if (!priv->base)
136 return -EINVAL;
137
138#if CONFIG_IS_ENABLED(CLK)
139 struct clk clk;
140
141 ret = clk_get_by_index(dev, 0, &clk);
142 if (ret)
143 return ret;
144
145 ret = clk_enable(&clk);
146 if (ret)
147 goto err;
148
149 priv->clk_khz = clk_get_rate(&clk) / 1000;
150 if (!priv->clk_khz) {
151 ret = -EINVAL;
152 goto err;
153 }
154#else
155 priv->clk_khz = CONFIG_DW_WDT_CLOCK_KHZ;
156#endif
157
158 if (CONFIG_IS_ENABLED(DM_RESET)) {
159 ret = reset_get_bulk(dev, &priv->resets);
160 if (ret)
161 goto err;
162
163 ret = reset_deassert_bulk(&priv->resets);
164 if (ret)
165 goto err;
166 }
167
168
169 return designware_wdt_stop(dev);
170
171err:
172#if CONFIG_IS_ENABLED(CLK)
173 clk_free(&clk);
174#endif
175 return ret;
176}
177
178static const struct wdt_ops designware_wdt_ops = {
179 .start = designware_wdt_start,
180 .reset = designware_wdt_reset,
181 .stop = designware_wdt_stop,
182};
183
184static const struct udevice_id designware_wdt_ids[] = {
185 { .compatible = "snps,dw-wdt"},
186 {}
187};
188
189U_BOOT_DRIVER(designware_wdt) = {
190 .name = "designware_wdt",
191 .id = UCLASS_WDT,
192 .of_match = designware_wdt_ids,
193 .priv_auto = sizeof(struct designware_wdt_priv),
194 .probe = designware_wdt_probe,
195 .ops = &designware_wdt_ops,
196 .flags = DM_FLAG_PRE_RELOC,
197};
198#endif
199