1e62748e3SManish V Badarkhe /* 2e62748e3SManish V Badarkhe * Copyright (c) 2022 Arm Limited. All rights reserved. 3e62748e3SManish V Badarkhe * 4e62748e3SManish V Badarkhe * SPDX-License-Identifier: BSD-3-Clause 5e62748e3SManish V Badarkhe * 6e62748e3SManish V Badarkhe * DRTM service 7e62748e3SManish V Badarkhe * 8e62748e3SManish V Badarkhe * Authors: 9e62748e3SManish V Badarkhe * Lucian Paul-Trifu <lucian.paultrifu@gmail.com> 10e62748e3SManish V Badarkhe * Brian Nezvadovitz <brinez@microsoft.com> 2021-02-01 11e62748e3SManish V Badarkhe */ 12e62748e3SManish V Badarkhe 13e62748e3SManish V Badarkhe #include <stdint.h> 14e62748e3SManish V Badarkhe 15d54792bdSManish V Badarkhe #include <arch.h> 16d54792bdSManish V Badarkhe #include <arch_helpers.h> 172a1cdee4Sjohpow01 #include <common/bl_common.h> 18e62748e3SManish V Badarkhe #include <common/debug.h> 19e62748e3SManish V Badarkhe #include <common/runtime_svc.h> 20d54792bdSManish V Badarkhe #include <drivers/auth/crypto_mod.h> 21e62748e3SManish V Badarkhe #include "drtm_main.h" 222090e552SManish V Badarkhe #include "drtm_measurements.h" 231436e37dSManish V Badarkhe #include "drtm_remediation.h" 24bd6cc0b2SManish Pandey #include <lib/psci/psci_lib.h> 252a1cdee4Sjohpow01 #include <lib/xlat_tables/xlat_tables_v2.h> 262a1cdee4Sjohpow01 #include <plat/common/platform.h> 27e62748e3SManish V Badarkhe #include <services/drtm_svc.h> 282a1cdee4Sjohpow01 #include <platform_def.h> 29e62748e3SManish V Badarkhe 302a1cdee4Sjohpow01 /* Structure to store DRTM features specific to the platform. */ 312a1cdee4Sjohpow01 static drtm_features_t plat_drtm_features; 322a1cdee4Sjohpow01 332a1cdee4Sjohpow01 /* DRTM-formatted memory map. */ 342a1cdee4Sjohpow01 static drtm_memory_region_descriptor_table_t *plat_drtm_mem_map; 35d54792bdSManish V Badarkhe 36*d42119ccSManish V Badarkhe /* DLME header */ 37*d42119ccSManish V Badarkhe struct_dlme_data_header dlme_data_hdr_init; 38*d42119ccSManish V Badarkhe 39*d42119ccSManish V Badarkhe /* Minimum data memory requirement */ 40*d42119ccSManish V Badarkhe uint64_t dlme_data_min_size; 41*d42119ccSManish V Badarkhe 42e62748e3SManish V Badarkhe int drtm_setup(void) 43e62748e3SManish V Badarkhe { 44d54792bdSManish V Badarkhe bool rc; 452a1cdee4Sjohpow01 const plat_drtm_tpm_features_t *plat_tpm_feat; 462a1cdee4Sjohpow01 const plat_drtm_dma_prot_features_t *plat_dma_prot_feat; 47d54792bdSManish V Badarkhe 48e62748e3SManish V Badarkhe INFO("DRTM service setup\n"); 49e62748e3SManish V Badarkhe 502a1cdee4Sjohpow01 /* Read boot PE ID from MPIDR */ 512a1cdee4Sjohpow01 plat_drtm_features.boot_pe_id = read_mpidr_el1() & MPIDR_AFFINITY_MASK; 52d54792bdSManish V Badarkhe 53d54792bdSManish V Badarkhe rc = drtm_dma_prot_init(); 54d54792bdSManish V Badarkhe if (rc) { 55d54792bdSManish V Badarkhe return INTERNAL_ERROR; 56d54792bdSManish V Badarkhe } 57d54792bdSManish V Badarkhe 58d54792bdSManish V Badarkhe /* 59d54792bdSManish V Badarkhe * initialise the platform supported crypto module that will 60d54792bdSManish V Badarkhe * be used by the DRTM-service to calculate hash of DRTM- 61d54792bdSManish V Badarkhe * implementation specific components 62d54792bdSManish V Badarkhe */ 63d54792bdSManish V Badarkhe crypto_mod_init(); 64d54792bdSManish V Badarkhe 652a1cdee4Sjohpow01 /* Build DRTM-compatible address map. */ 662a1cdee4Sjohpow01 plat_drtm_mem_map = drtm_build_address_map(); 672a1cdee4Sjohpow01 if (plat_drtm_mem_map == NULL) { 682a1cdee4Sjohpow01 return INTERNAL_ERROR; 692a1cdee4Sjohpow01 } 702a1cdee4Sjohpow01 712a1cdee4Sjohpow01 /* Get DRTM features from platform hooks. */ 722a1cdee4Sjohpow01 plat_tpm_feat = plat_drtm_get_tpm_features(); 732a1cdee4Sjohpow01 if (plat_tpm_feat == NULL) { 742a1cdee4Sjohpow01 return INTERNAL_ERROR; 752a1cdee4Sjohpow01 } 762a1cdee4Sjohpow01 772a1cdee4Sjohpow01 plat_dma_prot_feat = plat_drtm_get_dma_prot_features(); 782a1cdee4Sjohpow01 if (plat_dma_prot_feat == NULL) { 792a1cdee4Sjohpow01 return INTERNAL_ERROR; 802a1cdee4Sjohpow01 } 812a1cdee4Sjohpow01 822a1cdee4Sjohpow01 /* 832a1cdee4Sjohpow01 * Add up minimum DLME data memory. 842a1cdee4Sjohpow01 * 852a1cdee4Sjohpow01 * For systems with complete DMA protection there is only one entry in 862a1cdee4Sjohpow01 * the protected regions table. 872a1cdee4Sjohpow01 */ 882a1cdee4Sjohpow01 if (plat_dma_prot_feat->dma_protection_support == 892a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_DMA_SUPPORT_COMPLETE) { 902a1cdee4Sjohpow01 dlme_data_min_size = 912a1cdee4Sjohpow01 sizeof(drtm_memory_region_descriptor_table_t) + 922a1cdee4Sjohpow01 sizeof(drtm_mem_region_t); 93*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_prot_regions_size = dlme_data_min_size; 942a1cdee4Sjohpow01 } else { 952a1cdee4Sjohpow01 /* 962a1cdee4Sjohpow01 * TODO set protected regions table size based on platform DMA 972a1cdee4Sjohpow01 * protection configuration 982a1cdee4Sjohpow01 */ 992a1cdee4Sjohpow01 panic(); 1002a1cdee4Sjohpow01 } 1012a1cdee4Sjohpow01 102*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_addr_map_size = drtm_get_address_map_size(); 103*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_tcb_hashes_table_size = 104*d42119ccSManish V Badarkhe plat_drtm_get_tcb_hash_table_size(); 105*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_impdef_region_size = 106*d42119ccSManish V Badarkhe plat_drtm_get_imp_def_dlme_region_size(); 107*d42119ccSManish V Badarkhe 108*d42119ccSManish V Badarkhe dlme_data_min_size += dlme_data_hdr_init.dlme_addr_map_size + 1092a1cdee4Sjohpow01 PLAT_DRTM_EVENT_LOG_MAX_SIZE + 110*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_tcb_hashes_table_size + 111*d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_impdef_region_size; 1122a1cdee4Sjohpow01 1132a1cdee4Sjohpow01 dlme_data_min_size = page_align(dlme_data_min_size, UP)/PAGE_SIZE; 1142a1cdee4Sjohpow01 1152a1cdee4Sjohpow01 /* Fill out platform DRTM features structure */ 1162a1cdee4Sjohpow01 /* Only support default PCR schema (0x1) in this implementation. */ 1172a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_PCR_SCHEMA(plat_drtm_features.tpm_features, 1182a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_PCR_SCHEMA_DEFAULT); 1192a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_TPM_HASH(plat_drtm_features.tpm_features, 1202a1cdee4Sjohpow01 plat_tpm_feat->tpm_based_hash_support); 1212a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_FW_HASH(plat_drtm_features.tpm_features, 1222a1cdee4Sjohpow01 plat_tpm_feat->firmware_hash_algorithm); 1232a1cdee4Sjohpow01 ARM_DRTM_MIN_MEM_REQ_SET_MIN_DLME_DATA_SIZE(plat_drtm_features.minimum_memory_requirement, 1242a1cdee4Sjohpow01 dlme_data_min_size); 1252a1cdee4Sjohpow01 ARM_DRTM_MIN_MEM_REQ_SET_DCE_SIZE(plat_drtm_features.minimum_memory_requirement, 1262a1cdee4Sjohpow01 plat_drtm_get_min_size_normal_world_dce()); 1272a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_SET_MAX_REGIONS(plat_drtm_features.dma_prot_features, 1282a1cdee4Sjohpow01 plat_dma_prot_feat->max_num_mem_prot_regions); 1292a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_SET_DMA_SUPPORT(plat_drtm_features.dma_prot_features, 1302a1cdee4Sjohpow01 plat_dma_prot_feat->dma_protection_support); 1312a1cdee4Sjohpow01 ARM_DRTM_TCB_HASH_FEATURES_SET_MAX_NUM_HASHES(plat_drtm_features.tcb_hash_features, 1322a1cdee4Sjohpow01 plat_drtm_get_tcb_hash_features()); 1332a1cdee4Sjohpow01 134e62748e3SManish V Badarkhe return 0; 135e62748e3SManish V Badarkhe } 136e62748e3SManish V Badarkhe 137e9467afbSManish V Badarkhe static inline uint64_t drtm_features_tpm(void *ctx) 138e9467afbSManish V Badarkhe { 139e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* TPM feature is supported */ 140e9467afbSManish V Badarkhe plat_drtm_features.tpm_features); 141e9467afbSManish V Badarkhe } 142e9467afbSManish V Badarkhe 143e9467afbSManish V Badarkhe static inline uint64_t drtm_features_mem_req(void *ctx) 144e9467afbSManish V Badarkhe { 145e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* memory req Feature is supported */ 146e9467afbSManish V Badarkhe plat_drtm_features.minimum_memory_requirement); 147e9467afbSManish V Badarkhe } 148e9467afbSManish V Badarkhe 149e9467afbSManish V Badarkhe static inline uint64_t drtm_features_boot_pe_id(void *ctx) 150e9467afbSManish V Badarkhe { 151e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* Boot PE feature is supported */ 152e9467afbSManish V Badarkhe plat_drtm_features.boot_pe_id); 153e9467afbSManish V Badarkhe } 154e9467afbSManish V Badarkhe 155e9467afbSManish V Badarkhe static inline uint64_t drtm_features_dma_prot(void *ctx) 156e9467afbSManish V Badarkhe { 157e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* DMA protection feature is supported */ 158e9467afbSManish V Badarkhe plat_drtm_features.dma_prot_features); 159e9467afbSManish V Badarkhe } 160e9467afbSManish V Badarkhe 161e9467afbSManish V Badarkhe static inline uint64_t drtm_features_tcb_hashes(void *ctx) 162e9467afbSManish V Badarkhe { 163e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* TCB hash feature is supported */ 164e9467afbSManish V Badarkhe plat_drtm_features.tcb_hash_features); 165e9467afbSManish V Badarkhe } 166e9467afbSManish V Badarkhe 167bd6cc0b2SManish Pandey static enum drtm_retc drtm_dl_check_caller_el(void *ctx) 168bd6cc0b2SManish Pandey { 169bd6cc0b2SManish Pandey uint64_t spsr_el3 = read_ctx_reg(get_el3state_ctx(ctx), CTX_SPSR_EL3); 170bd6cc0b2SManish Pandey uint64_t dl_caller_el; 171bd6cc0b2SManish Pandey uint64_t dl_caller_aarch; 172bd6cc0b2SManish Pandey 173bd6cc0b2SManish Pandey dl_caller_el = spsr_el3 >> MODE_EL_SHIFT & MODE_EL_MASK; 174bd6cc0b2SManish Pandey dl_caller_aarch = spsr_el3 >> MODE_RW_SHIFT & MODE_RW_MASK; 175bd6cc0b2SManish Pandey 176bd6cc0b2SManish Pandey /* Caller's security state is checked from drtm_smc_handle function */ 177bd6cc0b2SManish Pandey 178bd6cc0b2SManish Pandey /* Caller can be NS-EL2/EL1 */ 179bd6cc0b2SManish Pandey if (dl_caller_el == MODE_EL3) { 180bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch from EL3\n"); 181bd6cc0b2SManish Pandey return DENIED; 182bd6cc0b2SManish Pandey } 183bd6cc0b2SManish Pandey 184bd6cc0b2SManish Pandey if (dl_caller_aarch != MODE_RW_64) { 185bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch from non-AArch64 execution state\n"); 186bd6cc0b2SManish Pandey return DENIED; 187bd6cc0b2SManish Pandey } 188bd6cc0b2SManish Pandey 189bd6cc0b2SManish Pandey return SUCCESS; 190bd6cc0b2SManish Pandey } 191bd6cc0b2SManish Pandey 192bd6cc0b2SManish Pandey static enum drtm_retc drtm_dl_check_cores(void) 193bd6cc0b2SManish Pandey { 194bd6cc0b2SManish Pandey bool running_on_single_core; 195bd6cc0b2SManish Pandey uint64_t this_pe_aff_value = read_mpidr_el1() & MPIDR_AFFINITY_MASK; 196bd6cc0b2SManish Pandey 197bd6cc0b2SManish Pandey if (this_pe_aff_value != plat_drtm_features.boot_pe_id) { 198bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch on a non-boot PE\n"); 199bd6cc0b2SManish Pandey return DENIED; 200bd6cc0b2SManish Pandey } 201bd6cc0b2SManish Pandey 202bd6cc0b2SManish Pandey running_on_single_core = psci_is_last_on_cpu_safe(); 203bd6cc0b2SManish Pandey if (!running_on_single_core) { 204bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch due to non-boot PE not being turned off\n"); 205bd6cc0b2SManish Pandey return DENIED; 206bd6cc0b2SManish Pandey } 207bd6cc0b2SManish Pandey 208bd6cc0b2SManish Pandey return SUCCESS; 209bd6cc0b2SManish Pandey } 210bd6cc0b2SManish Pandey 211*d42119ccSManish V Badarkhe static enum drtm_retc drtm_dl_prepare_dlme_data(const struct_drtm_dl_args *args) 21240e1fad6SManish Pandey { 213*d42119ccSManish V Badarkhe int rc; 214*d42119ccSManish V Badarkhe uint64_t dlme_data_paddr; 215*d42119ccSManish V Badarkhe size_t dlme_data_max_size; 216*d42119ccSManish V Badarkhe uintptr_t dlme_data_mapping; 217*d42119ccSManish V Badarkhe struct_dlme_data_header *dlme_data_hdr; 218*d42119ccSManish V Badarkhe uint8_t *dlme_data_cursor; 219*d42119ccSManish V Badarkhe size_t dlme_data_mapping_bytes; 220*d42119ccSManish V Badarkhe size_t serialised_bytes_actual; 22140e1fad6SManish Pandey 222*d42119ccSManish V Badarkhe dlme_data_paddr = args->dlme_paddr + args->dlme_data_off; 223*d42119ccSManish V Badarkhe dlme_data_max_size = args->dlme_size - args->dlme_data_off; 224*d42119ccSManish V Badarkhe 225*d42119ccSManish V Badarkhe /* 226*d42119ccSManish V Badarkhe * The capacity of the given DLME data region is checked when 227*d42119ccSManish V Badarkhe * the other dynamic launch arguments are. 228*d42119ccSManish V Badarkhe */ 229*d42119ccSManish V Badarkhe if (dlme_data_max_size < dlme_data_min_size) { 230*d42119ccSManish V Badarkhe ERROR("%s: assertion failed:" 231*d42119ccSManish V Badarkhe " dlme_data_max_size (%ld) < dlme_data_total_bytes_req (%ld)\n", 232*d42119ccSManish V Badarkhe __func__, dlme_data_max_size, dlme_data_min_size); 233*d42119ccSManish V Badarkhe panic(); 234*d42119ccSManish V Badarkhe } 235*d42119ccSManish V Badarkhe 236*d42119ccSManish V Badarkhe /* Map the DLME data region as NS memory. */ 237*d42119ccSManish V Badarkhe dlme_data_mapping_bytes = ALIGNED_UP(dlme_data_max_size, DRTM_PAGE_SIZE); 238*d42119ccSManish V Badarkhe rc = mmap_add_dynamic_region_alloc_va(dlme_data_paddr, 239*d42119ccSManish V Badarkhe &dlme_data_mapping, 240*d42119ccSManish V Badarkhe dlme_data_mapping_bytes, 241*d42119ccSManish V Badarkhe MT_RW_DATA | MT_NS | 242*d42119ccSManish V Badarkhe MT_SHAREABILITY_ISH); 243*d42119ccSManish V Badarkhe if (rc != 0) { 244*d42119ccSManish V Badarkhe WARN("DRTM: %s: mmap_add_dynamic_region() failed rc=%d\n", 245*d42119ccSManish V Badarkhe __func__, rc); 246*d42119ccSManish V Badarkhe return INTERNAL_ERROR; 247*d42119ccSManish V Badarkhe } 248*d42119ccSManish V Badarkhe dlme_data_hdr = (struct_dlme_data_header *)dlme_data_mapping; 249*d42119ccSManish V Badarkhe dlme_data_cursor = (uint8_t *)dlme_data_hdr + sizeof(*dlme_data_hdr); 250*d42119ccSManish V Badarkhe 251*d42119ccSManish V Badarkhe memcpy(dlme_data_hdr, (const void *)&dlme_data_hdr_init, 252*d42119ccSManish V Badarkhe sizeof(*dlme_data_hdr)); 253*d42119ccSManish V Badarkhe 254*d42119ccSManish V Badarkhe /* Set the header version and size. */ 255*d42119ccSManish V Badarkhe dlme_data_hdr->version = 1; 256*d42119ccSManish V Badarkhe dlme_data_hdr->this_hdr_size = sizeof(*dlme_data_hdr); 257*d42119ccSManish V Badarkhe 258*d42119ccSManish V Badarkhe /* Prepare DLME protected regions. */ 259*d42119ccSManish V Badarkhe drtm_dma_prot_serialise_table(dlme_data_cursor, 260*d42119ccSManish V Badarkhe &serialised_bytes_actual); 261*d42119ccSManish V Badarkhe assert(serialised_bytes_actual == 262*d42119ccSManish V Badarkhe dlme_data_hdr->dlme_prot_regions_size); 263*d42119ccSManish V Badarkhe dlme_data_cursor += serialised_bytes_actual; 264*d42119ccSManish V Badarkhe 265*d42119ccSManish V Badarkhe /* Prepare DLME address map. */ 266*d42119ccSManish V Badarkhe if (plat_drtm_mem_map != NULL) { 267*d42119ccSManish V Badarkhe memcpy(dlme_data_cursor, plat_drtm_mem_map, 268*d42119ccSManish V Badarkhe dlme_data_hdr->dlme_addr_map_size); 269*d42119ccSManish V Badarkhe } else { 270*d42119ccSManish V Badarkhe WARN("DRTM: DLME address map is not in the cache\n"); 271*d42119ccSManish V Badarkhe } 272*d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_addr_map_size; 273*d42119ccSManish V Badarkhe 274*d42119ccSManish V Badarkhe /* Prepare DRTM event log for DLME. */ 275*d42119ccSManish V Badarkhe drtm_serialise_event_log(dlme_data_cursor, &serialised_bytes_actual); 276*d42119ccSManish V Badarkhe assert(serialised_bytes_actual <= PLAT_DRTM_EVENT_LOG_MAX_SIZE); 277*d42119ccSManish V Badarkhe dlme_data_hdr->dlme_tpm_log_size = serialised_bytes_actual; 278*d42119ccSManish V Badarkhe dlme_data_cursor += serialised_bytes_actual; 279*d42119ccSManish V Badarkhe 280*d42119ccSManish V Badarkhe /* 281*d42119ccSManish V Badarkhe * TODO: Prepare the TCB hashes for DLME, currently its size 282*d42119ccSManish V Badarkhe * 0 283*d42119ccSManish V Badarkhe */ 284*d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_tcb_hashes_table_size; 285*d42119ccSManish V Badarkhe 286*d42119ccSManish V Badarkhe /* Implementation-specific region size is unused. */ 287*d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_impdef_region_size; 288*d42119ccSManish V Badarkhe 289*d42119ccSManish V Badarkhe /* 290*d42119ccSManish V Badarkhe * Prepare DLME data size, includes all data region referenced above 291*d42119ccSManish V Badarkhe * alongwith the DLME data header 292*d42119ccSManish V Badarkhe */ 293*d42119ccSManish V Badarkhe dlme_data_hdr->dlme_data_size = dlme_data_cursor - (uint8_t *)dlme_data_hdr; 294*d42119ccSManish V Badarkhe 295*d42119ccSManish V Badarkhe /* Unmap the DLME data region. */ 296*d42119ccSManish V Badarkhe rc = mmap_remove_dynamic_region(dlme_data_mapping, dlme_data_mapping_bytes); 297*d42119ccSManish V Badarkhe if (rc != 0) { 298*d42119ccSManish V Badarkhe ERROR("%s(): mmap_remove_dynamic_region() failed" 299*d42119ccSManish V Badarkhe " unexpectedly rc=%d\n", __func__, rc); 300*d42119ccSManish V Badarkhe panic(); 301*d42119ccSManish V Badarkhe } 30240e1fad6SManish Pandey 30340e1fad6SManish Pandey return SUCCESS; 30440e1fad6SManish Pandey } 30540e1fad6SManish Pandey 30640e1fad6SManish Pandey /* 30740e1fad6SManish Pandey * Note: accesses to the dynamic launch args, and to the DLME data are 30840e1fad6SManish Pandey * little-endian as required, thanks to TF-A BL31 init requirements. 30940e1fad6SManish Pandey */ 31040e1fad6SManish Pandey static enum drtm_retc drtm_dl_check_args(uint64_t x1, 31140e1fad6SManish Pandey struct_drtm_dl_args *a_out) 31240e1fad6SManish Pandey { 31340e1fad6SManish Pandey uint64_t dlme_start, dlme_end; 31440e1fad6SManish Pandey uint64_t dlme_img_start, dlme_img_ep, dlme_img_end; 31540e1fad6SManish Pandey uint64_t dlme_data_start, dlme_data_end; 31640e1fad6SManish Pandey uintptr_t args_mapping; 31740e1fad6SManish Pandey size_t args_mapping_size; 31840e1fad6SManish Pandey struct_drtm_dl_args *a; 31940e1fad6SManish Pandey struct_drtm_dl_args args_buf; 32040e1fad6SManish Pandey int rc; 32140e1fad6SManish Pandey 32240e1fad6SManish Pandey if (x1 % DRTM_PAGE_SIZE != 0) { 32340e1fad6SManish Pandey ERROR("DRTM: parameters structure is not " 32440e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 32540e1fad6SManish Pandey return INVALID_PARAMETERS; 32640e1fad6SManish Pandey } 32740e1fad6SManish Pandey 32840e1fad6SManish Pandey args_mapping_size = ALIGNED_UP(sizeof(struct_drtm_dl_args), DRTM_PAGE_SIZE); 32940e1fad6SManish Pandey rc = mmap_add_dynamic_region_alloc_va(x1, &args_mapping, args_mapping_size, 33040e1fad6SManish Pandey MT_MEMORY | MT_NS | MT_RO | 33140e1fad6SManish Pandey MT_SHAREABILITY_ISH); 33240e1fad6SManish Pandey if (rc != 0) { 33340e1fad6SManish Pandey WARN("DRTM: %s: mmap_add_dynamic_region() failed rc=%d\n", 33440e1fad6SManish Pandey __func__, rc); 33540e1fad6SManish Pandey return INTERNAL_ERROR; 33640e1fad6SManish Pandey } 33740e1fad6SManish Pandey a = (struct_drtm_dl_args *)args_mapping; 33840e1fad6SManish Pandey /* 33940e1fad6SManish Pandey * TODO: invalidate all data cache before reading the data passed by the 34040e1fad6SManish Pandey * DCE Preamble. This is required to avoid / defend against racing with 34140e1fad6SManish Pandey * cache evictions. 34240e1fad6SManish Pandey */ 34340e1fad6SManish Pandey args_buf = *a; 34440e1fad6SManish Pandey 34540e1fad6SManish Pandey rc = mmap_remove_dynamic_region(args_mapping, args_mapping_size); 34640e1fad6SManish Pandey if (rc) { 34740e1fad6SManish Pandey ERROR("%s(): mmap_remove_dynamic_region() failed unexpectedly" 34840e1fad6SManish Pandey " rc=%d\n", __func__, rc); 34940e1fad6SManish Pandey panic(); 35040e1fad6SManish Pandey } 35140e1fad6SManish Pandey a = &args_buf; 35240e1fad6SManish Pandey 35340e1fad6SManish Pandey if (a->version != 1) { 35440e1fad6SManish Pandey ERROR("DRTM: parameters structure incompatible with major version %d\n", 35540e1fad6SManish Pandey ARM_DRTM_VERSION_MAJOR); 35640e1fad6SManish Pandey return NOT_SUPPORTED; 35740e1fad6SManish Pandey } 35840e1fad6SManish Pandey 35940e1fad6SManish Pandey if (!(a->dlme_img_off < a->dlme_size && 36040e1fad6SManish Pandey a->dlme_data_off < a->dlme_size)) { 36140e1fad6SManish Pandey ERROR("DRTM: argument offset is outside of the DLME region\n"); 36240e1fad6SManish Pandey return INVALID_PARAMETERS; 36340e1fad6SManish Pandey } 36440e1fad6SManish Pandey dlme_start = a->dlme_paddr; 36540e1fad6SManish Pandey dlme_end = a->dlme_paddr + a->dlme_size; 36640e1fad6SManish Pandey dlme_img_start = a->dlme_paddr + a->dlme_img_off; 36740e1fad6SManish Pandey dlme_img_ep = dlme_img_start + a->dlme_img_ep_off; 36840e1fad6SManish Pandey dlme_img_end = dlme_img_start + a->dlme_img_size; 36940e1fad6SManish Pandey dlme_data_start = a->dlme_paddr + a->dlme_data_off; 37040e1fad6SManish Pandey dlme_data_end = dlme_end; 37140e1fad6SManish Pandey 37240e1fad6SManish Pandey /* 37340e1fad6SManish Pandey * TODO: validate that the DLME physical address range is all NS memory, 37440e1fad6SManish Pandey * return INVALID_PARAMETERS if it is not. 37540e1fad6SManish Pandey * Note that this check relies on platform-specific information. For 37640e1fad6SManish Pandey * examples, see psci_plat_pm_ops->validate_ns_entrypoint() or 37740e1fad6SManish Pandey * arm_validate_ns_entrypoint(). 37840e1fad6SManish Pandey */ 37940e1fad6SManish Pandey 38040e1fad6SManish Pandey /* Check the DLME regions arguments. */ 38140e1fad6SManish Pandey if ((dlme_start % DRTM_PAGE_SIZE) != 0) { 38240e1fad6SManish Pandey ERROR("DRTM: argument DLME region is not " 38340e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 38440e1fad6SManish Pandey return INVALID_PARAMETERS; 38540e1fad6SManish Pandey } 38640e1fad6SManish Pandey 38740e1fad6SManish Pandey if (!(dlme_start < dlme_end && 38840e1fad6SManish Pandey dlme_start <= dlme_img_start && dlme_img_start < dlme_img_end && 38940e1fad6SManish Pandey dlme_start <= dlme_data_start && dlme_data_start < dlme_data_end)) { 39040e1fad6SManish Pandey ERROR("DRTM: argument DLME region is discontiguous\n"); 39140e1fad6SManish Pandey return INVALID_PARAMETERS; 39240e1fad6SManish Pandey } 39340e1fad6SManish Pandey 39440e1fad6SManish Pandey if (dlme_img_start < dlme_data_end && dlme_data_start < dlme_img_end) { 39540e1fad6SManish Pandey ERROR("DRTM: argument DLME regions overlap\n"); 39640e1fad6SManish Pandey return INVALID_PARAMETERS; 39740e1fad6SManish Pandey } 39840e1fad6SManish Pandey 39940e1fad6SManish Pandey /* Check the DLME image region arguments. */ 40040e1fad6SManish Pandey if ((dlme_img_start % DRTM_PAGE_SIZE) != 0) { 40140e1fad6SManish Pandey ERROR("DRTM: argument DLME image region is not " 40240e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 40340e1fad6SManish Pandey return INVALID_PARAMETERS; 40440e1fad6SManish Pandey } 40540e1fad6SManish Pandey 40640e1fad6SManish Pandey if (!(dlme_img_start <= dlme_img_ep && dlme_img_ep < dlme_img_end)) { 40740e1fad6SManish Pandey ERROR("DRTM: DLME entry point is outside of the DLME image region\n"); 40840e1fad6SManish Pandey return INVALID_PARAMETERS; 40940e1fad6SManish Pandey } 41040e1fad6SManish Pandey 41140e1fad6SManish Pandey if ((dlme_img_ep % 4) != 0) { 41240e1fad6SManish Pandey ERROR("DRTM: DLME image entry point is not 4-byte-aligned\n"); 41340e1fad6SManish Pandey return INVALID_PARAMETERS; 41440e1fad6SManish Pandey } 41540e1fad6SManish Pandey 41640e1fad6SManish Pandey /* Check the DLME data region arguments. */ 41740e1fad6SManish Pandey if ((dlme_data_start % DRTM_PAGE_SIZE) != 0) { 41840e1fad6SManish Pandey ERROR("DRTM: argument DLME data region is not " 41940e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 42040e1fad6SManish Pandey return INVALID_PARAMETERS; 42140e1fad6SManish Pandey } 42240e1fad6SManish Pandey 423*d42119ccSManish V Badarkhe if (dlme_data_end - dlme_data_start < dlme_data_min_size) { 42440e1fad6SManish Pandey ERROR("DRTM: argument DLME data region is short of %lu bytes\n", 425*d42119ccSManish V Badarkhe dlme_data_min_size - (size_t)(dlme_data_end - dlme_data_start)); 42640e1fad6SManish Pandey return INVALID_PARAMETERS; 42740e1fad6SManish Pandey } 42840e1fad6SManish Pandey 42940e1fad6SManish Pandey /* Check the Normal World DCE region arguments. */ 43040e1fad6SManish Pandey if (a->dce_nwd_paddr != 0) { 43140e1fad6SManish Pandey uint32_t dce_nwd_start = a->dce_nwd_paddr; 43240e1fad6SManish Pandey uint32_t dce_nwd_end = dce_nwd_start + a->dce_nwd_size; 43340e1fad6SManish Pandey 43440e1fad6SManish Pandey if (!(dce_nwd_start < dce_nwd_end)) { 43540e1fad6SManish Pandey ERROR("DRTM: argument Normal World DCE region is dicontiguous\n"); 43640e1fad6SManish Pandey return INVALID_PARAMETERS; 43740e1fad6SManish Pandey } 43840e1fad6SManish Pandey 43940e1fad6SManish Pandey if (dce_nwd_start < dlme_end && dlme_start < dce_nwd_end) { 44040e1fad6SManish Pandey ERROR("DRTM: argument Normal World DCE regions overlap\n"); 44140e1fad6SManish Pandey return INVALID_PARAMETERS; 44240e1fad6SManish Pandey } 44340e1fad6SManish Pandey } 44440e1fad6SManish Pandey 44540e1fad6SManish Pandey *a_out = *a; 44640e1fad6SManish Pandey return SUCCESS; 44740e1fad6SManish Pandey } 44840e1fad6SManish Pandey 449bd6cc0b2SManish Pandey static uint64_t drtm_dynamic_launch(uint64_t x1, void *handle) 450bd6cc0b2SManish Pandey { 451bd6cc0b2SManish Pandey enum drtm_retc ret = SUCCESS; 4522b13a985SManish V Badarkhe enum drtm_retc dma_prot_ret; 45340e1fad6SManish Pandey struct_drtm_dl_args args; 454bd6cc0b2SManish Pandey 455bd6cc0b2SManish Pandey /* Ensure that only boot PE is powered on */ 456bd6cc0b2SManish Pandey ret = drtm_dl_check_cores(); 457bd6cc0b2SManish Pandey if (ret != SUCCESS) { 458bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 459bd6cc0b2SManish Pandey } 460bd6cc0b2SManish Pandey 461bd6cc0b2SManish Pandey /* 462bd6cc0b2SManish Pandey * Ensure that execution state is AArch64 and the caller 463bd6cc0b2SManish Pandey * is highest non-secure exception level 464bd6cc0b2SManish Pandey */ 465bd6cc0b2SManish Pandey ret = drtm_dl_check_caller_el(handle); 466bd6cc0b2SManish Pandey if (ret != SUCCESS) { 467bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 468bd6cc0b2SManish Pandey } 469bd6cc0b2SManish Pandey 47040e1fad6SManish Pandey ret = drtm_dl_check_args(x1, &args); 47140e1fad6SManish Pandey if (ret != SUCCESS) { 47240e1fad6SManish Pandey SMC_RET1(handle, ret); 47340e1fad6SManish Pandey } 47440e1fad6SManish Pandey 4752b13a985SManish V Badarkhe /* 4762b13a985SManish V Badarkhe * Engage the DMA protections. The launch cannot proceed without the DMA 4772b13a985SManish V Badarkhe * protections due to potential TOC/TOU vulnerabilities w.r.t. the DLME 4782b13a985SManish V Badarkhe * region (and to the NWd DCE region). 4792b13a985SManish V Badarkhe */ 4802b13a985SManish V Badarkhe ret = drtm_dma_prot_engage(&args.dma_prot_args, 4812b13a985SManish V Badarkhe DL_ARGS_GET_DMA_PROT_TYPE(&args)); 4822b13a985SManish V Badarkhe if (ret != SUCCESS) { 4832b13a985SManish V Badarkhe SMC_RET1(handle, ret); 4842b13a985SManish V Badarkhe } 4852b13a985SManish V Badarkhe 4862090e552SManish V Badarkhe /* 4872090e552SManish V Badarkhe * The DMA protection is now engaged. Note that any failure mode that 4882090e552SManish V Badarkhe * returns an error to the DRTM-launch caller must now disengage DMA 4892090e552SManish V Badarkhe * protections before returning to the caller. 4902090e552SManish V Badarkhe */ 4912090e552SManish V Badarkhe 4922090e552SManish V Badarkhe ret = drtm_take_measurements(&args); 4932090e552SManish V Badarkhe if (ret != SUCCESS) { 4942090e552SManish V Badarkhe goto err_undo_dma_prot; 4952090e552SManish V Badarkhe } 4962090e552SManish V Badarkhe 497*d42119ccSManish V Badarkhe ret = drtm_dl_prepare_dlme_data(&args); 498*d42119ccSManish V Badarkhe if (ret != SUCCESS) { 499*d42119ccSManish V Badarkhe goto err_undo_dma_prot; 500*d42119ccSManish V Badarkhe } 501*d42119ccSManish V Badarkhe 5022090e552SManish V Badarkhe SMC_RET1(handle, ret); 5032090e552SManish V Badarkhe 5042090e552SManish V Badarkhe err_undo_dma_prot: 5052090e552SManish V Badarkhe dma_prot_ret = drtm_dma_prot_disengage(); 5062090e552SManish V Badarkhe if (dma_prot_ret != SUCCESS) { 5072090e552SManish V Badarkhe ERROR("%s(): drtm_dma_prot_disengage() failed unexpectedly" 5082090e552SManish V Badarkhe " rc=%d\n", __func__, ret); 5092090e552SManish V Badarkhe panic(); 5102090e552SManish V Badarkhe } 5112090e552SManish V Badarkhe 512bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 513bd6cc0b2SManish Pandey } 514bd6cc0b2SManish Pandey 515e62748e3SManish V Badarkhe uint64_t drtm_smc_handler(uint32_t smc_fid, 516e62748e3SManish V Badarkhe uint64_t x1, 517e62748e3SManish V Badarkhe uint64_t x2, 518e62748e3SManish V Badarkhe uint64_t x3, 519e62748e3SManish V Badarkhe uint64_t x4, 520e62748e3SManish V Badarkhe void *cookie, 521e62748e3SManish V Badarkhe void *handle, 522e62748e3SManish V Badarkhe uint64_t flags) 523e62748e3SManish V Badarkhe { 524e62748e3SManish V Badarkhe /* Check that the SMC call is from the Normal World. */ 525e62748e3SManish V Badarkhe if (!is_caller_non_secure(flags)) { 526e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 527e62748e3SManish V Badarkhe } 528e62748e3SManish V Badarkhe 529e62748e3SManish V Badarkhe switch (smc_fid) { 530e62748e3SManish V Badarkhe case ARM_DRTM_SVC_VERSION: 531e62748e3SManish V Badarkhe INFO("DRTM service handler: version\n"); 532e62748e3SManish V Badarkhe /* Return the version of current implementation */ 533e62748e3SManish V Badarkhe SMC_RET1(handle, ARM_DRTM_VERSION); 534e62748e3SManish V Badarkhe break; /* not reached */ 535e62748e3SManish V Badarkhe 536e62748e3SManish V Badarkhe case ARM_DRTM_SVC_FEATURES: 537e62748e3SManish V Badarkhe if (((x1 >> ARM_DRTM_FUNC_SHIFT) & ARM_DRTM_FUNC_MASK) == 538e62748e3SManish V Badarkhe ARM_DRTM_FUNC_ID) { 539e62748e3SManish V Badarkhe /* Dispatch function-based queries. */ 540e62748e3SManish V Badarkhe switch (x1 & FUNCID_MASK) { 541e62748e3SManish V Badarkhe case ARM_DRTM_SVC_VERSION: 542e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 543e62748e3SManish V Badarkhe break; /* not reached */ 544e62748e3SManish V Badarkhe 545e62748e3SManish V Badarkhe case ARM_DRTM_SVC_FEATURES: 546e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 547e62748e3SManish V Badarkhe break; /* not reached */ 548e62748e3SManish V Badarkhe 549e62748e3SManish V Badarkhe case ARM_DRTM_SVC_UNPROTECT_MEM: 550e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 551e62748e3SManish V Badarkhe break; /* not reached */ 552e62748e3SManish V Badarkhe 553e62748e3SManish V Badarkhe case ARM_DRTM_SVC_DYNAMIC_LAUNCH: 554e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 555e62748e3SManish V Badarkhe break; /* not reached */ 556e62748e3SManish V Badarkhe 557e62748e3SManish V Badarkhe case ARM_DRTM_SVC_CLOSE_LOCALITY: 558e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_CLOSE_LOCALITY feature %s", 559e62748e3SManish V Badarkhe "is not supported\n"); 560e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 561e62748e3SManish V Badarkhe break; /* not reached */ 562e62748e3SManish V Badarkhe 563e62748e3SManish V Badarkhe case ARM_DRTM_SVC_GET_ERROR: 564e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 565e62748e3SManish V Badarkhe break; /* not reached */ 566e62748e3SManish V Badarkhe 567e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_ERROR: 568e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 569e62748e3SManish V Badarkhe break; /* not reached */ 570e62748e3SManish V Badarkhe 571e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_TCB_HASH: 572e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_TCB_HASH feature %s", 573e62748e3SManish V Badarkhe "is not supported\n"); 574e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 575e62748e3SManish V Badarkhe break; /* not reached */ 576e62748e3SManish V Badarkhe 577e62748e3SManish V Badarkhe case ARM_DRTM_SVC_LOCK_TCB_HASH: 578e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_LOCK_TCB_HASH feature %s", 579e62748e3SManish V Badarkhe "is not supported\n"); 580e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 581e62748e3SManish V Badarkhe break; /* not reached */ 582e62748e3SManish V Badarkhe 583e62748e3SManish V Badarkhe default: 584e62748e3SManish V Badarkhe ERROR("Unknown DRTM service function\n"); 585e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 586e62748e3SManish V Badarkhe break; /* not reached */ 587e62748e3SManish V Badarkhe } 588e9467afbSManish V Badarkhe } else { 589e9467afbSManish V Badarkhe /* Dispatch feature-based queries. */ 590e9467afbSManish V Badarkhe switch (x1 & ARM_DRTM_FEAT_ID_MASK) { 591e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_TPM: 592e9467afbSManish V Badarkhe INFO("++ DRTM service handler: TPM features\n"); 593e9467afbSManish V Badarkhe return drtm_features_tpm(handle); 594e9467afbSManish V Badarkhe break; /* not reached */ 595e9467afbSManish V Badarkhe 596e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_MEM_REQ: 597e9467afbSManish V Badarkhe INFO("++ DRTM service handler: Min. mem." 598e9467afbSManish V Badarkhe " requirement features\n"); 599e9467afbSManish V Badarkhe return drtm_features_mem_req(handle); 600e9467afbSManish V Badarkhe break; /* not reached */ 601e9467afbSManish V Badarkhe 602e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_DMA_PROT: 603e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 604e9467afbSManish V Badarkhe "DMA protection features\n"); 605e9467afbSManish V Badarkhe return drtm_features_dma_prot(handle); 606e9467afbSManish V Badarkhe break; /* not reached */ 607e9467afbSManish V Badarkhe 608e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_BOOT_PE_ID: 609e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 610e9467afbSManish V Badarkhe "Boot PE ID features\n"); 611e9467afbSManish V Badarkhe return drtm_features_boot_pe_id(handle); 612e9467afbSManish V Badarkhe break; /* not reached */ 613e9467afbSManish V Badarkhe 614e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_TCB_HASHES: 615e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 616e9467afbSManish V Badarkhe "TCB-hashes features\n"); 617e9467afbSManish V Badarkhe return drtm_features_tcb_hashes(handle); 618e9467afbSManish V Badarkhe break; /* not reached */ 619e9467afbSManish V Badarkhe 620e9467afbSManish V Badarkhe default: 621e9467afbSManish V Badarkhe ERROR("Unknown ARM DRTM service feature\n"); 622e9467afbSManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 623e9467afbSManish V Badarkhe break; /* not reached */ 624e9467afbSManish V Badarkhe } 625e62748e3SManish V Badarkhe } 626e62748e3SManish V Badarkhe 627e62748e3SManish V Badarkhe case ARM_DRTM_SVC_UNPROTECT_MEM: 628e62748e3SManish V Badarkhe INFO("DRTM service handler: unprotect mem\n"); 6292b13a985SManish V Badarkhe return drtm_unprotect_mem(handle); 630e62748e3SManish V Badarkhe break; /* not reached */ 631e62748e3SManish V Badarkhe 632e62748e3SManish V Badarkhe case ARM_DRTM_SVC_DYNAMIC_LAUNCH: 633e62748e3SManish V Badarkhe INFO("DRTM service handler: dynamic launch\n"); 634bd6cc0b2SManish Pandey return drtm_dynamic_launch(x1, handle); 635e62748e3SManish V Badarkhe break; /* not reached */ 636e62748e3SManish V Badarkhe 637e62748e3SManish V Badarkhe case ARM_DRTM_SVC_CLOSE_LOCALITY: 638e62748e3SManish V Badarkhe WARN("DRTM service handler: close locality %s\n", 639e62748e3SManish V Badarkhe "is not supported"); 640e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 641e62748e3SManish V Badarkhe break; /* not reached */ 642e62748e3SManish V Badarkhe 643e62748e3SManish V Badarkhe case ARM_DRTM_SVC_GET_ERROR: 644e62748e3SManish V Badarkhe INFO("DRTM service handler: get error\n"); 6451436e37dSManish V Badarkhe drtm_get_error(handle); 646e62748e3SManish V Badarkhe break; /* not reached */ 647e62748e3SManish V Badarkhe 648e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_ERROR: 649e62748e3SManish V Badarkhe INFO("DRTM service handler: set error\n"); 6501436e37dSManish V Badarkhe drtm_set_error(x1, handle); 651e62748e3SManish V Badarkhe break; /* not reached */ 652e62748e3SManish V Badarkhe 653e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_TCB_HASH: 654e62748e3SManish V Badarkhe WARN("DRTM service handler: set TCB hash %s\n", 655e62748e3SManish V Badarkhe "is not supported"); 656e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 657e62748e3SManish V Badarkhe break; /* not reached */ 658e62748e3SManish V Badarkhe 659e62748e3SManish V Badarkhe case ARM_DRTM_SVC_LOCK_TCB_HASH: 660e62748e3SManish V Badarkhe WARN("DRTM service handler: lock TCB hash %s\n", 661e62748e3SManish V Badarkhe "is not supported"); 662e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 663e62748e3SManish V Badarkhe break; /* not reached */ 664e62748e3SManish V Badarkhe 665e62748e3SManish V Badarkhe default: 666e62748e3SManish V Badarkhe ERROR("Unknown DRTM service function: 0x%x\n", smc_fid); 667e62748e3SManish V Badarkhe SMC_RET1(handle, SMC_UNK); 668e62748e3SManish V Badarkhe break; /* not reached */ 669e62748e3SManish V Badarkhe } 670e62748e3SManish V Badarkhe 671e62748e3SManish V Badarkhe /* not reached */ 672e62748e3SManish V Badarkhe SMC_RET1(handle, SMC_UNK); 673e62748e3SManish V Badarkhe } 674