blob: a02a8988a68a377f8339a8df4fdea11cbe449cb0 [file] [log] [blame]
Patrick Delaunaye0207372018-04-16 10:13:24 +02001// SPDX-License-Identifier: GPL-2.0+ OR BSD-3-Clause
2/*
3 * Copyright (C) 2018, STMicroelectronics - All Rights Reserved
4 */
5
6#include <config.h>
Patrick Delaunaye0207372018-04-16 10:13:24 +02007#include <asm/armv7.h>
Simon Glass274e0b02020-05-10 11:39:56 -06008#include <asm/cache.h>
Patrick Delaunaye0207372018-04-16 10:13:24 +02009#include <asm/gic.h>
10#include <asm/io.h>
11#include <asm/psci.h>
12#include <asm/secure.h>
Marek Vasut83ec9582022-02-25 02:15:59 +010013#include <hang.h>
Simon Glass4dcacfc2020-05-10 11:40:13 -060014#include <linux/bitops.h>
Tom Rinidec7ea02024-05-20 13:35:03 -060015#include <linux/errno.h>
Patrick Delaunaye0207372018-04-16 10:13:24 +020016
Marek Vasut83ec9582022-02-25 02:15:59 +010017/* PWR */
18#define PWR_CR3 0x0c
19#define PWR_MPUCR 0x10
Patrick Delaunaye0207372018-04-16 10:13:24 +020020
Marek Vasut83ec9582022-02-25 02:15:59 +010021#define PWR_CR3_DDRSREN BIT(10)
22#define PWR_CR3_DDRRETEN BIT(12)
Patrick Delaunaye0207372018-04-16 10:13:24 +020023
Marek Vasut83ec9582022-02-25 02:15:59 +010024#define PWR_MPUCR_PDDS BIT(0)
25#define PWR_MPUCR_CSTDBYDIS BIT(3)
26#define PWR_MPUCR_CSSF BIT(9)
Patrick Delaunaye0207372018-04-16 10:13:24 +020027
Marek Vasut83ec9582022-02-25 02:15:59 +010028/* RCC */
Marek Vasutcbf09002022-04-26 16:38:05 +020029#define RCC_MSSCKSELR 0x48
Marek Vasut83ec9582022-02-25 02:15:59 +010030#define RCC_DDRITFCR 0xd8
31
32#define RCC_DDRITFCR_DDRC1EN BIT(0)
33#define RCC_DDRITFCR_DDRC1LPEN BIT(1)
34#define RCC_DDRITFCR_DDRC2EN BIT(2)
35#define RCC_DDRITFCR_DDRC2LPEN BIT(3)
36#define RCC_DDRITFCR_DDRPHYCEN BIT(4)
37#define RCC_DDRITFCR_DDRPHYCLPEN BIT(5)
38#define RCC_DDRITFCR_DDRCAPBEN BIT(6)
39#define RCC_DDRITFCR_DDRCAPBLPEN BIT(7)
40#define RCC_DDRITFCR_AXIDCGEN BIT(8)
41#define RCC_DDRITFCR_DDRPHYCAPBEN BIT(9)
42#define RCC_DDRITFCR_DDRPHYCAPBLPEN BIT(10)
43#define RCC_DDRITFCR_DDRCKMOD_MASK GENMASK(22, 20)
44#define RCC_DDRITFCR_GSKPCTRL BIT(24)
45
46#define RCC_MP_SREQSETR 0x104
47#define RCC_MP_SREQCLRR 0x108
48
49#define RCC_MP_CIER 0x414
50#define RCC_MP_CIFR 0x418
51#define RCC_MP_CIFR_WKUPF BIT(20)
52
Marek Vasutcbf09002022-04-26 16:38:05 +020053#define RCC_MCUDIVR 0x830
54#define RCC_PLL3CR 0x880
55#define RCC_PLL4CR 0x894
56
Marek Vasut83ec9582022-02-25 02:15:59 +010057/* SYSCFG */
58#define SYSCFG_CMPCR 0x20
59#define SYSCFG_CMPCR_SW_CTRL BIT(2)
60#define SYSCFG_CMPENSETR 0x24
61#define SYSCFG_CMPENCLRR 0x28
62#define SYSCFG_CMPENR_MPUEN BIT(0)
63
64/* DDR Controller registers offsets */
65#define DDRCTRL_STAT 0x004
66#define DDRCTRL_PWRCTL 0x030
67#define DDRCTRL_PWRTMG 0x034
68#define DDRCTRL_HWLPCTL 0x038
69#define DDRCTRL_DFIMISC 0x1b0
70#define DDRCTRL_SWCTL 0x320
71#define DDRCTRL_SWSTAT 0x324
72#define DDRCTRL_PSTAT 0x3fc
73#define DDRCTRL_PCTRL_0 0x490
74#define DDRCTRL_PCTRL_1 0x540
75
76/* DDR Controller Register fields */
77#define DDRCTRL_STAT_OPERATING_MODE_MASK GENMASK(2, 0)
78#define DDRCTRL_STAT_OPERATING_MODE_NORMAL 0x1
79#define DDRCTRL_STAT_OPERATING_MODE_SR 0x3
80#define DDRCTRL_STAT_SELFREF_TYPE_MASK GENMASK(5, 4)
81#define DDRCTRL_STAT_SELFREF_TYPE_ASR (0x3 << 4)
82#define DDRCTRL_STAT_SELFREF_TYPE_SR (0x2 << 4)
83
84#define DDRCTRL_PWRCTL_SELFREF_EN BIT(0)
85#define DDRCTRL_PWRCTL_EN_DFI_DRAM_CLK_DISABLE BIT(3)
86#define DDRCTRL_PWRCTL_SELFREF_SW BIT(5)
87
88#define DDRCTRL_PWRTMG_SELFREF_TO_X32_MASK GENMASK(23, 16)
89#define DDRCTRL_PWRTMG_SELFREF_TO_X32_0 BIT(16)
90
91#define DDRCTRL_HWLPCTL_HW_LP_EN BIT(0)
92
93#define DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN BIT(0)
94
95#define DDRCTRL_SWCTL_SW_DONE BIT(0)
96
97#define DDRCTRL_SWSTAT_SW_DONE_ACK BIT(0)
98
99#define DDRCTRL_PSTAT_RD_PORT_BUSY_0 BIT(0)
100#define DDRCTRL_PSTAT_RD_PORT_BUSY_1 BIT(1)
101#define DDRCTRL_PSTAT_WR_PORT_BUSY_0 BIT(16)
102#define DDRCTRL_PSTAT_WR_PORT_BUSY_1 BIT(17)
103
104#define DDRCTRL_PCTRL_N_PORT_EN BIT(0)
105
106/* DDR PHY registers offsets */
107#define DDRPHYC_PIR 0x004
108#define DDRPHYC_PGSR 0x00c
109#define DDRPHYC_ACDLLCR 0x014
110#define DDRPHYC_ACIOCR 0x024
111#define DDRPHYC_DXCCR 0x028
112#define DDRPHYC_DSGCR 0x02c
113#define DDRPHYC_ZQ0CR0 0x180
114#define DDRPHYC_DX0DLLCR 0x1cc
115#define DDRPHYC_DX1DLLCR 0x20c
116#define DDRPHYC_DX2DLLCR 0x24c
117#define DDRPHYC_DX3DLLCR 0x28c
118
119/* DDR PHY Register fields */
120#define DDRPHYC_PIR_INIT BIT(0)
121#define DDRPHYC_PIR_DLLSRST BIT(1)
122#define DDRPHYC_PIR_DLLLOCK BIT(2)
123#define DDRPHYC_PIR_ITMSRST BIT(4)
124
125#define DDRPHYC_PGSR_IDONE BIT(0)
126
127#define DDRPHYC_ACDLLCR_DLLSRST BIT(30)
128#define DDRPHYC_ACDLLCR_DLLDIS BIT(31)
129
130#define DDRPHYC_ACIOCR_ACOE BIT(1)
131#define DDRPHYC_ACIOCR_ACPDD BIT(3)
132#define DDRPHYC_ACIOCR_ACPDR BIT(4)
133#define DDRPHYC_ACIOCR_CKPDD_MASK GENMASK(10, 8)
134#define DDRPHYC_ACIOCR_CKPDD_0 BIT(8)
135#define DDRPHYC_ACIOCR_CKPDR_MASK GENMASK(13, 11)
136#define DDRPHYC_ACIOCR_CKPDR_0 BIT(11)
137#define DDRPHYC_ACIOCR_CSPDD_MASK GENMASK(20, 18)
138#define DDRPHYC_ACIOCR_CSPDD_0 BIT(18)
139
140#define DDRPHYC_DXCCR_DXPDD BIT(2)
141#define DDRPHYC_DXCCR_DXPDR BIT(3)
142
143#define DDRPHYC_DSGCR_CKEPDD_MASK GENMASK(19, 16)
144#define DDRPHYC_DSGCR_CKEPDD_0 BIT(16)
145#define DDRPHYC_DSGCR_ODTPDD_MASK GENMASK(23, 20)
146#define DDRPHYC_DSGCR_ODTPDD_0 BIT(20)
147#define DDRPHYC_DSGCR_NL2PD BIT(24)
148#define DDRPHYC_DSGCR_CKOE BIT(28)
149
150#define DDRPHYC_ZQ0CRN_ZQPD BIT(31)
151
152#define DDRPHYC_DXNDLLCR_DLLDIS BIT(31)
153
154#define BOOT_API_A7_CORE0_MAGIC_NUMBER 0xca7face0
155#define BOOT_API_A7_CORE1_MAGIC_NUMBER 0xca7face1
156
157#define MPIDR_AFF0 GENMASK(7, 0)
158
159#define RCC_MP_GRSTCSETR (STM32_RCC_BASE + 0x0404)
160#define RCC_MP_GRSTCSETR_MPSYSRST BIT(0)
161#define RCC_MP_GRSTCSETR_MPUP0RST BIT(4)
162#define RCC_MP_GRSTCSETR_MPUP1RST BIT(5)
163
Marek Vasutc49678e2023-05-11 21:55:45 +0200164/* IWDG */
165#define IWDG_KR 0x00
166#define IWDG_KR_RELOAD_KEY 0xaaaa
167#define IWDG_EWCR 0x14
168#define IWDG_EWCR_EWIC BIT(14)
169
Marek Vasut83ec9582022-02-25 02:15:59 +0100170#define STM32MP1_PSCI_NR_CPUS 2
Patrick Delaunaye0207372018-04-16 10:13:24 +0200171#if STM32MP1_PSCI_NR_CPUS > CONFIG_ARMV7_PSCI_NR_CPUS
172#error "invalid value for CONFIG_ARMV7_PSCI_NR_CPUS"
173#endif
174
175u8 psci_state[STM32MP1_PSCI_NR_CPUS] __secure_data = {
176 PSCI_AFFINITY_LEVEL_ON,
177 PSCI_AFFINITY_LEVEL_OFF};
178
Ludovic Barre5a40c512020-03-02 11:27:02 +0100179static u32 __secure_data cntfrq;
180
181static u32 __secure cp15_read_cntfrq(void)
182{
183 u32 frq;
184
185 asm volatile("mrc p15, 0, %0, c14, c0, 0" : "=r" (frq));
186
187 return frq;
188}
189
190static void __secure cp15_write_cntfrq(u32 frq)
191{
192 asm volatile ("mcr p15, 0, %0, c14, c0, 0" : : "r" (frq));
193}
194
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200195static inline void psci_set_state(int cpu, u8 state)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200196{
197 psci_state[cpu] = state;
198 dsb();
199 isb();
200}
201
202static u32 __secure stm32mp_get_gicd_base_address(void)
203{
204 u32 periphbase;
205
206 /* get the GIC base address from the CBAR register */
207 asm("mrc p15, 4, %0, c15, c0, 0\n" : "=r" (periphbase));
208
209 return (periphbase & CBAR_MASK) + GIC_DIST_OFFSET;
210}
211
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200212static void __secure stm32mp_raise_sgi0(int cpu)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200213{
214 u32 gic_dist_addr;
215
216 gic_dist_addr = stm32mp_get_gicd_base_address();
217
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200218 /* ask cpu with SGI0 */
219 writel((BIT(cpu) << 16), gic_dist_addr + GICD_SGIR);
Patrick Delaunaye0207372018-04-16 10:13:24 +0200220}
221
222void __secure psci_arch_cpu_entry(void)
223{
224 u32 cpu = psci_get_cpu_id();
225
226 psci_set_state(cpu, PSCI_AFFINITY_LEVEL_ON);
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200227
Ludovic Barre5a40c512020-03-02 11:27:02 +0100228 /* write the saved cntfrq */
229 cp15_write_cntfrq(cntfrq);
230
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200231 /* reset magic in TAMP register */
232 writel(0xFFFFFFFF, TAMP_BACKUP_MAGIC_NUMBER);
Patrick Delaunaye0207372018-04-16 10:13:24 +0200233}
234
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200235s32 __secure psci_features(u32 function_id, u32 psci_fid)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200236{
237 switch (psci_fid) {
238 case ARM_PSCI_0_2_FN_PSCI_VERSION:
239 case ARM_PSCI_0_2_FN_CPU_OFF:
240 case ARM_PSCI_0_2_FN_CPU_ON:
241 case ARM_PSCI_0_2_FN_AFFINITY_INFO:
242 case ARM_PSCI_0_2_FN_MIGRATE_INFO_TYPE:
243 case ARM_PSCI_0_2_FN_SYSTEM_OFF:
244 case ARM_PSCI_0_2_FN_SYSTEM_RESET:
Marek Vasut83ec9582022-02-25 02:15:59 +0100245 case ARM_PSCI_1_0_FN_SYSTEM_SUSPEND:
Patrick Delaunaye0207372018-04-16 10:13:24 +0200246 return 0x0;
247 }
248 return ARM_PSCI_RET_NI;
249}
250
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200251u32 __secure psci_version(void)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200252{
253 return ARM_PSCI_VER_1_0;
254}
255
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200256s32 __secure psci_affinity_info(u32 function_id, u32 target_affinity,
Patrick Delaunaye0207372018-04-16 10:13:24 +0200257 u32 lowest_affinity_level)
258{
259 u32 cpu = target_affinity & MPIDR_AFF0;
260
261 if (lowest_affinity_level > 0)
262 return ARM_PSCI_RET_INVAL;
263
264 if (target_affinity & ~MPIDR_AFF0)
265 return ARM_PSCI_RET_INVAL;
266
267 if (cpu >= STM32MP1_PSCI_NR_CPUS)
268 return ARM_PSCI_RET_INVAL;
269
270 return psci_state[cpu];
271}
272
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200273u32 __secure psci_migrate_info_type(void)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200274{
Patrick Delaunaydde32d62019-02-27 17:01:16 +0100275 /*
276 * in Power_State_Coordination_Interface_PDD_v1_1_DEN0022D.pdf
277 * return 2 = Trusted OS is either not present or does not require
278 * migration, system of this type does not require the caller
279 * to use the MIGRATE function.
280 * MIGRATE function calls return NOT_SUPPORTED.
281 */
Patrick Delaunaye0207372018-04-16 10:13:24 +0200282 return 2;
283}
284
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200285s32 __secure psci_cpu_on(u32 function_id, u32 target_cpu, u32 pc,
Patrick Delaunaye0207372018-04-16 10:13:24 +0200286 u32 context_id)
287{
288 u32 cpu = target_cpu & MPIDR_AFF0;
289
290 if (target_cpu & ~MPIDR_AFF0)
291 return ARM_PSCI_RET_INVAL;
292
293 if (cpu >= STM32MP1_PSCI_NR_CPUS)
294 return ARM_PSCI_RET_INVAL;
295
296 if (psci_state[cpu] == PSCI_AFFINITY_LEVEL_ON)
297 return ARM_PSCI_RET_ALREADY_ON;
298
Ludovic Barre5a40c512020-03-02 11:27:02 +0100299 /* read and save cntfrq of current cpu to write on target cpu */
300 cntfrq = cp15_read_cntfrq();
301
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200302 /* reset magic in TAMP register */
303 if (readl(TAMP_BACKUP_MAGIC_NUMBER))
304 writel(0xFFFFFFFF, TAMP_BACKUP_MAGIC_NUMBER);
305 /*
306 * ROM code need a first SGI0 after core reset
307 * core is ready when magic is set to 0 in ROM code
308 */
309 while (readl(TAMP_BACKUP_MAGIC_NUMBER))
310 stm32mp_raise_sgi0(cpu);
311
Patrick Delaunaye0207372018-04-16 10:13:24 +0200312 /* store target PC and context id*/
313 psci_save(cpu, pc, context_id);
314
315 /* write entrypoint in backup RAM register */
316 writel((u32)&psci_cpu_entry, TAMP_BACKUP_BRANCH_ADDRESS);
317 psci_set_state(cpu, PSCI_AFFINITY_LEVEL_ON_PENDING);
318
319 /* write magic number in backup register */
320 if (cpu == 0x01)
321 writel(BOOT_API_A7_CORE1_MAGIC_NUMBER,
322 TAMP_BACKUP_MAGIC_NUMBER);
323 else
324 writel(BOOT_API_A7_CORE0_MAGIC_NUMBER,
325 TAMP_BACKUP_MAGIC_NUMBER);
326
Patrick Delaunayad1bc0a2019-04-18 17:32:40 +0200327 /* Generate an IT to start the core */
328 stm32mp_raise_sgi0(cpu);
Patrick Delaunaye0207372018-04-16 10:13:24 +0200329
330 return ARM_PSCI_RET_SUCCESS;
331}
332
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200333s32 __secure psci_cpu_off(void)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200334{
335 u32 cpu;
336
337 cpu = psci_get_cpu_id();
338
339 psci_cpu_off_common();
340 psci_set_state(cpu, PSCI_AFFINITY_LEVEL_OFF);
341
342 /* reset core: wfi is managed by BootRom */
343 if (cpu == 0x01)
344 writel(RCC_MP_GRSTCSETR_MPUP1RST, RCC_MP_GRSTCSETR);
345 else
346 writel(RCC_MP_GRSTCSETR_MPUP0RST, RCC_MP_GRSTCSETR);
347
348 /* just waiting reset */
349 while (1)
350 wfi();
351}
352
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200353void __secure psci_system_reset(void)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200354{
355 /* System reset */
356 writel(RCC_MP_GRSTCSETR_MPSYSRST, RCC_MP_GRSTCSETR);
357 /* just waiting reset */
358 while (1)
359 wfi();
360}
361
Patrick Delaunay9c59d862019-07-22 14:19:20 +0200362void __secure psci_system_off(void)
Patrick Delaunaye0207372018-04-16 10:13:24 +0200363{
364 /* System Off is not managed, waiting user power off
365 * TODO: handle I2C write in PMIC Main Control register bit 0 = SWOFF
366 */
367 while (1)
368 wfi();
369}
Marek Vasut83ec9582022-02-25 02:15:59 +0100370
371static void __secure secure_udelay(unsigned int delay)
372{
373 u32 freq = cp15_read_cntfrq() / 1000000;
374 u64 start, end;
375
376 delay *= freq;
377
378 asm volatile("mrrc p15, 0, %Q0, %R0, c14" : "=r" (start));
379 for (;;) {
380 asm volatile("mrrc p15, 0, %Q0, %R0, c14" : "=r" (end));
381 if ((end - start) > delay)
382 break;
383 }
384}
385
386static int __secure secure_waitbits(u32 reg, u32 mask, u32 val)
387{
388 u32 freq = cp15_read_cntfrq() / 1000000;
389 u32 delay = 500 * freq; /* 500 us */
390 u64 start, end;
391 u32 tmp;
392
393 asm volatile("mrrc p15, 0, %Q0, %R0, c14" : "=r" (start));
394 for (;;) {
395 tmp = readl(reg);
Marek Vasutbb2c3872024-07-08 13:43:23 +0200396 if ((tmp & mask) == val)
Marek Vasut83ec9582022-02-25 02:15:59 +0100397 return 0;
398 asm volatile("mrrc p15, 0, %Q0, %R0, c14" : "=r" (end));
399 if ((end - start) > delay)
400 return -ETIMEDOUT;
401 }
402}
403
404static void __secure ddr_sr_mode_ssr(u32 *saved_pwrctl)
405{
406 setbits_le32(STM32_RCC_BASE + RCC_DDRITFCR,
407 RCC_DDRITFCR_DDRC1LPEN | RCC_DDRITFCR_DDRC1EN |
408 RCC_DDRITFCR_DDRC2LPEN | RCC_DDRITFCR_DDRC2EN |
409 RCC_DDRITFCR_DDRCAPBLPEN | RCC_DDRITFCR_DDRPHYCAPBLPEN |
410 RCC_DDRITFCR_DDRCAPBEN | RCC_DDRITFCR_DDRPHYCAPBEN |
411 RCC_DDRITFCR_DDRPHYCEN);
412
413 clrbits_le32(STM32_RCC_BASE + RCC_DDRITFCR,
414 RCC_DDRITFCR_AXIDCGEN | RCC_DDRITFCR_DDRCKMOD_MASK);
415
416 /* Disable HW LP interface of uMCTL2 */
417 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_HWLPCTL,
418 DDRCTRL_HWLPCTL_HW_LP_EN);
419
420 /* Configure Automatic LP modes of uMCTL2 */
421 clrsetbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRTMG,
422 DDRCTRL_PWRTMG_SELFREF_TO_X32_MASK,
423 DDRCTRL_PWRTMG_SELFREF_TO_X32_0);
424
425 /* Save PWRCTL register to restart ASR after suspend (if applicable) */
426 *saved_pwrctl = readl(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL);
427
428 /*
429 * Disable Clock disable with LP modes
430 * (used in RUN mode for LPDDR2 with specific timing).
431 */
432 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL,
433 DDRCTRL_PWRCTL_EN_DFI_DRAM_CLK_DISABLE);
434
435 /* Disable automatic Self-Refresh mode */
436 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL,
437 DDRCTRL_PWRCTL_SELFREF_EN);
438}
439
440static void __secure ddr_sr_mode_restore(u32 saved_pwrctl)
441{
442 saved_pwrctl &= DDRCTRL_PWRCTL_EN_DFI_DRAM_CLK_DISABLE |
443 DDRCTRL_PWRCTL_SELFREF_EN;
444
445 /* Restore ASR mode in case it was enabled before suspend. */
446 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL, saved_pwrctl);
447}
448
449static int __secure ddr_sw_self_refresh_in(void)
450{
451 int ret;
452
453 clrbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_AXIDCGEN);
454
455 /* Blocks AXI ports from taking anymore transactions */
456 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_0,
457 DDRCTRL_PCTRL_N_PORT_EN);
458 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_1,
459 DDRCTRL_PCTRL_N_PORT_EN);
460
461 /*
462 * Waits unit all AXI ports are idle
463 * Poll PSTAT.rd_port_busy_n = 0
464 * Poll PSTAT.wr_port_busy_n = 0
465 */
466 ret = secure_waitbits(STM32_DDRCTRL_BASE + DDRCTRL_PSTAT,
467 DDRCTRL_PSTAT_RD_PORT_BUSY_0 |
468 DDRCTRL_PSTAT_RD_PORT_BUSY_1 |
469 DDRCTRL_PSTAT_WR_PORT_BUSY_0 |
470 DDRCTRL_PSTAT_WR_PORT_BUSY_1, 0);
471 if (ret)
472 goto pstat_failed;
473
474 /* SW Self-Refresh entry */
475 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL, DDRCTRL_PWRCTL_SELFREF_SW);
476
477 /*
478 * Wait operating mode change in self-refresh mode
479 * with STAT.operating_mode[1:0]==11.
480 * Ensure transition to self-refresh was due to software
481 * by checking also that STAT.selfref_type[1:0]=2.
482 */
483 ret = secure_waitbits(STM32_DDRCTRL_BASE + DDRCTRL_STAT,
484 DDRCTRL_STAT_OPERATING_MODE_MASK |
485 DDRCTRL_STAT_SELFREF_TYPE_MASK,
486 DDRCTRL_STAT_OPERATING_MODE_SR |
487 DDRCTRL_STAT_SELFREF_TYPE_SR);
488 if (ret)
489 goto selfref_sw_failed;
490
491 /* IOs powering down (PUBL registers) */
492 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_ACPDD);
493 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_ACPDR);
494
495 clrsetbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR,
496 DDRPHYC_ACIOCR_CKPDD_MASK,
497 DDRPHYC_ACIOCR_CKPDD_0);
498
499 clrsetbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR,
500 DDRPHYC_ACIOCR_CKPDR_MASK,
501 DDRPHYC_ACIOCR_CKPDR_0);
502
503 clrsetbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR,
504 DDRPHYC_ACIOCR_CSPDD_MASK,
505 DDRPHYC_ACIOCR_CSPDD_0);
506
507 /* Disable command/address output driver */
508 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_ACOE);
509
510 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DXCCR, DDRPHYC_DXCCR_DXPDD);
511
512 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DXCCR, DDRPHYC_DXCCR_DXPDR);
513
514 clrsetbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR,
515 DDRPHYC_DSGCR_ODTPDD_MASK,
516 DDRPHYC_DSGCR_ODTPDD_0);
517
518 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_NL2PD);
519
520 clrsetbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR,
521 DDRPHYC_DSGCR_CKEPDD_MASK,
522 DDRPHYC_DSGCR_CKEPDD_0);
523
524 /* Disable PZQ cell (PUBL register) */
525 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ZQ0CR0, DDRPHYC_ZQ0CRN_ZQPD);
526
527 /* Set latch */
528 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_CKOE);
529
530 /* Additional delay to avoid early latch */
531 secure_udelay(10);
532
533 /* Activate sw retention in PWRCTRL */
534 setbits_le32(STM32_PWR_BASE + PWR_CR3, PWR_CR3_DDRRETEN);
535
536 /* Switch controller clocks (uMCTL2/PUBL) to DLL ref clock */
537 setbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_GSKPCTRL);
538
539 /* Disable all DLLs: GLITCH window */
540 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACDLLCR, DDRPHYC_ACDLLCR_DLLDIS);
541
542 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX0DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
543
544 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX1DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
545
546 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX2DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
547
548 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX3DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
549
550 /* Switch controller clocks (uMCTL2/PUBL) to DLL output clock */
551 clrbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_GSKPCTRL);
552
553 /* Deactivate all DDR clocks */
554 clrbits_le32(STM32_RCC_BASE + RCC_DDRITFCR,
555 RCC_DDRITFCR_DDRC1EN | RCC_DDRITFCR_DDRC2EN |
556 RCC_DDRITFCR_DDRCAPBEN | RCC_DDRITFCR_DDRPHYCAPBEN);
557
558 return 0;
559
560selfref_sw_failed:
561 /* This bit should be cleared to restore DDR in its previous state */
562 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL,
563 DDRCTRL_PWRCTL_SELFREF_SW);
564
565pstat_failed:
566 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_0,
567 DDRCTRL_PCTRL_N_PORT_EN);
568 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_1,
569 DDRCTRL_PCTRL_N_PORT_EN);
570
571 return -EINVAL;
572};
573
574static void __secure ddr_sw_self_refresh_exit(void)
575{
576 int ret;
577
578 /* Enable all clocks */
579 setbits_le32(STM32_RCC_BASE + RCC_DDRITFCR,
580 RCC_DDRITFCR_DDRC1EN | RCC_DDRITFCR_DDRC2EN |
581 RCC_DDRITFCR_DDRPHYCEN | RCC_DDRITFCR_DDRPHYCAPBEN |
582 RCC_DDRITFCR_DDRCAPBEN);
583
584 /* Handshake */
585 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_SWCTL, DDRCTRL_SWCTL_SW_DONE);
586
587 /* Mask dfi_init_complete_en */
588 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_DFIMISC,
589 DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
590
591 /* Ack */
592 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_SWCTL, DDRCTRL_SWCTL_SW_DONE);
593 ret = secure_waitbits(STM32_DDRCTRL_BASE + DDRCTRL_SWSTAT,
594 DDRCTRL_SWSTAT_SW_DONE_ACK,
595 DDRCTRL_SWSTAT_SW_DONE_ACK);
596 if (ret)
597 hang();
598
599 /* Switch controller clocks (uMCTL2/PUBL) to DLL ref clock */
600 setbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_GSKPCTRL);
601
602 /* Enable all DLLs: GLITCH window */
603 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACDLLCR,
604 DDRPHYC_ACDLLCR_DLLDIS);
605
606 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX0DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
607
608 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX1DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
609
610 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX2DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
611
612 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DX3DLLCR, DDRPHYC_DXNDLLCR_DLLDIS);
613
614 /* Additional delay to avoid early DLL clock switch */
615 secure_udelay(50);
616
617 /* Switch controller clocks (uMCTL2/PUBL) to DLL ref clock */
618 clrbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_GSKPCTRL);
619
620 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACDLLCR, DDRPHYC_ACDLLCR_DLLSRST);
621
622 secure_udelay(10);
623
624 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACDLLCR, DDRPHYC_ACDLLCR_DLLSRST);
625
626 /* PHY partial init: (DLL lock and ITM reset) */
627 writel(DDRPHYC_PIR_DLLSRST | DDRPHYC_PIR_DLLLOCK |
628 DDRPHYC_PIR_ITMSRST | DDRPHYC_PIR_INIT,
629 STM32_DDRPHYC_BASE + DDRPHYC_PIR);
630
631 /* Need to wait at least 10 clock cycles before accessing PGSR */
632 secure_udelay(1);
633
634 /* Pool end of init */
635 ret = secure_waitbits(STM32_DDRPHYC_BASE + DDRPHYC_PGSR,
636 DDRPHYC_PGSR_IDONE, DDRPHYC_PGSR_IDONE);
637 if (ret)
638 hang();
639
640 /* Handshake */
641 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_SWCTL, DDRCTRL_SWCTL_SW_DONE);
642
643 /* Unmask dfi_init_complete_en to uMCTL2 */
644 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_DFIMISC, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
645
646 /* Ack */
647 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_SWCTL, DDRCTRL_SWCTL_SW_DONE);
648 ret = secure_waitbits(STM32_DDRCTRL_BASE + DDRCTRL_SWSTAT,
649 DDRCTRL_SWSTAT_SW_DONE_ACK,
650 DDRCTRL_SWSTAT_SW_DONE_ACK);
651 if (ret)
652 hang();
653
654 /* Deactivate sw retention in PWR */
655 clrbits_le32(STM32_PWR_BASE + PWR_CR3, PWR_CR3_DDRRETEN);
656
657 /* Enable PZQ cell (PUBL register) */
658 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ZQ0CR0, DDRPHYC_ZQ0CRN_ZQPD);
659
660 /* Enable pad drivers */
661 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_ACPDD);
662
663 /* Enable command/address output driver */
664 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_ACOE);
665
666 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_CKPDD_MASK);
667
668 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_ACIOCR, DDRPHYC_ACIOCR_CSPDD_MASK);
669
670 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DXCCR, DDRPHYC_DXCCR_DXPDD);
671
672 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DXCCR, DDRPHYC_DXCCR_DXPDR);
673
674 /* Release latch */
675 setbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_CKOE);
676
677 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_ODTPDD_MASK);
678
679 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_NL2PD);
680
681 clrbits_le32(STM32_DDRPHYC_BASE + DDRPHYC_DSGCR, DDRPHYC_DSGCR_CKEPDD_MASK);
682
683 /* Remove selfrefresh */
684 clrbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PWRCTL, DDRCTRL_PWRCTL_SELFREF_SW);
685
686 /* Wait operating_mode == normal */
687 ret = secure_waitbits(STM32_DDRCTRL_BASE + DDRCTRL_STAT,
688 DDRCTRL_STAT_OPERATING_MODE_MASK,
689 DDRCTRL_STAT_OPERATING_MODE_NORMAL);
690 if (ret)
691 hang();
692
693 /* AXI ports are no longer blocked from taking transactions */
694 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_0, DDRCTRL_PCTRL_N_PORT_EN);
695 setbits_le32(STM32_DDRCTRL_BASE + DDRCTRL_PCTRL_1, DDRCTRL_PCTRL_N_PORT_EN);
696
697 setbits_le32(STM32_RCC_BASE + RCC_DDRITFCR, RCC_DDRITFCR_AXIDCGEN);
698}
699
700void __secure psci_system_suspend(u32 __always_unused function_id,
701 u32 ep, u32 context_id)
702{
Marek Vasutcbf09002022-04-26 16:38:05 +0200703 u32 saved_mcudivr, saved_pll3cr, saved_pll4cr, saved_mssckselr;
Marek Vasutc49678e2023-05-11 21:55:45 +0200704 u32 gicd_addr = stm32mp_get_gicd_base_address();
Marek Vasut4f9de152024-04-07 22:21:07 +0200705 u32 cpu = psci_get_cpu_id();
706 u32 sp = (u32)__secure_stack_end - (cpu << ARM_PSCI_STACK_SHIFT);
Marek Vasutc49678e2023-05-11 21:55:45 +0200707 bool iwdg1_wake = false;
708 bool iwdg2_wake = false;
709 bool other_wake = false;
Marek Vasut83ec9582022-02-25 02:15:59 +0100710 u32 saved_pwrctl, reg;
Marek Vasutc49678e2023-05-11 21:55:45 +0200711 u32 gic_enabled[8];
712 u32 irqs;
713 int i;
714
715 /* Cache enable mask of all 256 SPI */
716 for (i = 0; i < ARRAY_SIZE(gic_enabled); i++)
717 gic_enabled[i] = readl(gicd_addr + GICD_ISENABLERn + 0x4 + 4 * i);
Marek Vasut83ec9582022-02-25 02:15:59 +0100718
719 /* Disable IO compensation */
720
721 /* Place current APSRC/ANSRC into RAPSRC/RANSRC */
722 reg = readl(STM32_SYSCFG_BASE + SYSCFG_CMPCR);
723 reg >>= 8;
724 reg &= 0xff << 16;
725 reg |= SYSCFG_CMPCR_SW_CTRL;
726 writel(reg, STM32_SYSCFG_BASE + SYSCFG_CMPCR);
727 writel(SYSCFG_CMPENR_MPUEN, STM32_SYSCFG_BASE + SYSCFG_CMPENCLRR);
728
729 writel(RCC_MP_CIFR_WKUPF, STM32_RCC_BASE + RCC_MP_CIFR);
730 setbits_le32(STM32_RCC_BASE + RCC_MP_CIER, RCC_MP_CIFR_WKUPF);
731
732 setbits_le32(STM32_PWR_BASE + PWR_MPUCR,
Marek Vasut30294422023-07-06 23:32:27 +0200733 PWR_MPUCR_CSSF | PWR_MPUCR_CSTDBYDIS);
Marek Vasut83ec9582022-02-25 02:15:59 +0100734
Marek Vasutcbf09002022-04-26 16:38:05 +0200735 saved_mcudivr = readl(STM32_RCC_BASE + RCC_MCUDIVR);
736 saved_pll3cr = readl(STM32_RCC_BASE + RCC_PLL3CR);
737 saved_pll4cr = readl(STM32_RCC_BASE + RCC_PLL4CR);
738 saved_mssckselr = readl(STM32_RCC_BASE + RCC_MSSCKSELR);
739
Marek Vasut83ec9582022-02-25 02:15:59 +0100740 psci_v7_flush_dcache_all();
741 ddr_sr_mode_ssr(&saved_pwrctl);
742 ddr_sw_self_refresh_in();
743 setbits_le32(STM32_PWR_BASE + PWR_CR3, PWR_CR3_DDRSREN);
744 writel(0x3, STM32_RCC_BASE + RCC_MP_SREQSETR);
745
Marek Vasutc49678e2023-05-11 21:55:45 +0200746 /* Ping the IWDG before entering suspend */
747 iwdg1_wake = !!(gic_enabled[4] & BIT(22)); /* SPI 150 */
748 iwdg2_wake = !!(gic_enabled[4] & BIT(23)); /* SPI 151 */
749
750 for (;;) {
751 /* Ping IWDG1 and ACK pretimer IRQ */
752 if (iwdg1_wake) {
753 writel(IWDG_KR_RELOAD_KEY, STM32_IWDG1_BASE + IWDG_KR);
754 writel(IWDG_EWCR_EWIC, STM32_IWDG1_BASE + IWDG_EWCR);
755 }
756
757 /* Ping IWDG2 and ACK pretimer IRQ */
758 if (iwdg2_wake) {
759 writel(IWDG_KR_RELOAD_KEY, STM32_IWDG2_BASE + IWDG_KR);
760 writel(IWDG_EWCR_EWIC, STM32_IWDG2_BASE + IWDG_EWCR);
761 }
762
763 iwdg1_wake = false;
764 iwdg2_wake = false;
765
766 /* Zzz, enter stop mode */
767 asm volatile(
768 "isb\n"
769 "dsb\n"
770 "wfi\n");
771
772 /* Determine the wake up source */
773 for (i = 0; i < ARRAY_SIZE(gic_enabled); i++) {
774 irqs = readl(gicd_addr + GICR_IGROUPMODRn + 0x4 + 4 * i);
775 irqs &= gic_enabled[i];
776 if (!irqs)
777 continue;
778
779 /* Test whether IWDG pretimeout triggered the wake up. */
780 if (i == 4) { /* SPI Num 128..159 */
781 iwdg1_wake = !!(irqs & BIT(22)); /* SPI 150 */
782 iwdg2_wake = !!(irqs & BIT(23)); /* SPI 151 */
783 irqs &= ~(BIT(22) | BIT(23));
784 }
785
786 /* Test whether there is any other wake up trigger. */
787 if (irqs) {
788 other_wake = true;
789 break;
790 }
791 }
792
793 /* Other wake up triggers pending, let OS deal with all of it. */
794 if (other_wake)
795 break;
796 }
Marek Vasut83ec9582022-02-25 02:15:59 +0100797
798 writel(0x3, STM32_RCC_BASE + RCC_MP_SREQCLRR);
799 ddr_sw_self_refresh_exit();
800 ddr_sr_mode_restore(saved_pwrctl);
801
Marek Vasutcbf09002022-04-26 16:38:05 +0200802 writel(saved_mcudivr, STM32_RCC_BASE + RCC_MCUDIVR);
803 writel(saved_pll3cr, STM32_RCC_BASE + RCC_PLL3CR);
804 writel(saved_pll4cr, STM32_RCC_BASE + RCC_PLL4CR);
805 writel(saved_mssckselr, STM32_RCC_BASE + RCC_MSSCKSELR);
806
Marek Vasut83ec9582022-02-25 02:15:59 +0100807 writel(SYSCFG_CMPENR_MPUEN, STM32_SYSCFG_BASE + SYSCFG_CMPENSETR);
808 clrbits_le32(STM32_SYSCFG_BASE + SYSCFG_CMPCR, SYSCFG_CMPCR_SW_CTRL);
Marek Vasut4f9de152024-04-07 22:21:07 +0200809
810 /*
Marek Vasut1edcc112024-04-20 00:03:09 +0200811 * Make sure the OS would not get any spurious IWDG pretimeout IRQ
812 * right after the system wakes up. This may happen in case the SoC
813 * got woken up by another source than the IWDG pretimeout and the
814 * pretimeout IRQ arrived immediately afterward, but too late to be
815 * handled by the main loop above. In case either of the IWDG is
816 * enabled, ping it first and then return to the OS.
817 */
818
819 /* Ping IWDG1 and ACK pretimer IRQ */
820 if (gic_enabled[4] & BIT(22)) {
821 writel(IWDG_KR_RELOAD_KEY, STM32_IWDG1_BASE + IWDG_KR);
822 writel(IWDG_EWCR_EWIC, STM32_IWDG1_BASE + IWDG_EWCR);
823 }
824
825 /* Ping IWDG2 and ACK pretimer IRQ */
826 if (gic_enabled[4] & BIT(23)) {
827 writel(IWDG_KR_RELOAD_KEY, STM32_IWDG2_BASE + IWDG_KR);
828 writel(IWDG_EWCR_EWIC, STM32_IWDG2_BASE + IWDG_EWCR);
829 }
830
831 /*
Marek Vasut4f9de152024-04-07 22:21:07 +0200832 * The system has resumed successfully. Rewrite LR register stored
833 * on stack with 'ep' value, so that on return from this PSCI call,
834 * the code would jump to that 'ep' resume entry point code path
835 * instead of the previous 'lr' register content which (e.g. with
836 * Linux) points to resume failure code path.
837 *
838 * See arch/arm/cpu/armv7/psci.S _smc_psci: for the stack layout
839 * used here, SP-4 is PC, SP-8 is LR, SP-12 is R7, and so on.
840 */
841 writel(ep, sp - 8);
Marek Vasut83ec9582022-02-25 02:15:59 +0100842}