1a2847172SGrzegorz Jaszczyk /* 2a2847172SGrzegorz Jaszczyk * Copyright (C) 2018 Marvell International Ltd. 3a2847172SGrzegorz Jaszczyk * 4a2847172SGrzegorz Jaszczyk * SPDX-License-Identifier: BSD-3-Clause 5a2847172SGrzegorz Jaszczyk * https://spdx.org/licenses 6a2847172SGrzegorz Jaszczyk */ 7a2847172SGrzegorz Jaszczyk 8a2847172SGrzegorz Jaszczyk #include <common/debug.h> 9a2847172SGrzegorz Jaszczyk #include <common/runtime_svc.h> 10a2847172SGrzegorz Jaszczyk #include <drivers/marvell/cache_llc.h> 11a2847172SGrzegorz Jaszczyk #include <drivers/marvell/mochi/ap_setup.h> 12*6d5fad8dSWilson Ding #include <drivers/marvell/trng.h> 13a2847172SGrzegorz Jaszczyk #include <lib/smccc.h> 14a2847172SGrzegorz Jaszczyk 15a2847172SGrzegorz Jaszczyk #include <marvell_plat_priv.h> 16a2847172SGrzegorz Jaszczyk #include <plat_marvell.h> 17a2847172SGrzegorz Jaszczyk 18a2847172SGrzegorz Jaszczyk #include "comphy/phy-comphy-cp110.h" 19ad416958SGrzegorz Jaszczyk #include "secure_dfx_access/dfx.h" 20b81444e8SAlex Leibovich #include "ddr_phy_access.h" 21a2847172SGrzegorz Jaszczyk #include <stdbool.h> 22a2847172SGrzegorz Jaszczyk 23a2847172SGrzegorz Jaszczyk /* #define DEBUG_COMPHY */ 24a2847172SGrzegorz Jaszczyk #ifdef DEBUG_COMPHY 25a2847172SGrzegorz Jaszczyk #define debug(format...) NOTICE(format) 26a2847172SGrzegorz Jaszczyk #else 27a2847172SGrzegorz Jaszczyk #define debug(format, arg...) 28a2847172SGrzegorz Jaszczyk #endif 29a2847172SGrzegorz Jaszczyk 30a2847172SGrzegorz Jaszczyk /* Comphy related FID's */ 31a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_POWER_ON 0x82000001 32a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_POWER_OFF 0x82000002 33a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_PLL_LOCK 0x82000003 34a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_XFI_TRAIN 0x82000004 35a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_DIG_RESET 0x82000005 36a2847172SGrzegorz Jaszczyk 37a2847172SGrzegorz Jaszczyk /* Miscellaneous FID's' */ 38a2847172SGrzegorz Jaszczyk #define MV_SIP_DRAM_SIZE 0x82000010 39a2847172SGrzegorz Jaszczyk #define MV_SIP_LLC_ENABLE 0x82000011 40a2847172SGrzegorz Jaszczyk #define MV_SIP_PMU_IRQ_ENABLE 0x82000012 41a2847172SGrzegorz Jaszczyk #define MV_SIP_PMU_IRQ_DISABLE 0x82000013 42ad416958SGrzegorz Jaszczyk #define MV_SIP_DFX 0x82000014 43b81444e8SAlex Leibovich #define MV_SIP_DDR_PHY_WRITE 0x82000015 44b81444e8SAlex Leibovich #define MV_SIP_DDR_PHY_READ 0x82000016 45a2847172SGrzegorz Jaszczyk 46*6d5fad8dSWilson Ding /* TRNG */ 47*6d5fad8dSWilson Ding #define MV_SIP_RNG_64 0xC200FF11 48*6d5fad8dSWilson Ding 49a2847172SGrzegorz Jaszczyk #define MAX_LANE_NR 6 50a2847172SGrzegorz Jaszczyk #define MVEBU_COMPHY_OFFSET 0x441000 51a2847172SGrzegorz Jaszczyk #define MVEBU_CP_BASE_MASK (~0xffffff) 52a2847172SGrzegorz Jaszczyk 53550a06dfSAlex Evraev /* Common PHY register */ 54550a06dfSAlex Evraev #define COMPHY_TRX_TRAIN_CTRL_REG_0_OFFS 0x120a2c 55550a06dfSAlex Evraev 56a2847172SGrzegorz Jaszczyk /* This macro is used to identify COMPHY related calls from SMC function ID */ 57a2847172SGrzegorz Jaszczyk #define is_comphy_fid(fid) \ 58a2847172SGrzegorz Jaszczyk ((fid) >= MV_SIP_COMPHY_POWER_ON && (fid) <= MV_SIP_COMPHY_DIG_RESET) 59a2847172SGrzegorz Jaszczyk 60a2847172SGrzegorz Jaszczyk _Bool is_cp_range_valid(u_register_t *addr) 61a2847172SGrzegorz Jaszczyk { 62a2847172SGrzegorz Jaszczyk int cp_nr; 63a2847172SGrzegorz Jaszczyk 64a2847172SGrzegorz Jaszczyk *addr &= MVEBU_CP_BASE_MASK; 65a2847172SGrzegorz Jaszczyk for (cp_nr = 0; cp_nr < CP_NUM; cp_nr++) { 66a2847172SGrzegorz Jaszczyk if (*addr == MVEBU_CP_REGS_BASE(cp_nr)) 67a2847172SGrzegorz Jaszczyk return true; 68a2847172SGrzegorz Jaszczyk } 69a2847172SGrzegorz Jaszczyk 70a2847172SGrzegorz Jaszczyk return false; 71a2847172SGrzegorz Jaszczyk } 72a2847172SGrzegorz Jaszczyk 73a2847172SGrzegorz Jaszczyk uintptr_t mrvl_sip_smc_handler(uint32_t smc_fid, 74a2847172SGrzegorz Jaszczyk u_register_t x1, 75a2847172SGrzegorz Jaszczyk u_register_t x2, 76a2847172SGrzegorz Jaszczyk u_register_t x3, 77a2847172SGrzegorz Jaszczyk u_register_t x4, 78a2847172SGrzegorz Jaszczyk void *cookie, 79a2847172SGrzegorz Jaszczyk void *handle, 80a2847172SGrzegorz Jaszczyk u_register_t flags) 81a2847172SGrzegorz Jaszczyk { 82550a06dfSAlex Evraev u_register_t ret, read, x5 = x1; 83a2847172SGrzegorz Jaszczyk int i; 84a2847172SGrzegorz Jaszczyk 85a2847172SGrzegorz Jaszczyk debug("%s: got SMC (0x%x) x1 0x%lx, x2 0x%lx, x3 0x%lx\n", 86a2847172SGrzegorz Jaszczyk __func__, smc_fid, x1, x2, x3); 87a2847172SGrzegorz Jaszczyk 88a2847172SGrzegorz Jaszczyk if (is_comphy_fid(smc_fid)) { 89a2847172SGrzegorz Jaszczyk /* validate address passed via x1 */ 90a2847172SGrzegorz Jaszczyk if (!is_cp_range_valid(&x1)) { 91a2847172SGrzegorz Jaszczyk ERROR("%s: Wrong smc (0x%x) address: %lx\n", 92a2847172SGrzegorz Jaszczyk __func__, smc_fid, x1); 93a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 94a2847172SGrzegorz Jaszczyk } 95a2847172SGrzegorz Jaszczyk 96550a06dfSAlex Evraev x5 = x1 + COMPHY_TRX_TRAIN_CTRL_REG_0_OFFS; 97a2847172SGrzegorz Jaszczyk x1 += MVEBU_COMPHY_OFFSET; 98a2847172SGrzegorz Jaszczyk 99a2847172SGrzegorz Jaszczyk if (x2 >= MAX_LANE_NR) { 100a2847172SGrzegorz Jaszczyk ERROR("%s: Wrong smc (0x%x) lane nr: %lx\n", 101a2847172SGrzegorz Jaszczyk __func__, smc_fid, x2); 102a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 103a2847172SGrzegorz Jaszczyk } 104a2847172SGrzegorz Jaszczyk } 105a2847172SGrzegorz Jaszczyk 106a2847172SGrzegorz Jaszczyk switch (smc_fid) { 107a2847172SGrzegorz Jaszczyk 108a2847172SGrzegorz Jaszczyk /* Comphy related FID's */ 109a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_POWER_ON: 110a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index, x3: comphy_mode */ 111550a06dfSAlex Evraev ret = mvebu_cp110_comphy_power_on(x1, x2, x3, x5); 112a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 113a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_POWER_OFF: 114a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 115a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_power_off(x1, x2, x3); 116a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 117a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_PLL_LOCK: 118a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 119a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_is_pll_locked(x1, x2); 120a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 121a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_XFI_TRAIN: 122a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 123a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_xfi_rx_training(x1, x2); 124a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 125a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_DIG_RESET: 126a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index, x3: mode, x4: command */ 127a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_digital_reset(x1, x2, x3, x4); 128a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 129a2847172SGrzegorz Jaszczyk 130a2847172SGrzegorz Jaszczyk /* Miscellaneous FID's' */ 131a2847172SGrzegorz Jaszczyk case MV_SIP_DRAM_SIZE: 132a2847172SGrzegorz Jaszczyk ret = mvebu_get_dram_size(MVEBU_REGS_BASE); 133a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 134a2847172SGrzegorz Jaszczyk case MV_SIP_LLC_ENABLE: 135a2847172SGrzegorz Jaszczyk for (i = 0; i < ap_get_count(); i++) 136a2847172SGrzegorz Jaszczyk llc_runtime_enable(i); 137a2847172SGrzegorz Jaszczyk 138a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 139a2847172SGrzegorz Jaszczyk #ifdef MVEBU_PMU_IRQ_WA 140a2847172SGrzegorz Jaszczyk case MV_SIP_PMU_IRQ_ENABLE: 141a2847172SGrzegorz Jaszczyk mvebu_pmu_interrupt_enable(); 142a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 143a2847172SGrzegorz Jaszczyk case MV_SIP_PMU_IRQ_DISABLE: 144a2847172SGrzegorz Jaszczyk mvebu_pmu_interrupt_disable(); 145a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 146a2847172SGrzegorz Jaszczyk #endif 147ad416958SGrzegorz Jaszczyk case MV_SIP_DFX: 1480cedca63SGrzegorz Jaszczyk if (x1 >= MV_SIP_DFX_THERMAL_INIT && 1490cedca63SGrzegorz Jaszczyk x1 <= MV_SIP_DFX_THERMAL_SEL_CHANNEL) { 1500cedca63SGrzegorz Jaszczyk ret = mvebu_dfx_thermal_handle(x1, &read, x2, x3); 151ad416958SGrzegorz Jaszczyk SMC_RET2(handle, ret, read); 1520cedca63SGrzegorz Jaszczyk } 15381c2a044SGrzegorz Jaszczyk if (x1 >= MV_SIP_DFX_SREAD && x1 <= MV_SIP_DFX_SWRITE) { 15481c2a044SGrzegorz Jaszczyk ret = mvebu_dfx_misc_handle(x1, &read, x2, x3); 15581c2a044SGrzegorz Jaszczyk SMC_RET2(handle, ret, read); 15681c2a044SGrzegorz Jaszczyk } 15781c2a044SGrzegorz Jaszczyk 1580cedca63SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 159b81444e8SAlex Leibovich case MV_SIP_DDR_PHY_WRITE: 160b81444e8SAlex Leibovich ret = mvebu_ddr_phy_write(x1, x2); 161b81444e8SAlex Leibovich SMC_RET1(handle, ret); 162b81444e8SAlex Leibovich case MV_SIP_DDR_PHY_READ: 163b81444e8SAlex Leibovich read = 0; 164b81444e8SAlex Leibovich ret = mvebu_ddr_phy_read(x1, (uint16_t *)&read); 165b81444e8SAlex Leibovich SMC_RET2(handle, ret, read); 166*6d5fad8dSWilson Ding case MV_SIP_RNG_64: 167*6d5fad8dSWilson Ding if (x1 > 1) { 168*6d5fad8dSWilson Ding SMC_RET1(handle, SMC_UNK); 169*6d5fad8dSWilson Ding } 170*6d5fad8dSWilson Ding 171*6d5fad8dSWilson Ding ret = mv_trng_get_random32((uint32_t *)&read, 172*6d5fad8dSWilson Ding ((uint8_t)x1 + 1)); 173*6d5fad8dSWilson Ding SMC_RET2(handle, ret, read); 174a2847172SGrzegorz Jaszczyk default: 175a2847172SGrzegorz Jaszczyk ERROR("%s: unhandled SMC (0x%x)\n", __func__, smc_fid); 176a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 177a2847172SGrzegorz Jaszczyk } 178a2847172SGrzegorz Jaszczyk } 179a2847172SGrzegorz Jaszczyk 180a2847172SGrzegorz Jaszczyk /* Define a runtime service descriptor for fast SMC calls */ 181a2847172SGrzegorz Jaszczyk DECLARE_RT_SVC( 182a2847172SGrzegorz Jaszczyk marvell_sip_svc, 183a2847172SGrzegorz Jaszczyk OEN_SIP_START, 184a2847172SGrzegorz Jaszczyk OEN_SIP_END, 185a2847172SGrzegorz Jaszczyk SMC_TYPE_FAST, 186a2847172SGrzegorz Jaszczyk NULL, 187a2847172SGrzegorz Jaszczyk mrvl_sip_smc_handler 188a2847172SGrzegorz Jaszczyk ); 189