1*91f16700Schasinglulu /* 2*91f16700Schasinglulu * Copyright 2018-2021 NXP 3*91f16700Schasinglulu * 4*91f16700Schasinglulu * SPDX-License-Identifier: BSD-3-Clause 5*91f16700Schasinglulu * 6*91f16700Schasinglulu */ 7*91f16700Schasinglulu 8*91f16700Schasinglulu #include <assert.h> 9*91f16700Schasinglulu #include <string.h> 10*91f16700Schasinglulu 11*91f16700Schasinglulu #include <caam.h> 12*91f16700Schasinglulu #include <common/runtime_svc.h> 13*91f16700Schasinglulu #include <dcfg.h> 14*91f16700Schasinglulu #include <lib/mmio.h> 15*91f16700Schasinglulu #include <tools_share/uuid.h> 16*91f16700Schasinglulu 17*91f16700Schasinglulu #include <plat_common.h> 18*91f16700Schasinglulu #include <sipsvc.h> 19*91f16700Schasinglulu 20*91f16700Schasinglulu /* Layerscape SiP Service UUID */ 21*91f16700Schasinglulu DEFINE_SVC_UUID2(nxp_sip_svc_uid, 22*91f16700Schasinglulu 0x871de4ef, 0xedfc, 0x4209, 0xa4, 0x23, 23*91f16700Schasinglulu 0x8d, 0x23, 0x75, 0x9d, 0x3b, 0x9f); 24*91f16700Schasinglulu 25*91f16700Schasinglulu #pragma weak nxp_plat_sip_handler 26*91f16700Schasinglulu static uintptr_t nxp_plat_sip_handler(unsigned int smc_fid, 27*91f16700Schasinglulu u_register_t x1, 28*91f16700Schasinglulu u_register_t x2, 29*91f16700Schasinglulu u_register_t x3, 30*91f16700Schasinglulu u_register_t x4, 31*91f16700Schasinglulu void *cookie, 32*91f16700Schasinglulu void *handle, 33*91f16700Schasinglulu u_register_t flags) 34*91f16700Schasinglulu { 35*91f16700Schasinglulu ERROR("%s: unhandled SMC (0x%x)\n", __func__, smc_fid); 36*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 37*91f16700Schasinglulu } 38*91f16700Schasinglulu 39*91f16700Schasinglulu uint64_t el2_2_aarch32(u_register_t smc_id, u_register_t start_addr, 40*91f16700Schasinglulu u_register_t parm1, u_register_t parm2); 41*91f16700Schasinglulu 42*91f16700Schasinglulu uint64_t prefetch_disable(u_register_t smc_id, u_register_t mask); 43*91f16700Schasinglulu uint64_t bl31_get_porsr1(void); 44*91f16700Schasinglulu 45*91f16700Schasinglulu static void clean_top_32b_of_param(uint32_t smc_fid, 46*91f16700Schasinglulu u_register_t *px1, 47*91f16700Schasinglulu u_register_t *px2, 48*91f16700Schasinglulu u_register_t *px3, 49*91f16700Schasinglulu u_register_t *px4) 50*91f16700Schasinglulu { 51*91f16700Schasinglulu /* if parameters from SMC32. Clean top 32 bits */ 52*91f16700Schasinglulu if (GET_SMC_CC(smc_fid) == SMC_32) { 53*91f16700Schasinglulu *px1 = *px1 & SMC32_PARAM_MASK; 54*91f16700Schasinglulu *px2 = *px2 & SMC32_PARAM_MASK; 55*91f16700Schasinglulu *px3 = *px3 & SMC32_PARAM_MASK; 56*91f16700Schasinglulu *px4 = *px4 & SMC32_PARAM_MASK; 57*91f16700Schasinglulu } 58*91f16700Schasinglulu } 59*91f16700Schasinglulu 60*91f16700Schasinglulu /* This function handles Layerscape defined SiP Calls */ 61*91f16700Schasinglulu static uintptr_t nxp_sip_handler(unsigned int smc_fid, 62*91f16700Schasinglulu u_register_t x1, 63*91f16700Schasinglulu u_register_t x2, 64*91f16700Schasinglulu u_register_t x3, 65*91f16700Schasinglulu u_register_t x4, 66*91f16700Schasinglulu void *cookie, 67*91f16700Schasinglulu void *handle, 68*91f16700Schasinglulu u_register_t flags) 69*91f16700Schasinglulu { 70*91f16700Schasinglulu uint32_t ns; 71*91f16700Schasinglulu uint64_t ret; 72*91f16700Schasinglulu dram_regions_info_t *info_dram_regions; 73*91f16700Schasinglulu 74*91f16700Schasinglulu /* if parameter is sent from SMC32. Clean top 32 bits */ 75*91f16700Schasinglulu clean_top_32b_of_param(smc_fid, &x1, &x2, &x3, &x4); 76*91f16700Schasinglulu 77*91f16700Schasinglulu /* Determine which security state this SMC originated from */ 78*91f16700Schasinglulu ns = is_caller_non_secure(flags); 79*91f16700Schasinglulu if (ns == 0) { 80*91f16700Schasinglulu /* SiP SMC service secure world's call */ 81*91f16700Schasinglulu ; 82*91f16700Schasinglulu } else { 83*91f16700Schasinglulu /* SiP SMC service normal world's call */ 84*91f16700Schasinglulu ; 85*91f16700Schasinglulu } 86*91f16700Schasinglulu 87*91f16700Schasinglulu switch (smc_fid & SMC_FUNC_MASK) { 88*91f16700Schasinglulu case SIP_SVC_RNG: 89*91f16700Schasinglulu if (is_sec_enabled() == false) { 90*91f16700Schasinglulu NOTICE("SEC is disabled.\n"); 91*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 92*91f16700Schasinglulu } 93*91f16700Schasinglulu 94*91f16700Schasinglulu /* Return zero on failure */ 95*91f16700Schasinglulu ret = get_random((int)x1); 96*91f16700Schasinglulu if (ret != 0) { 97*91f16700Schasinglulu SMC_RET2(handle, SMC_OK, ret); 98*91f16700Schasinglulu } else { 99*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 100*91f16700Schasinglulu } 101*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 102*91f16700Schasinglulu case SIP_SVC_HUK: 103*91f16700Schasinglulu if (is_sec_enabled() == false) { 104*91f16700Schasinglulu NOTICE("SEC is disabled.\n"); 105*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 106*91f16700Schasinglulu } 107*91f16700Schasinglulu ret = get_hw_unq_key_blob_hw((uint8_t *) x1, (uint32_t) x2); 108*91f16700Schasinglulu 109*91f16700Schasinglulu if (ret == SMC_OK) { 110*91f16700Schasinglulu SMC_RET1(handle, SMC_OK); 111*91f16700Schasinglulu } else { 112*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 113*91f16700Schasinglulu } 114*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 115*91f16700Schasinglulu case SIP_SVC_MEM_BANK: 116*91f16700Schasinglulu VERBOSE("Handling SMC SIP_SVC_MEM_BANK.\n"); 117*91f16700Schasinglulu info_dram_regions = get_dram_regions_info(); 118*91f16700Schasinglulu 119*91f16700Schasinglulu if (x1 == -1) { 120*91f16700Schasinglulu SMC_RET2(handle, SMC_OK, 121*91f16700Schasinglulu info_dram_regions->total_dram_size); 122*91f16700Schasinglulu } else if (x1 >= info_dram_regions->num_dram_regions) { 123*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 124*91f16700Schasinglulu } else { 125*91f16700Schasinglulu SMC_RET3(handle, SMC_OK, 126*91f16700Schasinglulu info_dram_regions->region[x1].addr, 127*91f16700Schasinglulu info_dram_regions->region[x1].size); 128*91f16700Schasinglulu } 129*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 130*91f16700Schasinglulu case SIP_SVC_PREFETCH_DIS: 131*91f16700Schasinglulu VERBOSE("In SIP_SVC_PREFETCH_DIS call\n"); 132*91f16700Schasinglulu ret = prefetch_disable(smc_fid, x1); 133*91f16700Schasinglulu if (ret == SMC_OK) { 134*91f16700Schasinglulu SMC_RET1(handle, SMC_OK); 135*91f16700Schasinglulu } else { 136*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 137*91f16700Schasinglulu } 138*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 139*91f16700Schasinglulu case SIP_SVC_2_AARCH32: 140*91f16700Schasinglulu ret = el2_2_aarch32(smc_fid, x1, x2, x3); 141*91f16700Schasinglulu 142*91f16700Schasinglulu /* In success case, control should not reach here. */ 143*91f16700Schasinglulu NOTICE("SMC: SIP_SVC_2_AARCH32 Failed.\n"); 144*91f16700Schasinglulu SMC_RET1(handle, SMC_UNK); 145*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 146*91f16700Schasinglulu case SIP_SVC_PORSR1: 147*91f16700Schasinglulu ret = bl31_get_porsr1(); 148*91f16700Schasinglulu SMC_RET2(handle, SMC_OK, ret); 149*91f16700Schasinglulu /* break is not required as SMC_RETx return */ 150*91f16700Schasinglulu default: 151*91f16700Schasinglulu return nxp_plat_sip_handler(smc_fid, x1, x2, x3, x4, 152*91f16700Schasinglulu cookie, handle, flags); 153*91f16700Schasinglulu } 154*91f16700Schasinglulu } 155*91f16700Schasinglulu 156*91f16700Schasinglulu /* This function is responsible for handling all SiP calls */ 157*91f16700Schasinglulu static uintptr_t sip_smc_handler(unsigned int smc_fid, 158*91f16700Schasinglulu u_register_t x1, 159*91f16700Schasinglulu u_register_t x2, 160*91f16700Schasinglulu u_register_t x3, 161*91f16700Schasinglulu u_register_t x4, 162*91f16700Schasinglulu void *cookie, 163*91f16700Schasinglulu void *handle, 164*91f16700Schasinglulu u_register_t flags) 165*91f16700Schasinglulu { 166*91f16700Schasinglulu switch (smc_fid & SMC_FUNC_MASK) { 167*91f16700Schasinglulu case SIP_SVC_CALL_COUNT: 168*91f16700Schasinglulu /* Return the number of Layerscape SiP Service Calls. */ 169*91f16700Schasinglulu SMC_RET1(handle, LS_COMMON_SIP_NUM_CALLS); 170*91f16700Schasinglulu break; 171*91f16700Schasinglulu case SIP_SVC_UID: 172*91f16700Schasinglulu /* Return UID to the caller */ 173*91f16700Schasinglulu SMC_UUID_RET(handle, nxp_sip_svc_uid); 174*91f16700Schasinglulu break; 175*91f16700Schasinglulu case SIP_SVC_VERSION: 176*91f16700Schasinglulu /* Return the version of current implementation */ 177*91f16700Schasinglulu SMC_RET2(handle, LS_SIP_SVC_VERSION_MAJOR, 178*91f16700Schasinglulu LS_SIP_SVC_VERSION_MINOR); 179*91f16700Schasinglulu break; 180*91f16700Schasinglulu default: 181*91f16700Schasinglulu return nxp_sip_handler(smc_fid, x1, x2, x3, x4, 182*91f16700Schasinglulu cookie, handle, flags); 183*91f16700Schasinglulu } 184*91f16700Schasinglulu } 185*91f16700Schasinglulu 186*91f16700Schasinglulu /* Define a runtime service descriptor for fast SMC calls */ 187*91f16700Schasinglulu DECLARE_RT_SVC( 188*91f16700Schasinglulu nxp_sip_svc, 189*91f16700Schasinglulu OEN_SIP_START, 190*91f16700Schasinglulu OEN_SIP_END, 191*91f16700Schasinglulu SMC_TYPE_FAST, 192*91f16700Schasinglulu NULL, 193*91f16700Schasinglulu sip_smc_handler 194*91f16700Schasinglulu ); 195