/* * Copyright (c) 2023, Linaro Limited and Contributors. All rights reserved. * * SPDX-License-Identifier: BSD-3-Clause */ #include #include #include #include #include /* default platform version is 0.0 */ static int platform_version_major; static int platform_version_minor; #define SMC_FASTCALL 0x80000000 #define SMC64_FUNCTION (SMC_FASTCALL | 0x40000000) #define SIP_FUNCTION (SMC64_FUNCTION | 0x02000000) #define SIP_FUNCTION_ID(n) (SIP_FUNCTION | (n)) /* * We do not use SMCCC_ARCH_SOC_ID here because qemu_sbsa is virtual platform * which uses SoC present in QEMU. And they can change on their own while we * need version of whole 'virtual hardware platform'. */ #define SIP_SVC_VERSION SIP_FUNCTION_ID(1) #define SIP_SVC_GET_GIC SIP_FUNCTION_ID(100) #define SIP_SVC_GET_GIC_ITS SIP_FUNCTION_ID(101) #define SIP_SVC_GET_CPU_COUNT SIP_FUNCTION_ID(200) #define SIP_SVC_GET_CPU_NODE SIP_FUNCTION_ID(201) #define SIP_SVC_GET_CPU_TOPOLOGY SIP_FUNCTION_ID(202) #define SIP_SVC_GET_MEMORY_NODE_COUNT SIP_FUNCTION_ID(300) #define SIP_SVC_GET_MEMORY_NODE SIP_FUNCTION_ID(301) static uint64_t gic_its_addr; typedef struct { uint32_t nodeid; uint32_t mpidr; } cpu_data; typedef struct{ uint32_t nodeid; uint64_t addr_base; uint64_t addr_size; } memory_data; /* * sockets: the number of sockets on sbsa-ref platform. * clusters: the number of clusters in one socket. * cores: the number of cores in one cluster. * threads: the number of threads in one core. */ typedef struct { uint32_t sockets; uint32_t clusters; uint32_t cores; uint32_t threads; } cpu_topology; static struct { uint32_t num_cpus; uint32_t num_memnodes; cpu_data cpu[PLATFORM_CORE_COUNT]; cpu_topology cpu_topo; memory_data memory[PLAT_MAX_MEM_NODES]; } dynamic_platform_info; void sbsa_set_gic_bases(const uintptr_t gicd_base, const uintptr_t gicr_base); uintptr_t sbsa_get_gicd(void); uintptr_t sbsa_get_gicr(void); /* * QEMU provides us with minimal information about hardware platform using * minimalistic DeviceTree. This is not a Linux DeviceTree. It is not even * a firmware DeviceTree. * * It is information passed from QEMU to describe the information a hardware * platform would have other mechanisms to discover at runtime, that are * affected by the QEMU command line. * * Ultimately this device tree will be replaced by IPC calls to an emulated SCP. * And when we do that, we won't then have to rewrite Normal world firmware to * cope. */ static void read_cpu_topology_from_dt(void *dtb) { int node; /* * QEMU gives us this DeviceTree node when we config: * -smp 16,sockets=2,clusters=2,cores=2,threads=2 * * topology { * threads = <0x02>; * cores = <0x02>; * clusters = <0x02>; * sockets = <0x02>; * }; */ node = fdt_path_offset(dtb, "/cpus/topology"); if (node > 0) { dynamic_platform_info.cpu_topo.sockets = fdt_read_uint32_default(dtb, node, "sockets", 0); dynamic_platform_info.cpu_topo.clusters = fdt_read_uint32_default(dtb, node, "clusters", 0); dynamic_platform_info.cpu_topo.cores = fdt_read_uint32_default(dtb, node, "cores", 0); dynamic_platform_info.cpu_topo.threads = fdt_read_uint32_default(dtb, node, "threads", 0); } INFO("Cpu topology: sockets: %d, clusters: %d, cores: %d, threads: %d\n", dynamic_platform_info.cpu_topo.sockets, dynamic_platform_info.cpu_topo.clusters, dynamic_platform_info.cpu_topo.cores, dynamic_platform_info.cpu_topo.threads); } void read_cpuinfo_from_dt(void *dtb) { int node; int prev; int cpu = 0; uintptr_t mpidr; /* * QEMU gives us this DeviceTree node: * numa-node-id entries are only when NUMA config is used * * cpus { * #size-cells = <0x00>; * #address-cells = <0x02>; * * cpu@0 { * numa-node-id = <0x00>; * reg = <0x00 0x00>; * }; * * cpu@1 { * numa-node-id = <0x03>; * reg = <0x00 0x01>; * }; * }; */ node = fdt_path_offset(dtb, "/cpus"); if (node < 0) { ERROR("No information about cpus in DeviceTree.\n"); panic(); } /* * QEMU numbers cpus from 0 and there can be /cpus/cpu-map present so we * cannot use fdt_first_subnode() here */ node = fdt_path_offset(dtb, "/cpus/cpu@0"); while (node > 0) { if (fdt_getprop(dtb, node, "reg", NULL)) { fdt_get_reg_props_by_index(dtb, node, 0, &mpidr, NULL); } else { ERROR("Incomplete information for cpu %d in DeviceTree.\n", cpu); panic(); } dynamic_platform_info.cpu[cpu].mpidr = mpidr; dynamic_platform_info.cpu[cpu].nodeid = fdt_read_uint32_default(dtb, node, "numa-node-id", 0); INFO("CPU %d: node-id: %d, mpidr: %ld\n", cpu, dynamic_platform_info.cpu[cpu].nodeid, mpidr); cpu++; prev = node; node = fdt_next_subnode(dtb, prev); } dynamic_platform_info.num_cpus = cpu; INFO("Found %d cpus\n", dynamic_platform_info.num_cpus); read_cpu_topology_from_dt(dtb); } void read_meminfo_from_dt(void *dtb) { const fdt32_t *prop; const char *type; int prev, node; int len; uint32_t memnode = 0; uint32_t higher_value, lower_value; uint64_t cur_base, cur_size; /* * QEMU gives us this DeviceTree node: * * memory@100c0000000 { * numa-node-id = <0x01>; * reg = <0x100 0xc0000000 0x00 0x40000000>; * device_type = "memory"; * }; * * memory@10000000000 { * numa-node-id = <0x00>; * reg = <0x100 0x00 0x00 0xc0000000>; * device_type = "memory"; * } */ for (prev = 0;; prev = node) { node = fdt_next_node(dtb, prev, NULL); if (node < 0) { break; } type = fdt_getprop(dtb, node, "device_type", &len); if (type && strncmp(type, "memory", len) == 0) { dynamic_platform_info.memory[memnode].nodeid = fdt_read_uint32_default(dtb, node, "numa-node-id", 0); /* * Get the 'reg' property of this node and * assume two 8 bytes for base and size. */ prop = fdt_getprop(dtb, node, "reg", &len); if (prop != 0 && len == (2 * sizeof(int64_t))) { higher_value = fdt32_to_cpu(*prop); lower_value = fdt32_to_cpu(*(prop + 1)); cur_base = (uint64_t)(lower_value | ((uint64_t)higher_value) << 32); higher_value = fdt32_to_cpu(*(prop + 2)); lower_value = fdt32_to_cpu(*(prop + 3)); cur_size = (uint64_t)(lower_value | ((uint64_t)higher_value) << 32); dynamic_platform_info.memory[memnode].addr_base = cur_base; dynamic_platform_info.memory[memnode].addr_size = cur_size; INFO("RAM %d: node-id: %d, address: 0x%lx - 0x%lx\n", memnode, dynamic_platform_info.memory[memnode].nodeid, dynamic_platform_info.memory[memnode].addr_base, dynamic_platform_info.memory[memnode].addr_base + dynamic_platform_info.memory[memnode].addr_size - 1); } memnode++; } } dynamic_platform_info.num_memnodes = memnode; } void read_platform_config_from_dt(void *dtb) { int node; const fdt64_t *data; int err; uintptr_t gicd_base; uintptr_t gicr_base; /* * QEMU gives us this DeviceTree node: * * intc { * reg = < 0x00 0x40060000 0x00 0x10000 * 0x00 0x40080000 0x00 0x4000000>; * its { * reg = <0x00 0x44081000 0x00 0x20000>; * }; * }; */ node = fdt_path_offset(dtb, "/intc"); if (node < 0) { return; } data = fdt_getprop(dtb, node, "reg", NULL); if (data == NULL) { return; } err = fdt_get_reg_props_by_index(dtb, node, 0, &gicd_base, NULL); if (err < 0) { ERROR("Failed to read GICD reg property of GIC node\n"); return; } INFO("GICD base = 0x%lx\n", gicd_base); err = fdt_get_reg_props_by_index(dtb, node, 1, &gicr_base, NULL); if (err < 0) { ERROR("Failed to read GICR reg property of GIC node\n"); return; } INFO("GICR base = 0x%lx\n", gicr_base); sbsa_set_gic_bases(gicd_base, gicr_base); node = fdt_path_offset(dtb, "/intc/its"); if (node < 0) { return; } err = fdt_get_reg_props_by_index(dtb, node, 0, &gic_its_addr, NULL); if (err < 0) { ERROR("Failed to read GICI reg property of GIC node\n"); return; } INFO("GICI base = 0x%lx\n", gic_its_addr); } void read_platform_version(void *dtb) { int node; node = fdt_path_offset(dtb, "/"); if (node >= 0) { platform_version_major = fdt_read_uint32_default(dtb, node, "machine-version-major", 0); platform_version_minor = fdt_read_uint32_default(dtb, node, "machine-version-minor", 0); } } void sip_svc_init(void) { /* Read DeviceTree data before MMU is enabled */ void *dtb = (void *)(uintptr_t)ARM_PRELOADED_DTB_BASE; int err; err = fdt_open_into(dtb, dtb, PLAT_QEMU_DT_MAX_SIZE); if (err < 0) { ERROR("Invalid Device Tree at %p: error %d\n", dtb, err); return; } err = fdt_check_header(dtb); if (err < 0) { ERROR("Invalid DTB file passed\n"); return; } read_platform_version(dtb); INFO("Platform version: %d.%d\n", platform_version_major, platform_version_minor); read_platform_config_from_dt(dtb); read_cpuinfo_from_dt(dtb); read_meminfo_from_dt(dtb); } /* * This function is responsible for handling all SiP calls from the NS world */ uintptr_t sbsa_sip_smc_handler(uint32_t smc_fid, u_register_t x1, u_register_t x2, u_register_t x3, u_register_t x4, void *cookie, void *handle, u_register_t flags) { uint32_t ns; uint64_t index; /* Determine which security state this SMC originated from */ ns = is_caller_non_secure(flags); if (!ns) { ERROR("%s: wrong world SMC (0x%x)\n", __func__, smc_fid); SMC_RET1(handle, SMC_UNK); } switch (smc_fid) { case SIP_SVC_VERSION: INFO("Platform version requested\n"); SMC_RET3(handle, NULL, platform_version_major, platform_version_minor); case SIP_SVC_GET_GIC: SMC_RET3(handle, NULL, sbsa_get_gicd(), sbsa_get_gicr()); case SIP_SVC_GET_GIC_ITS: SMC_RET2(handle, NULL, gic_its_addr); case SIP_SVC_GET_CPU_COUNT: SMC_RET2(handle, NULL, dynamic_platform_info.num_cpus); case SIP_SVC_GET_CPU_NODE: index = x1; if (index < PLATFORM_CORE_COUNT) { SMC_RET3(handle, NULL, dynamic_platform_info.cpu[index].nodeid, dynamic_platform_info.cpu[index].mpidr); } else { SMC_RET1(handle, SMC_ARCH_CALL_INVAL_PARAM); } case SIP_SVC_GET_CPU_TOPOLOGY: if (dynamic_platform_info.cpu_topo.cores > 0) { SMC_RET5(handle, NULL, dynamic_platform_info.cpu_topo.sockets, dynamic_platform_info.cpu_topo.clusters, dynamic_platform_info.cpu_topo.cores, dynamic_platform_info.cpu_topo.threads); } else { /* we do not know topology so we report SMC as unknown */ SMC_RET1(handle, SMC_UNK); } case SIP_SVC_GET_MEMORY_NODE_COUNT: SMC_RET2(handle, NULL, dynamic_platform_info.num_memnodes); case SIP_SVC_GET_MEMORY_NODE: index = x1; if (index < PLAT_MAX_MEM_NODES) { SMC_RET4(handle, NULL, dynamic_platform_info.memory[index].nodeid, dynamic_platform_info.memory[index].addr_base, dynamic_platform_info.memory[index].addr_size); } else { SMC_RET1(handle, SMC_ARCH_CALL_INVAL_PARAM); } default: ERROR("%s: unhandled SMC (0x%x) (function id: %d)\n", __func__, smc_fid, smc_fid - SIP_FUNCTION); SMC_RET1(handle, SMC_UNK); } } int sbsa_sip_smc_setup(void) { return 0; } /* Define a runtime service descriptor for fast SMC calls */ DECLARE_RT_SVC( sbsa_sip_svc, OEN_SIP_START, OEN_SIP_END, SMC_TYPE_FAST, sbsa_sip_smc_setup, sbsa_sip_smc_handler );