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