blob: 3bf14ca78b62b830b4321af0d3e5543ef7823a56 [file] [log] [blame]
Krzysztof Kozlowski347863d2017-12-25 20:54:31 +01001// SPDX-License-Identifier: GPL-2.0
2//
3// Copyright (c) 2011-2014 Samsung Electronics Co., Ltd.
4// http://www.samsung.com
5//
Krzysztof Kozlowski45984f02020-01-04 16:20:51 +01006// Exynos - Suspend support
Krzysztof Kozlowski347863d2017-12-25 20:54:31 +01007//
8// Based on arch/arm/mach-s3c2410/pm.c
9// Copyright (c) 2006 Simtec Electronics
10// Ben Dooks <ben@simtec.co.uk>
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090011
12#include <linux/init.h>
13#include <linux/suspend.h>
14#include <linux/syscore_ops.h>
15#include <linux/cpu_pm.h>
16#include <linux/io.h>
Marc Zyngier8b283c02015-03-11 15:44:52 +000017#include <linux/irq.h>
Marc Zyngier0cc09e82015-10-16 15:21:10 +010018#include <linux/irqchip.h>
Marc Zyngier8b283c02015-03-11 15:44:52 +000019#include <linux/irqdomain.h>
20#include <linux/of_address.h>
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090021#include <linux/err.h>
Javier Martinez Canillasc645a592014-11-13 11:14:40 +090022#include <linux/regulator/machine.h>
Pankaj Dubey2262d6e2015-12-18 09:02:11 +053023#include <linux/soc/samsung/exynos-pmu.h>
24#include <linux/soc/samsung/exynos-regs-pmu.h>
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090025
26#include <asm/cacheflush.h>
27#include <asm/hardware/cache-l2x0.h>
28#include <asm/firmware.h>
Abhilash Kesavanadc548d2014-11-07 09:20:16 +090029#include <asm/mcpm.h>
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090030#include <asm/smp_scu.h>
31#include <asm/suspend.h>
32
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090033#include "common.h"
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +010034#include "smc.h"
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090035
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090036#define REG_TABLE_END (-1U)
37
Vikas Sajjan0fdf0882014-11-07 09:17:36 +090038#define EXYNOS5420_CPU_STATE 0x28
39
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090040/**
Marc Zyngier8b283c02015-03-11 15:44:52 +000041 * struct exynos_wkup_irq - PMU IRQ to mask mapping
42 * @hwirq: Hardware IRQ signal of the PMU
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090043 * @mask: Mask in PMU wake-up mask register
44 */
45struct exynos_wkup_irq {
46 unsigned int hwirq;
47 u32 mask;
48};
49
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090050struct exynos_pm_data {
51 const struct exynos_wkup_irq *wkup_irq;
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090052 unsigned int wake_disable_mask;
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090053
54 void (*pm_prepare)(void);
Abhilash Kesavanadc548d2014-11-07 09:20:16 +090055 void (*pm_resume_prepare)(void);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090056 void (*pm_resume)(void);
57 int (*pm_suspend)(void);
58 int (*cpu_suspend)(unsigned long);
59};
60
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +020061/* Used only on Exynos542x/5800 */
62struct exynos_pm_state {
63 int cpu_state;
64 unsigned int pmu_spare3;
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +020065 void __iomem *sysram_base;
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +010066 phys_addr_t sysram_phys;
67 bool secure_firmware;
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +020068};
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090069
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +020070static const struct exynos_pm_data *pm_data __ro_after_init;
71static struct exynos_pm_state pm_state;
Vikas Sajjan0fdf0882014-11-07 09:17:36 +090072
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090073/*
74 * GIC wake-up support
75 */
76
77static u32 exynos_irqwake_intmask = 0xffffffff;
78
Chanwoo Choia4f582f2015-01-12 17:41:34 +090079static const struct exynos_wkup_irq exynos3250_wkup_irq[] = {
Marc Zyngierfe931222015-04-22 18:40:52 +010080 { 73, BIT(1) }, /* RTC alarm */
81 { 74, BIT(2) }, /* RTC tick */
Chanwoo Choia4f582f2015-01-12 17:41:34 +090082 { /* sentinel */ },
83};
84
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090085static const struct exynos_wkup_irq exynos4_wkup_irq[] = {
Marc Zyngier8b283c02015-03-11 15:44:52 +000086 { 44, BIT(1) }, /* RTC alarm */
87 { 45, BIT(2) }, /* RTC tick */
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090088 { /* sentinel */ },
89};
90
91static const struct exynos_wkup_irq exynos5250_wkup_irq[] = {
Marc Zyngier8b283c02015-03-11 15:44:52 +000092 { 43, BIT(1) }, /* RTC alarm */
93 { 44, BIT(2) }, /* RTC tick */
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +090094 { /* sentinel */ },
95};
96
Krzysztof Kozlowski2c809202018-07-23 19:53:01 +020097static u32 exynos_read_eint_wakeup_mask(void)
98{
99 return pmu_raw_readl(EXYNOS_EINT_WAKEUP_MASK);
100}
101
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900102static int exynos_irq_set_wake(struct irq_data *data, unsigned int state)
103{
104 const struct exynos_wkup_irq *wkup_irq;
105
106 if (!pm_data->wkup_irq)
107 return -ENOENT;
108 wkup_irq = pm_data->wkup_irq;
109
110 while (wkup_irq->mask) {
111 if (wkup_irq->hwirq == data->hwirq) {
112 if (!state)
113 exynos_irqwake_intmask |= wkup_irq->mask;
114 else
115 exynos_irqwake_intmask &= ~wkup_irq->mask;
116 return 0;
117 }
118 ++wkup_irq;
119 }
120
121 return -ENOENT;
122}
123
Marc Zyngier8b283c02015-03-11 15:44:52 +0000124static struct irq_chip exynos_pmu_chip = {
125 .name = "PMU",
126 .irq_eoi = irq_chip_eoi_parent,
127 .irq_mask = irq_chip_mask_parent,
128 .irq_unmask = irq_chip_unmask_parent,
129 .irq_retrigger = irq_chip_retrigger_hierarchy,
130 .irq_set_wake = exynos_irq_set_wake,
131#ifdef CONFIG_SMP
132 .irq_set_affinity = irq_chip_set_affinity_parent,
133#endif
134};
135
Marc Zyngierf833f572015-10-13 12:51:33 +0100136static int exynos_pmu_domain_translate(struct irq_domain *d,
137 struct irq_fwspec *fwspec,
138 unsigned long *hwirq,
139 unsigned int *type)
Marc Zyngier8b283c02015-03-11 15:44:52 +0000140{
Marc Zyngierf833f572015-10-13 12:51:33 +0100141 if (is_of_node(fwspec->fwnode)) {
142 if (fwspec->param_count != 3)
143 return -EINVAL;
Marc Zyngier8b283c02015-03-11 15:44:52 +0000144
Marc Zyngierf833f572015-10-13 12:51:33 +0100145 /* No PPI should point to this domain */
146 if (fwspec->param[0] != 0)
147 return -EINVAL;
148
149 *hwirq = fwspec->param[1];
150 *type = fwspec->param[2];
151 return 0;
152 }
153
154 return -EINVAL;
Marc Zyngier8b283c02015-03-11 15:44:52 +0000155}
156
157static int exynos_pmu_domain_alloc(struct irq_domain *domain,
158 unsigned int virq,
159 unsigned int nr_irqs, void *data)
160{
Marc Zyngierf833f572015-10-13 12:51:33 +0100161 struct irq_fwspec *fwspec = data;
162 struct irq_fwspec parent_fwspec;
Marc Zyngier8b283c02015-03-11 15:44:52 +0000163 irq_hw_number_t hwirq;
164 int i;
165
Marc Zyngierf833f572015-10-13 12:51:33 +0100166 if (fwspec->param_count != 3)
Marc Zyngier8b283c02015-03-11 15:44:52 +0000167 return -EINVAL; /* Not GIC compliant */
Marc Zyngierf833f572015-10-13 12:51:33 +0100168 if (fwspec->param[0] != 0)
Marc Zyngier8b283c02015-03-11 15:44:52 +0000169 return -EINVAL; /* No PPI should point to this domain */
170
Marc Zyngierf833f572015-10-13 12:51:33 +0100171 hwirq = fwspec->param[1];
Marc Zyngier8b283c02015-03-11 15:44:52 +0000172
173 for (i = 0; i < nr_irqs; i++)
174 irq_domain_set_hwirq_and_chip(domain, virq + i, hwirq + i,
175 &exynos_pmu_chip, NULL);
176
Marc Zyngierf833f572015-10-13 12:51:33 +0100177 parent_fwspec = *fwspec;
178 parent_fwspec.fwnode = domain->parent->fwnode;
179 return irq_domain_alloc_irqs_parent(domain, virq, nr_irqs,
180 &parent_fwspec);
Marc Zyngier8b283c02015-03-11 15:44:52 +0000181}
182
Krzysztof Kozlowskifc4a2cc2015-04-27 19:48:59 +0900183static const struct irq_domain_ops exynos_pmu_domain_ops = {
Marc Zyngierf833f572015-10-13 12:51:33 +0100184 .translate = exynos_pmu_domain_translate,
185 .alloc = exynos_pmu_domain_alloc,
186 .free = irq_domain_free_irqs_common,
Marc Zyngier8b283c02015-03-11 15:44:52 +0000187};
188
189static int __init exynos_pmu_irq_init(struct device_node *node,
190 struct device_node *parent)
191{
192 struct irq_domain *parent_domain, *domain;
193
194 if (!parent) {
Rob Herringa8e65e02017-07-21 14:28:32 -0500195 pr_err("%pOF: no parent, giving up\n", node);
Marc Zyngier8b283c02015-03-11 15:44:52 +0000196 return -ENODEV;
197 }
198
199 parent_domain = irq_find_host(parent);
200 if (!parent_domain) {
Rob Herringa8e65e02017-07-21 14:28:32 -0500201 pr_err("%pOF: unable to obtain parent domain\n", node);
Marc Zyngier8b283c02015-03-11 15:44:52 +0000202 return -ENXIO;
203 }
204
205 pmu_base_addr = of_iomap(node, 0);
206
207 if (!pmu_base_addr) {
Rob Herringa8e65e02017-07-21 14:28:32 -0500208 pr_err("%pOF: failed to find exynos pmu register\n", node);
Marc Zyngier8b283c02015-03-11 15:44:52 +0000209 return -ENOMEM;
210 }
211
212 domain = irq_domain_add_hierarchy(parent_domain, 0, 0,
213 node, &exynos_pmu_domain_ops,
214 NULL);
215 if (!domain) {
216 iounmap(pmu_base_addr);
Krzysztof Kozlowskicd480692018-07-24 18:48:14 +0200217 pmu_base_addr = NULL;
Marc Zyngier8b283c02015-03-11 15:44:52 +0000218 return -ENOMEM;
219 }
220
Javier Martinez Canillasb0304852016-08-21 03:27:45 -0400221 /*
222 * Clear the OF_POPULATED flag set in of_irq_init so that
223 * later the Exynos PMU platform device won't be skipped.
224 */
225 of_node_clear_flag(node, OF_POPULATED);
226
Marc Zyngier8b283c02015-03-11 15:44:52 +0000227 return 0;
228}
229
Marc Zyngier0cc09e82015-10-16 15:21:10 +0100230#define EXYNOS_PMU_IRQ(symbol, name) IRQCHIP_DECLARE(symbol, name, exynos_pmu_irq_init)
Marc Zyngier8b283c02015-03-11 15:44:52 +0000231
232EXYNOS_PMU_IRQ(exynos3250_pmu_irq, "samsung,exynos3250-pmu");
233EXYNOS_PMU_IRQ(exynos4210_pmu_irq, "samsung,exynos4210-pmu");
Marc Zyngier8b283c02015-03-11 15:44:52 +0000234EXYNOS_PMU_IRQ(exynos4412_pmu_irq, "samsung,exynos4412-pmu");
Marc Zyngier8b283c02015-03-11 15:44:52 +0000235EXYNOS_PMU_IRQ(exynos5250_pmu_irq, "samsung,exynos5250-pmu");
236EXYNOS_PMU_IRQ(exynos5420_pmu_irq, "samsung,exynos5420-pmu");
237
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900238static int exynos_cpu_do_idle(void)
239{
240 /* issue the standby signal into the pm unit. */
241 cpu_do_idle();
242
243 pr_info("Failed to suspend the system\n");
244 return 1; /* Aborting suspend */
245}
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900246static void exynos_flush_cache_all(void)
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900247{
248 flush_cache_all();
249 outer_flush_all();
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900250}
251
252static int exynos_cpu_suspend(unsigned long arg)
253{
254 exynos_flush_cache_all();
255 return exynos_cpu_do_idle();
256}
257
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900258static int exynos3250_cpu_suspend(unsigned long arg)
259{
260 flush_cache_all();
261 return exynos_cpu_do_idle();
262}
263
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900264static int exynos5420_cpu_suspend(unsigned long arg)
265{
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900266 /* MCPM works with HW CPU identifiers */
267 unsigned int mpidr = read_cpuid_mpidr();
268 unsigned int cluster = MPIDR_AFFINITY_LEVEL(mpidr, 1);
269 unsigned int cpu = MPIDR_AFFINITY_LEVEL(mpidr, 0);
270
Arnd Bergmann24d2c732019-06-19 14:55:29 +0200271 if (IS_ENABLED(CONFIG_EXYNOS_MCPM)) {
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900272 mcpm_set_entry_vector(cpu, cluster, exynos_cpu_resume);
Nicolas Pitre7895f732015-04-28 15:51:19 -0400273 mcpm_cpu_suspend();
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900274 }
275
276 pr_info("Failed to suspend the system\n");
277
278 /* return value != 0 means failure */
279 return 1;
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900280}
281
282static void exynos_pm_set_wakeup_mask(void)
283{
Krzysztof Kozlowski2c809202018-07-23 19:53:01 +0200284 /*
285 * Set wake-up mask registers
286 * EXYNOS_EINT_WAKEUP_MASK is set by pinctrl driver in late suspend.
287 */
Phong Trana55e0402019-06-25 11:03:45 +0700288 pmu_raw_writel(exynos_irqwake_intmask & ~BIT(31), S5P_WAKEUP_MASK);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900289}
290
291static void exynos_pm_enter_sleep_mode(void)
292{
293 /* Set value of power down register for sleep mode */
294 exynos_sys_powerdown_conf(SYS_SLEEP);
Krzysztof Kozlowski054e6aa2015-06-14 13:38:23 +0900295 pmu_raw_writel(EXYNOS_SLEEP_MAGIC, S5P_INFORM1);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900296}
297
298static void exynos_pm_prepare(void)
299{
Krzysztof Kozlowski6f024972015-03-11 11:13:57 +0100300 exynos_set_delayed_reset_assertion(false);
301
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900302 /* Set wake-up mask registers */
303 exynos_pm_set_wakeup_mask();
304
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900305 exynos_pm_enter_sleep_mode();
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900306
307 /* ensure at least INFORM0 has the resume address */
Florian Fainelli64fc2a92017-01-15 03:59:29 +0100308 pmu_raw_writel(__pa_symbol(exynos_cpu_resume), S5P_INFORM0);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900309}
310
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900311static void exynos3250_pm_prepare(void)
312{
313 unsigned int tmp;
314
315 /* Set wake-up mask registers */
316 exynos_pm_set_wakeup_mask();
317
318 tmp = pmu_raw_readl(EXYNOS3_ARM_L2_OPTION);
319 tmp &= ~EXYNOS5_OPTION_USE_RETENTION;
320 pmu_raw_writel(tmp, EXYNOS3_ARM_L2_OPTION);
321
322 exynos_pm_enter_sleep_mode();
323
324 /* ensure at least INFORM0 has the resume address */
Florian Fainelli64fc2a92017-01-15 03:59:29 +0100325 pmu_raw_writel(__pa_symbol(exynos_cpu_resume), S5P_INFORM0);
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900326}
327
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900328static void exynos5420_pm_prepare(void)
329{
330 unsigned int tmp;
331
332 /* Set wake-up mask registers */
333 exynos_pm_set_wakeup_mask();
334
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +0200335 pm_state.pmu_spare3 = pmu_raw_readl(S5P_PMU_SPARE3);
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900336 /*
337 * The cpu state needs to be saved and restored so that the
338 * secondary CPUs will enter low power start. Though the U-Boot
339 * is setting the cpu state with low power flag, the kernel
340 * needs to restore it back in case, the primary cpu fails to
341 * suspend for any reason.
342 */
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +0200343 pm_state.cpu_state = readl_relaxed(pm_state.sysram_base +
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +0200344 EXYNOS5420_CPU_STATE);
Marek Szyprowskie7467312019-02-18 15:34:09 +0100345 writel_relaxed(0x0, pm_state.sysram_base + EXYNOS5420_CPU_STATE);
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +0100346 if (pm_state.secure_firmware)
347 exynos_smc(SMC_CMD_REG, SMC_REG_ID_SFR_W(pm_state.sysram_phys +
348 EXYNOS5420_CPU_STATE),
349 0, 0);
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900350
351 exynos_pm_enter_sleep_mode();
352
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900353 /* ensure at least INFORM0 has the resume address */
Arnd Bergmann24d2c732019-06-19 14:55:29 +0200354 if (IS_ENABLED(CONFIG_EXYNOS_MCPM))
Florian Fainelli64fc2a92017-01-15 03:59:29 +0100355 pmu_raw_writel(__pa_symbol(mcpm_entry_point), S5P_INFORM0);
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900356
Krzysztof Kozlowskiee55ae62017-01-25 21:09:44 +0200357 tmp = pmu_raw_readl(EXYNOS_L2_OPTION(0));
358 tmp &= ~EXYNOS_L2_USE_RETENTION;
359 pmu_raw_writel(tmp, EXYNOS_L2_OPTION(0));
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900360
361 tmp = pmu_raw_readl(EXYNOS5420_SFR_AXI_CGDIS1);
362 tmp |= EXYNOS5420_UFS;
363 pmu_raw_writel(tmp, EXYNOS5420_SFR_AXI_CGDIS1);
364
365 tmp = pmu_raw_readl(EXYNOS5420_ARM_COMMON_OPTION);
366 tmp &= ~EXYNOS5420_L2RSTDISABLE_VALUE;
367 pmu_raw_writel(tmp, EXYNOS5420_ARM_COMMON_OPTION);
368
369 tmp = pmu_raw_readl(EXYNOS5420_FSYS2_OPTION);
370 tmp |= EXYNOS5420_EMULATION;
371 pmu_raw_writel(tmp, EXYNOS5420_FSYS2_OPTION);
372
373 tmp = pmu_raw_readl(EXYNOS5420_PSGEN_OPTION);
374 tmp |= EXYNOS5420_EMULATION;
375 pmu_raw_writel(tmp, EXYNOS5420_PSGEN_OPTION);
376}
377
378
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900379static int exynos_pm_suspend(void)
380{
381 exynos_pm_central_suspend();
382
Bartlomiej Zolnierkiewicz865e8b72015-01-24 14:05:50 +0900383 /* Setting SEQ_OPTION register */
384 pmu_raw_writel(S5P_USE_STANDBY_WFI0 | S5P_USE_STANDBY_WFE0,
385 S5P_CENTRAL_SEQ_OPTION);
386
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900387 if (read_cpuid_part() == ARM_CPU_PART_CORTEX_A9)
388 exynos_cpu_save_register();
389
390 return 0;
391}
392
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900393static int exynos5420_pm_suspend(void)
394{
395 u32 this_cluster;
396
397 exynos_pm_central_suspend();
398
399 /* Setting SEQ_OPTION register */
400
401 this_cluster = MPIDR_AFFINITY_LEVEL(read_cpuid_mpidr(), 1);
402 if (!this_cluster)
403 pmu_raw_writel(EXYNOS5420_ARM_USE_STANDBY_WFI0,
404 S5P_CENTRAL_SEQ_OPTION);
405 else
406 pmu_raw_writel(EXYNOS5420_KFC_USE_STANDBY_WFI0,
407 S5P_CENTRAL_SEQ_OPTION);
408 return 0;
409}
410
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900411static void exynos_pm_resume(void)
412{
413 u32 cpuid = read_cpuid_part();
414
415 if (exynos_pm_central_resume())
416 goto early_wakeup;
417
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900418 if (cpuid == ARM_CPU_PART_CORTEX_A9)
Pankaj Dubey3c337102018-05-10 13:02:54 +0200419 exynos_scu_enable();
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900420
421 if (call_firmware_op(resume) == -ENOSYS
422 && cpuid == ARM_CPU_PART_CORTEX_A9)
423 exynos_cpu_restore_register();
424
425early_wakeup:
426
427 /* Clear SLEEP mode set in INFORM1 */
428 pmu_raw_writel(0x0, S5P_INFORM1);
Krzysztof Kozlowski6f024972015-03-11 11:13:57 +0100429 exynos_set_delayed_reset_assertion(true);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900430}
431
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900432static void exynos3250_pm_resume(void)
433{
434 u32 cpuid = read_cpuid_part();
435
436 if (exynos_pm_central_resume())
437 goto early_wakeup;
438
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900439 pmu_raw_writel(S5P_USE_STANDBY_WFI_ALL, S5P_CENTRAL_SEQ_OPTION);
440
441 if (call_firmware_op(resume) == -ENOSYS
442 && cpuid == ARM_CPU_PART_CORTEX_A9)
443 exynos_cpu_restore_register();
444
445early_wakeup:
446
447 /* Clear SLEEP mode set in INFORM1 */
448 pmu_raw_writel(0x0, S5P_INFORM1);
449}
450
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900451static void exynos5420_prepare_pm_resume(void)
452{
Marek Szyprowski4d8e3e92019-02-18 15:34:12 +0100453 unsigned int mpidr, cluster;
454
455 mpidr = read_cpuid_mpidr();
456 cluster = MPIDR_AFFINITY_LEVEL(mpidr, 1);
457
Arnd Bergmann24d2c732019-06-19 14:55:29 +0200458 if (IS_ENABLED(CONFIG_EXYNOS_MCPM))
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900459 WARN_ON(mcpm_cpu_powered_up());
Marek Szyprowski4d8e3e92019-02-18 15:34:12 +0100460
461 if (IS_ENABLED(CONFIG_HW_PERF_EVENTS) && cluster != 0) {
462 /*
463 * When system is resumed on the LITTLE/KFC core (cluster 1),
464 * the DSCR is not properly updated until the power is turned
465 * on also for the cluster 0. Enable it for a while to
466 * propagate the SPNIDEN and SPIDEN signals from Secure JTAG
467 * block and avoid undefined instruction issue on CP14 reset.
468 */
469 pmu_raw_writel(S5P_CORE_LOCAL_PWR_EN,
470 EXYNOS_COMMON_CONFIGURATION(0));
471 pmu_raw_writel(0,
472 EXYNOS_COMMON_CONFIGURATION(0));
473 }
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900474}
475
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900476static void exynos5420_pm_resume(void)
477{
478 unsigned long tmp;
479
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900480 /* Restore the CPU0 low power state register */
481 tmp = pmu_raw_readl(EXYNOS5_ARM_CORE0_SYS_PWR_REG);
482 pmu_raw_writel(tmp | S5P_CORE_LOCAL_PWR_EN,
Ben Dooks17e06452016-06-21 11:20:28 +0100483 EXYNOS5_ARM_CORE0_SYS_PWR_REG);
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900484
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900485 /* Restore the sysram cpu state register */
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +0200486 writel_relaxed(pm_state.cpu_state,
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +0200487 pm_state.sysram_base + EXYNOS5420_CPU_STATE);
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +0100488 if (pm_state.secure_firmware)
489 exynos_smc(SMC_CMD_REG,
490 SMC_REG_ID_SFR_W(pm_state.sysram_phys +
491 EXYNOS5420_CPU_STATE),
492 EXYNOS_AFTR_MAGIC, 0);
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900493
494 pmu_raw_writel(EXYNOS5420_USE_STANDBY_WFI_ALL,
495 S5P_CENTRAL_SEQ_OPTION);
496
497 if (exynos_pm_central_resume())
498 goto early_wakeup;
499
Krzysztof Kozlowski687b5ae2018-07-24 18:49:44 +0200500 pmu_raw_writel(pm_state.pmu_spare3, S5P_PMU_SPARE3);
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900501
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900502early_wakeup:
503
504 tmp = pmu_raw_readl(EXYNOS5420_SFR_AXI_CGDIS1);
505 tmp &= ~EXYNOS5420_UFS;
506 pmu_raw_writel(tmp, EXYNOS5420_SFR_AXI_CGDIS1);
507
508 tmp = pmu_raw_readl(EXYNOS5420_FSYS2_OPTION);
509 tmp &= ~EXYNOS5420_EMULATION;
510 pmu_raw_writel(tmp, EXYNOS5420_FSYS2_OPTION);
511
512 tmp = pmu_raw_readl(EXYNOS5420_PSGEN_OPTION);
513 tmp &= ~EXYNOS5420_EMULATION;
514 pmu_raw_writel(tmp, EXYNOS5420_PSGEN_OPTION);
515
516 /* Clear SLEEP mode set in INFORM1 */
517 pmu_raw_writel(0x0, S5P_INFORM1);
518}
519
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900520/*
521 * Suspend Ops
522 */
523
524static int exynos_suspend_enter(suspend_state_t state)
525{
Krzysztof Kozlowski2c809202018-07-23 19:53:01 +0200526 u32 eint_wakeup_mask = exynos_read_eint_wakeup_mask();
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900527 int ret;
528
Bartlomiej Zolnierkiewiczb1658852018-11-14 16:30:58 +0100529 pr_debug("%s: suspending the system...\n", __func__);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900530
Bartlomiej Zolnierkiewiczb1658852018-11-14 16:30:58 +0100531 pr_debug("%s: wakeup masks: %08x,%08x\n", __func__,
Krzysztof Kozlowski2c809202018-07-23 19:53:01 +0200532 exynos_irqwake_intmask, eint_wakeup_mask);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900533
534 if (exynos_irqwake_intmask == -1U
Krzysztof Kozlowski2c809202018-07-23 19:53:01 +0200535 && eint_wakeup_mask == EXYNOS_EINT_WAKEUP_MASK_DISABLED) {
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900536 pr_err("%s: No wake-up sources!\n", __func__);
537 pr_err("%s: Aborting sleep\n", __func__);
538 return -EINVAL;
539 }
540
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900541 if (pm_data->pm_prepare)
542 pm_data->pm_prepare();
543 flush_cache_all();
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900544
545 ret = call_firmware_op(suspend);
546 if (ret == -ENOSYS)
547 ret = cpu_suspend(0, pm_data->cpu_suspend);
548 if (ret)
549 return ret;
550
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900551 if (pm_data->pm_resume_prepare)
552 pm_data->pm_resume_prepare();
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900553
Bartlomiej Zolnierkiewiczb1658852018-11-14 16:30:58 +0100554 pr_debug("%s: wakeup stat: %08x\n", __func__,
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900555 pmu_raw_readl(S5P_WAKEUP_STAT));
556
Bartlomiej Zolnierkiewiczb1658852018-11-14 16:30:58 +0100557 pr_debug("%s: resuming the system...\n", __func__);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900558
559 return 0;
560}
561
562static int exynos_suspend_prepare(void)
563{
Javier Martinez Canillasc645a592014-11-13 11:14:40 +0900564 int ret;
565
566 /*
567 * REVISIT: It would be better if struct platform_suspend_ops
568 * .prepare handler get the suspend_state_t as a parameter to
569 * avoid hard-coding the suspend to mem state. It's safe to do
570 * it now only because the suspend_valid_only_mem function is
571 * used as the .valid callback used to check if a given state
572 * is supported by the platform anyways.
573 */
574 ret = regulator_suspend_prepare(PM_SUSPEND_MEM);
575 if (ret) {
576 pr_err("Failed to prepare regulators for suspend (%d)\n", ret);
577 return ret;
578 }
579
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900580 return 0;
581}
582
583static void exynos_suspend_finish(void)
584{
Javier Martinez Canillasc645a592014-11-13 11:14:40 +0900585 int ret;
586
Javier Martinez Canillasc645a592014-11-13 11:14:40 +0900587 ret = regulator_suspend_finish();
588 if (ret)
589 pr_warn("Failed to resume regulators from suspend (%d)\n", ret);
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900590}
591
592static const struct platform_suspend_ops exynos_suspend_ops = {
593 .enter = exynos_suspend_enter,
594 .prepare = exynos_suspend_prepare,
595 .finish = exynos_suspend_finish,
596 .valid = suspend_valid_only_mem,
597};
598
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900599static const struct exynos_pm_data exynos3250_pm_data = {
600 .wkup_irq = exynos3250_wkup_irq,
601 .wake_disable_mask = ((0xFF << 8) | (0x1F << 1)),
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900602 .pm_suspend = exynos_pm_suspend,
603 .pm_resume = exynos3250_pm_resume,
604 .pm_prepare = exynos3250_pm_prepare,
605 .cpu_suspend = exynos3250_cpu_suspend,
606};
607
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900608static const struct exynos_pm_data exynos4_pm_data = {
609 .wkup_irq = exynos4_wkup_irq,
610 .wake_disable_mask = ((0xFF << 8) | (0x1F << 1)),
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900611 .pm_suspend = exynos_pm_suspend,
612 .pm_resume = exynos_pm_resume,
613 .pm_prepare = exynos_pm_prepare,
614 .cpu_suspend = exynos_cpu_suspend,
615};
616
617static const struct exynos_pm_data exynos5250_pm_data = {
618 .wkup_irq = exynos5250_wkup_irq,
619 .wake_disable_mask = ((0xFF << 8) | (0x1F << 1)),
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900620 .pm_suspend = exynos_pm_suspend,
621 .pm_resume = exynos_pm_resume,
622 .pm_prepare = exynos_pm_prepare,
623 .cpu_suspend = exynos_cpu_suspend,
624};
625
Krzysztof Kozlowski73838332015-03-18 02:34:37 +0900626static const struct exynos_pm_data exynos5420_pm_data = {
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900627 .wkup_irq = exynos5250_wkup_irq,
628 .wake_disable_mask = (0x7F << 7) | (0x1F << 1),
Abhilash Kesavanadc548d2014-11-07 09:20:16 +0900629 .pm_resume_prepare = exynos5420_prepare_pm_resume,
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900630 .pm_resume = exynos5420_pm_resume,
631 .pm_suspend = exynos5420_pm_suspend,
632 .pm_prepare = exynos5420_pm_prepare,
633 .cpu_suspend = exynos5420_cpu_suspend,
634};
635
Uwe Kleine-König444d2d32015-02-18 21:19:56 +0100636static const struct of_device_id exynos_pmu_of_device_ids[] __initconst = {
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900637 {
Chanwoo Choia4f582f2015-01-12 17:41:34 +0900638 .compatible = "samsung,exynos3250-pmu",
639 .data = &exynos3250_pm_data,
640 }, {
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900641 .compatible = "samsung,exynos4210-pmu",
642 .data = &exynos4_pm_data,
643 }, {
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900644 .compatible = "samsung,exynos4412-pmu",
645 .data = &exynos4_pm_data,
646 }, {
647 .compatible = "samsung,exynos5250-pmu",
648 .data = &exynos5250_pm_data,
Vikas Sajjan0fdf0882014-11-07 09:17:36 +0900649 }, {
650 .compatible = "samsung,exynos5420-pmu",
651 .data = &exynos5420_pm_data,
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900652 },
653 { /*sentinel*/ },
654};
655
656static struct syscore_ops exynos_pm_syscore_ops;
657
658void __init exynos_pm_init(void)
659{
660 const struct of_device_id *match;
Marc Zyngier8b283c02015-03-11 15:44:52 +0000661 struct device_node *np;
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900662 u32 tmp;
663
Marc Zyngier8b283c02015-03-11 15:44:52 +0000664 np = of_find_matching_node_and_match(NULL, exynos_pmu_of_device_ids, &match);
665 if (!np) {
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900666 pr_err("Failed to find PMU node\n");
667 return;
668 }
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900669
Julien Gralle5cbec62015-05-13 03:49:04 +0900670 if (WARN_ON(!of_find_property(np, "interrupt-controller", NULL))) {
Marc Zyngier8b283c02015-03-11 15:44:52 +0000671 pr_warn("Outdated DT detected, suspend/resume will NOT work\n");
Wen Yang629266b2019-03-05 19:33:54 +0800672 of_node_put(np);
Julien Gralle5cbec62015-05-13 03:49:04 +0900673 return;
674 }
Wen Yang629266b2019-03-05 19:33:54 +0800675 of_node_put(np);
Marc Zyngier8b283c02015-03-11 15:44:52 +0000676
Linus Torvaldse6c81cc2015-04-22 09:08:39 -0700677 pm_data = (const struct exynos_pm_data *) match->data;
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900678
679 /* All wakeup disable */
680 tmp = pmu_raw_readl(S5P_WAKEUP_MASK);
681 tmp |= pm_data->wake_disable_mask;
682 pmu_raw_writel(tmp, S5P_WAKEUP_MASK);
683
684 exynos_pm_syscore_ops.suspend = pm_data->pm_suspend;
685 exynos_pm_syscore_ops.resume = pm_data->pm_resume;
686
687 register_syscore_ops(&exynos_pm_syscore_ops);
688 suspend_set_ops(&exynos_suspend_ops);
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +0200689
690 /*
691 * Applicable as of now only to Exynos542x. If booted under secure
692 * firmware, the non-secure region of sysram should be used.
693 */
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +0100694 if (exynos_secure_firmware_available()) {
695 pm_state.sysram_phys = sysram_base_phys;
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +0200696 pm_state.sysram_base = sysram_ns_base_addr;
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +0100697 pm_state.secure_firmware = true;
698 } else {
Krzysztof Kozlowskie0b35c12018-07-24 18:49:46 +0200699 pm_state.sysram_base = sysram_base_addr;
Joonyoung Shim3a1f2f32019-02-18 15:34:11 +0100700 }
Bartlomiej Zolnierkiewicz0d713cf2014-09-25 18:02:45 +0900701}