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