xref: /arm-trusted-firmware/plat/qemu/qemu_sbsa/sbsa_sip_svc.c (revision 91f16700b400a8c0651d24a598fc48ee2997a0d7)
1*91f16700Schasinglulu /*
2*91f16700Schasinglulu  * Copyright (c) 2023, Linaro Limited and Contributors. All rights reserved.
3*91f16700Schasinglulu  *
4*91f16700Schasinglulu  * SPDX-License-Identifier: BSD-3-Clause
5*91f16700Schasinglulu  */
6*91f16700Schasinglulu 
7*91f16700Schasinglulu #include <assert.h>
8*91f16700Schasinglulu 
9*91f16700Schasinglulu #include <common/fdt_wrappers.h>
10*91f16700Schasinglulu #include <common/runtime_svc.h>
11*91f16700Schasinglulu #include <libfdt.h>
12*91f16700Schasinglulu #include <smccc_helpers.h>
13*91f16700Schasinglulu 
14*91f16700Schasinglulu /* default platform version is 0.0 */
15*91f16700Schasinglulu static int platform_version_major;
16*91f16700Schasinglulu static int platform_version_minor;
17*91f16700Schasinglulu 
18*91f16700Schasinglulu #define SMC_FASTCALL       0x80000000
19*91f16700Schasinglulu #define SMC64_FUNCTION     (SMC_FASTCALL   | 0x40000000)
20*91f16700Schasinglulu #define SIP_FUNCTION       (SMC64_FUNCTION | 0x02000000)
21*91f16700Schasinglulu #define SIP_FUNCTION_ID(n) (SIP_FUNCTION   | (n))
22*91f16700Schasinglulu 
23*91f16700Schasinglulu /*
24*91f16700Schasinglulu  * We do not use SMCCC_ARCH_SOC_ID here because qemu_sbsa is virtual platform
25*91f16700Schasinglulu  * which uses SoC present in QEMU. And they can change on their own while we
26*91f16700Schasinglulu  * need version of whole 'virtual hardware platform'.
27*91f16700Schasinglulu  */
28*91f16700Schasinglulu #define SIP_SVC_VERSION  SIP_FUNCTION_ID(1)
29*91f16700Schasinglulu #define SIP_SVC_GET_GIC  SIP_FUNCTION_ID(100)
30*91f16700Schasinglulu #define SIP_SVC_GET_GIC_ITS SIP_FUNCTION_ID(101)
31*91f16700Schasinglulu 
32*91f16700Schasinglulu static uint64_t gic_its_addr;
33*91f16700Schasinglulu 
34*91f16700Schasinglulu void sbsa_set_gic_bases(const uintptr_t gicd_base, const uintptr_t gicr_base);
35*91f16700Schasinglulu uintptr_t sbsa_get_gicd(void);
36*91f16700Schasinglulu uintptr_t sbsa_get_gicr(void);
37*91f16700Schasinglulu 
38*91f16700Schasinglulu void read_platform_config_from_dt(void *dtb)
39*91f16700Schasinglulu {
40*91f16700Schasinglulu 	int node;
41*91f16700Schasinglulu 	const fdt64_t *data;
42*91f16700Schasinglulu 	int err;
43*91f16700Schasinglulu 	uintptr_t gicd_base;
44*91f16700Schasinglulu 	uintptr_t gicr_base;
45*91f16700Schasinglulu 
46*91f16700Schasinglulu 	/*
47*91f16700Schasinglulu 	 * QEMU gives us this DeviceTree node:
48*91f16700Schasinglulu 	 *
49*91f16700Schasinglulu 	 * intc {
50*91f16700Schasinglulu 	 *	 reg = < 0x00 0x40060000 0x00 0x10000
51*91f16700Schasinglulu 	 *		 0x00 0x40080000 0x00 0x4000000>;
52*91f16700Schasinglulu 	 *       its {
53*91f16700Schasinglulu 	 *               reg = <0x00 0x44081000 0x00 0x20000>;
54*91f16700Schasinglulu 	 *       };
55*91f16700Schasinglulu 	 * };
56*91f16700Schasinglulu 	 */
57*91f16700Schasinglulu 	node = fdt_path_offset(dtb, "/intc");
58*91f16700Schasinglulu 	if (node < 0) {
59*91f16700Schasinglulu 		return;
60*91f16700Schasinglulu 	}
61*91f16700Schasinglulu 
62*91f16700Schasinglulu 	data = fdt_getprop(dtb, node, "reg", NULL);
63*91f16700Schasinglulu 	if (data == NULL) {
64*91f16700Schasinglulu 		return;
65*91f16700Schasinglulu 	}
66*91f16700Schasinglulu 
67*91f16700Schasinglulu 	err = fdt_get_reg_props_by_index(dtb, node, 0, &gicd_base, NULL);
68*91f16700Schasinglulu 	if (err < 0) {
69*91f16700Schasinglulu 		ERROR("Failed to read GICD reg property of GIC node\n");
70*91f16700Schasinglulu 		return;
71*91f16700Schasinglulu 	}
72*91f16700Schasinglulu 	INFO("GICD base = 0x%lx\n", gicd_base);
73*91f16700Schasinglulu 
74*91f16700Schasinglulu 	err = fdt_get_reg_props_by_index(dtb, node, 1, &gicr_base, NULL);
75*91f16700Schasinglulu 	if (err < 0) {
76*91f16700Schasinglulu 		ERROR("Failed to read GICR reg property of GIC node\n");
77*91f16700Schasinglulu 		return;
78*91f16700Schasinglulu 	}
79*91f16700Schasinglulu 	INFO("GICR base = 0x%lx\n", gicr_base);
80*91f16700Schasinglulu 
81*91f16700Schasinglulu 	sbsa_set_gic_bases(gicd_base, gicr_base);
82*91f16700Schasinglulu 
83*91f16700Schasinglulu 	node = fdt_path_offset(dtb, "/intc/its");
84*91f16700Schasinglulu 	if (node < 0) {
85*91f16700Schasinglulu 		return;
86*91f16700Schasinglulu 	}
87*91f16700Schasinglulu 
88*91f16700Schasinglulu 	err = fdt_get_reg_props_by_index(dtb, node, 0, &gic_its_addr, NULL);
89*91f16700Schasinglulu 	if (err < 0) {
90*91f16700Schasinglulu 		ERROR("Failed to read GICI reg property of GIC node\n");
91*91f16700Schasinglulu 		return;
92*91f16700Schasinglulu 	}
93*91f16700Schasinglulu 	INFO("GICI base = 0x%lx\n", gic_its_addr);
94*91f16700Schasinglulu }
95*91f16700Schasinglulu 
96*91f16700Schasinglulu void read_platform_version(void *dtb)
97*91f16700Schasinglulu {
98*91f16700Schasinglulu 	int node;
99*91f16700Schasinglulu 
100*91f16700Schasinglulu 	node = fdt_path_offset(dtb, "/");
101*91f16700Schasinglulu 	if (node >= 0) {
102*91f16700Schasinglulu 		platform_version_major = fdt32_ld(fdt_getprop(dtb, node,
103*91f16700Schasinglulu 							      "machine-version-major", NULL));
104*91f16700Schasinglulu 		platform_version_minor = fdt32_ld(fdt_getprop(dtb, node,
105*91f16700Schasinglulu 							      "machine-version-minor", NULL));
106*91f16700Schasinglulu 	}
107*91f16700Schasinglulu }
108*91f16700Schasinglulu 
109*91f16700Schasinglulu void sip_svc_init(void)
110*91f16700Schasinglulu {
111*91f16700Schasinglulu 	/* Read DeviceTree data before MMU is enabled */
112*91f16700Schasinglulu 
113*91f16700Schasinglulu 	void *dtb = (void *)(uintptr_t)ARM_PRELOADED_DTB_BASE;
114*91f16700Schasinglulu 	int err;
115*91f16700Schasinglulu 
116*91f16700Schasinglulu 	err = fdt_open_into(dtb, dtb, PLAT_QEMU_DT_MAX_SIZE);
117*91f16700Schasinglulu 	if (err < 0) {
118*91f16700Schasinglulu 		ERROR("Invalid Device Tree at %p: error %d\n", dtb, err);
119*91f16700Schasinglulu 		return;
120*91f16700Schasinglulu 	}
121*91f16700Schasinglulu 
122*91f16700Schasinglulu 	err = fdt_check_header(dtb);
123*91f16700Schasinglulu 	if (err < 0) {
124*91f16700Schasinglulu 		ERROR("Invalid DTB file passed\n");
125*91f16700Schasinglulu 		return;
126*91f16700Schasinglulu 	}
127*91f16700Schasinglulu 
128*91f16700Schasinglulu 	read_platform_version(dtb);
129*91f16700Schasinglulu 	INFO("Platform version: %d.%d\n", platform_version_major, platform_version_minor);
130*91f16700Schasinglulu 
131*91f16700Schasinglulu 	read_platform_config_from_dt(dtb);
132*91f16700Schasinglulu }
133*91f16700Schasinglulu 
134*91f16700Schasinglulu /*
135*91f16700Schasinglulu  * This function is responsible for handling all SiP calls from the NS world
136*91f16700Schasinglulu  */
137*91f16700Schasinglulu uintptr_t sbsa_sip_smc_handler(uint32_t smc_fid,
138*91f16700Schasinglulu 			       u_register_t x1,
139*91f16700Schasinglulu 			       u_register_t x2,
140*91f16700Schasinglulu 			       u_register_t x3,
141*91f16700Schasinglulu 			       u_register_t x4,
142*91f16700Schasinglulu 			       void *cookie,
143*91f16700Schasinglulu 			       void *handle,
144*91f16700Schasinglulu 			       u_register_t flags)
145*91f16700Schasinglulu {
146*91f16700Schasinglulu 	uint32_t ns;
147*91f16700Schasinglulu 
148*91f16700Schasinglulu 	/* Determine which security state this SMC originated from */
149*91f16700Schasinglulu 	ns = is_caller_non_secure(flags);
150*91f16700Schasinglulu 	if (!ns) {
151*91f16700Schasinglulu 		ERROR("%s: wrong world SMC (0x%x)\n", __func__, smc_fid);
152*91f16700Schasinglulu 		SMC_RET1(handle, SMC_UNK);
153*91f16700Schasinglulu 	}
154*91f16700Schasinglulu 
155*91f16700Schasinglulu 	switch (smc_fid) {
156*91f16700Schasinglulu 	case SIP_SVC_VERSION:
157*91f16700Schasinglulu 		INFO("Platform version requested\n");
158*91f16700Schasinglulu 		SMC_RET3(handle, NULL, platform_version_major, platform_version_minor);
159*91f16700Schasinglulu 
160*91f16700Schasinglulu 	case SIP_SVC_GET_GIC:
161*91f16700Schasinglulu 		SMC_RET3(handle, NULL, sbsa_get_gicd(), sbsa_get_gicr());
162*91f16700Schasinglulu 
163*91f16700Schasinglulu 	case SIP_SVC_GET_GIC_ITS:
164*91f16700Schasinglulu 		SMC_RET2(handle, NULL, gic_its_addr);
165*91f16700Schasinglulu 
166*91f16700Schasinglulu 	default:
167*91f16700Schasinglulu 		ERROR("%s: unhandled SMC (0x%x) (function id: %d)\n", __func__, smc_fid,
168*91f16700Schasinglulu 		      smc_fid - SIP_FUNCTION);
169*91f16700Schasinglulu 		SMC_RET1(handle, SMC_UNK);
170*91f16700Schasinglulu 	}
171*91f16700Schasinglulu }
172*91f16700Schasinglulu 
173*91f16700Schasinglulu int sbsa_sip_smc_setup(void)
174*91f16700Schasinglulu {
175*91f16700Schasinglulu 	return 0;
176*91f16700Schasinglulu }
177*91f16700Schasinglulu 
178*91f16700Schasinglulu /* Define a runtime service descriptor for fast SMC calls */
179*91f16700Schasinglulu DECLARE_RT_SVC(
180*91f16700Schasinglulu 	sbsa_sip_svc,
181*91f16700Schasinglulu 	OEN_SIP_START,
182*91f16700Schasinglulu 	OEN_SIP_END,
183*91f16700Schasinglulu 	SMC_TYPE_FAST,
184*91f16700Schasinglulu 	sbsa_sip_smc_setup,
185*91f16700Schasinglulu 	sbsa_sip_smc_handler
186*91f16700Schasinglulu );
187