2 * Atmel SAMA5D2-Compatible Shutdown Controller (SHDWC) driver.
3 * Found on some SoCs as the sama5d2 (obviously).
5 * Copyright (C) 2015 Atmel Corporation,
6 * Nicolas Ferre <nicolas.ferre@atmel.com>
8 * Evolved from driver at91-poweroff.c.
10 * This file is licensed under the terms of the GNU General Public
11 * License version 2. This program is licensed "as is" without any
12 * warranty of any kind, whether express or implied.
15 * - addition to status of other wake-up inputs [1 - 15]
16 * - Analog Comparator wake-up alarm
17 * - Serial RX wake-up alarm
18 * - low power debouncer
21 #include <linux/clk.h>
22 #include <linux/clk/at91_pmc.h>
24 #include <linux/module.h>
26 #include <linux/of_address.h>
27 #include <linux/platform_device.h>
28 #include <linux/printk.h>
30 #include <soc/at91/at91sam9_ddrsdr.h>
32 #define SLOW_CLOCK_FREQ 32768
34 #define AT91_SHDW_CR 0x00 /* Shut Down Control Register */
35 #define AT91_SHDW_SHDW BIT(0) /* Shut Down command */
36 #define AT91_SHDW_KEY (0xa5UL << 24) /* KEY Password */
38 #define AT91_SHDW_MR 0x04 /* Shut Down Mode Register */
39 #define AT91_SHDW_WKUPDBC_SHIFT 24
40 #define AT91_SHDW_WKUPDBC_MASK GENMASK(31, 16)
41 #define AT91_SHDW_WKUPDBC(x) (((x) << AT91_SHDW_WKUPDBC_SHIFT) \
42 & AT91_SHDW_WKUPDBC_MASK)
44 #define AT91_SHDW_SR 0x08 /* Shut Down Status Register */
45 #define AT91_SHDW_WKUPIS_SHIFT 16
46 #define AT91_SHDW_WKUPIS_MASK GENMASK(31, 16)
47 #define AT91_SHDW_WKUPIS(x) ((1 << (x)) << AT91_SHDW_WKUPIS_SHIFT \
48 & AT91_SHDW_WKUPIS_MASK)
50 #define AT91_SHDW_WUIR 0x0c /* Shutdown Wake-up Inputs Register */
51 #define AT91_SHDW_WKUPEN_MASK GENMASK(15, 0)
52 #define AT91_SHDW_WKUPEN(x) ((1 << (x)) & AT91_SHDW_WKUPEN_MASK)
53 #define AT91_SHDW_WKUPT_SHIFT 16
54 #define AT91_SHDW_WKUPT_MASK GENMASK(31, 16)
55 #define AT91_SHDW_WKUPT(x) ((1 << (x)) << AT91_SHDW_WKUPT_SHIFT \
56 & AT91_SHDW_WKUPT_MASK)
58 #define SHDW_WK_PIN(reg, cfg) ((reg) & AT91_SHDW_WKUPIS((cfg)->wkup_pin_input))
59 #define SHDW_RTCWK(reg, cfg) (((reg) >> ((cfg)->sr_rtcwk_shift)) & 0x1)
60 #define SHDW_RTTWK(reg, cfg) (((reg) >> ((cfg)->sr_rttwk_shift)) & 0x1)
61 #define SHDW_RTCWKEN(cfg) (1 << ((cfg)->mr_rtcwk_shift))
62 #define SHDW_RTTWKEN(cfg) (1 << ((cfg)->mr_rttwk_shift))
64 #define DBC_PERIOD_US(x) DIV_ROUND_UP_ULL((1000000 * (x)), \
67 #define SHDW_CFG_NOT_USED (32)
69 struct shdwc_reg_config
{
77 struct pmc_reg_config
{
82 struct shdwc_reg_config shdwc
;
83 struct pmc_reg_config pmc
;
87 const struct reg_config
*rcfg
;
89 void __iomem
*shdwc_base
;
90 void __iomem
*mpddrc_base
;
91 void __iomem
*pmc_base
;
95 * Hold configuration here, cannot be more than one instance of the driver
96 * since pm_power_off itself is global.
98 static struct shdwc
*at91_shdwc
;
100 static const unsigned long long sdwc_dbc_period
[] = {
101 0, 3, 32, 512, 4096, 32768,
104 static void __init
at91_wakeup_status(struct platform_device
*pdev
)
106 struct shdwc
*shdw
= platform_get_drvdata(pdev
);
107 const struct reg_config
*rcfg
= shdw
->rcfg
;
109 char *reason
= "unknown";
111 reg
= readl(shdw
->shdwc_base
+ AT91_SHDW_SR
);
113 dev_dbg(&pdev
->dev
, "%s: status = %#x\n", __func__
, reg
);
115 /* Simple power-on, just bail out */
119 if (SHDW_WK_PIN(reg
, &rcfg
->shdwc
))
121 else if (SHDW_RTCWK(reg
, &rcfg
->shdwc
))
123 else if (SHDW_RTTWK(reg
, &rcfg
->shdwc
))
126 pr_info("AT91: Wake-Up source: %s\n", reason
);
129 static void at91_poweroff(void)
132 /* Align to cache lines */
135 /* Ensure AT91_SHDW_CR is in the TLB by reading it */
136 " ldr r6, [%2, #" __stringify(AT91_SHDW_CR
) "]\n\t"
138 /* Power down SDRAM0 */
141 " str %1, [%0, #" __stringify(AT91_DDRSDRC_LPR
) "]\n\t"
143 /* Switch the master clock source to slow clock. */
144 "1: ldr r6, [%4, %5]\n\t"
145 " bic r6, r6, #" __stringify(AT91_PMC_CSS
) "\n\t"
146 " str r6, [%4, %5]\n\t"
147 /* Wait for clock switch. */
148 "2: ldr r6, [%4, #" __stringify(AT91_PMC_SR
) "]\n\t"
149 " tst r6, #" __stringify(AT91_PMC_MCKRDY
) "\n\t"
153 " str %3, [%2, #" __stringify(AT91_SHDW_CR
) "]\n\t"
157 : "r" (at91_shdwc
->mpddrc_base
),
158 "r" cpu_to_le32(AT91_DDRSDRC_LPDDR2_PWOFF
),
159 "r" (at91_shdwc
->shdwc_base
),
160 "r" cpu_to_le32(AT91_SHDW_KEY
| AT91_SHDW_SHDW
),
161 "r" (at91_shdwc
->pmc_base
),
162 "r" (at91_shdwc
->rcfg
->pmc
.mckr
)
166 static u32
at91_shdwc_debouncer_value(struct platform_device
*pdev
,
170 int max_idx
= ARRAY_SIZE(sdwc_dbc_period
) - 1;
171 unsigned long long period_us
;
172 unsigned long long max_period_us
= DBC_PERIOD_US(sdwc_dbc_period
[max_idx
]);
174 if (in_period_us
> max_period_us
) {
176 "debouncer period %u too big, reduced to %llu us\n",
177 in_period_us
, max_period_us
);
181 for (i
= max_idx
- 1; i
> 0; i
--) {
182 period_us
= DBC_PERIOD_US(sdwc_dbc_period
[i
]);
183 dev_dbg(&pdev
->dev
, "%s: ref[%d] = %llu\n",
184 __func__
, i
, period_us
);
185 if (in_period_us
> period_us
)
192 static u32
at91_shdwc_get_wakeup_input(struct platform_device
*pdev
,
193 struct device_node
*np
)
195 struct device_node
*cnp
;
200 for_each_child_of_node(np
, cnp
) {
201 if (of_property_read_u32(cnp
, "reg", &wk_input
)) {
202 dev_warn(&pdev
->dev
, "reg property is missing for %pOF\n",
207 wk_input_mask
= 1 << wk_input
;
208 if (!(wk_input_mask
& AT91_SHDW_WKUPEN_MASK
)) {
210 "wake-up input %d out of bounds ignore\n",
214 wuir
|= wk_input_mask
;
216 if (of_property_read_bool(cnp
, "atmel,wakeup-active-high"))
217 wuir
|= AT91_SHDW_WKUPT(wk_input
);
219 dev_dbg(&pdev
->dev
, "%s: (child %d) wuir = %#x\n",
220 __func__
, wk_input
, wuir
);
226 static void at91_shdwc_dt_configure(struct platform_device
*pdev
)
228 struct shdwc
*shdw
= platform_get_drvdata(pdev
);
229 const struct reg_config
*rcfg
= shdw
->rcfg
;
230 struct device_node
*np
= pdev
->dev
.of_node
;
231 u32 mode
= 0, tmp
, input
;
234 dev_err(&pdev
->dev
, "device node not found\n");
238 if (!of_property_read_u32(np
, "debounce-delay-us", &tmp
))
239 mode
|= AT91_SHDW_WKUPDBC(at91_shdwc_debouncer_value(pdev
, tmp
));
241 if (of_property_read_bool(np
, "atmel,wakeup-rtc-timer"))
242 mode
|= SHDW_RTCWKEN(&rcfg
->shdwc
);
244 if (of_property_read_bool(np
, "atmel,wakeup-rtt-timer"))
245 mode
|= SHDW_RTTWKEN(&rcfg
->shdwc
);
247 dev_dbg(&pdev
->dev
, "%s: mode = %#x\n", __func__
, mode
);
248 writel(mode
, shdw
->shdwc_base
+ AT91_SHDW_MR
);
250 input
= at91_shdwc_get_wakeup_input(pdev
, np
);
251 writel(input
, shdw
->shdwc_base
+ AT91_SHDW_WUIR
);
254 static const struct reg_config sama5d2_reg_config
= {
257 .mr_rtcwk_shift
= 17,
258 .mr_rttwk_shift
= SHDW_CFG_NOT_USED
,
260 .sr_rttwk_shift
= SHDW_CFG_NOT_USED
,
267 static const struct reg_config sam9x60_reg_config
= {
270 .mr_rtcwk_shift
= 17,
271 .mr_rttwk_shift
= 16,
280 static const struct of_device_id at91_shdwc_of_match
[] = {
282 .compatible
= "atmel,sama5d2-shdwc",
283 .data
= &sama5d2_reg_config
,
286 .compatible
= "microchip,sam9x60-shdwc",
287 .data
= &sam9x60_reg_config
,
292 MODULE_DEVICE_TABLE(of
, at91_shdwc_of_match
);
294 static const struct of_device_id at91_pmc_ids
[] = {
295 { .compatible
= "atmel,sama5d2-pmc" },
296 { .compatible
= "microchip,sam9x60-pmc" },
300 static int __init
at91_shdwc_probe(struct platform_device
*pdev
)
302 struct resource
*res
;
303 const struct of_device_id
*match
;
304 struct device_node
*np
;
308 if (!pdev
->dev
.of_node
)
314 at91_shdwc
= devm_kzalloc(&pdev
->dev
, sizeof(*at91_shdwc
), GFP_KERNEL
);
318 platform_set_drvdata(pdev
, at91_shdwc
);
320 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
321 at91_shdwc
->shdwc_base
= devm_ioremap_resource(&pdev
->dev
, res
);
322 if (IS_ERR(at91_shdwc
->shdwc_base
)) {
323 dev_err(&pdev
->dev
, "Could not map reset controller address\n");
324 return PTR_ERR(at91_shdwc
->shdwc_base
);
327 match
= of_match_node(at91_shdwc_of_match
, pdev
->dev
.of_node
);
328 at91_shdwc
->rcfg
= match
->data
;
330 at91_shdwc
->sclk
= devm_clk_get(&pdev
->dev
, NULL
);
331 if (IS_ERR(at91_shdwc
->sclk
))
332 return PTR_ERR(at91_shdwc
->sclk
);
334 ret
= clk_prepare_enable(at91_shdwc
->sclk
);
336 dev_err(&pdev
->dev
, "Could not enable slow clock\n");
340 at91_wakeup_status(pdev
);
342 at91_shdwc_dt_configure(pdev
);
344 np
= of_find_matching_node(NULL
, at91_pmc_ids
);
350 at91_shdwc
->pmc_base
= of_iomap(np
, 0);
353 if (!at91_shdwc
->pmc_base
) {
358 np
= of_find_compatible_node(NULL
, NULL
, "atmel,sama5d3-ddramc");
364 at91_shdwc
->mpddrc_base
= of_iomap(np
, 0);
367 if (!at91_shdwc
->mpddrc_base
) {
372 pm_power_off
= at91_poweroff
;
374 ddr_type
= readl(at91_shdwc
->mpddrc_base
+ AT91_DDRSDRC_MDR
) &
376 if (ddr_type
!= AT91_DDRSDRC_MD_LPDDR2
&&
377 ddr_type
!= AT91_DDRSDRC_MD_LPDDR3
) {
378 iounmap(at91_shdwc
->mpddrc_base
);
379 at91_shdwc
->mpddrc_base
= NULL
;
385 iounmap(at91_shdwc
->pmc_base
);
387 clk_disable_unprepare(at91_shdwc
->sclk
);
392 static int __exit
at91_shdwc_remove(struct platform_device
*pdev
)
394 struct shdwc
*shdw
= platform_get_drvdata(pdev
);
396 if (pm_power_off
== at91_poweroff
)
399 /* Reset values to disable wake-up features */
400 writel(0, shdw
->shdwc_base
+ AT91_SHDW_MR
);
401 writel(0, shdw
->shdwc_base
+ AT91_SHDW_WUIR
);
403 if (shdw
->mpddrc_base
)
404 iounmap(shdw
->mpddrc_base
);
405 iounmap(shdw
->pmc_base
);
407 clk_disable_unprepare(shdw
->sclk
);
412 static struct platform_driver at91_shdwc_driver
= {
413 .remove
= __exit_p(at91_shdwc_remove
),
415 .name
= "at91-shdwc",
416 .of_match_table
= at91_shdwc_of_match
,
419 module_platform_driver_probe(at91_shdwc_driver
, at91_shdwc_probe
);
421 MODULE_AUTHOR("Nicolas Ferre <nicolas.ferre@atmel.com>");
422 MODULE_DESCRIPTION("Atmel shutdown controller driver");
423 MODULE_LICENSE("GPL v2");