xref: /rk3399_ARM-atf/plat/imx/imx7/common/imx7_bl2_el3_common.c (revision caf8fdb712e3491737be2e75e3fb2c3390cf03e0)
1c5937f2dSJun Nie /*
2c5937f2dSJun Nie  * Copyright (c) 2018-2019, ARM Limited and Contributors. All rights reserved.
3c5937f2dSJun Nie  *
4c5937f2dSJun Nie  * SPDX-License-Identifier: BSD-3-Clause
5c5937f2dSJun Nie  */
6c5937f2dSJun Nie 
7c5937f2dSJun Nie #include <assert.h>
8c5937f2dSJun Nie 
9c5937f2dSJun Nie #include <platform_def.h>
10c5937f2dSJun Nie 
11c5937f2dSJun Nie #include <arch_helpers.h>
12c5937f2dSJun Nie #include <common/bl_common.h>
13c5937f2dSJun Nie #include <common/debug.h>
14c5937f2dSJun Nie #include <common/desc_image_load.h>
15c5937f2dSJun Nie #include <drivers/mmc.h>
16c5937f2dSJun Nie #include <lib/xlat_tables/xlat_mmu_helpers.h>
17c5937f2dSJun Nie #include <lib/xlat_tables/xlat_tables_defs.h>
18c5937f2dSJun Nie #include <lib/mmio.h>
19c5937f2dSJun Nie #include <lib/optee_utils.h>
20c5937f2dSJun Nie #include <lib/utils.h>
21c5937f2dSJun Nie 
22c5937f2dSJun Nie #include <imx_aips.h>
23c5937f2dSJun Nie #include <imx_caam.h>
24c5937f2dSJun Nie #include <imx_clock.h>
25c5937f2dSJun Nie #include <imx_csu.h>
26c5937f2dSJun Nie #include <imx_gpt.h>
27c5937f2dSJun Nie #include <imx_uart.h>
28c5937f2dSJun Nie #include <imx_snvs.h>
29c5937f2dSJun Nie #include <imx_wdog.h>
30c5937f2dSJun Nie #include <imx7_def.h>
31c5937f2dSJun Nie 
32c5937f2dSJun Nie #ifndef AARCH32_SP_OPTEE
33c5937f2dSJun Nie #error "Must build with OPTEE support included"
34c5937f2dSJun Nie #endif
35c5937f2dSJun Nie 
plat_get_ns_image_entrypoint(void)36c5937f2dSJun Nie uintptr_t plat_get_ns_image_entrypoint(void)
37c5937f2dSJun Nie {
38c5937f2dSJun Nie 	return IMX7_UBOOT_BASE;
39c5937f2dSJun Nie }
40c5937f2dSJun Nie 
imx7_get_spsr_for_bl32_entry(void)41c5937f2dSJun Nie static uint32_t imx7_get_spsr_for_bl32_entry(void)
42c5937f2dSJun Nie {
43c5937f2dSJun Nie 	return SPSR_MODE32(MODE32_svc, SPSR_T_ARM, SPSR_E_LITTLE,
44c5937f2dSJun Nie 			   DISABLE_ALL_EXCEPTIONS);
45c5937f2dSJun Nie }
46c5937f2dSJun Nie 
imx7_get_spsr_for_bl33_entry(void)47c5937f2dSJun Nie static uint32_t imx7_get_spsr_for_bl33_entry(void)
48c5937f2dSJun Nie {
49c5937f2dSJun Nie 	return SPSR_MODE32(MODE32_svc,
50c5937f2dSJun Nie 			   plat_get_ns_image_entrypoint() & 0x1,
51c5937f2dSJun Nie 			   SPSR_E_LITTLE, DISABLE_ALL_EXCEPTIONS);
52c5937f2dSJun Nie }
53c5937f2dSJun Nie 
bl2_plat_handle_post_image_load(unsigned int image_id)54c5937f2dSJun Nie int bl2_plat_handle_post_image_load(unsigned int image_id)
55c5937f2dSJun Nie {
56c5937f2dSJun Nie 	int err = 0;
57c5937f2dSJun Nie 	bl_mem_params_node_t *bl_mem_params = get_bl_mem_params_node(image_id);
58c5937f2dSJun Nie 	bl_mem_params_node_t *hw_cfg_mem_params = NULL;
59c5937f2dSJun Nie 
60c5937f2dSJun Nie 	bl_mem_params_node_t *pager_mem_params = NULL;
61c5937f2dSJun Nie 	bl_mem_params_node_t *paged_mem_params = NULL;
62c5937f2dSJun Nie 
63c5937f2dSJun Nie 	assert(bl_mem_params);
64c5937f2dSJun Nie 
65c5937f2dSJun Nie 	switch (image_id) {
66c5937f2dSJun Nie 	case BL32_IMAGE_ID:
67c5937f2dSJun Nie 		pager_mem_params = get_bl_mem_params_node(BL32_EXTRA1_IMAGE_ID);
68c5937f2dSJun Nie 		assert(pager_mem_params);
69c5937f2dSJun Nie 
70c5937f2dSJun Nie 		paged_mem_params = get_bl_mem_params_node(BL32_EXTRA2_IMAGE_ID);
71c5937f2dSJun Nie 		assert(paged_mem_params);
72c5937f2dSJun Nie 
73c5937f2dSJun Nie 		err = parse_optee_header(&bl_mem_params->ep_info,
74c5937f2dSJun Nie 					 &pager_mem_params->image_info,
75c5937f2dSJun Nie 					 &paged_mem_params->image_info);
76c5937f2dSJun Nie 		if (err != 0)
77c5937f2dSJun Nie 			WARN("OPTEE header parse error.\n");
78c5937f2dSJun Nie 
79c5937f2dSJun Nie 		/*
80c5937f2dSJun Nie 		 * When ATF loads the DTB the address of the DTB is passed in
81c5937f2dSJun Nie 		 * arg2, if an hw config image is present use the base address
82c5937f2dSJun Nie 		 * as DTB address an pass it as arg2
83c5937f2dSJun Nie 		 */
84c5937f2dSJun Nie 		hw_cfg_mem_params = get_bl_mem_params_node(HW_CONFIG_ID);
85c5937f2dSJun Nie 
86c5937f2dSJun Nie 		bl_mem_params->ep_info.args.arg0 =
87c5937f2dSJun Nie 					bl_mem_params->ep_info.args.arg1;
88c5937f2dSJun Nie 		bl_mem_params->ep_info.args.arg1 = 0;
89c5937f2dSJun Nie 		if (hw_cfg_mem_params)
90c5937f2dSJun Nie 			bl_mem_params->ep_info.args.arg2 =
91c5937f2dSJun Nie 					hw_cfg_mem_params->image_info.image_base;
92c5937f2dSJun Nie 		else
93c5937f2dSJun Nie 			bl_mem_params->ep_info.args.arg2 = 0;
94c5937f2dSJun Nie 		bl_mem_params->ep_info.args.arg3 = 0;
95c5937f2dSJun Nie 		bl_mem_params->ep_info.spsr = imx7_get_spsr_for_bl32_entry();
96c5937f2dSJun Nie 		break;
97c5937f2dSJun Nie 
98c5937f2dSJun Nie 	case BL33_IMAGE_ID:
99c5937f2dSJun Nie 		/* AArch32 only core: OP-TEE expects NSec EP in register LR */
100c5937f2dSJun Nie 		pager_mem_params = get_bl_mem_params_node(BL32_IMAGE_ID);
101c5937f2dSJun Nie 		assert(pager_mem_params);
102c5937f2dSJun Nie 		pager_mem_params->ep_info.lr_svc = bl_mem_params->ep_info.pc;
103c5937f2dSJun Nie 
104c5937f2dSJun Nie 		/* BL33 expects to receive the primary CPU MPID (through r0) */
105c5937f2dSJun Nie 		bl_mem_params->ep_info.args.arg0 = 0xffff & read_mpidr();
106c5937f2dSJun Nie 		bl_mem_params->ep_info.spsr = imx7_get_spsr_for_bl33_entry();
107c5937f2dSJun Nie 		break;
108c5937f2dSJun Nie 
109c5937f2dSJun Nie 	default:
110c5937f2dSJun Nie 		/* Do nothing in default case */
111c5937f2dSJun Nie 		break;
112c5937f2dSJun Nie 	}
113c5937f2dSJun Nie 
114c5937f2dSJun Nie 	return err;
115c5937f2dSJun Nie }
116c5937f2dSJun Nie 
bl2_el3_plat_arch_setup(void)117c5937f2dSJun Nie void bl2_el3_plat_arch_setup(void)
118c5937f2dSJun Nie {
119c5937f2dSJun Nie 	/* Setup the MMU here */
120c5937f2dSJun Nie }
121c5937f2dSJun Nie 
imx7_setup_system_counter(void)122c5937f2dSJun Nie static void imx7_setup_system_counter(void)
123c5937f2dSJun Nie {
124c5937f2dSJun Nie 	unsigned long freq = SYS_COUNTER_FREQ_IN_TICKS;
125c5937f2dSJun Nie 
126c5937f2dSJun Nie 	/* Set the frequency table index to our target frequency */
127c5937f2dSJun Nie 	write_cntfrq(freq);
128c5937f2dSJun Nie 
129c5937f2dSJun Nie 	/* Enable system counter @ frequency table index 0, halt on debug */
130c5937f2dSJun Nie 	mmio_write_32(SYS_CNTCTL_BASE + CNTCR_OFF,
131c5937f2dSJun Nie 		      CNTCR_FCREQ(0) | CNTCR_HDBG | CNTCR_EN);
132c5937f2dSJun Nie }
133c5937f2dSJun Nie 
imx7_setup_wdog_clocks(void)134c5937f2dSJun Nie static void imx7_setup_wdog_clocks(void)
135c5937f2dSJun Nie {
136c5937f2dSJun Nie 	uint32_t wdog_en_bits = (uint32_t)WDOG_DEFAULT_CLK_SELECT;
137c5937f2dSJun Nie 
138c5937f2dSJun Nie 	imx_clock_set_wdog_clk_root_bits(wdog_en_bits);
139c5937f2dSJun Nie 	imx_clock_enable_wdog(0);
140c5937f2dSJun Nie 	imx_clock_enable_wdog(1);
141c5937f2dSJun Nie 	imx_clock_enable_wdog(2);
142c5937f2dSJun Nie 	imx_clock_enable_wdog(3);
143c5937f2dSJun Nie }
144c5937f2dSJun Nie 
145c5937f2dSJun Nie 
146c5937f2dSJun Nie /*
147c5937f2dSJun Nie  * bl2_el3_early_platform_setup()
148c5937f2dSJun Nie  * MMU off
149c5937f2dSJun Nie  */
bl2_el3_early_platform_setup(u_register_t arg1,u_register_t arg2,u_register_t arg3,u_register_t arg4)150c5937f2dSJun Nie void bl2_el3_early_platform_setup(u_register_t arg1, u_register_t arg2,
151c5937f2dSJun Nie 				  u_register_t arg3, u_register_t arg4)
152c5937f2dSJun Nie {
153d7873bcdSAndre Przywara 	static console_t console;
154c5937f2dSJun Nie 	int console_scope = CONSOLE_FLAG_BOOT | CONSOLE_FLAG_RUNTIME;
155c5937f2dSJun Nie 
156c5937f2dSJun Nie 	/* Initialize common components */
157c5937f2dSJun Nie 	imx_aips_init();
158c5937f2dSJun Nie 	imx_csu_init();
159c5937f2dSJun Nie 	imx_snvs_init();
160c5937f2dSJun Nie 	imx_gpt_ops_init(GPT1_BASE_ADDR);
161c5937f2dSJun Nie 	imx_clock_init();
162c5937f2dSJun Nie 	imx7_setup_system_counter();
163c5937f2dSJun Nie 	imx7_setup_wdog_clocks();
164c5937f2dSJun Nie 
165c5937f2dSJun Nie 	/* Platform specific setup */
166c5937f2dSJun Nie 	imx7_platform_setup(arg1, arg2, arg3, arg4);
167c5937f2dSJun Nie 
168c5937f2dSJun Nie 	/* Init UART, clock should be enabled in imx7_platform_setup() */
169c5937f2dSJun Nie 	console_imx_uart_register(PLAT_IMX7_BOOT_UART_BASE,
170c5937f2dSJun Nie 				  PLAT_IMX7_BOOT_UART_CLK_IN_HZ,
171c5937f2dSJun Nie 				  PLAT_IMX7_CONSOLE_BAUDRATE,
172c5937f2dSJun Nie 				  &console);
173d7873bcdSAndre Przywara 	console_set_scope(&console, console_scope);
174c5937f2dSJun Nie 
175c5937f2dSJun Nie 	/* Open handles to persistent storage */
176*81d1d86cSYing-Chun Liu (PaulLiu) 	plat_imx_io_setup();
177c5937f2dSJun Nie 
178c5937f2dSJun Nie 	/* Setup higher-level functionality CAAM, RTC etc */
179c5937f2dSJun Nie 	imx_caam_init();
180c5937f2dSJun Nie 	imx_wdog_init();
181c5937f2dSJun Nie 
182c5937f2dSJun Nie 	/* Print out the expected memory map */
183c5937f2dSJun Nie 	VERBOSE("\tOPTEE       0x%08x-0x%08x\n", IMX7_OPTEE_BASE, IMX7_OPTEE_LIMIT);
184c5937f2dSJun Nie 	VERBOSE("\tATF/BL2     0x%08x-0x%08x\n", BL2_RAM_BASE, BL2_RAM_LIMIT);
185c5937f2dSJun Nie 	VERBOSE("\tSHRAM       0x%08x-0x%08x\n", SHARED_RAM_BASE, SHARED_RAM_LIMIT);
186*81d1d86cSYing-Chun Liu (PaulLiu) 	VERBOSE("\tFIP         0x%08x-0x%08x\n", IMX_FIP_BASE, IMX_FIP_LIMIT);
187c5937f2dSJun Nie 	VERBOSE("\tDTB-OVERLAY 0x%08x-0x%08x\n", IMX7_DTB_OVERLAY_BASE, IMX7_DTB_OVERLAY_LIMIT);
188c5937f2dSJun Nie 	VERBOSE("\tDTB         0x%08x-0x%08x\n", IMX7_DTB_BASE, IMX7_DTB_LIMIT);
189c5937f2dSJun Nie 	VERBOSE("\tUBOOT/BL33  0x%08x-0x%08x\n", IMX7_UBOOT_BASE, IMX7_UBOOT_LIMIT);
190c5937f2dSJun Nie }
191c5937f2dSJun Nie 
192c5937f2dSJun Nie /*
193c5937f2dSJun Nie  * bl2_platform_setup()
194c5937f2dSJun Nie  * MMU on - enabled by bl2_el3_plat_arch_setup()
195c5937f2dSJun Nie  */
bl2_platform_setup(void)196c5937f2dSJun Nie void bl2_platform_setup(void)
197c5937f2dSJun Nie {
198c5937f2dSJun Nie }
199