1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright 2016 Freescale Semiconductor, Inc.
3*4882a593Smuzhiyun * Author: Hongbo Zhang <hongbo.zhang@nxp.com>
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * SPDX-License-Identifier: GPL-2.0+
6*4882a593Smuzhiyun * This file implements LS102X platform PSCI SYSTEM-SUSPEND function
7*4882a593Smuzhiyun */
8*4882a593Smuzhiyun
9*4882a593Smuzhiyun #include <config.h>
10*4882a593Smuzhiyun #include <asm/io.h>
11*4882a593Smuzhiyun #include <asm/psci.h>
12*4882a593Smuzhiyun #include <asm/arch/immap_ls102xa.h>
13*4882a593Smuzhiyun #include <fsl_immap.h>
14*4882a593Smuzhiyun #include "fsl_epu.h"
15*4882a593Smuzhiyun
16*4882a593Smuzhiyun #define __secure __attribute__((section("._secure.text")))
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #define CCSR_GICD_CTLR 0x1000
19*4882a593Smuzhiyun #define CCSR_GICC_CTLR 0x2000
20*4882a593Smuzhiyun #define DCSR_RCPM_CG1CR0 0x31c
21*4882a593Smuzhiyun #define DCSR_RCPM_CSTTACR0 0xb00
22*4882a593Smuzhiyun #define DCFG_CRSTSR_WDRFR 0x8
23*4882a593Smuzhiyun #define DDR_RESV_LEN 128
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun #ifdef CONFIG_LS1_DEEP_SLEEP
26*4882a593Smuzhiyun /*
27*4882a593Smuzhiyun * DDR controller initialization training breaks the first 128 bytes of DDR,
28*4882a593Smuzhiyun * save them so that the bootloader can restore them while resuming.
29*4882a593Smuzhiyun */
ls1_save_ddr_head(void)30*4882a593Smuzhiyun static void __secure ls1_save_ddr_head(void)
31*4882a593Smuzhiyun {
32*4882a593Smuzhiyun const char *src = (const char *)CONFIG_SYS_SDRAM_BASE;
33*4882a593Smuzhiyun char *dest = (char *)(OCRAM_BASE_S_ADDR + OCRAM_S_SIZE - DDR_RESV_LEN);
34*4882a593Smuzhiyun struct ccsr_scfg __iomem *scfg = (void *)CONFIG_SYS_FSL_SCFG_ADDR;
35*4882a593Smuzhiyun int i;
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun out_le32(&scfg->sparecr[2], dest);
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun for (i = 0; i < DDR_RESV_LEN; i++)
40*4882a593Smuzhiyun *dest++ = *src++;
41*4882a593Smuzhiyun }
42*4882a593Smuzhiyun
ls1_fsm_setup(void)43*4882a593Smuzhiyun static void __secure ls1_fsm_setup(void)
44*4882a593Smuzhiyun {
45*4882a593Smuzhiyun void *dcsr_epu_base = (void *)(CONFIG_SYS_DCSRBAR + EPU_BLOCK_OFFSET);
46*4882a593Smuzhiyun void *dcsr_rcpm_base = (void *)SYS_FSL_DCSR_RCPM_ADDR;
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun out_be32(dcsr_rcpm_base + DCSR_RCPM_CSTTACR0, 0x00001001);
49*4882a593Smuzhiyun out_be32(dcsr_rcpm_base + DCSR_RCPM_CG1CR0, 0x00000001);
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun fsl_epu_setup((void *)dcsr_epu_base);
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun /* Pull MCKE signal low before enabling deep sleep signal in FPGA */
54*4882a593Smuzhiyun out_be32(dcsr_epu_base + EPECR0, 0x5);
55*4882a593Smuzhiyun out_be32(dcsr_epu_base + EPSMCR15, 0x76300000);
56*4882a593Smuzhiyun }
57*4882a593Smuzhiyun
ls1_deepsleep_irq_cfg(void)58*4882a593Smuzhiyun static void __secure ls1_deepsleep_irq_cfg(void)
59*4882a593Smuzhiyun {
60*4882a593Smuzhiyun struct ccsr_scfg __iomem *scfg = (void *)CONFIG_SYS_FSL_SCFG_ADDR;
61*4882a593Smuzhiyun struct ccsr_rcpm __iomem *rcpm = (void *)CONFIG_SYS_FSL_RCPM_ADDR;
62*4882a593Smuzhiyun u32 ippdexpcr0, ippdexpcr1, pmcintecr = 0;
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun /* Mask interrupts from GIC */
65*4882a593Smuzhiyun out_be32(&rcpm->nfiqoutr, 0x0ffffffff);
66*4882a593Smuzhiyun out_be32(&rcpm->nirqoutr, 0x0ffffffff);
67*4882a593Smuzhiyun /* Mask deep sleep wake-up interrupts while entering deep sleep */
68*4882a593Smuzhiyun out_be32(&rcpm->dsimskr, 0x0ffffffff);
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun ippdexpcr0 = in_be32(&rcpm->ippdexpcr0);
71*4882a593Smuzhiyun /*
72*4882a593Smuzhiyun * Workaround: There is bug of register ippdexpcr1, when read it always
73*4882a593Smuzhiyun * returns zero, so its value is saved to a scrachpad register to be
74*4882a593Smuzhiyun * read, that is why we don't read it from register ippdexpcr1 itself.
75*4882a593Smuzhiyun */
76*4882a593Smuzhiyun ippdexpcr1 = in_le32(&scfg->sparecr[7]);
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun if (ippdexpcr0 & RCPM_IPPDEXPCR0_ETSEC)
79*4882a593Smuzhiyun pmcintecr |= SCFG_PMCINTECR_ETSECRXG0 |
80*4882a593Smuzhiyun SCFG_PMCINTECR_ETSECRXG1 |
81*4882a593Smuzhiyun SCFG_PMCINTECR_ETSECERRG0 |
82*4882a593Smuzhiyun SCFG_PMCINTECR_ETSECERRG1;
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun if (ippdexpcr0 & RCPM_IPPDEXPCR0_GPIO)
85*4882a593Smuzhiyun pmcintecr |= SCFG_PMCINTECR_GPIO;
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun if (ippdexpcr1 & RCPM_IPPDEXPCR1_LPUART)
88*4882a593Smuzhiyun pmcintecr |= SCFG_PMCINTECR_LPUART;
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun if (ippdexpcr1 & RCPM_IPPDEXPCR1_FLEXTIMER)
91*4882a593Smuzhiyun pmcintecr |= SCFG_PMCINTECR_FTM;
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun /* Always set external IRQ pins as wakeup source */
94*4882a593Smuzhiyun pmcintecr |= SCFG_PMCINTECR_IRQ0 | SCFG_PMCINTECR_IRQ1;
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun out_be32(&scfg->pmcintlecr, 0);
97*4882a593Smuzhiyun /* Clear PMC interrupt status */
98*4882a593Smuzhiyun out_be32(&scfg->pmcintsr, 0xffffffff);
99*4882a593Smuzhiyun /* Enable wakeup interrupt during deep sleep */
100*4882a593Smuzhiyun out_be32(&scfg->pmcintecr, pmcintecr);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
ls1_delay(unsigned int loop)103*4882a593Smuzhiyun static void __secure ls1_delay(unsigned int loop)
104*4882a593Smuzhiyun {
105*4882a593Smuzhiyun while (loop--) {
106*4882a593Smuzhiyun int i = 1000;
107*4882a593Smuzhiyun while (i--)
108*4882a593Smuzhiyun ;
109*4882a593Smuzhiyun }
110*4882a593Smuzhiyun }
111*4882a593Smuzhiyun
ls1_start_fsm(void)112*4882a593Smuzhiyun static void __secure ls1_start_fsm(void)
113*4882a593Smuzhiyun {
114*4882a593Smuzhiyun void *dcsr_epu_base = (void *)(CONFIG_SYS_DCSRBAR + EPU_BLOCK_OFFSET);
115*4882a593Smuzhiyun void *ccsr_gic_base = (void *)SYS_FSL_GIC_ADDR;
116*4882a593Smuzhiyun struct ccsr_scfg __iomem *scfg = (void *)CONFIG_SYS_FSL_SCFG_ADDR;
117*4882a593Smuzhiyun struct ccsr_ddr __iomem *ddr = (void *)CONFIG_SYS_FSL_DDR_ADDR;
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun /* Set HRSTCR */
120*4882a593Smuzhiyun setbits_be32(&scfg->hrstcr, 0x80000000);
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun /* Place DDR controller in self refresh mode */
123*4882a593Smuzhiyun setbits_be32(&ddr->sdram_cfg_2, 0x80000000);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun ls1_delay(2000);
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun /* Set EVT4_B to lock the signal MCKE down */
128*4882a593Smuzhiyun out_be32(dcsr_epu_base + EPECR0, 0x0);
129*4882a593Smuzhiyun
130*4882a593Smuzhiyun ls1_delay(2000);
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun out_be32(ccsr_gic_base + CCSR_GICD_CTLR, 0x0);
133*4882a593Smuzhiyun out_be32(ccsr_gic_base + CCSR_GICC_CTLR, 0x0);
134*4882a593Smuzhiyun
135*4882a593Smuzhiyun /* Enable all EPU Counters */
136*4882a593Smuzhiyun setbits_be32(dcsr_epu_base + EPGCR, 0x80000000);
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun /* Enable SCU15 */
139*4882a593Smuzhiyun setbits_be32(dcsr_epu_base + EPECR15, 0x90000004);
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun /* Enter WFI mode, and EPU FSM will start */
142*4882a593Smuzhiyun __asm__ __volatile__ ("wfi" : : : "memory");
143*4882a593Smuzhiyun
144*4882a593Smuzhiyun /* NEVER ENTER HERE */
145*4882a593Smuzhiyun while (1)
146*4882a593Smuzhiyun ;
147*4882a593Smuzhiyun }
148*4882a593Smuzhiyun
ls1_deep_sleep(u32 entry_point)149*4882a593Smuzhiyun static void __secure ls1_deep_sleep(u32 entry_point)
150*4882a593Smuzhiyun {
151*4882a593Smuzhiyun struct ccsr_scfg __iomem *scfg = (void *)CONFIG_SYS_FSL_SCFG_ADDR;
152*4882a593Smuzhiyun struct ccsr_gur __iomem *gur = (void *)CONFIG_SYS_FSL_GUTS_ADDR;
153*4882a593Smuzhiyun struct ccsr_rcpm __iomem *rcpm = (void *)CONFIG_SYS_FSL_RCPM_ADDR;
154*4882a593Smuzhiyun #ifdef QIXIS_BASE
155*4882a593Smuzhiyun u32 tmp;
156*4882a593Smuzhiyun void *qixis_base = (void *)QIXIS_BASE;
157*4882a593Smuzhiyun #endif
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun /* Enable cluster to enter the PCL10 state */
160*4882a593Smuzhiyun out_be32(&scfg->clusterpmcr, SCFG_CLUSTERPMCR_WFIL2EN);
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun /* Save the first 128 bytes of DDR data */
163*4882a593Smuzhiyun ls1_save_ddr_head();
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun /* Save the kernel resume entry */
166*4882a593Smuzhiyun out_le32(&scfg->sparecr[3], entry_point);
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun /* Request to put cluster 0 in PCL10 state */
169*4882a593Smuzhiyun setbits_be32(&rcpm->clpcl10setr, RCPM_CLPCL10SETR_C0);
170*4882a593Smuzhiyun
171*4882a593Smuzhiyun /* Setup the registers of the EPU FSM for deep sleep */
172*4882a593Smuzhiyun ls1_fsm_setup();
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun #ifdef QIXIS_BASE
175*4882a593Smuzhiyun /* Connect the EVENT button to IRQ in FPGA */
176*4882a593Smuzhiyun tmp = in_8(qixis_base + QIXIS_CTL_SYS);
177*4882a593Smuzhiyun tmp &= ~QIXIS_CTL_SYS_EVTSW_MASK;
178*4882a593Smuzhiyun tmp |= QIXIS_CTL_SYS_EVTSW_IRQ;
179*4882a593Smuzhiyun out_8(qixis_base + QIXIS_CTL_SYS, tmp);
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun /* Enable deep sleep signals in FPGA */
182*4882a593Smuzhiyun tmp = in_8(qixis_base + QIXIS_PWR_CTL2);
183*4882a593Smuzhiyun tmp |= QIXIS_PWR_CTL2_PCTL;
184*4882a593Smuzhiyun out_8(qixis_base + QIXIS_PWR_CTL2, tmp);
185*4882a593Smuzhiyun
186*4882a593Smuzhiyun /* Pull down PCIe RST# */
187*4882a593Smuzhiyun tmp = in_8(qixis_base + QIXIS_RST_FORCE_3);
188*4882a593Smuzhiyun tmp |= QIXIS_RST_FORCE_3_PCIESLOT1;
189*4882a593Smuzhiyun out_8(qixis_base + QIXIS_RST_FORCE_3, tmp);
190*4882a593Smuzhiyun #endif
191*4882a593Smuzhiyun
192*4882a593Smuzhiyun /* Enable Warm Device Reset */
193*4882a593Smuzhiyun setbits_be32(&scfg->dpslpcr, SCFG_DPSLPCR_WDRR_EN);
194*4882a593Smuzhiyun setbits_be32(&gur->crstsr, DCFG_CRSTSR_WDRFR);
195*4882a593Smuzhiyun
196*4882a593Smuzhiyun ls1_deepsleep_irq_cfg();
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun psci_v7_flush_dcache_all();
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun ls1_start_fsm();
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun #else
ls1_sleep(void)204*4882a593Smuzhiyun static void __secure ls1_sleep(void)
205*4882a593Smuzhiyun {
206*4882a593Smuzhiyun struct ccsr_scfg __iomem *scfg = (void *)CONFIG_SYS_FSL_SCFG_ADDR;
207*4882a593Smuzhiyun struct ccsr_rcpm __iomem *rcpm = (void *)CONFIG_SYS_FSL_RCPM_ADDR;
208*4882a593Smuzhiyun
209*4882a593Smuzhiyun #ifdef QIXIS_BASE
210*4882a593Smuzhiyun u32 tmp;
211*4882a593Smuzhiyun void *qixis_base = (void *)QIXIS_BASE;
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun /* Connect the EVENT button to IRQ in FPGA */
214*4882a593Smuzhiyun tmp = in_8(qixis_base + QIXIS_CTL_SYS);
215*4882a593Smuzhiyun tmp &= ~QIXIS_CTL_SYS_EVTSW_MASK;
216*4882a593Smuzhiyun tmp |= QIXIS_CTL_SYS_EVTSW_IRQ;
217*4882a593Smuzhiyun out_8(qixis_base + QIXIS_CTL_SYS, tmp);
218*4882a593Smuzhiyun #endif
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun /* Enable cluster to enter the PCL10 state */
221*4882a593Smuzhiyun out_be32(&scfg->clusterpmcr, SCFG_CLUSTERPMCR_WFIL2EN);
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun setbits_be32(&rcpm->powmgtcsr, RCPM_POWMGTCSR_LPM20_REQ);
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun __asm__ __volatile__ ("wfi" : : : "memory");
226*4882a593Smuzhiyun }
227*4882a593Smuzhiyun #endif
228*4882a593Smuzhiyun
ls1_system_suspend(u32 fn,u32 entry_point,u32 context_id)229*4882a593Smuzhiyun void __secure ls1_system_suspend(u32 fn, u32 entry_point, u32 context_id)
230*4882a593Smuzhiyun {
231*4882a593Smuzhiyun #ifdef CONFIG_LS1_DEEP_SLEEP
232*4882a593Smuzhiyun ls1_deep_sleep(entry_point);
233*4882a593Smuzhiyun #else
234*4882a593Smuzhiyun ls1_sleep();
235*4882a593Smuzhiyun #endif
236*4882a593Smuzhiyun }
237