Merge tag 'efi-2020-04-rc2' of https://gitlab.denx.de/u-boot/custodians/u-boot-efi
[oweals/u-boot.git] / drivers / watchdog / designware_wdt.c
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Copyright (C) 2013 Altera Corporation <www.altera.com>
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 <asm/utils.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
22 struct designware_wdt_priv {
23         void __iomem    *base;
24         unsigned int    clk_khz;
25 };
26
27 /*
28  * Set the watchdog time interval.
29  * Counter is 32 bit.
30  */
31 static int designware_wdt_settimeout(void __iomem *base, unsigned int clk_khz,
32                                      unsigned int timeout)
33 {
34         signed int i;
35
36         /* calculate the timeout range value */
37         i = log_2_n_round_up(timeout * clk_khz) - 16;
38         i = clamp(i, 0, 15);
39
40         writel(i | (i << 4), base + DW_WDT_TORR);
41
42         return 0;
43 }
44
45 static void designware_wdt_enable(void __iomem *base)
46 {
47         writel(BIT(DW_WDT_CR_EN_OFFSET), base + DW_WDT_CR);
48 }
49
50 static unsigned int designware_wdt_is_enabled(void __iomem *base)
51 {
52         return readl(base + DW_WDT_CR) & BIT(0);
53 }
54
55 static void designware_wdt_reset_common(void __iomem *base)
56 {
57         if (designware_wdt_is_enabled(base))
58                 /* restart the watchdog counter */
59                 writel(DW_WDT_CRR_RESTART_VAL, base + DW_WDT_CRR);
60 }
61
62 #if !CONFIG_IS_ENABLED(WDT)
63 void hw_watchdog_reset(void)
64 {
65         designware_wdt_reset_common((void __iomem *)CONFIG_DW_WDT_BASE);
66 }
67
68 void hw_watchdog_init(void)
69 {
70         /* reset to disable the watchdog */
71         hw_watchdog_reset();
72         /* set timer in miliseconds */
73         designware_wdt_settimeout((void __iomem *)CONFIG_DW_WDT_BASE,
74                                   CONFIG_DW_WDT_CLOCK_KHZ,
75                                   CONFIG_WATCHDOG_TIMEOUT_MSECS);
76         /* enable the watchdog */
77         designware_wdt_enable((void __iomem *)CONFIG_DW_WDT_BASE);
78         /* reset the watchdog */
79         hw_watchdog_reset();
80 }
81 #else
82 static int designware_wdt_reset(struct udevice *dev)
83 {
84         struct designware_wdt_priv *priv = dev_get_priv(dev);
85
86         designware_wdt_reset_common(priv->base);
87
88         return 0;
89 }
90
91 static int designware_wdt_stop(struct udevice *dev)
92 {
93         struct designware_wdt_priv *priv = dev_get_priv(dev);
94
95         designware_wdt_reset(dev);
96         writel(0, priv->base + DW_WDT_CR);
97
98         return 0;
99 }
100
101 static int designware_wdt_start(struct udevice *dev, u64 timeout, ulong flags)
102 {
103         struct designware_wdt_priv *priv = dev_get_priv(dev);
104
105         designware_wdt_stop(dev);
106
107         /* set timer in miliseconds */
108         designware_wdt_settimeout(priv->base, priv->clk_khz, timeout);
109
110         designware_wdt_enable(priv->base);
111
112         /* reset the watchdog */
113         return designware_wdt_reset(dev);
114 }
115
116 static int designware_wdt_probe(struct udevice *dev)
117 {
118         struct designware_wdt_priv *priv = dev_get_priv(dev);
119         __maybe_unused int ret;
120
121         priv->base = dev_remap_addr(dev);
122         if (!priv->base)
123                 return -EINVAL;
124
125 #if CONFIG_IS_ENABLED(CLK)
126         struct clk clk;
127
128         ret = clk_get_by_index(dev, 0, &clk);
129         if (ret)
130                 return ret;
131
132         priv->clk_khz = clk_get_rate(&clk);
133         if (!priv->clk_khz)
134                 return -EINVAL;
135 #else
136         priv->clk_khz = CONFIG_DW_WDT_CLOCK_KHZ;
137 #endif
138
139 #if CONFIG_IS_ENABLED(DM_RESET)
140         struct reset_ctl_bulk resets;
141
142         ret = reset_get_bulk(dev, &resets);
143         if (ret)
144                 return ret;
145
146         ret = reset_deassert_bulk(&resets);
147         if (ret)
148                 return ret;
149 #endif
150
151         /* reset to disable the watchdog */
152         return designware_wdt_stop(dev);
153 }
154
155 static const struct wdt_ops designware_wdt_ops = {
156         .start = designware_wdt_start,
157         .reset = designware_wdt_reset,
158         .stop = designware_wdt_stop,
159 };
160
161 static const struct udevice_id designware_wdt_ids[] = {
162         { .compatible = "snps,dw-wdt"},
163         {}
164 };
165
166 U_BOOT_DRIVER(designware_wdt) = {
167         .name = "designware_wdt",
168         .id = UCLASS_WDT,
169         .of_match = designware_wdt_ids,
170         .priv_auto_alloc_size = sizeof(struct designware_wdt_priv),
171         .probe = designware_wdt_probe,
172         .ops = &designware_wdt_ops,
173         .flags = DM_FLAG_PRE_RELOC,
174 };
175 #endif