1e62748e3SManish V Badarkhe /* 228e8f9d9SManish V Badarkhe * Copyright (c) 2022-2025 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" 24d1747e1bSManish Pandey #include <lib/el3_runtime/context_mgmt.h> 25bd6cc0b2SManish Pandey #include <lib/psci/psci_lib.h> 262a1cdee4Sjohpow01 #include <lib/xlat_tables/xlat_tables_v2.h> 272a1cdee4Sjohpow01 #include <plat/common/platform.h> 28e62748e3SManish V Badarkhe #include <services/drtm_svc.h> 29b1392f42SManish Pandey #include <services/sdei.h> 302a1cdee4Sjohpow01 #include <platform_def.h> 31e62748e3SManish V Badarkhe 322a1cdee4Sjohpow01 /* Structure to store DRTM features specific to the platform. */ 332a1cdee4Sjohpow01 static drtm_features_t plat_drtm_features; 342a1cdee4Sjohpow01 352a1cdee4Sjohpow01 /* DRTM-formatted memory map. */ 362a1cdee4Sjohpow01 static drtm_memory_region_descriptor_table_t *plat_drtm_mem_map; 37*8666bcfaSManish V Badarkhe static const plat_drtm_dma_prot_features_t *plat_dma_prot_feat; 38*8666bcfaSManish V Badarkhe static const plat_drtm_tpm_features_t *plat_tpm_feat; 39d54792bdSManish V Badarkhe 40d42119ccSManish V Badarkhe /* DLME header */ 41d42119ccSManish V Badarkhe struct_dlme_data_header dlme_data_hdr_init; 42d42119ccSManish V Badarkhe 43d42119ccSManish V Badarkhe /* Minimum data memory requirement */ 44d42119ccSManish V Badarkhe uint64_t dlme_data_min_size; 45d42119ccSManish V Badarkhe 46e62748e3SManish V Badarkhe int drtm_setup(void) 47e62748e3SManish V Badarkhe { 48d54792bdSManish V Badarkhe bool rc; 49d54792bdSManish V Badarkhe 50e62748e3SManish V Badarkhe INFO("DRTM service setup\n"); 51e62748e3SManish V Badarkhe 522a1cdee4Sjohpow01 /* Read boot PE ID from MPIDR */ 532a1cdee4Sjohpow01 plat_drtm_features.boot_pe_id = read_mpidr_el1() & MPIDR_AFFINITY_MASK; 54d54792bdSManish V Badarkhe 55d54792bdSManish V Badarkhe rc = drtm_dma_prot_init(); 56d54792bdSManish V Badarkhe if (rc) { 57d54792bdSManish V Badarkhe return INTERNAL_ERROR; 58d54792bdSManish V Badarkhe } 59d54792bdSManish V Badarkhe 60d54792bdSManish V Badarkhe /* 61d54792bdSManish V Badarkhe * initialise the platform supported crypto module that will 62d54792bdSManish V Badarkhe * be used by the DRTM-service to calculate hash of DRTM- 63d54792bdSManish V Badarkhe * implementation specific components 64d54792bdSManish V Badarkhe */ 65d54792bdSManish V Badarkhe crypto_mod_init(); 66d54792bdSManish V Badarkhe 672a1cdee4Sjohpow01 /* Build DRTM-compatible address map. */ 682a1cdee4Sjohpow01 plat_drtm_mem_map = drtm_build_address_map(); 692a1cdee4Sjohpow01 if (plat_drtm_mem_map == NULL) { 702a1cdee4Sjohpow01 return INTERNAL_ERROR; 712a1cdee4Sjohpow01 } 722a1cdee4Sjohpow01 732a1cdee4Sjohpow01 /* Get DRTM features from platform hooks. */ 742a1cdee4Sjohpow01 plat_tpm_feat = plat_drtm_get_tpm_features(); 752a1cdee4Sjohpow01 if (plat_tpm_feat == NULL) { 762a1cdee4Sjohpow01 return INTERNAL_ERROR; 772a1cdee4Sjohpow01 } 782a1cdee4Sjohpow01 792a1cdee4Sjohpow01 plat_dma_prot_feat = plat_drtm_get_dma_prot_features(); 802a1cdee4Sjohpow01 if (plat_dma_prot_feat == NULL) { 812a1cdee4Sjohpow01 return INTERNAL_ERROR; 822a1cdee4Sjohpow01 } 832a1cdee4Sjohpow01 842a1cdee4Sjohpow01 /* 852a1cdee4Sjohpow01 * Add up minimum DLME data memory. 862a1cdee4Sjohpow01 * 872a1cdee4Sjohpow01 * For systems with complete DMA protection there is only one entry in 882a1cdee4Sjohpow01 * the protected regions table. 892a1cdee4Sjohpow01 */ 902a1cdee4Sjohpow01 if (plat_dma_prot_feat->dma_protection_support == 912a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_DMA_SUPPORT_COMPLETE) { 922a1cdee4Sjohpow01 dlme_data_min_size = 932a1cdee4Sjohpow01 sizeof(drtm_memory_region_descriptor_table_t) + 942a1cdee4Sjohpow01 sizeof(drtm_mem_region_t); 95d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_prot_regions_size = dlme_data_min_size; 962a1cdee4Sjohpow01 } else { 972a1cdee4Sjohpow01 /* 982a1cdee4Sjohpow01 * TODO set protected regions table size based on platform DMA 992a1cdee4Sjohpow01 * protection configuration 1002a1cdee4Sjohpow01 */ 1012a1cdee4Sjohpow01 panic(); 1022a1cdee4Sjohpow01 } 1032a1cdee4Sjohpow01 104d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_addr_map_size = drtm_get_address_map_size(); 105d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_tcb_hashes_table_size = 106d42119ccSManish V Badarkhe plat_drtm_get_tcb_hash_table_size(); 107a65fa57bSManish V Badarkhe dlme_data_hdr_init.dlme_acpi_tables_region_size = 108a65fa57bSManish V Badarkhe plat_drtm_get_acpi_tables_region_size(); 109d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_impdef_region_size = 110d42119ccSManish V Badarkhe plat_drtm_get_imp_def_dlme_region_size(); 111d42119ccSManish V Badarkhe 112a65fa57bSManish V Badarkhe dlme_data_min_size += sizeof(struct_dlme_data_header) + 113a65fa57bSManish V Badarkhe dlme_data_hdr_init.dlme_addr_map_size + 11463d2020fSManish V Badarkhe ARM_DRTM_MIN_EVENT_LOG_SIZE + 115d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_tcb_hashes_table_size + 116a65fa57bSManish V Badarkhe dlme_data_hdr_init.dlme_acpi_tables_region_size + 117d42119ccSManish V Badarkhe dlme_data_hdr_init.dlme_impdef_region_size; 1182a1cdee4Sjohpow01 1192a1cdee4Sjohpow01 /* Fill out platform DRTM features structure */ 1202a1cdee4Sjohpow01 /* Only support default PCR schema (0x1) in this implementation. */ 1212a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_PCR_SCHEMA(plat_drtm_features.tpm_features, 1222a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_PCR_SCHEMA_DEFAULT); 1232a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_TPM_HASH(plat_drtm_features.tpm_features, 1242a1cdee4Sjohpow01 plat_tpm_feat->tpm_based_hash_support); 1252a1cdee4Sjohpow01 ARM_DRTM_TPM_FEATURES_SET_FW_HASH(plat_drtm_features.tpm_features, 1262a1cdee4Sjohpow01 plat_tpm_feat->firmware_hash_algorithm); 1272a1cdee4Sjohpow01 ARM_DRTM_MIN_MEM_REQ_SET_MIN_DLME_DATA_SIZE(plat_drtm_features.minimum_memory_requirement, 12828e8f9d9SManish V Badarkhe page_align(dlme_data_min_size, UP)/PAGE_SIZE); 1292a1cdee4Sjohpow01 ARM_DRTM_MIN_MEM_REQ_SET_DCE_SIZE(plat_drtm_features.minimum_memory_requirement, 1302a1cdee4Sjohpow01 plat_drtm_get_min_size_normal_world_dce()); 1312a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_SET_MAX_REGIONS(plat_drtm_features.dma_prot_features, 1322a1cdee4Sjohpow01 plat_dma_prot_feat->max_num_mem_prot_regions); 1332a1cdee4Sjohpow01 ARM_DRTM_DMA_PROT_FEATURES_SET_DMA_SUPPORT(plat_drtm_features.dma_prot_features, 1342a1cdee4Sjohpow01 plat_dma_prot_feat->dma_protection_support); 1352a1cdee4Sjohpow01 ARM_DRTM_TCB_HASH_FEATURES_SET_MAX_NUM_HASHES(plat_drtm_features.tcb_hash_features, 1362a1cdee4Sjohpow01 plat_drtm_get_tcb_hash_features()); 13794127ae2SManish V Badarkhe ARM_DRTM_DLME_IMG_AUTH_SUPPORT(plat_drtm_features.dlme_image_auth_features, 13894127ae2SManish V Badarkhe plat_drtm_get_dlme_img_auth_features()); 1392a1cdee4Sjohpow01 140e62748e3SManish V Badarkhe return 0; 141e62748e3SManish V Badarkhe } 142e62748e3SManish V Badarkhe 1432c265975SManish Pandey static inline void invalidate_icache_all(void) 1442c265975SManish Pandey { 1452c265975SManish Pandey __asm__ volatile("ic ialluis"); 1462c265975SManish Pandey dsb(); 1472c265975SManish Pandey isb(); 1482c265975SManish Pandey } 1492c265975SManish Pandey 150e9467afbSManish V Badarkhe static inline uint64_t drtm_features_tpm(void *ctx) 151e9467afbSManish V Badarkhe { 152e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* TPM feature is supported */ 153e9467afbSManish V Badarkhe plat_drtm_features.tpm_features); 154e9467afbSManish V Badarkhe } 155e9467afbSManish V Badarkhe 156e9467afbSManish V Badarkhe static inline uint64_t drtm_features_mem_req(void *ctx) 157e9467afbSManish V Badarkhe { 158e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* memory req Feature is supported */ 159e9467afbSManish V Badarkhe plat_drtm_features.minimum_memory_requirement); 160e9467afbSManish V Badarkhe } 161e9467afbSManish V Badarkhe 162e9467afbSManish V Badarkhe static inline uint64_t drtm_features_boot_pe_id(void *ctx) 163e9467afbSManish V Badarkhe { 164e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* Boot PE feature is supported */ 165e9467afbSManish V Badarkhe plat_drtm_features.boot_pe_id); 166e9467afbSManish V Badarkhe } 167e9467afbSManish V Badarkhe 168e9467afbSManish V Badarkhe static inline uint64_t drtm_features_dma_prot(void *ctx) 169e9467afbSManish V Badarkhe { 170e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* DMA protection feature is supported */ 171e9467afbSManish V Badarkhe plat_drtm_features.dma_prot_features); 172e9467afbSManish V Badarkhe } 173e9467afbSManish V Badarkhe 174e9467afbSManish V Badarkhe static inline uint64_t drtm_features_tcb_hashes(void *ctx) 175e9467afbSManish V Badarkhe { 176e9467afbSManish V Badarkhe SMC_RET2(ctx, 1ULL, /* TCB hash feature is supported */ 177e9467afbSManish V Badarkhe plat_drtm_features.tcb_hash_features); 178e9467afbSManish V Badarkhe } 179e9467afbSManish V Badarkhe 18094127ae2SManish V Badarkhe static inline uint64_t drtm_features_dlme_img_auth_features(void *ctx) 18194127ae2SManish V Badarkhe { 18294127ae2SManish V Badarkhe SMC_RET2(ctx, 1ULL, /* DLME Image auth is supported */ 18394127ae2SManish V Badarkhe plat_drtm_features.dlme_image_auth_features); 18494127ae2SManish V Badarkhe } 18594127ae2SManish V Badarkhe 186bd6cc0b2SManish Pandey static enum drtm_retc drtm_dl_check_caller_el(void *ctx) 187bd6cc0b2SManish Pandey { 188bd6cc0b2SManish Pandey uint64_t spsr_el3 = read_ctx_reg(get_el3state_ctx(ctx), CTX_SPSR_EL3); 189bd6cc0b2SManish Pandey uint64_t dl_caller_el; 190bd6cc0b2SManish Pandey uint64_t dl_caller_aarch; 191bd6cc0b2SManish Pandey 192bd6cc0b2SManish Pandey dl_caller_el = spsr_el3 >> MODE_EL_SHIFT & MODE_EL_MASK; 193bd6cc0b2SManish Pandey dl_caller_aarch = spsr_el3 >> MODE_RW_SHIFT & MODE_RW_MASK; 194bd6cc0b2SManish Pandey 195bd6cc0b2SManish Pandey /* Caller's security state is checked from drtm_smc_handle function */ 196bd6cc0b2SManish Pandey 197bd6cc0b2SManish Pandey /* Caller can be NS-EL2/EL1 */ 198bd6cc0b2SManish Pandey if (dl_caller_el == MODE_EL3) { 199bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch from EL3\n"); 200bd6cc0b2SManish Pandey return DENIED; 201bd6cc0b2SManish Pandey } 202bd6cc0b2SManish Pandey 203bd6cc0b2SManish Pandey if (dl_caller_aarch != MODE_RW_64) { 204bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch from non-AArch64 execution state\n"); 205bd6cc0b2SManish Pandey return DENIED; 206bd6cc0b2SManish Pandey } 207bd6cc0b2SManish Pandey 208bd6cc0b2SManish Pandey return SUCCESS; 209bd6cc0b2SManish Pandey } 210bd6cc0b2SManish Pandey 211bd6cc0b2SManish Pandey static enum drtm_retc drtm_dl_check_cores(void) 212bd6cc0b2SManish Pandey { 213bd6cc0b2SManish Pandey bool running_on_single_core; 214bd6cc0b2SManish Pandey uint64_t this_pe_aff_value = read_mpidr_el1() & MPIDR_AFFINITY_MASK; 215bd6cc0b2SManish Pandey 216bd6cc0b2SManish Pandey if (this_pe_aff_value != plat_drtm_features.boot_pe_id) { 217bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch on a non-boot PE\n"); 218bd6cc0b2SManish Pandey return DENIED; 219bd6cc0b2SManish Pandey } 220bd6cc0b2SManish Pandey 2213b802105SBoyan Karatotev running_on_single_core = psci_is_last_on_cpu_safe(plat_my_core_pos()); 222bd6cc0b2SManish Pandey if (!running_on_single_core) { 223bd6cc0b2SManish Pandey ERROR("DRTM: invalid launch due to non-boot PE not being turned off\n"); 224bc9064aeSStuart Yoder return SECONDARY_PE_NOT_OFF; 225bd6cc0b2SManish Pandey } 226bd6cc0b2SManish Pandey 227bd6cc0b2SManish Pandey return SUCCESS; 228bd6cc0b2SManish Pandey } 229bd6cc0b2SManish Pandey 230d42119ccSManish V Badarkhe static enum drtm_retc drtm_dl_prepare_dlme_data(const struct_drtm_dl_args *args) 23140e1fad6SManish Pandey { 232d42119ccSManish V Badarkhe int rc; 233d42119ccSManish V Badarkhe uint64_t dlme_data_paddr; 234d42119ccSManish V Badarkhe size_t dlme_data_max_size; 235d42119ccSManish V Badarkhe uintptr_t dlme_data_mapping; 236d42119ccSManish V Badarkhe struct_dlme_data_header *dlme_data_hdr; 237d42119ccSManish V Badarkhe uint8_t *dlme_data_cursor; 238d42119ccSManish V Badarkhe size_t dlme_data_mapping_bytes; 239d42119ccSManish V Badarkhe size_t serialised_bytes_actual; 24040e1fad6SManish Pandey 241d42119ccSManish V Badarkhe dlme_data_paddr = args->dlme_paddr + args->dlme_data_off; 242d42119ccSManish V Badarkhe dlme_data_max_size = args->dlme_size - args->dlme_data_off; 243d42119ccSManish V Badarkhe 244d42119ccSManish V Badarkhe /* 245d42119ccSManish V Badarkhe * The capacity of the given DLME data region is checked when 246d42119ccSManish V Badarkhe * the other dynamic launch arguments are. 247d42119ccSManish V Badarkhe */ 248d42119ccSManish V Badarkhe if (dlme_data_max_size < dlme_data_min_size) { 249d42119ccSManish V Badarkhe ERROR("%s: assertion failed:" 25028e8f9d9SManish V Badarkhe " dlme_data_max_size (%ld) < dlme_data_min_size (%ld)\n", 251d42119ccSManish V Badarkhe __func__, dlme_data_max_size, dlme_data_min_size); 252d42119ccSManish V Badarkhe panic(); 253d42119ccSManish V Badarkhe } 254d42119ccSManish V Badarkhe 255d42119ccSManish V Badarkhe /* Map the DLME data region as NS memory. */ 256d42119ccSManish V Badarkhe dlme_data_mapping_bytes = ALIGNED_UP(dlme_data_max_size, DRTM_PAGE_SIZE); 257d42119ccSManish V Badarkhe rc = mmap_add_dynamic_region_alloc_va(dlme_data_paddr, 258d42119ccSManish V Badarkhe &dlme_data_mapping, 259d42119ccSManish V Badarkhe dlme_data_mapping_bytes, 260d42119ccSManish V Badarkhe MT_RW_DATA | MT_NS | 261d42119ccSManish V Badarkhe MT_SHAREABILITY_ISH); 262d42119ccSManish V Badarkhe if (rc != 0) { 263d42119ccSManish V Badarkhe WARN("DRTM: %s: mmap_add_dynamic_region() failed rc=%d\n", 264d42119ccSManish V Badarkhe __func__, rc); 265d42119ccSManish V Badarkhe return INTERNAL_ERROR; 266d42119ccSManish V Badarkhe } 267d42119ccSManish V Badarkhe dlme_data_hdr = (struct_dlme_data_header *)dlme_data_mapping; 268d42119ccSManish V Badarkhe dlme_data_cursor = (uint8_t *)dlme_data_hdr + sizeof(*dlme_data_hdr); 269d42119ccSManish V Badarkhe 270d42119ccSManish V Badarkhe memcpy(dlme_data_hdr, (const void *)&dlme_data_hdr_init, 271d42119ccSManish V Badarkhe sizeof(*dlme_data_hdr)); 272d42119ccSManish V Badarkhe 273d42119ccSManish V Badarkhe /* Set the header version and size. */ 274d42119ccSManish V Badarkhe dlme_data_hdr->version = 1; 275d42119ccSManish V Badarkhe dlme_data_hdr->this_hdr_size = sizeof(*dlme_data_hdr); 276d42119ccSManish V Badarkhe 277d42119ccSManish V Badarkhe /* Prepare DLME protected regions. */ 278d42119ccSManish V Badarkhe drtm_dma_prot_serialise_table(dlme_data_cursor, 279d42119ccSManish V Badarkhe &serialised_bytes_actual); 280d42119ccSManish V Badarkhe assert(serialised_bytes_actual == 281d42119ccSManish V Badarkhe dlme_data_hdr->dlme_prot_regions_size); 282d42119ccSManish V Badarkhe dlme_data_cursor += serialised_bytes_actual; 283d42119ccSManish V Badarkhe 284d42119ccSManish V Badarkhe /* Prepare DLME address map. */ 285d42119ccSManish V Badarkhe if (plat_drtm_mem_map != NULL) { 286d42119ccSManish V Badarkhe memcpy(dlme_data_cursor, plat_drtm_mem_map, 287d42119ccSManish V Badarkhe dlme_data_hdr->dlme_addr_map_size); 288d42119ccSManish V Badarkhe } else { 289d42119ccSManish V Badarkhe WARN("DRTM: DLME address map is not in the cache\n"); 290d42119ccSManish V Badarkhe } 291d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_addr_map_size; 292d42119ccSManish V Badarkhe 293d42119ccSManish V Badarkhe /* Prepare DRTM event log for DLME. */ 294d42119ccSManish V Badarkhe drtm_serialise_event_log(dlme_data_cursor, &serialised_bytes_actual); 29563d2020fSManish V Badarkhe assert(serialised_bytes_actual <= ARM_DRTM_MIN_EVENT_LOG_SIZE); 29663d2020fSManish V Badarkhe dlme_data_hdr->dlme_tpm_log_size = ARM_DRTM_MIN_EVENT_LOG_SIZE; 29763d2020fSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_tpm_log_size; 298d42119ccSManish V Badarkhe 299d42119ccSManish V Badarkhe /* 300d42119ccSManish V Badarkhe * TODO: Prepare the TCB hashes for DLME, currently its size 301d42119ccSManish V Badarkhe * 0 302d42119ccSManish V Badarkhe */ 303d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_tcb_hashes_table_size; 304d42119ccSManish V Badarkhe 305d42119ccSManish V Badarkhe /* Implementation-specific region size is unused. */ 306d42119ccSManish V Badarkhe dlme_data_cursor += dlme_data_hdr->dlme_impdef_region_size; 307d42119ccSManish V Badarkhe 308d42119ccSManish V Badarkhe /* 309d42119ccSManish V Badarkhe * Prepare DLME data size, includes all data region referenced above 310d42119ccSManish V Badarkhe * alongwith the DLME data header 311d42119ccSManish V Badarkhe */ 312d42119ccSManish V Badarkhe dlme_data_hdr->dlme_data_size = dlme_data_cursor - (uint8_t *)dlme_data_hdr; 313d42119ccSManish V Badarkhe 314d42119ccSManish V Badarkhe /* Unmap the DLME data region. */ 315d42119ccSManish V Badarkhe rc = mmap_remove_dynamic_region(dlme_data_mapping, dlme_data_mapping_bytes); 316d42119ccSManish V Badarkhe if (rc != 0) { 317d42119ccSManish V Badarkhe ERROR("%s(): mmap_remove_dynamic_region() failed" 318d42119ccSManish V Badarkhe " unexpectedly rc=%d\n", __func__, rc); 319d42119ccSManish V Badarkhe panic(); 320d42119ccSManish V Badarkhe } 32140e1fad6SManish Pandey 32240e1fad6SManish Pandey return SUCCESS; 32340e1fad6SManish Pandey } 32440e1fad6SManish Pandey 325*8666bcfaSManish V Badarkhe /* Function to check if the value is valid for each bit field */ 326*8666bcfaSManish V Badarkhe static int drtm_dl_check_features_sanity(uint32_t val) 327*8666bcfaSManish V Badarkhe { 328*8666bcfaSManish V Badarkhe /** 329*8666bcfaSManish V Badarkhe * Ensure that if DLME Authorities Schema (Bits [2:1]) is set, then 330*8666bcfaSManish V Badarkhe * DLME image authentication (Bit[6]) must also be set 331*8666bcfaSManish V Badarkhe */ 332*8666bcfaSManish V Badarkhe if ((EXTRACT_FIELD(val, DRTM_LAUNCH_FEAT_PCR_USAGE_SCHEMA_MASK, 333*8666bcfaSManish V Badarkhe DRTM_LAUNCH_FEAT_PCR_USAGE_SCHEMA_SHIFT) == DLME_AUTH_SCHEMA) && 334*8666bcfaSManish V Badarkhe (EXTRACT_FIELD(val, DRTM_LAUNCH_FEAT_DLME_IMG_AUTH_MASK, 335*8666bcfaSManish V Badarkhe DRTM_LAUNCH_FEAT_DLME_IMG_AUTH_SHIFT) != DLME_IMG_AUTH)) { 336*8666bcfaSManish V Badarkhe return INVALID_PARAMETERS; 337*8666bcfaSManish V Badarkhe } 338*8666bcfaSManish V Badarkhe 339*8666bcfaSManish V Badarkhe /** 340*8666bcfaSManish V Badarkhe * Check if Bits [5:3] (Memory protection type) matches with platform's 341*8666bcfaSManish V Badarkhe * memory protection type 342*8666bcfaSManish V Badarkhe */ 343*8666bcfaSManish V Badarkhe if (EXTRACT_FIELD(val, DRTM_LAUNCH_FEAT_MEM_PROTECTION_TYPE_MASK, 344*8666bcfaSManish V Badarkhe DRTM_LAUNCH_FEAT_MEM_PROTECTION_TYPE_SHIFT) != 345*8666bcfaSManish V Badarkhe __builtin_ctz(plat_dma_prot_feat->dma_protection_support)) { 346*8666bcfaSManish V Badarkhe return INVALID_PARAMETERS; 347*8666bcfaSManish V Badarkhe } 348*8666bcfaSManish V Badarkhe 349*8666bcfaSManish V Badarkhe /** 350*8666bcfaSManish V Badarkhe * Check if Bits [0] (Type of hashing) matches with platform's 351*8666bcfaSManish V Badarkhe * supported hash type. 352*8666bcfaSManish V Badarkhe */ 353*8666bcfaSManish V Badarkhe if (EXTRACT_FIELD(val, DRTM_LAUNCH_FEAT_HASHING_TYPE_MASK, 354*8666bcfaSManish V Badarkhe DRTM_LAUNCH_FEAT_HASHING_TYPE_SHIFT) != 355*8666bcfaSManish V Badarkhe plat_tpm_feat->tpm_based_hash_support) { 356*8666bcfaSManish V Badarkhe return INVALID_PARAMETERS; 357*8666bcfaSManish V Badarkhe } 358*8666bcfaSManish V Badarkhe 359*8666bcfaSManish V Badarkhe return 0; 360*8666bcfaSManish V Badarkhe } 361*8666bcfaSManish V Badarkhe 36240e1fad6SManish Pandey /* 36340e1fad6SManish Pandey * Note: accesses to the dynamic launch args, and to the DLME data are 36440e1fad6SManish Pandey * little-endian as required, thanks to TF-A BL31 init requirements. 36540e1fad6SManish Pandey */ 36640e1fad6SManish Pandey static enum drtm_retc drtm_dl_check_args(uint64_t x1, 36740e1fad6SManish Pandey struct_drtm_dl_args *a_out) 36840e1fad6SManish Pandey { 36940e1fad6SManish Pandey uint64_t dlme_start, dlme_end; 37040e1fad6SManish Pandey uint64_t dlme_img_start, dlme_img_ep, dlme_img_end; 37140e1fad6SManish Pandey uint64_t dlme_data_start, dlme_data_end; 37267471e75SManish Pandey uintptr_t va_mapping; 37367471e75SManish Pandey size_t va_mapping_size; 37440e1fad6SManish Pandey struct_drtm_dl_args *a; 37540e1fad6SManish Pandey struct_drtm_dl_args args_buf; 37640e1fad6SManish Pandey int rc; 37740e1fad6SManish Pandey 37840e1fad6SManish Pandey if (x1 % DRTM_PAGE_SIZE != 0) { 37940e1fad6SManish Pandey ERROR("DRTM: parameters structure is not " 38040e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 38140e1fad6SManish Pandey return INVALID_PARAMETERS; 38240e1fad6SManish Pandey } 38340e1fad6SManish Pandey 38467471e75SManish Pandey va_mapping_size = ALIGNED_UP(sizeof(struct_drtm_dl_args), DRTM_PAGE_SIZE); 385764aa951SManish V Badarkhe 386764aa951SManish V Badarkhe /* check DRTM parameters are within NS address region */ 38767471e75SManish Pandey rc = plat_drtm_validate_ns_region(x1, va_mapping_size); 388764aa951SManish V Badarkhe if (rc != 0) { 389764aa951SManish V Badarkhe ERROR("DRTM: parameters lies within secure memory\n"); 390764aa951SManish V Badarkhe return INVALID_PARAMETERS; 391764aa951SManish V Badarkhe } 392764aa951SManish V Badarkhe 39367471e75SManish Pandey rc = mmap_add_dynamic_region_alloc_va(x1, &va_mapping, va_mapping_size, 39440e1fad6SManish Pandey MT_MEMORY | MT_NS | MT_RO | 39540e1fad6SManish Pandey MT_SHAREABILITY_ISH); 39640e1fad6SManish Pandey if (rc != 0) { 39740e1fad6SManish Pandey WARN("DRTM: %s: mmap_add_dynamic_region() failed rc=%d\n", 39840e1fad6SManish Pandey __func__, rc); 39940e1fad6SManish Pandey return INTERNAL_ERROR; 40040e1fad6SManish Pandey } 40167471e75SManish Pandey a = (struct_drtm_dl_args *)va_mapping; 40267471e75SManish Pandey 40367471e75SManish Pandey /* Sanitize cache of data passed in args by the DCE Preamble. */ 40467471e75SManish Pandey flush_dcache_range(va_mapping, va_mapping_size); 40567471e75SManish Pandey 40640e1fad6SManish Pandey args_buf = *a; 40740e1fad6SManish Pandey 40867471e75SManish Pandey rc = mmap_remove_dynamic_region(va_mapping, va_mapping_size); 409*8666bcfaSManish V Badarkhe if (rc != 0) { 41040e1fad6SManish Pandey ERROR("%s(): mmap_remove_dynamic_region() failed unexpectedly" 41140e1fad6SManish Pandey " rc=%d\n", __func__, rc); 41240e1fad6SManish Pandey panic(); 41340e1fad6SManish Pandey } 41440e1fad6SManish Pandey a = &args_buf; 41540e1fad6SManish Pandey 416c503ded2SManish V Badarkhe if (!((a->version >= ARM_DRTM_PARAMS_MIN_VERSION) && 417c503ded2SManish V Badarkhe (a->version <= ARM_DRTM_PARAMS_MAX_VERSION))) { 418c503ded2SManish V Badarkhe ERROR("DRTM: parameters structure version %u is unsupported\n", 419c503ded2SManish V Badarkhe a->version); 42040e1fad6SManish Pandey return NOT_SUPPORTED; 42140e1fad6SManish Pandey } 42240e1fad6SManish Pandey 423*8666bcfaSManish V Badarkhe rc = drtm_dl_check_features_sanity(a->features); 424*8666bcfaSManish V Badarkhe if (rc != 0) { 425*8666bcfaSManish V Badarkhe ERROR("%s(): drtm_dl_check_features_sanity() failed.\n" 426*8666bcfaSManish V Badarkhe " rc=%d\n", __func__, rc); 427*8666bcfaSManish V Badarkhe return rc; 428*8666bcfaSManish V Badarkhe } 429*8666bcfaSManish V Badarkhe 43040e1fad6SManish Pandey if (!(a->dlme_img_off < a->dlme_size && 43140e1fad6SManish Pandey a->dlme_data_off < a->dlme_size)) { 43240e1fad6SManish Pandey ERROR("DRTM: argument offset is outside of the DLME region\n"); 43340e1fad6SManish Pandey return INVALID_PARAMETERS; 43440e1fad6SManish Pandey } 43540e1fad6SManish Pandey dlme_start = a->dlme_paddr; 43640e1fad6SManish Pandey dlme_end = a->dlme_paddr + a->dlme_size; 43740e1fad6SManish Pandey dlme_img_start = a->dlme_paddr + a->dlme_img_off; 43840e1fad6SManish Pandey dlme_img_ep = dlme_img_start + a->dlme_img_ep_off; 43940e1fad6SManish Pandey dlme_img_end = dlme_img_start + a->dlme_img_size; 44040e1fad6SManish Pandey dlme_data_start = a->dlme_paddr + a->dlme_data_off; 44140e1fad6SManish Pandey dlme_data_end = dlme_end; 44240e1fad6SManish Pandey 44340e1fad6SManish Pandey /* Check the DLME regions arguments. */ 44440e1fad6SManish Pandey if ((dlme_start % DRTM_PAGE_SIZE) != 0) { 44540e1fad6SManish Pandey ERROR("DRTM: argument DLME region is not " 44640e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 44740e1fad6SManish Pandey return INVALID_PARAMETERS; 44840e1fad6SManish Pandey } 44940e1fad6SManish Pandey 45040e1fad6SManish Pandey if (!(dlme_start < dlme_end && 45140e1fad6SManish Pandey dlme_start <= dlme_img_start && dlme_img_start < dlme_img_end && 45240e1fad6SManish Pandey dlme_start <= dlme_data_start && dlme_data_start < dlme_data_end)) { 45340e1fad6SManish Pandey ERROR("DRTM: argument DLME region is discontiguous\n"); 45440e1fad6SManish Pandey return INVALID_PARAMETERS; 45540e1fad6SManish Pandey } 45640e1fad6SManish Pandey 45740e1fad6SManish Pandey if (dlme_img_start < dlme_data_end && dlme_data_start < dlme_img_end) { 45840e1fad6SManish Pandey ERROR("DRTM: argument DLME regions overlap\n"); 45940e1fad6SManish Pandey return INVALID_PARAMETERS; 46040e1fad6SManish Pandey } 46140e1fad6SManish Pandey 46240e1fad6SManish Pandey /* Check the DLME image region arguments. */ 46340e1fad6SManish Pandey if ((dlme_img_start % DRTM_PAGE_SIZE) != 0) { 46440e1fad6SManish Pandey ERROR("DRTM: argument DLME image region is not " 46540e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 46640e1fad6SManish Pandey return INVALID_PARAMETERS; 46740e1fad6SManish Pandey } 46840e1fad6SManish Pandey 46940e1fad6SManish Pandey if (!(dlme_img_start <= dlme_img_ep && dlme_img_ep < dlme_img_end)) { 47040e1fad6SManish Pandey ERROR("DRTM: DLME entry point is outside of the DLME image region\n"); 47140e1fad6SManish Pandey return INVALID_PARAMETERS; 47240e1fad6SManish Pandey } 47340e1fad6SManish Pandey 47440e1fad6SManish Pandey if ((dlme_img_ep % 4) != 0) { 47540e1fad6SManish Pandey ERROR("DRTM: DLME image entry point is not 4-byte-aligned\n"); 47640e1fad6SManish Pandey return INVALID_PARAMETERS; 47740e1fad6SManish Pandey } 47840e1fad6SManish Pandey 47940e1fad6SManish Pandey /* Check the DLME data region arguments. */ 48040e1fad6SManish Pandey if ((dlme_data_start % DRTM_PAGE_SIZE) != 0) { 48140e1fad6SManish Pandey ERROR("DRTM: argument DLME data region is not " 48240e1fad6SManish Pandey DRTM_PAGE_SIZE_STR "-aligned\n"); 48340e1fad6SManish Pandey return INVALID_PARAMETERS; 48440e1fad6SManish Pandey } 48540e1fad6SManish Pandey 486d42119ccSManish V Badarkhe if (dlme_data_end - dlme_data_start < dlme_data_min_size) { 48740e1fad6SManish Pandey ERROR("DRTM: argument DLME data region is short of %lu bytes\n", 488d42119ccSManish V Badarkhe dlme_data_min_size - (size_t)(dlme_data_end - dlme_data_start)); 48940e1fad6SManish Pandey return INVALID_PARAMETERS; 49040e1fad6SManish Pandey } 49140e1fad6SManish Pandey 492764aa951SManish V Badarkhe /* check DLME region (paddr + size) is within a NS address region */ 493764aa951SManish V Badarkhe rc = plat_drtm_validate_ns_region(dlme_start, (size_t)a->dlme_size); 494764aa951SManish V Badarkhe if (rc != 0) { 495764aa951SManish V Badarkhe ERROR("DRTM: DLME region lies within secure memory\n"); 496764aa951SManish V Badarkhe return INVALID_PARAMETERS; 497764aa951SManish V Badarkhe } 498764aa951SManish V Badarkhe 49940e1fad6SManish Pandey /* Check the Normal World DCE region arguments. */ 50040e1fad6SManish Pandey if (a->dce_nwd_paddr != 0) { 50140e1fad6SManish Pandey uint32_t dce_nwd_start = a->dce_nwd_paddr; 50240e1fad6SManish Pandey uint32_t dce_nwd_end = dce_nwd_start + a->dce_nwd_size; 50340e1fad6SManish Pandey 50440e1fad6SManish Pandey if (!(dce_nwd_start < dce_nwd_end)) { 50540e1fad6SManish Pandey ERROR("DRTM: argument Normal World DCE region is dicontiguous\n"); 50640e1fad6SManish Pandey return INVALID_PARAMETERS; 50740e1fad6SManish Pandey } 50840e1fad6SManish Pandey 50940e1fad6SManish Pandey if (dce_nwd_start < dlme_end && dlme_start < dce_nwd_end) { 51040e1fad6SManish Pandey ERROR("DRTM: argument Normal World DCE regions overlap\n"); 51140e1fad6SManish Pandey return INVALID_PARAMETERS; 51240e1fad6SManish Pandey } 51340e1fad6SManish Pandey } 51440e1fad6SManish Pandey 51567471e75SManish Pandey /* 51667471e75SManish Pandey * Map and sanitize the cache of data range passed by DCE Preamble. This 51767471e75SManish Pandey * is required to avoid / defend against racing with cache evictions 51867471e75SManish Pandey */ 51967471e75SManish Pandey va_mapping_size = ALIGNED_UP((dlme_end - dlme_start), DRTM_PAGE_SIZE); 52023378ae0SManish V Badarkhe rc = mmap_add_dynamic_region_alloc_va(dlme_start, &va_mapping, va_mapping_size, 52167471e75SManish Pandey MT_MEMORY | MT_NS | MT_RO | 52267471e75SManish Pandey MT_SHAREABILITY_ISH); 52367471e75SManish Pandey if (rc != 0) { 52467471e75SManish Pandey ERROR("DRTM: %s: mmap_add_dynamic_region_alloc_va() failed rc=%d\n", 52567471e75SManish Pandey __func__, rc); 52667471e75SManish Pandey return INTERNAL_ERROR; 52767471e75SManish Pandey } 52867471e75SManish Pandey flush_dcache_range(va_mapping, va_mapping_size); 52967471e75SManish Pandey 53067471e75SManish Pandey rc = mmap_remove_dynamic_region(va_mapping, va_mapping_size); 53167471e75SManish Pandey if (rc) { 53267471e75SManish Pandey ERROR("%s(): mmap_remove_dynamic_region() failed unexpectedly" 53367471e75SManish Pandey " rc=%d\n", __func__, rc); 53467471e75SManish Pandey panic(); 53567471e75SManish Pandey } 53667471e75SManish Pandey 53740e1fad6SManish Pandey *a_out = *a; 53840e1fad6SManish Pandey return SUCCESS; 53940e1fad6SManish Pandey } 54040e1fad6SManish Pandey 541d1747e1bSManish Pandey static void drtm_dl_reset_dlme_el_state(enum drtm_dlme_el dlme_el) 542d1747e1bSManish Pandey { 543d1747e1bSManish Pandey uint64_t sctlr; 544d1747e1bSManish Pandey 545d1747e1bSManish Pandey /* 546d1747e1bSManish Pandey * TODO: Set PE state according to the PSCI's specification of the initial 547d1747e1bSManish Pandey * state after CPU_ON, or to reset values if unspecified, where they exist, 548d1747e1bSManish Pandey * or define sensible values otherwise. 549d1747e1bSManish Pandey */ 550d1747e1bSManish Pandey 551d1747e1bSManish Pandey switch (dlme_el) { 552d1747e1bSManish Pandey case DLME_AT_EL1: 553d1747e1bSManish Pandey sctlr = read_sctlr_el1(); 554d1747e1bSManish Pandey break; 555d1747e1bSManish Pandey 556d1747e1bSManish Pandey case DLME_AT_EL2: 557d1747e1bSManish Pandey sctlr = read_sctlr_el2(); 558d1747e1bSManish Pandey break; 559d1747e1bSManish Pandey 560d1747e1bSManish Pandey default: /* Not reached */ 561d1747e1bSManish Pandey ERROR("%s(): dlme_el has the unexpected value %d\n", 562d1747e1bSManish Pandey __func__, dlme_el); 563d1747e1bSManish Pandey panic(); 564d1747e1bSManish Pandey } 565d1747e1bSManish Pandey 566d1747e1bSManish Pandey sctlr &= ~(/* Disable DLME's EL MMU, since the existing page-tables are untrusted. */ 567d1747e1bSManish Pandey SCTLR_M_BIT 568d1747e1bSManish Pandey | SCTLR_EE_BIT /* Little-endian data accesses. */ 56923378ae0SManish V Badarkhe | SCTLR_C_BIT /* disable data caching */ 57023378ae0SManish V Badarkhe | SCTLR_I_BIT /* disable instruction caching */ 571d1747e1bSManish Pandey ); 572d1747e1bSManish Pandey 573d1747e1bSManish Pandey switch (dlme_el) { 574d1747e1bSManish Pandey case DLME_AT_EL1: 575d1747e1bSManish Pandey write_sctlr_el1(sctlr); 576d1747e1bSManish Pandey break; 577d1747e1bSManish Pandey 578d1747e1bSManish Pandey case DLME_AT_EL2: 579d1747e1bSManish Pandey write_sctlr_el2(sctlr); 580d1747e1bSManish Pandey break; 581d1747e1bSManish Pandey } 582d1747e1bSManish Pandey } 583d1747e1bSManish Pandey 584d1747e1bSManish Pandey static void drtm_dl_reset_dlme_context(enum drtm_dlme_el dlme_el) 585d1747e1bSManish Pandey { 586d1747e1bSManish Pandey void *ns_ctx = cm_get_context(NON_SECURE); 587d1747e1bSManish Pandey gp_regs_t *gpregs = get_gpregs_ctx(ns_ctx); 588d1747e1bSManish Pandey uint64_t spsr_el3 = read_ctx_reg(get_el3state_ctx(ns_ctx), CTX_SPSR_EL3); 589d1747e1bSManish Pandey 590d1747e1bSManish Pandey /* Reset all gpregs, including SP_EL0. */ 591d1747e1bSManish Pandey memset(gpregs, 0, sizeof(*gpregs)); 592d1747e1bSManish Pandey 593d1747e1bSManish Pandey /* Reset SP_ELx. */ 594d1747e1bSManish Pandey switch (dlme_el) { 595d1747e1bSManish Pandey case DLME_AT_EL1: 596d1747e1bSManish Pandey write_sp_el1(0); 597d1747e1bSManish Pandey break; 598d1747e1bSManish Pandey 599d1747e1bSManish Pandey case DLME_AT_EL2: 600d1747e1bSManish Pandey write_sp_el2(0); 601d1747e1bSManish Pandey break; 602d1747e1bSManish Pandey } 603d1747e1bSManish Pandey 604d1747e1bSManish Pandey /* 605d1747e1bSManish Pandey * DLME's async exceptions are masked to avoid a NWd attacker's timed 606d1747e1bSManish Pandey * interference with any state we established trust in or measured. 607d1747e1bSManish Pandey */ 608d1747e1bSManish Pandey spsr_el3 |= SPSR_DAIF_MASK << SPSR_DAIF_SHIFT; 609d1747e1bSManish Pandey 610d1747e1bSManish Pandey write_ctx_reg(get_el3state_ctx(ns_ctx), CTX_SPSR_EL3, spsr_el3); 611d1747e1bSManish Pandey } 612d1747e1bSManish Pandey 613d1747e1bSManish Pandey static void drtm_dl_prepare_eret_to_dlme(const struct_drtm_dl_args *args, enum drtm_dlme_el dlme_el) 614d1747e1bSManish Pandey { 615d1747e1bSManish Pandey void *ctx = cm_get_context(NON_SECURE); 616d1747e1bSManish Pandey uint64_t dlme_ep = DL_ARGS_GET_DLME_ENTRY_POINT(args); 617d1747e1bSManish Pandey uint64_t spsr_el3 = read_ctx_reg(get_el3state_ctx(ctx), CTX_SPSR_EL3); 618d1747e1bSManish Pandey 619d1747e1bSManish Pandey /* Next ERET is to the DLME's EL. */ 620d1747e1bSManish Pandey spsr_el3 &= ~(MODE_EL_MASK << MODE_EL_SHIFT); 621d1747e1bSManish Pandey switch (dlme_el) { 622d1747e1bSManish Pandey case DLME_AT_EL1: 623d1747e1bSManish Pandey spsr_el3 |= MODE_EL1 << MODE_EL_SHIFT; 624d1747e1bSManish Pandey break; 625d1747e1bSManish Pandey 626d1747e1bSManish Pandey case DLME_AT_EL2: 627d1747e1bSManish Pandey spsr_el3 |= MODE_EL2 << MODE_EL_SHIFT; 628d1747e1bSManish Pandey break; 629d1747e1bSManish Pandey } 630d1747e1bSManish Pandey 631d1747e1bSManish Pandey /* Next ERET is to the DLME entry point. */ 632d1747e1bSManish Pandey cm_set_elr_spsr_el3(NON_SECURE, dlme_ep, spsr_el3); 633d1747e1bSManish Pandey } 634d1747e1bSManish Pandey 635bd6cc0b2SManish Pandey static uint64_t drtm_dynamic_launch(uint64_t x1, void *handle) 636bd6cc0b2SManish Pandey { 637bd6cc0b2SManish Pandey enum drtm_retc ret = SUCCESS; 6382b13a985SManish V Badarkhe enum drtm_retc dma_prot_ret; 63940e1fad6SManish Pandey struct_drtm_dl_args args; 640d1747e1bSManish Pandey /* DLME should be highest NS exception level */ 641d1747e1bSManish Pandey enum drtm_dlme_el dlme_el = (el_implemented(2) != EL_IMPL_NONE) ? MODE_EL2 : MODE_EL1; 642bd6cc0b2SManish Pandey 643bd6cc0b2SManish Pandey /* Ensure that only boot PE is powered on */ 644bd6cc0b2SManish Pandey ret = drtm_dl_check_cores(); 645bd6cc0b2SManish Pandey if (ret != SUCCESS) { 646bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 647bd6cc0b2SManish Pandey } 648bd6cc0b2SManish Pandey 649bd6cc0b2SManish Pandey /* 650bd6cc0b2SManish Pandey * Ensure that execution state is AArch64 and the caller 651bd6cc0b2SManish Pandey * is highest non-secure exception level 652bd6cc0b2SManish Pandey */ 653bd6cc0b2SManish Pandey ret = drtm_dl_check_caller_el(handle); 654bd6cc0b2SManish Pandey if (ret != SUCCESS) { 655bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 656bd6cc0b2SManish Pandey } 657bd6cc0b2SManish Pandey 65840e1fad6SManish Pandey ret = drtm_dl_check_args(x1, &args); 65940e1fad6SManish Pandey if (ret != SUCCESS) { 66040e1fad6SManish Pandey SMC_RET1(handle, ret); 66140e1fad6SManish Pandey } 66240e1fad6SManish Pandey 663b1392f42SManish Pandey /* Ensure that there are no SDEI event registered */ 664b1392f42SManish Pandey #if SDEI_SUPPORT 665b1392f42SManish Pandey if (sdei_get_registered_event_count() != 0) { 666b1392f42SManish Pandey SMC_RET1(handle, DENIED); 667b1392f42SManish Pandey } 668b1392f42SManish Pandey #endif /* SDEI_SUPPORT */ 669b1392f42SManish Pandey 6702b13a985SManish V Badarkhe /* 6712b13a985SManish V Badarkhe * Engage the DMA protections. The launch cannot proceed without the DMA 6722b13a985SManish V Badarkhe * protections due to potential TOC/TOU vulnerabilities w.r.t. the DLME 6732b13a985SManish V Badarkhe * region (and to the NWd DCE region). 6742b13a985SManish V Badarkhe */ 6752b13a985SManish V Badarkhe ret = drtm_dma_prot_engage(&args.dma_prot_args, 6762b13a985SManish V Badarkhe DL_ARGS_GET_DMA_PROT_TYPE(&args)); 6772b13a985SManish V Badarkhe if (ret != SUCCESS) { 6782b13a985SManish V Badarkhe SMC_RET1(handle, ret); 6792b13a985SManish V Badarkhe } 6802b13a985SManish V Badarkhe 6812090e552SManish V Badarkhe /* 6822090e552SManish V Badarkhe * The DMA protection is now engaged. Note that any failure mode that 6832090e552SManish V Badarkhe * returns an error to the DRTM-launch caller must now disengage DMA 6842090e552SManish V Badarkhe * protections before returning to the caller. 6852090e552SManish V Badarkhe */ 6862090e552SManish V Badarkhe 6872090e552SManish V Badarkhe ret = drtm_take_measurements(&args); 6882090e552SManish V Badarkhe if (ret != SUCCESS) { 6892090e552SManish V Badarkhe goto err_undo_dma_prot; 6902090e552SManish V Badarkhe } 6912090e552SManish V Badarkhe 692d42119ccSManish V Badarkhe ret = drtm_dl_prepare_dlme_data(&args); 693d42119ccSManish V Badarkhe if (ret != SUCCESS) { 694d42119ccSManish V Badarkhe goto err_undo_dma_prot; 695d42119ccSManish V Badarkhe } 696d42119ccSManish V Badarkhe 697d1747e1bSManish Pandey /* 698d1747e1bSManish Pandey * Note that, at the time of writing, the DRTM spec allows a successful 699d1747e1bSManish Pandey * launch from NS-EL1 to return to a DLME in NS-EL2. The practical risk 700d1747e1bSManish Pandey * of a privilege escalation, e.g. due to a compromised hypervisor, is 701d1747e1bSManish Pandey * considered small enough not to warrant the specification of additional 702d1747e1bSManish Pandey * DRTM conduits that would be necessary to maintain OSs' abstraction from 703d1747e1bSManish Pandey * the presence of EL2 were the dynamic launch only be allowed from the 704d1747e1bSManish Pandey * highest NS EL. 705d1747e1bSManish Pandey */ 706d1747e1bSManish Pandey 707d1747e1bSManish Pandey dlme_el = (el_implemented(2) != EL_IMPL_NONE) ? MODE_EL2 : MODE_EL1; 708d1747e1bSManish Pandey 709d1747e1bSManish Pandey drtm_dl_reset_dlme_el_state(dlme_el); 710d1747e1bSManish Pandey drtm_dl_reset_dlme_context(dlme_el); 711d1747e1bSManish Pandey 71223378ae0SManish V Badarkhe /* 71323378ae0SManish V Badarkhe * Setting the Generic Timer frequency is required before launching 71423378ae0SManish V Badarkhe * DLME and is already done for running CPU during PSCI setup. 71523378ae0SManish V Badarkhe */ 716d1747e1bSManish Pandey drtm_dl_prepare_eret_to_dlme(&args, dlme_el); 717d1747e1bSManish Pandey 718d1747e1bSManish Pandey /* 719b94d5909SStuart Yoder * As per DRTM 1.0 spec table #30 invalidate the instruction cache 7202c265975SManish Pandey * before jumping to the DLME. This is required to defend against 7212c265975SManish Pandey * potentially-malicious cache contents. 722d1747e1bSManish Pandey */ 7232c265975SManish Pandey invalidate_icache_all(); 724d1747e1bSManish Pandey 725d1747e1bSManish Pandey /* Return the DLME region's address in x0, and the DLME data offset in x1.*/ 726d1747e1bSManish Pandey SMC_RET2(handle, args.dlme_paddr, args.dlme_data_off); 7272090e552SManish V Badarkhe 7282090e552SManish V Badarkhe err_undo_dma_prot: 7292090e552SManish V Badarkhe dma_prot_ret = drtm_dma_prot_disengage(); 7302090e552SManish V Badarkhe if (dma_prot_ret != SUCCESS) { 7312090e552SManish V Badarkhe ERROR("%s(): drtm_dma_prot_disengage() failed unexpectedly" 7322090e552SManish V Badarkhe " rc=%d\n", __func__, ret); 7332090e552SManish V Badarkhe panic(); 7342090e552SManish V Badarkhe } 7352090e552SManish V Badarkhe 736bd6cc0b2SManish Pandey SMC_RET1(handle, ret); 737bd6cc0b2SManish Pandey } 738bd6cc0b2SManish Pandey 739e62748e3SManish V Badarkhe uint64_t drtm_smc_handler(uint32_t smc_fid, 740e62748e3SManish V Badarkhe uint64_t x1, 741e62748e3SManish V Badarkhe uint64_t x2, 742e62748e3SManish V Badarkhe uint64_t x3, 743e62748e3SManish V Badarkhe uint64_t x4, 744e62748e3SManish V Badarkhe void *cookie, 745e62748e3SManish V Badarkhe void *handle, 746e62748e3SManish V Badarkhe uint64_t flags) 747e62748e3SManish V Badarkhe { 748e62748e3SManish V Badarkhe /* Check that the SMC call is from the Normal World. */ 749e62748e3SManish V Badarkhe if (!is_caller_non_secure(flags)) { 750e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 751e62748e3SManish V Badarkhe } 752e62748e3SManish V Badarkhe 753e62748e3SManish V Badarkhe switch (smc_fid) { 754e62748e3SManish V Badarkhe case ARM_DRTM_SVC_VERSION: 755e62748e3SManish V Badarkhe INFO("DRTM service handler: version\n"); 756e62748e3SManish V Badarkhe /* Return the version of current implementation */ 757e62748e3SManish V Badarkhe SMC_RET1(handle, ARM_DRTM_VERSION); 758e62748e3SManish V Badarkhe break; /* not reached */ 759e62748e3SManish V Badarkhe 760e62748e3SManish V Badarkhe case ARM_DRTM_SVC_FEATURES: 761e62748e3SManish V Badarkhe if (((x1 >> ARM_DRTM_FUNC_SHIFT) & ARM_DRTM_FUNC_MASK) == 762e62748e3SManish V Badarkhe ARM_DRTM_FUNC_ID) { 763e62748e3SManish V Badarkhe /* Dispatch function-based queries. */ 764e62748e3SManish V Badarkhe switch (x1 & FUNCID_MASK) { 765e62748e3SManish V Badarkhe case ARM_DRTM_SVC_VERSION: 766e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 767e62748e3SManish V Badarkhe break; /* not reached */ 768e62748e3SManish V Badarkhe 769e62748e3SManish V Badarkhe case ARM_DRTM_SVC_FEATURES: 770e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 771e62748e3SManish V Badarkhe break; /* not reached */ 772e62748e3SManish V Badarkhe 773e62748e3SManish V Badarkhe case ARM_DRTM_SVC_UNPROTECT_MEM: 774e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 775e62748e3SManish V Badarkhe break; /* not reached */ 776e62748e3SManish V Badarkhe 777e62748e3SManish V Badarkhe case ARM_DRTM_SVC_DYNAMIC_LAUNCH: 778e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 779e62748e3SManish V Badarkhe break; /* not reached */ 780e62748e3SManish V Badarkhe 781e62748e3SManish V Badarkhe case ARM_DRTM_SVC_CLOSE_LOCALITY: 782e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_CLOSE_LOCALITY feature %s", 783e62748e3SManish V Badarkhe "is not supported\n"); 784e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 785e62748e3SManish V Badarkhe break; /* not reached */ 786e62748e3SManish V Badarkhe 787e62748e3SManish V Badarkhe case ARM_DRTM_SVC_GET_ERROR: 788e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 789e62748e3SManish V Badarkhe break; /* not reached */ 790e62748e3SManish V Badarkhe 791e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_ERROR: 792e62748e3SManish V Badarkhe SMC_RET1(handle, SUCCESS); 793e62748e3SManish V Badarkhe break; /* not reached */ 794e62748e3SManish V Badarkhe 795e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_TCB_HASH: 796e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_TCB_HASH feature %s", 797e62748e3SManish V Badarkhe "is not supported\n"); 798e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 799e62748e3SManish V Badarkhe break; /* not reached */ 800e62748e3SManish V Badarkhe 801e62748e3SManish V Badarkhe case ARM_DRTM_SVC_LOCK_TCB_HASH: 802e62748e3SManish V Badarkhe WARN("ARM_DRTM_SVC_LOCK_TCB_HASH feature %s", 803e62748e3SManish V Badarkhe "is not supported\n"); 804e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 805e62748e3SManish V Badarkhe break; /* not reached */ 806e62748e3SManish V Badarkhe 807e62748e3SManish V Badarkhe default: 808e62748e3SManish V Badarkhe ERROR("Unknown DRTM service function\n"); 809e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 810e62748e3SManish V Badarkhe break; /* not reached */ 811e62748e3SManish V Badarkhe } 812e9467afbSManish V Badarkhe } else { 813e9467afbSManish V Badarkhe /* Dispatch feature-based queries. */ 814e9467afbSManish V Badarkhe switch (x1 & ARM_DRTM_FEAT_ID_MASK) { 815e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_TPM: 816e9467afbSManish V Badarkhe INFO("++ DRTM service handler: TPM features\n"); 817e9467afbSManish V Badarkhe return drtm_features_tpm(handle); 818e9467afbSManish V Badarkhe break; /* not reached */ 819e9467afbSManish V Badarkhe 820e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_MEM_REQ: 821e9467afbSManish V Badarkhe INFO("++ DRTM service handler: Min. mem." 822e9467afbSManish V Badarkhe " requirement features\n"); 823e9467afbSManish V Badarkhe return drtm_features_mem_req(handle); 824e9467afbSManish V Badarkhe break; /* not reached */ 825e9467afbSManish V Badarkhe 826e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_DMA_PROT: 827e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 828e9467afbSManish V Badarkhe "DMA protection features\n"); 829e9467afbSManish V Badarkhe return drtm_features_dma_prot(handle); 830e9467afbSManish V Badarkhe break; /* not reached */ 831e9467afbSManish V Badarkhe 832e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_BOOT_PE_ID: 833e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 834e9467afbSManish V Badarkhe "Boot PE ID features\n"); 835e9467afbSManish V Badarkhe return drtm_features_boot_pe_id(handle); 836e9467afbSManish V Badarkhe break; /* not reached */ 837e9467afbSManish V Badarkhe 838e9467afbSManish V Badarkhe case ARM_DRTM_FEATURES_TCB_HASHES: 839e9467afbSManish V Badarkhe INFO("++ DRTM service handler: " 840e9467afbSManish V Badarkhe "TCB-hashes features\n"); 841e9467afbSManish V Badarkhe return drtm_features_tcb_hashes(handle); 842e9467afbSManish V Badarkhe break; /* not reached */ 843e9467afbSManish V Badarkhe 84494127ae2SManish V Badarkhe case ARM_DRTM_FEATURES_DLME_IMG_AUTH: 84594127ae2SManish V Badarkhe INFO("++ DRTM service handler: " 84694127ae2SManish V Badarkhe "DLME Image authentication features\n"); 84794127ae2SManish V Badarkhe return drtm_features_dlme_img_auth_features(handle); 84894127ae2SManish V Badarkhe break; /* not reached */ 84994127ae2SManish V Badarkhe 850e9467afbSManish V Badarkhe default: 851e9467afbSManish V Badarkhe ERROR("Unknown ARM DRTM service feature\n"); 852e9467afbSManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 853e9467afbSManish V Badarkhe break; /* not reached */ 854e9467afbSManish V Badarkhe } 855e62748e3SManish V Badarkhe } 856e62748e3SManish V Badarkhe 857e62748e3SManish V Badarkhe case ARM_DRTM_SVC_UNPROTECT_MEM: 858e62748e3SManish V Badarkhe INFO("DRTM service handler: unprotect mem\n"); 8592b13a985SManish V Badarkhe return drtm_unprotect_mem(handle); 860e62748e3SManish V Badarkhe break; /* not reached */ 861e62748e3SManish V Badarkhe 862e62748e3SManish V Badarkhe case ARM_DRTM_SVC_DYNAMIC_LAUNCH: 863e62748e3SManish V Badarkhe INFO("DRTM service handler: dynamic launch\n"); 864bd6cc0b2SManish Pandey return drtm_dynamic_launch(x1, handle); 865e62748e3SManish V Badarkhe break; /* not reached */ 866e62748e3SManish V Badarkhe 867e62748e3SManish V Badarkhe case ARM_DRTM_SVC_CLOSE_LOCALITY: 868e62748e3SManish V Badarkhe WARN("DRTM service handler: close locality %s\n", 869e62748e3SManish V Badarkhe "is not supported"); 870e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 871e62748e3SManish V Badarkhe break; /* not reached */ 872e62748e3SManish V Badarkhe 873e62748e3SManish V Badarkhe case ARM_DRTM_SVC_GET_ERROR: 874e62748e3SManish V Badarkhe INFO("DRTM service handler: get error\n"); 8755e1fa574SManish V Badarkhe return drtm_get_error(handle); 876e62748e3SManish V Badarkhe break; /* not reached */ 877e62748e3SManish V Badarkhe 878e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_ERROR: 879e62748e3SManish V Badarkhe INFO("DRTM service handler: set error\n"); 8805e1fa574SManish V Badarkhe return drtm_set_error(x1, handle); 881e62748e3SManish V Badarkhe break; /* not reached */ 882e62748e3SManish V Badarkhe 883e62748e3SManish V Badarkhe case ARM_DRTM_SVC_SET_TCB_HASH: 884e62748e3SManish V Badarkhe WARN("DRTM service handler: set TCB hash %s\n", 885e62748e3SManish V Badarkhe "is not supported"); 886e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 887e62748e3SManish V Badarkhe break; /* not reached */ 888e62748e3SManish V Badarkhe 889e62748e3SManish V Badarkhe case ARM_DRTM_SVC_LOCK_TCB_HASH: 890e62748e3SManish V Badarkhe WARN("DRTM service handler: lock TCB hash %s\n", 891e62748e3SManish V Badarkhe "is not supported"); 892e62748e3SManish V Badarkhe SMC_RET1(handle, NOT_SUPPORTED); 893e62748e3SManish V Badarkhe break; /* not reached */ 894e62748e3SManish V Badarkhe 895e62748e3SManish V Badarkhe default: 896e62748e3SManish V Badarkhe ERROR("Unknown DRTM service function: 0x%x\n", smc_fid); 897e62748e3SManish V Badarkhe SMC_RET1(handle, SMC_UNK); 898e62748e3SManish V Badarkhe break; /* not reached */ 899e62748e3SManish V Badarkhe } 900e62748e3SManish V Badarkhe 901e62748e3SManish V Badarkhe /* not reached */ 902e62748e3SManish V Badarkhe SMC_RET1(handle, SMC_UNK); 903e62748e3SManish V Badarkhe } 904