256 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 * Copyright (c) 2019, ARM Limited and Contributors. All rights reserved.
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-3-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <arch_helpers.h>
 | 
						|
#include <common/debug.h>
 | 
						|
#include <drivers/delay_timer.h>
 | 
						|
#include <mt_gic_v3.h>
 | 
						|
#include <lib/mmio.h>
 | 
						|
#include <platform_def.h>
 | 
						|
#include <pmic.h>
 | 
						|
#include <spm.h>
 | 
						|
#include <uart.h>
 | 
						|
 | 
						|
#define SPM_SYSCLK_SETTLE       99
 | 
						|
 | 
						|
#define WAKE_SRC_FOR_SUSPEND \
 | 
						|
	(WAKE_SRC_R12_PCM_TIMER | \
 | 
						|
	 WAKE_SRC_R12_SSPM_WDT_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_KP_IRQ_B | \
 | 
						|
	 WAKE_SRC_R12_CONN2AP_SPM_WAKEUP_B | \
 | 
						|
	 WAKE_SRC_R12_EINT_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_CONN_WDT_IRQ_B | \
 | 
						|
	 WAKE_SRC_R12_CCIF0_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_SSPM_SPM_IRQ_B | \
 | 
						|
	 WAKE_SRC_R12_SCP_SPM_IRQ_B | \
 | 
						|
	 WAKE_SRC_R12_SCP_WDT_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_USB_CDSC_B | \
 | 
						|
	 WAKE_SRC_R12_USB_POWERDWN_B | \
 | 
						|
	 WAKE_SRC_R12_SYS_TIMER_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_EINT_EVENT_SECURE_B | \
 | 
						|
	 WAKE_SRC_R12_CCIF1_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_MD2AP_PEER_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_MD1_WDT_B | \
 | 
						|
	 WAKE_SRC_R12_CLDMA_EVENT_B | \
 | 
						|
	 WAKE_SRC_R12_SEJ_WDT_GPT_B)
 | 
						|
 | 
						|
#define SLP_PCM_FLAGS \
 | 
						|
	(SPM_FLAG_DIS_VCORE_DVS	| SPM_FLAG_DIS_VCORE_DFS | \
 | 
						|
	 SPM_FLAG_DIS_ATF_ABORT | SPM_FLAG_DISABLE_MMSYS_DVFS | \
 | 
						|
	 SPM_FLAG_DIS_INFRA_PDN | SPM_FLAG_SUSPEND_OPTION)
 | 
						|
 | 
						|
#define SLP_PCM_FLAGS1 \
 | 
						|
	(SPM_FLAG1_DISABLE_MCDSR)
 | 
						|
 | 
						|
