xref: /arm-trusted-firmware/plat/mediatek/mt8183/drivers/spm/spm_suspend.c (revision 91f16700b400a8c0651d24a598fc48ee2997a0d7)
1*91f16700Schasinglulu /*
2*91f16700Schasinglulu  * Copyright (c) 2019, ARM Limited and Contributors. All rights reserved.
3*91f16700Schasinglulu  *
4*91f16700Schasinglulu  * SPDX-License-Identifier: BSD-3-Clause
5*91f16700Schasinglulu  */
6*91f16700Schasinglulu 
7*91f16700Schasinglulu #include <arch_helpers.h>
8*91f16700Schasinglulu #include <common/debug.h>
9*91f16700Schasinglulu #include <drivers/delay_timer.h>
10*91f16700Schasinglulu #include <mt_gic_v3.h>
11*91f16700Schasinglulu #include <lib/mmio.h>
12*91f16700Schasinglulu #include <platform_def.h>
13*91f16700Schasinglulu #include <pmic.h>
14*91f16700Schasinglulu #include <spm.h>
15*91f16700Schasinglulu #include <uart.h>
16*91f16700Schasinglulu 
17*91f16700Schasinglulu #define SPM_SYSCLK_SETTLE       99
18*91f16700Schasinglulu 
19*91f16700Schasinglulu #define WAKE_SRC_FOR_SUSPEND \
20*91f16700Schasinglulu 	(WAKE_SRC_R12_PCM_TIMER | \
21*91f16700Schasinglulu 	 WAKE_SRC_R12_SSPM_WDT_EVENT_B | \
22*91f16700Schasinglulu 	 WAKE_SRC_R12_KP_IRQ_B | \
23*91f16700Schasinglulu 	 WAKE_SRC_R12_CONN2AP_SPM_WAKEUP_B | \
24*91f16700Schasinglulu 	 WAKE_SRC_R12_EINT_EVENT_B | \
25*91f16700Schasinglulu 	 WAKE_SRC_R12_CONN_WDT_IRQ_B | \
26*91f16700Schasinglulu 	 WAKE_SRC_R12_CCIF0_EVENT_B | \
27*91f16700Schasinglulu 	 WAKE_SRC_R12_SSPM_SPM_IRQ_B | \
28*91f16700Schasinglulu 	 WAKE_SRC_R12_SCP_SPM_IRQ_B | \
29*91f16700Schasinglulu 	 WAKE_SRC_R12_SCP_WDT_EVENT_B | \
30*91f16700Schasinglulu 	 WAKE_SRC_R12_USB_CDSC_B | \
31*91f16700Schasinglulu 	 WAKE_SRC_R12_USB_POWERDWN_B | \
32*91f16700Schasinglulu 	 WAKE_SRC_R12_SYS_TIMER_EVENT_B | \
33*91f16700Schasinglulu 	 WAKE_SRC_R12_EINT_EVENT_SECURE_B | \
34*91f16700Schasinglulu 	 WAKE_SRC_R12_CCIF1_EVENT_B | \
35*91f16700Schasinglulu 	 WAKE_SRC_R12_MD2AP_PEER_EVENT_B | \
36*91f16700Schasinglulu 	 WAKE_SRC_R12_MD1_WDT_B | \
37*91f16700Schasinglulu 	 WAKE_SRC_R12_CLDMA_EVENT_B | \
38*91f16700Schasinglulu 	 WAKE_SRC_R12_SEJ_WDT_GPT_B)
39*91f16700Schasinglulu 
40*91f16700Schasinglulu #define SLP_PCM_FLAGS \
41*91f16700Schasinglulu 	(SPM_FLAG_DIS_VCORE_DVS	| SPM_FLAG_DIS_VCORE_DFS | \
42*91f16700Schasinglulu 	 SPM_FLAG_DIS_ATF_ABORT | SPM_FLAG_DISABLE_MMSYS_DVFS | \
43*91f16700Schasinglulu 	 SPM_FLAG_DIS_INFRA_PDN | SPM_FLAG_SUSPEND_OPTION)
44*91f16700Schasinglulu 
45*91f16700Schasinglulu #define SLP_PCM_FLAGS1 \
46*91f16700Schasinglulu 	(SPM_FLAG1_DISABLE_MCDSR)
47*91f16700Schasinglulu 
48*91f16700Schasinglulu static const struct pwr_ctrl suspend_ctrl = {
49*91f16700Schasinglulu 	.wake_src = WAKE_SRC_FOR_SUSPEND,
50*91f16700Schasinglulu 	.pcm_flags = SLP_PCM_FLAGS,
51*91f16700Schasinglulu 	.pcm_flags1 = SLP_PCM_FLAGS1,
52*91f16700Schasinglulu 
53*91f16700Schasinglulu 	/* SPM_AP_STANDBY_CON */
54*91f16700Schasinglulu 	.wfi_op = 0x1,
55*91f16700Schasinglulu 	.mp0_cputop_idle_mask = 0,
56*91f16700Schasinglulu 	.mp1_cputop_idle_mask = 0,
57*91f16700Schasinglulu 	.mcusys_idle_mask = 0,
58*91f16700Schasinglulu 	.mm_mask_b = 0,
59*91f16700Schasinglulu 	.md_ddr_en_0_dbc_en = 0x1,
60*91f16700Schasinglulu 	.md_ddr_en_1_dbc_en = 0,
61*91f16700Schasinglulu 	.md_mask_b = 0x1,
62*91f16700Schasinglulu 	.sspm_mask_b = 0x1,
63*91f16700Schasinglulu 	.scp_mask_b = 0x1,
64*91f16700Schasinglulu 	.srcclkeni_mask_b = 0x1,
65*91f16700Schasinglulu 	.md_apsrc_1_sel = 0,
66*91f16700Schasinglulu 	.md_apsrc_0_sel = 0,
67*91f16700Schasinglulu 	.conn_ddr_en_dbc_en = 0x1,
68*91f16700Schasinglulu 	.conn_mask_b = 0x1,
69*91f16700Schasinglulu 	.conn_apsrc_sel = 0,
70*91f16700Schasinglulu 
71*91f16700Schasinglulu 	/* SPM_SRC_REQ */
72*91f16700Schasinglulu 	.spm_apsrc_req = 0,
73*91f16700Schasinglulu 	.spm_f26m_req = 0,
74*91f16700Schasinglulu 	.spm_infra_req = 0,
75*91f16700Schasinglulu 	.spm_vrf18_req = 0,
76*91f16700Schasinglulu 	.spm_ddren_req = 0,
77*91f16700Schasinglulu 	.spm_rsv_src_req = 0,
78*91f16700Schasinglulu 	.spm_ddren_2_req = 0,
79*91f16700Schasinglulu 	.cpu_md_dvfs_sop_force_on = 0,
80*91f16700Schasinglulu 
81*91f16700Schasinglulu 	/* SPM_SRC_MASK */
82*91f16700Schasinglulu 	.csyspwreq_mask = 0x1,
83*91f16700Schasinglulu 	.ccif0_md_event_mask_b = 0x1,
84*91f16700Schasinglulu 	.ccif0_ap_event_mask_b = 0x1,
85*91f16700Schasinglulu 	.ccif1_md_event_mask_b = 0x1,
86*91f16700Schasinglulu 	.ccif1_ap_event_mask_b = 0x1,
87*91f16700Schasinglulu 	.ccif2_md_event_mask_b = 0x1,
88*91f16700Schasinglulu 	.ccif2_ap_event_mask_b = 0x1,
89*91f16700Schasinglulu 	.ccif3_md_event_mask_b = 0x1,
90*91f16700Schasinglulu 	.ccif3_ap_event_mask_b = 0x1,
91*91f16700Schasinglulu 	.md_srcclkena_0_infra_mask_b = 0x1,
92*91f16700Schasinglulu 	.md_srcclkena_1_infra_mask_b = 0,
93*91f16700Schasinglulu 	.conn_srcclkena_infra_mask_b = 0,
94*91f16700Schasinglulu 	.ufs_infra_req_mask_b = 0,
95*91f16700Schasinglulu 	.srcclkeni_infra_mask_b = 0,
96*91f16700Schasinglulu 	.md_apsrc_req_0_infra_mask_b = 0x1,
97*91f16700Schasinglulu 	.md_apsrc_req_1_infra_mask_b = 0x1,
98*91f16700Schasinglulu 	.conn_apsrcreq_infra_mask_b = 0x1,
99*91f16700Schasinglulu 	.ufs_srcclkena_mask_b = 0,
100*91f16700Schasinglulu 	.md_vrf18_req_0_mask_b = 0,
101*91f16700Schasinglulu 	.md_vrf18_req_1_mask_b = 0,
102*91f16700Schasinglulu 	.ufs_vrf18_req_mask_b = 0,
103*91f16700Schasinglulu 	.gce_vrf18_req_mask_b = 0,
104*91f16700Schasinglulu 	.conn_infra_req_mask_b = 0x1,
105*91f16700Schasinglulu 	.gce_apsrc_req_mask_b = 0,
106*91f16700Schasinglulu 	.disp0_apsrc_req_mask_b = 0,
107*91f16700Schasinglulu 	.disp1_apsrc_req_mask_b = 0,
108*91f16700Schasinglulu 	.mfg_req_mask_b = 0,
109*91f16700Schasinglulu 	.vdec_req_mask_b = 0,
110*91f16700Schasinglulu 
111*91f16700Schasinglulu 	/* SPM_SRC2_MASK */
112*91f16700Schasinglulu 	.md_ddr_en_0_mask_b = 0x1,
113*91f16700Schasinglulu 	.md_ddr_en_1_mask_b = 0,
114*91f16700Schasinglulu 	.conn_ddr_en_mask_b = 0x1,
115*91f16700Schasinglulu 	.ddren_sspm_apsrc_req_mask_b = 0x1,
116*91f16700Schasinglulu 	.ddren_scp_apsrc_req_mask_b = 0x1,
117*91f16700Schasinglulu 	.disp0_ddren_mask_b = 0x1,
118*91f16700Schasinglulu 	.disp1_ddren_mask_b = 0x1,
119*91f16700Schasinglulu 	.gce_ddren_mask_b = 0x1,
120*91f16700Schasinglulu 	.ddren_emi_self_refresh_ch0_mask_b = 0,
121*91f16700Schasinglulu 	.ddren_emi_self_refresh_ch1_mask_b = 0,
122*91f16700Schasinglulu 
123*91f16700Schasinglulu 	/* SPM_WAKEUP_EVENT_MASK */
124*91f16700Schasinglulu 	.spm_wakeup_event_mask = 0xF1782218,
125*91f16700Schasinglulu 
126*91f16700Schasinglulu 	/* SPM_WAKEUP_EVENT_EXT_MASK */
127*91f16700Schasinglulu 	.spm_wakeup_event_ext_mask = 0xFFFFFFFF,
128*91f16700Schasinglulu 
129*91f16700Schasinglulu 	/* SPM_SRC3_MASK */
130*91f16700Schasinglulu 	.md_ddr_en_2_0_mask_b = 0x1,
131*91f16700Schasinglulu 	.md_ddr_en_2_1_mask_b = 0,
132*91f16700Schasinglulu 	.conn_ddr_en_2_mask_b = 0x1,
133*91f16700Schasinglulu 	.ddren2_sspm_apsrc_req_mask_b = 0x1,
134*91f16700Schasinglulu 	.ddren2_scp_apsrc_req_mask_b = 0x1,
135*91f16700Schasinglulu 	.disp0_ddren2_mask_b = 0,
136*91f16700Schasinglulu 	.disp1_ddren2_mask_b = 0,
137*91f16700Schasinglulu 	.gce_ddren2_mask_b = 0,
138*91f16700Schasinglulu 	.ddren2_emi_self_refresh_ch0_mask_b = 0,
139*91f16700Schasinglulu 	.ddren2_emi_self_refresh_ch1_mask_b = 0,
140*91f16700Schasinglulu 
141*91f16700Schasinglulu 	.mp0_cpu0_wfi_en = 0x1,
142*91f16700Schasinglulu 	.mp0_cpu1_wfi_en = 0x1,
143*91f16700Schasinglulu 	.mp0_cpu2_wfi_en = 0x1,
144*91f16700Schasinglulu 	.mp0_cpu3_wfi_en = 0x1,
145*91f16700Schasinglulu 
146*91f16700Schasinglulu 	.mp1_cpu0_wfi_en = 0x1,
147*91f16700Schasinglulu 	.mp1_cpu1_wfi_en = 0x1,
148*91f16700Schasinglulu 	.mp1_cpu2_wfi_en = 0x1,
149*91f16700Schasinglulu 	.mp1_cpu3_wfi_en = 0x1
150*91f16700Schasinglulu };
151*91f16700Schasinglulu 
152*91f16700Schasinglulu static uint32_t spm_set_sysclk_settle(void)
153*91f16700Schasinglulu {
154*91f16700Schasinglulu 	mmio_write_32(SPM_CLK_SETTLE, SPM_SYSCLK_SETTLE);
155*91f16700Schasinglulu 	return mmio_read_32(SPM_CLK_SETTLE);
156*91f16700Schasinglulu }
157*91f16700Schasinglulu 
158*91f16700Schasinglulu void go_to_sleep_before_wfi(void)
159*91f16700Schasinglulu {
160*91f16700Schasinglulu 	int cpu = MPIDR_AFFLVL0_VAL(read_mpidr());
161*91f16700Schasinglulu 	uint32_t settle;
162*91f16700Schasinglulu 
163*91f16700Schasinglulu 	settle = spm_set_sysclk_settle();
164*91f16700Schasinglulu 	spm_set_cpu_status(cpu);
165*91f16700Schasinglulu 	spm_set_power_control(&suspend_ctrl);
166*91f16700Schasinglulu 	spm_set_wakeup_event(&suspend_ctrl);
167*91f16700Schasinglulu 	spm_set_pcm_flags(&suspend_ctrl);
168*91f16700Schasinglulu 	spm_send_cpu_wakeup_event();
169*91f16700Schasinglulu 	spm_set_pcm_wdt(0);
170*91f16700Schasinglulu 	spm_disable_pcm_timer();
171*91f16700Schasinglulu 
172*91f16700Schasinglulu 	if (is_infra_pdn(suspend_ctrl.pcm_flags))
173*91f16700Schasinglulu 		mt_uart_save();
174*91f16700Schasinglulu 
175*91f16700Schasinglulu 	if (!mt_console_uart_cg_status())
176*91f16700Schasinglulu 		console_switch_state(CONSOLE_FLAG_BOOT);
177*91f16700Schasinglulu 
178*91f16700Schasinglulu 	INFO("cpu%d: \"%s\", wakesrc = 0x%x, pcm_con1 = 0x%x\n",
179*91f16700Schasinglulu 	     cpu, spm_get_firmware_version(), suspend_ctrl.wake_src,
180*91f16700Schasinglulu 	     mmio_read_32(PCM_CON1));
181*91f16700Schasinglulu 	INFO("settle = %u, sec = %u, sw_flag = 0x%x 0x%x, src_req = 0x%x\n",
182*91f16700Schasinglulu 	     settle, mmio_read_32(PCM_TIMER_VAL) / 32768,
183*91f16700Schasinglulu 	     suspend_ctrl.pcm_flags, suspend_ctrl.pcm_flags1,
184*91f16700Schasinglulu 	     mmio_read_32(SPM_SRC_REQ));
185*91f16700Schasinglulu 
186*91f16700Schasinglulu 	if (!mt_console_uart_cg_status())
187*91f16700Schasinglulu 		console_switch_state(CONSOLE_FLAG_RUNTIME);
188*91f16700Schasinglulu }
189*91f16700Schasinglulu 
190*91f16700Schasinglulu static void go_to_sleep_after_wfi(void)
191*91f16700Schasinglulu {
192*91f16700Schasinglulu 	struct wake_status spm_wakesta;
193*91f16700Schasinglulu 
194*91f16700Schasinglulu 	if (is_infra_pdn(suspend_ctrl.pcm_flags))
195*91f16700Schasinglulu 		mt_uart_restore();
196*91f16700Schasinglulu 
197*91f16700Schasinglulu 	spm_set_pcm_wdt(0);
198*91f16700Schasinglulu 	spm_get_wakeup_status(&spm_wakesta);
199*91f16700Schasinglulu 	spm_clean_after_wakeup();
200*91f16700Schasinglulu 
201*91f16700Schasinglulu 	if (!mt_console_uart_cg_status())
202*91f16700Schasinglulu 		console_switch_state(CONSOLE_FLAG_BOOT);
203*91f16700Schasinglulu 
204*91f16700Schasinglulu 	spm_output_wake_reason(&spm_wakesta, "suspend");
205*91f16700Schasinglulu 
206*91f16700Schasinglulu 	if (!mt_console_uart_cg_status())
207*91f16700Schasinglulu 		console_switch_state(CONSOLE_FLAG_RUNTIME);
208*91f16700Schasinglulu }
209*91f16700Schasinglulu 
210*91f16700Schasinglulu static void spm_enable_armpll_l(void)
211*91f16700Schasinglulu {
212*91f16700Schasinglulu 	/* power on */
213*91f16700Schasinglulu 	mmio_setbits_32(ARMPLL_L_PWR_CON0, 0x1);
214*91f16700Schasinglulu 
215*91f16700Schasinglulu 	/* clear isolation */
216*91f16700Schasinglulu 	mmio_clrbits_32(ARMPLL_L_PWR_CON0, 0x2);
217*91f16700Schasinglulu 
218*91f16700Schasinglulu 	/* enable pll */
219*91f16700Schasinglulu 	mmio_setbits_32(ARMPLL_L_CON0, 0x1);
220*91f16700Schasinglulu 
221*91f16700Schasinglulu 	/* Add 20us delay for turning on PLL */
222*91f16700Schasinglulu 	udelay(20);
223*91f16700Schasinglulu }
224*91f16700Schasinglulu 
225*91f16700Schasinglulu static void spm_disable_armpll_l(void)
226*91f16700Schasinglulu {
227*91f16700Schasinglulu 	/* disable pll */
228*91f16700Schasinglulu 	mmio_clrbits_32(ARMPLL_L_CON0, 0x1);
229*91f16700Schasinglulu 
230*91f16700Schasinglulu 	/* isolation */
231*91f16700Schasinglulu 	mmio_setbits_32(ARMPLL_L_PWR_CON0, 0x2);
232*91f16700Schasinglulu 
233*91f16700Schasinglulu 	/* power off */
234*91f16700Schasinglulu 	mmio_clrbits_32(ARMPLL_L_PWR_CON0, 0x1);
235*91f16700Schasinglulu }
236*91f16700Schasinglulu 
237*91f16700Schasinglulu void spm_system_suspend(void)
238*91f16700Schasinglulu {
239*91f16700Schasinglulu 	spm_disable_armpll_l();
240*91f16700Schasinglulu 	bcpu_enable(0);
241*91f16700Schasinglulu 	bcpu_sram_enable(0);
242*91f16700Schasinglulu 	spm_lock_get();
243*91f16700Schasinglulu 	go_to_sleep_before_wfi();
244*91f16700Schasinglulu 	spm_lock_release();
245*91f16700Schasinglulu }
246*91f16700Schasinglulu 
247*91f16700Schasinglulu void spm_system_suspend_finish(void)
248*91f16700Schasinglulu {
249*91f16700Schasinglulu 	spm_lock_get();
250*91f16700Schasinglulu 	go_to_sleep_after_wfi();
251*91f16700Schasinglulu 	spm_lock_release();
252*91f16700Schasinglulu 	spm_enable_armpll_l();
253*91f16700Schasinglulu 	bcpu_sram_enable(1);
254*91f16700Schasinglulu 	bcpu_enable(1);
255*91f16700Schasinglulu }
256