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