1*a2847172SGrzegorz Jaszczyk /* 2*a2847172SGrzegorz Jaszczyk * Copyright (C) 2018 Marvell International Ltd. 3*a2847172SGrzegorz Jaszczyk * 4*a2847172SGrzegorz Jaszczyk * SPDX-License-Identifier: BSD-3-Clause 5*a2847172SGrzegorz Jaszczyk * https://spdx.org/licenses 6*a2847172SGrzegorz Jaszczyk */ 7*a2847172SGrzegorz Jaszczyk 8*a2847172SGrzegorz Jaszczyk #include <common/debug.h> 9*a2847172SGrzegorz Jaszczyk #include <common/runtime_svc.h> 10*a2847172SGrzegorz Jaszczyk #include <drivers/marvell/cache_llc.h> 11*a2847172SGrzegorz Jaszczyk #include <drivers/marvell/mochi/ap_setup.h> 12*a2847172SGrzegorz Jaszczyk #include <lib/smccc.h> 13*a2847172SGrzegorz Jaszczyk 14*a2847172SGrzegorz Jaszczyk #include <marvell_plat_priv.h> 15*a2847172SGrzegorz Jaszczyk #include <plat_marvell.h> 16*a2847172SGrzegorz Jaszczyk 17*a2847172SGrzegorz Jaszczyk #include "comphy/phy-comphy-cp110.h" 18*a2847172SGrzegorz Jaszczyk #include <stdbool.h> 19*a2847172SGrzegorz Jaszczyk 20*a2847172SGrzegorz Jaszczyk /* #define DEBUG_COMPHY */ 21*a2847172SGrzegorz Jaszczyk #ifdef DEBUG_COMPHY 22*a2847172SGrzegorz Jaszczyk #define debug(format...) NOTICE(format) 23*a2847172SGrzegorz Jaszczyk #else 24*a2847172SGrzegorz Jaszczyk #define debug(format, arg...) 25*a2847172SGrzegorz Jaszczyk #endif 26*a2847172SGrzegorz Jaszczyk 27*a2847172SGrzegorz Jaszczyk /* Comphy related FID's */ 28*a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_POWER_ON 0x82000001 29*a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_POWER_OFF 0x82000002 30*a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_PLL_LOCK 0x82000003 31*a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_XFI_TRAIN 0x82000004 32*a2847172SGrzegorz Jaszczyk #define MV_SIP_COMPHY_DIG_RESET 0x82000005 33*a2847172SGrzegorz Jaszczyk 34*a2847172SGrzegorz Jaszczyk /* Miscellaneous FID's' */ 35*a2847172SGrzegorz Jaszczyk #define MV_SIP_DRAM_SIZE 0x82000010 36*a2847172SGrzegorz Jaszczyk #define MV_SIP_LLC_ENABLE 0x82000011 37*a2847172SGrzegorz Jaszczyk #define MV_SIP_PMU_IRQ_ENABLE 0x82000012 38*a2847172SGrzegorz Jaszczyk #define MV_SIP_PMU_IRQ_DISABLE 0x82000013 39*a2847172SGrzegorz Jaszczyk 40*a2847172SGrzegorz Jaszczyk #define MAX_LANE_NR 6 41*a2847172SGrzegorz Jaszczyk #define MVEBU_COMPHY_OFFSET 0x441000 42*a2847172SGrzegorz Jaszczyk #define MVEBU_CP_BASE_MASK (~0xffffff) 43*a2847172SGrzegorz Jaszczyk 44*a2847172SGrzegorz Jaszczyk /* This macro is used to identify COMPHY related calls from SMC function ID */ 45*a2847172SGrzegorz Jaszczyk #define is_comphy_fid(fid) \ 46*a2847172SGrzegorz Jaszczyk ((fid) >= MV_SIP_COMPHY_POWER_ON && (fid) <= MV_SIP_COMPHY_DIG_RESET) 47*a2847172SGrzegorz Jaszczyk 48*a2847172SGrzegorz Jaszczyk _Bool is_cp_range_valid(u_register_t *addr) 49*a2847172SGrzegorz Jaszczyk { 50*a2847172SGrzegorz Jaszczyk int cp_nr; 51*a2847172SGrzegorz Jaszczyk 52*a2847172SGrzegorz Jaszczyk *addr &= MVEBU_CP_BASE_MASK; 53*a2847172SGrzegorz Jaszczyk for (cp_nr = 0; cp_nr < CP_NUM; cp_nr++) { 54*a2847172SGrzegorz Jaszczyk if (*addr == MVEBU_CP_REGS_BASE(cp_nr)) 55*a2847172SGrzegorz Jaszczyk return true; 56*a2847172SGrzegorz Jaszczyk } 57*a2847172SGrzegorz Jaszczyk 58*a2847172SGrzegorz Jaszczyk return false; 59*a2847172SGrzegorz Jaszczyk } 60*a2847172SGrzegorz Jaszczyk 61*a2847172SGrzegorz Jaszczyk uintptr_t mrvl_sip_smc_handler(uint32_t smc_fid, 62*a2847172SGrzegorz Jaszczyk u_register_t x1, 63*a2847172SGrzegorz Jaszczyk u_register_t x2, 64*a2847172SGrzegorz Jaszczyk u_register_t x3, 65*a2847172SGrzegorz Jaszczyk u_register_t x4, 66*a2847172SGrzegorz Jaszczyk void *cookie, 67*a2847172SGrzegorz Jaszczyk void *handle, 68*a2847172SGrzegorz Jaszczyk u_register_t flags) 69*a2847172SGrzegorz Jaszczyk { 70*a2847172SGrzegorz Jaszczyk u_register_t ret; 71*a2847172SGrzegorz Jaszczyk int i; 72*a2847172SGrzegorz Jaszczyk 73*a2847172SGrzegorz Jaszczyk debug("%s: got SMC (0x%x) x1 0x%lx, x2 0x%lx, x3 0x%lx\n", 74*a2847172SGrzegorz Jaszczyk __func__, smc_fid, x1, x2, x3); 75*a2847172SGrzegorz Jaszczyk 76*a2847172SGrzegorz Jaszczyk if (is_comphy_fid(smc_fid)) { 77*a2847172SGrzegorz Jaszczyk /* validate address passed via x1 */ 78*a2847172SGrzegorz Jaszczyk if (!is_cp_range_valid(&x1)) { 79*a2847172SGrzegorz Jaszczyk ERROR("%s: Wrong smc (0x%x) address: %lx\n", 80*a2847172SGrzegorz Jaszczyk __func__, smc_fid, x1); 81*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 82*a2847172SGrzegorz Jaszczyk } 83*a2847172SGrzegorz Jaszczyk 84*a2847172SGrzegorz Jaszczyk x1 += MVEBU_COMPHY_OFFSET; 85*a2847172SGrzegorz Jaszczyk 86*a2847172SGrzegorz Jaszczyk if (x2 >= MAX_LANE_NR) { 87*a2847172SGrzegorz Jaszczyk ERROR("%s: Wrong smc (0x%x) lane nr: %lx\n", 88*a2847172SGrzegorz Jaszczyk __func__, smc_fid, x2); 89*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 90*a2847172SGrzegorz Jaszczyk } 91*a2847172SGrzegorz Jaszczyk } 92*a2847172SGrzegorz Jaszczyk 93*a2847172SGrzegorz Jaszczyk switch (smc_fid) { 94*a2847172SGrzegorz Jaszczyk 95*a2847172SGrzegorz Jaszczyk /* Comphy related FID's */ 96*a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_POWER_ON: 97*a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index, x3: comphy_mode */ 98*a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_power_on(x1, x2, x3); 99*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 100*a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_POWER_OFF: 101*a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 102*a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_power_off(x1, x2, x3); 103*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 104*a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_PLL_LOCK: 105*a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 106*a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_is_pll_locked(x1, x2); 107*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 108*a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_XFI_TRAIN: 109*a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index */ 110*a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_xfi_rx_training(x1, x2); 111*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 112*a2847172SGrzegorz Jaszczyk case MV_SIP_COMPHY_DIG_RESET: 113*a2847172SGrzegorz Jaszczyk /* x1: comphy_base, x2: comphy_index, x3: mode, x4: command */ 114*a2847172SGrzegorz Jaszczyk ret = mvebu_cp110_comphy_digital_reset(x1, x2, x3, x4); 115*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 116*a2847172SGrzegorz Jaszczyk 117*a2847172SGrzegorz Jaszczyk /* Miscellaneous FID's' */ 118*a2847172SGrzegorz Jaszczyk case MV_SIP_DRAM_SIZE: 119*a2847172SGrzegorz Jaszczyk ret = mvebu_get_dram_size(MVEBU_REGS_BASE); 120*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, ret); 121*a2847172SGrzegorz Jaszczyk case MV_SIP_LLC_ENABLE: 122*a2847172SGrzegorz Jaszczyk for (i = 0; i < ap_get_count(); i++) 123*a2847172SGrzegorz Jaszczyk llc_runtime_enable(i); 124*a2847172SGrzegorz Jaszczyk 125*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 126*a2847172SGrzegorz Jaszczyk #ifdef MVEBU_PMU_IRQ_WA 127*a2847172SGrzegorz Jaszczyk case MV_SIP_PMU_IRQ_ENABLE: 128*a2847172SGrzegorz Jaszczyk mvebu_pmu_interrupt_enable(); 129*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 130*a2847172SGrzegorz Jaszczyk case MV_SIP_PMU_IRQ_DISABLE: 131*a2847172SGrzegorz Jaszczyk mvebu_pmu_interrupt_disable(); 132*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, 0); 133*a2847172SGrzegorz Jaszczyk #endif 134*a2847172SGrzegorz Jaszczyk 135*a2847172SGrzegorz Jaszczyk default: 136*a2847172SGrzegorz Jaszczyk ERROR("%s: unhandled SMC (0x%x)\n", __func__, smc_fid); 137*a2847172SGrzegorz Jaszczyk SMC_RET1(handle, SMC_UNK); 138*a2847172SGrzegorz Jaszczyk } 139*a2847172SGrzegorz Jaszczyk } 140*a2847172SGrzegorz Jaszczyk 141*a2847172SGrzegorz Jaszczyk /* Define a runtime service descriptor for fast SMC calls */ 142*a2847172SGrzegorz Jaszczyk DECLARE_RT_SVC( 143*a2847172SGrzegorz Jaszczyk marvell_sip_svc, 144*a2847172SGrzegorz Jaszczyk OEN_SIP_START, 145*a2847172SGrzegorz Jaszczyk OEN_SIP_END, 146*a2847172SGrzegorz Jaszczyk SMC_TYPE_FAST, 147*a2847172SGrzegorz Jaszczyk NULL, 148*a2847172SGrzegorz Jaszczyk mrvl_sip_smc_handler 149*a2847172SGrzegorz Jaszczyk ); 150