xref: /rk3399_ARM-atf/plat/qemu/qemu_sbsa/sbsa_sip_svc.c (revision 4171e981d13e6aa764c2520a2b513beafe449818)
1c681d02cSMarcin Juszkiewicz /*
2c681d02cSMarcin Juszkiewicz  * Copyright (c) 2023, Linaro Limited and Contributors. All rights reserved.
3c681d02cSMarcin Juszkiewicz  *
4c681d02cSMarcin Juszkiewicz  * SPDX-License-Identifier: BSD-3-Clause
5c681d02cSMarcin Juszkiewicz  */
6c681d02cSMarcin Juszkiewicz 
7c681d02cSMarcin Juszkiewicz #include <assert.h>
8c681d02cSMarcin Juszkiewicz 
91e67b1b1SMarcin Juszkiewicz #include <common/fdt_wrappers.h>
10c681d02cSMarcin Juszkiewicz #include <common/runtime_svc.h>
11c681d02cSMarcin Juszkiewicz #include <libfdt.h>
12c681d02cSMarcin Juszkiewicz #include <smccc_helpers.h>
13c681d02cSMarcin Juszkiewicz 
14c681d02cSMarcin Juszkiewicz /* default platform version is 0.0 */
15c681d02cSMarcin Juszkiewicz static int platform_version_major;
16c681d02cSMarcin Juszkiewicz static int platform_version_minor;
17c681d02cSMarcin Juszkiewicz 
18c681d02cSMarcin Juszkiewicz #define SMC_FASTCALL       0x80000000
19c681d02cSMarcin Juszkiewicz #define SMC64_FUNCTION     (SMC_FASTCALL   | 0x40000000)
20c681d02cSMarcin Juszkiewicz #define SIP_FUNCTION       (SMC64_FUNCTION | 0x02000000)
21c681d02cSMarcin Juszkiewicz #define SIP_FUNCTION_ID(n) (SIP_FUNCTION   | (n))
22c681d02cSMarcin Juszkiewicz 
23c681d02cSMarcin Juszkiewicz /*
24c681d02cSMarcin Juszkiewicz  * We do not use SMCCC_ARCH_SOC_ID here because qemu_sbsa is virtual platform
25c681d02cSMarcin Juszkiewicz  * which uses SoC present in QEMU. And they can change on their own while we
26c681d02cSMarcin Juszkiewicz  * need version of whole 'virtual hardware platform'.
27c681d02cSMarcin Juszkiewicz  */
28c681d02cSMarcin Juszkiewicz #define SIP_SVC_VERSION  SIP_FUNCTION_ID(1)
291e67b1b1SMarcin Juszkiewicz #define SIP_SVC_GET_GIC  SIP_FUNCTION_ID(100)
30*4171e981SMarcin Juszkiewicz #define SIP_SVC_GET_GIC_ITS SIP_FUNCTION_ID(101)
31*4171e981SMarcin Juszkiewicz 
32*4171e981SMarcin Juszkiewicz static uint64_t gic_its_addr;
331e67b1b1SMarcin Juszkiewicz 
341e67b1b1SMarcin Juszkiewicz void sbsa_set_gic_bases(const uintptr_t gicd_base, const uintptr_t gicr_base);
351e67b1b1SMarcin Juszkiewicz uintptr_t sbsa_get_gicd(void);
361e67b1b1SMarcin Juszkiewicz uintptr_t sbsa_get_gicr(void);
371e67b1b1SMarcin Juszkiewicz 
381e67b1b1SMarcin Juszkiewicz void read_platform_config_from_dt(void *dtb)
391e67b1b1SMarcin Juszkiewicz {
401e67b1b1SMarcin Juszkiewicz 	int node;
411e67b1b1SMarcin Juszkiewicz 	const fdt64_t *data;
421e67b1b1SMarcin Juszkiewicz 	int err;
431e67b1b1SMarcin Juszkiewicz 	uintptr_t gicd_base;
441e67b1b1SMarcin Juszkiewicz 	uintptr_t gicr_base;
451e67b1b1SMarcin Juszkiewicz 
461e67b1b1SMarcin Juszkiewicz 	/*
471e67b1b1SMarcin Juszkiewicz 	 * QEMU gives us this DeviceTree node:
481e67b1b1SMarcin Juszkiewicz 	 *
491e67b1b1SMarcin Juszkiewicz 	 * intc {
50*4171e981SMarcin Juszkiewicz 	 *	 reg = < 0x00 0x40060000 0x00 0x10000
51*4171e981SMarcin Juszkiewicz 	 *		 0x00 0x40080000 0x00 0x4000000>;
52*4171e981SMarcin Juszkiewicz 	 *       its {
53*4171e981SMarcin Juszkiewicz 	 *               reg = <0x00 0x44081000 0x00 0x20000>;
54*4171e981SMarcin Juszkiewicz 	 *       };
55*4171e981SMarcin Juszkiewicz 	 * };
561e67b1b1SMarcin Juszkiewicz 	 */
571e67b1b1SMarcin Juszkiewicz 	node = fdt_path_offset(dtb, "/intc");
581e67b1b1SMarcin Juszkiewicz 	if (node < 0) {
591e67b1b1SMarcin Juszkiewicz 		return;
601e67b1b1SMarcin Juszkiewicz 	}
611e67b1b1SMarcin Juszkiewicz 
621e67b1b1SMarcin Juszkiewicz 	data = fdt_getprop(dtb, node, "reg", NULL);
631e67b1b1SMarcin Juszkiewicz 	if (data == NULL) {
641e67b1b1SMarcin Juszkiewicz 		return;
651e67b1b1SMarcin Juszkiewicz 	}
661e67b1b1SMarcin Juszkiewicz 
671e67b1b1SMarcin Juszkiewicz 	err = fdt_get_reg_props_by_index(dtb, node, 0, &gicd_base, NULL);
681e67b1b1SMarcin Juszkiewicz 	if (err < 0) {
691e67b1b1SMarcin Juszkiewicz 		ERROR("Failed to read GICD reg property of GIC node\n");
701e67b1b1SMarcin Juszkiewicz 		return;
711e67b1b1SMarcin Juszkiewicz 	}
721e67b1b1SMarcin Juszkiewicz 	INFO("GICD base = 0x%lx\n", gicd_base);
731e67b1b1SMarcin Juszkiewicz 
741e67b1b1SMarcin Juszkiewicz 	err = fdt_get_reg_props_by_index(dtb, node, 1, &gicr_base, NULL);
751e67b1b1SMarcin Juszkiewicz 	if (err < 0) {
761e67b1b1SMarcin Juszkiewicz 		ERROR("Failed to read GICR reg property of GIC node\n");
771e67b1b1SMarcin Juszkiewicz 		return;
781e67b1b1SMarcin Juszkiewicz 	}
791e67b1b1SMarcin Juszkiewicz 	INFO("GICR base = 0x%lx\n", gicr_base);
801e67b1b1SMarcin Juszkiewicz 
811e67b1b1SMarcin Juszkiewicz 	sbsa_set_gic_bases(gicd_base, gicr_base);
82*4171e981SMarcin Juszkiewicz 
83*4171e981SMarcin Juszkiewicz 	node = fdt_path_offset(dtb, "/intc/its");
84*4171e981SMarcin Juszkiewicz 	if (node < 0) {
85*4171e981SMarcin Juszkiewicz 		return;
86*4171e981SMarcin Juszkiewicz 	}
87*4171e981SMarcin Juszkiewicz 
88*4171e981SMarcin Juszkiewicz 	err = fdt_get_reg_props_by_index(dtb, node, 0, &gic_its_addr, NULL);
89*4171e981SMarcin Juszkiewicz 	if (err < 0) {
90*4171e981SMarcin Juszkiewicz 		ERROR("Failed to read GICI reg property of GIC node\n");
91*4171e981SMarcin Juszkiewicz 		return;
92*4171e981SMarcin Juszkiewicz 	}
93*4171e981SMarcin Juszkiewicz 	INFO("GICI base = 0x%lx\n", gic_its_addr);
941e67b1b1SMarcin Juszkiewicz }
951e67b1b1SMarcin Juszkiewicz 
96c681d02cSMarcin Juszkiewicz void read_platform_version(void *dtb)
97c681d02cSMarcin Juszkiewicz {
98c681d02cSMarcin Juszkiewicz 	int node;
99c681d02cSMarcin Juszkiewicz 
100c681d02cSMarcin Juszkiewicz 	node = fdt_path_offset(dtb, "/");
101c681d02cSMarcin Juszkiewicz 	if (node >= 0) {
102c681d02cSMarcin Juszkiewicz 		platform_version_major = fdt32_ld(fdt_getprop(dtb, node,
103c681d02cSMarcin Juszkiewicz 							      "machine-version-major", NULL));
104c681d02cSMarcin Juszkiewicz 		platform_version_minor = fdt32_ld(fdt_getprop(dtb, node,
105c681d02cSMarcin Juszkiewicz 							      "machine-version-minor", NULL));
106c681d02cSMarcin Juszkiewicz 	}
107c681d02cSMarcin Juszkiewicz }
108c681d02cSMarcin Juszkiewicz 
109c681d02cSMarcin Juszkiewicz void sip_svc_init(void)
110c681d02cSMarcin Juszkiewicz {
111c681d02cSMarcin Juszkiewicz 	/* Read DeviceTree data before MMU is enabled */
112c681d02cSMarcin Juszkiewicz 
113c681d02cSMarcin Juszkiewicz 	void *dtb = (void *)(uintptr_t)ARM_PRELOADED_DTB_BASE;
114c681d02cSMarcin Juszkiewicz 	int err;
115c681d02cSMarcin Juszkiewicz 
116c681d02cSMarcin Juszkiewicz 	err = fdt_open_into(dtb, dtb, PLAT_QEMU_DT_MAX_SIZE);
117c681d02cSMarcin Juszkiewicz 	if (err < 0) {
118c681d02cSMarcin Juszkiewicz 		ERROR("Invalid Device Tree at %p: error %d\n", dtb, err);
119c681d02cSMarcin Juszkiewicz 		return;
120c681d02cSMarcin Juszkiewicz 	}
121c681d02cSMarcin Juszkiewicz 
122c681d02cSMarcin Juszkiewicz 	err = fdt_check_header(dtb);
123c681d02cSMarcin Juszkiewicz 	if (err < 0) {
124c681d02cSMarcin Juszkiewicz 		ERROR("Invalid DTB file passed\n");
125c681d02cSMarcin Juszkiewicz 		return;
126c681d02cSMarcin Juszkiewicz 	}
127c681d02cSMarcin Juszkiewicz 
128c681d02cSMarcin Juszkiewicz 	read_platform_version(dtb);
129c681d02cSMarcin Juszkiewicz 	INFO("Platform version: %d.%d\n", platform_version_major, platform_version_minor);
1301e67b1b1SMarcin Juszkiewicz 
1311e67b1b1SMarcin Juszkiewicz 	read_platform_config_from_dt(dtb);
132c681d02cSMarcin Juszkiewicz }
133c681d02cSMarcin Juszkiewicz 
134c681d02cSMarcin Juszkiewicz /*
135c681d02cSMarcin Juszkiewicz  * This function is responsible for handling all SiP calls from the NS world
136c681d02cSMarcin Juszkiewicz  */
137c681d02cSMarcin Juszkiewicz uintptr_t sbsa_sip_smc_handler(uint32_t smc_fid,
138c681d02cSMarcin Juszkiewicz 			       u_register_t x1,
139c681d02cSMarcin Juszkiewicz 			       u_register_t x2,
140c681d02cSMarcin Juszkiewicz 			       u_register_t x3,
141c681d02cSMarcin Juszkiewicz 			       u_register_t x4,
142c681d02cSMarcin Juszkiewicz 			       void *cookie,
143c681d02cSMarcin Juszkiewicz 			       void *handle,
144c681d02cSMarcin Juszkiewicz 			       u_register_t flags)
145c681d02cSMarcin Juszkiewicz {
146c681d02cSMarcin Juszkiewicz 	uint32_t ns;
147c681d02cSMarcin Juszkiewicz 
148c681d02cSMarcin Juszkiewicz 	/* Determine which security state this SMC originated from */
149c681d02cSMarcin Juszkiewicz 	ns = is_caller_non_secure(flags);
150c681d02cSMarcin Juszkiewicz 	if (!ns) {
151c681d02cSMarcin Juszkiewicz 		ERROR("%s: wrong world SMC (0x%x)\n", __func__, smc_fid);
152c681d02cSMarcin Juszkiewicz 		SMC_RET1(handle, SMC_UNK);
153c681d02cSMarcin Juszkiewicz 	}
154c681d02cSMarcin Juszkiewicz 
155c681d02cSMarcin Juszkiewicz 	switch (smc_fid) {
156c681d02cSMarcin Juszkiewicz 	case SIP_SVC_VERSION:
157c681d02cSMarcin Juszkiewicz 		INFO("Platform version requested\n");
158c681d02cSMarcin Juszkiewicz 		SMC_RET3(handle, NULL, platform_version_major, platform_version_minor);
159c681d02cSMarcin Juszkiewicz 
1601e67b1b1SMarcin Juszkiewicz 	case SIP_SVC_GET_GIC:
1611e67b1b1SMarcin Juszkiewicz 		SMC_RET3(handle, NULL, sbsa_get_gicd(), sbsa_get_gicr());
1621e67b1b1SMarcin Juszkiewicz 
163*4171e981SMarcin Juszkiewicz 	case SIP_SVC_GET_GIC_ITS:
164*4171e981SMarcin Juszkiewicz 		SMC_RET2(handle, NULL, gic_its_addr);
165*4171e981SMarcin Juszkiewicz 
166c681d02cSMarcin Juszkiewicz 	default:
167c681d02cSMarcin Juszkiewicz 		ERROR("%s: unhandled SMC (0x%x) (function id: %d)\n", __func__, smc_fid,
168c681d02cSMarcin Juszkiewicz 		      smc_fid - SIP_FUNCTION);
169c681d02cSMarcin Juszkiewicz 		SMC_RET1(handle, SMC_UNK);
170c681d02cSMarcin Juszkiewicz 	}
171c681d02cSMarcin Juszkiewicz }
172c681d02cSMarcin Juszkiewicz 
173c681d02cSMarcin Juszkiewicz int sbsa_sip_smc_setup(void)
174c681d02cSMarcin Juszkiewicz {
175c681d02cSMarcin Juszkiewicz 	return 0;
176c681d02cSMarcin Juszkiewicz }
177c681d02cSMarcin Juszkiewicz 
178c681d02cSMarcin Juszkiewicz /* Define a runtime service descriptor for fast SMC calls */
179c681d02cSMarcin Juszkiewicz DECLARE_RT_SVC(
180c681d02cSMarcin Juszkiewicz 	sbsa_sip_svc,
181c681d02cSMarcin Juszkiewicz 	OEN_SIP_START,
182c681d02cSMarcin Juszkiewicz 	OEN_SIP_END,
183c681d02cSMarcin Juszkiewicz 	SMC_TYPE_FAST,
184c681d02cSMarcin Juszkiewicz 	sbsa_sip_smc_setup,
185c681d02cSMarcin Juszkiewicz 	sbsa_sip_smc_handler
186c681d02cSMarcin Juszkiewicz );
187