static const struct pwr_ctrl suspend_ctrl = {
 | 
						|
	.wake_src = WAKE_SRC_FOR_SUSPEND,
 | 
						|
	.pcm_flags = SLP_PCM_FLAGS,
 | 
						|
	.pcm_flags1 = SLP_PCM_FLAGS1,
 | 
						|
 | 
						|
	/* SPM_AP_STANDBY_CON */
 | 
						|
	.wfi_op = 0x1,
 | 
						|
	.mp0_cputop_idle_mask = 0,
 | 
						|
	.mp1_cputop_idle_mask = 0,
 | 
						|
	.mcusys_idle_mask = 0,
 | 
						|
	.mm_mask_b = 0,
 | 
						|
	.md_ddr_en_0_dbc_en = 0x1,
 | 
						|
	.md_ddr_en_1_dbc_en = 0,
 | 
						|
	.md_mask_b = 0x1,
 | 
						|
	.sspm_mask_b = 0x1,
 | 
						|
	.scp_mask_b = 0x1,
 | 
						|
	.srcclkeni_mask_b = 0x1,
 | 
						|
	.md_apsrc_1_sel = 0,
 | 
						|
	.md_apsrc_0_sel = 0,
 | 
						|
	.conn_ddr_en_dbc_en = 0x1,
 | 
						|
	.conn_mask_b = 0x1,
 | 
						|
	.conn_apsrc_sel = 0,
 | 
						|
 | 
						|
	/* SPM_SRC_REQ */
 | 
						|
	.spm_apsrc_req = 0,
 | 
						|
	.spm_f26m_req = 0,
 | 
						|
	.spm_infra_req = 0,
 | 
						|
	.spm_vrf18_req = 0,
 | 
						|
	.spm_ddren_req = 0,
 | 
						|
	.spm_rsv_src_req = 0,
 | 
						|
	.spm_ddren_2_req = 0,
 | 
						|
	.cpu_md_dvfs_sop_force_on = 0,
 | 
						|
 | 
						|
	/* SPM_SRC_MASK */
 | 
						|
	.csyspwreq_mask = 0x1,
 | 
						|
	.ccif0_md_event_mask_b = 0x1,
 | 
						|
	.ccif0_ap_event_mask_b = 0x1,
 | 
						|
	.ccif1_md_event_mask_b = 0x1,
 | 
						|
	.ccif1_ap_event_mask_b = 0x1,
 | 
						|
	.ccif2_md_event_mask_b = 0x1,
 | 
						|
	.ccif2_ap_event_mask_b = 0x1,
 | 
						|
	.ccif3_md_event_mask_b = 0x1,
 | 
						|
	.ccif3_ap_event_mask_b = 0x1,
 | 
						|
	.md_srcclkena_0_infra_mask_b = 0x1,
 | 
						|
	.md_srcclkena_1_infra_mask_b = 0,
 | 
						|
	.conn_srcclkena_infra_mask_b = 0,
 | 
						|
	.ufs_infra_req_mask_b = 0,
 | 
						|
	.srcclkeni_infra_mask_b = 0,
 | 
						|
	.md_apsrc_req_0_infra_mask_b = 0x1,
 | 
						|
	.md_apsrc_req_1_infra_mask_b = 0x1,
 | 
						|
	.conn_apsrcreq_infra_mask_b = 0x1,
 | 
						|
	.ufs_srcclkena_mask_b = 0,
 | 
						|
	.md_vrf18_req_0_mask_b = 0,
 | 
						|
	.md_vrf18_req_1_mask_b = 0,
 | 
						|
	.ufs_vrf18_req_mask_b = 0,
 | 
						|
	.gce_vrf18_req_mask_b = 0,
 | 
						|
	.conn_infra_req_mask_b = 0x1,
 | 
						|
	.gce_apsrc_req_mask_b = 0,
 | 
						|
	.disp0_apsrc_req_mask_b = 0,
 | 
						|
	.disp1_apsrc_req_mask_b = 0,
 | 
						|
	.mfg_req_mask_b = 0,
 | 
						|
	.vdec_req_mask_b = 0,
 | 
						|
 | 
						|
	/* SPM_SRC2_MASK */
 | 
						|
	.md_ddr_en_0_mask_b = 0x1,
 | 
						|
	.md_ddr_en_1_mask_b = 0,
 | 
						|
	.conn_ddr_en_mask_b = 0x1,
 | 
						|
	.ddren_sspm_apsrc_req_mask_b = 0x1,
 | 
						|
	.ddren_scp_apsrc_req_mask_b = 0x1,
 | 
						|
	.disp0_ddren_mask_b = 0x1,
 | 
						|
	.disp1_ddren_mask_b = 0x1,
 | 
						|
	.gce_ddren_mask_b = 0x1,
 | 
						|
	.ddren_emi_self_refresh_ch0_mask_b = 0,
 | 
						|
	.ddren_emi_self_refresh_ch1_mask_b = 0,
 | 
						|
 | 
						|
	/* SPM_WAKEUP_EVENT_MASK */
 | 
						|
	.spm_wakeup_event_mask = 0xF1782218,
 | 
						|
 | 
						|
	/* SPM_WAKEUP_EVENT_EXT_MASK */
 | 
						|
	.spm_wakeup_event_ext_mask = 0xFFFFFFFF,
 | 
						|
 | 
						|
	/* SPM_SRC3_MASK */
 | 
						|
	.md_ddr_en_2_0_mask_b = 0x1,
 | 
						|
	.md_ddr_en_2_1_mask_b = 0,
 | 
						|
	.conn_ddr_en_2_mask_b = 0x1,
 | 
						|
	.ddren2_sspm_apsrc_req_mask_b = 0x1,
 | 
						|
	.ddren2_scp_apsrc_req_mask_b = 0x1,
 | 
						|
	.disp0_ddren2_mask_b = 0,
 | 
						|
	.disp1_ddren2_mask_b = 0,
 | 
						|
	.gce_ddren2_mask_b = 0,
 | 
						|
	.ddren2_emi_self_refresh_ch0_mask_b = 0,
 | 
						|
	.ddren2_emi_self_refresh_ch1_mask_b = 0,
 | 
						|
 | 
						|
	.mp0_cpu0_wfi_en = 0x1,
 | 
						|
	.mp0_cpu1_wfi_en = 0x1,
 | 
						|
	.mp0_cpu2_wfi_en = 0x1,
 | 
						|
	.mp0_cpu3_wfi_en = 0x1,
 | 
						|
 | 
						|
	.mp1_cpu0_wfi_en = 0x1,
 | 
						|
	.mp1_cpu1_wfi_en = 0x1,
 | 
						|
	.mp1_cpu2_wfi_en = 0x1,
 | 
						|
	.mp1_cpu3_wfi_en = 0x1
 | 
						|
};
 | 
						|
 | 
						|
static uint32_t spm_set_sysclk_settle(void)
 | 
						|
{
 | 
						|
	mmio_write_32(SPM_CLK_SETTLE, SPM_SYSCLK_SETTLE);
 | 
						|
	return mmio_read_32(SPM_CLK_SETTLE);
 | 
						|
}
 | 
						|
 | 
						|
void go_to_sleep_before_wfi(void)
 | 
						|
{
 | 
						|
	int cpu = MPIDR_AFFLVL0_VAL(read_mpidr());
 | 
						|
	uint32_t settle;
 | 
						|
 | 
						|
	settle = spm_set_sysclk_settle();
 | 
						|
	spm_set_cpu_status(cpu);
 | 
						|
	spm_set_power_control(&suspend_ctrl);
 | 
						|
	spm_set_wakeup_event(&suspend_ctrl);
 | 
						|
	spm_set_pcm_flags(&suspend_ctrl);
 | 
						|
	spm_send_cpu_wakeup_event();
 | 
						|
	spm_set_pcm_wdt(0);
 | 
						|
	spm_disable_pcm_timer();
 | 
						|
 | 
						|
	if (is_infra_pdn(suspend_ctrl.pcm_flags))
 | 
						|
		mt_uart_save();
 | 
						|
 | 
						|
	if (!mt_console_uart_cg_status())
 | 
						|
		console_switch_state(CONSOLE_FLAG_BOOT);
 | 
						|
 | 
						|
	INFO("cpu%d: \"%s\", wakesrc = 0x%x, pcm_con1 = 0x%x\n",
 | 
						|
	     cpu, spm_get_firmware_version(), suspend_ctrl.wake_src,
 | 
						|
	     mmio_read_32(PCM_CON1));
 | 
						|
	INFO("settle = %u, sec = %u, sw_flag = 0x%x 0x%x, src_req = 0x%x\n",
 | 
						|
	     settle, mmio_read_32(PCM_TIMER_VAL) / 32768,
 | 
						|
	     suspend_ctrl.pcm_flags, suspend_ctrl.pcm_flags1,
 | 
						|
	     mmio_read_32(SPM_SRC_REQ));
 | 
						|
 | 
						|
	if (!mt_console_uart_cg_status())
 | 
						|
		console_switch_state(CONSOLE_FLAG_RUNTIME);
 | 
						|
}
 | 
						|
 | 
						|
static void go_to_sleep_after_wfi(void)
 | 
						|
{
 | 
						|
	struct wake_status spm_wakesta;
 | 
						|
 | 
						|
	if (is_infra_pdn(suspend_ctrl.pcm_flags))
 | 
						|
		mt_uart_restore();
 | 
						|
 | 
						|
	spm_set_pcm_wdt(0);
 | 
						|
	spm_get_wakeup_status(&spm_wakesta);
 | 
						|
	spm_clean_after_wakeup();
 | 
						|
 | 
						|
	if (!mt_console_uart_cg_status())
 | 
						|
		console_switch_state(CONSOLE_FLAG_BOOT);
 | 
						|
 | 
						|
	spm_output_wake_reason(&spm_wakesta, "suspend");
 | 
						|
 | 
						|
	if (!mt_console_uart_cg_status())
 | 
						|
		console_switch_state(CONSOLE_FLAG_RUNTIME);
 | 
						|
}
 | 
						|
 | 
						|
static void spm_enable_armpll_l(void)
 | 
						|
{
 | 
						|
	/* power on */
 | 
						|
	mmio_setbits_32(ARMPLL_L_PWR_CON0, 0x1);
 | 
						|
 | 
						|
	/* clear isolation */
 | 
						|
	mmio_clrbits_32(ARMPLL_L_PWR_CON0, 0x2);
 | 
						|
 | 
						|
	/* enable pll */
 | 
						|
	mmio_setbits_32(ARMPLL_L_CON0, 0x1);
 | 
						|
 | 
						|
	/* Add 20us delay for turning on PLL */
 | 
						|
	udelay(20);
 | 
						|
}
 | 
						|
 | 
						|
static void spm_disable_armpll_l(void)
 | 
						|
{
 | 
						|
	/* disable pll */
 | 
						|
	mmio_clrbits_32(ARMPLL_L_CON0, 0x1);
 | 
						|
 | 
						|
	/* isolation */
 | 
						|
	mmio_setbits_32(ARMPLL_L_PWR_CON0, 0x2);
 | 
						|
 | 
						|
	/* power off */
 | 
						|
	mmio_clrbits_32(ARMPLL_L_PWR_CON0, 0x1);
 | 
						|
}
 | 
						|
 | 
						|
void spm_system_suspend(void)
 | 
						|
{
 | 
						|
	spm_disable_armpll_l();
 | 
						|
	bcpu_enable(0);
 | 
						|
	bcpu_sram_enable(0);
 | 
						|
	spm_lock_get();
 | 
						|
	go_to_sleep_before_wfi();
 | 
						|
	spm_lock_release();
 | 
						|
}
 | 
						|
 | 
						|
void spm_system_suspend_finish(void)
 | 
						|
{
 | 
						|
	spm_lock_get();
 | 
						|
	go_to_sleep_after_wfi();
 | 
						|
	spm_lock_release();
 | 
						|
	spm_enable_armpll_l();
 | 
						|
	bcpu_sram_enable(1);
 | 
						|
	bcpu_enable(1);
 | 
						|
}
 |