1b5c850d4SMarcin Wojtas /* 2b5c850d4SMarcin Wojtas * Copyright (C) 2019 Repk repk@triplefau.lt 3b5c850d4SMarcin Wojtas * 4b5c850d4SMarcin Wojtas * SPDX-License-Identifier: BSD-3-Clause 5b5c850d4SMarcin Wojtas * https://spdx.org/licenses 6b5c850d4SMarcin Wojtas */ 7*4ce3e99aSScott Branden 8*4ce3e99aSScott Branden #include <inttypes.h> 9*4ce3e99aSScott Branden #include <stdint.h> 10*4ce3e99aSScott Branden 11b5c850d4SMarcin Wojtas #include <common/bl_common.h> 12b5c850d4SMarcin Wojtas #include <common/debug.h> 13b5c850d4SMarcin Wojtas #include <arch_helpers.h> 1430e8fa7eSPali Rohár #include <plat/common/platform.h> 15068fe919SPali Rohár #include <bl31/ea_handle.h> 16b5c850d4SMarcin Wojtas 17068fe919SPali Rohár #define A53_SERR_INT_AXI_SLVERR_ON_EXTERNAL_ACCESS 0xbf000002 18068fe919SPali Rohár 19068fe919SPali Rohár #if !ENABLE_BACKTRACE 20068fe919SPali Rohár static const char *get_el_str(unsigned int el) 21068fe919SPali Rohár { 22068fe919SPali Rohár if (el == MODE_EL3) { 23068fe919SPali Rohár return "EL3"; 24068fe919SPali Rohár } else if (el == MODE_EL2) { 25068fe919SPali Rohár return "EL2"; 26068fe919SPali Rohár } 27068fe919SPali Rohár return "S-EL1"; 28068fe919SPali Rohár } 29068fe919SPali Rohár #endif /* !ENABLE_BACKTRACE */ 30b5c850d4SMarcin Wojtas 313017e932SPali Rohár /* 323017e932SPali Rohár * This source file with custom plat_ea_handler function is compiled only when 333017e932SPali Rohár * building TF-A with compile option HANDLE_EA_EL3_FIRST=1 343017e932SPali Rohár */ 35b5c850d4SMarcin Wojtas void plat_ea_handler(unsigned int ea_reason, uint64_t syndrome, void *cookie, 36b5c850d4SMarcin Wojtas void *handle, uint64_t flags) 37b5c850d4SMarcin Wojtas { 38068fe919SPali Rohár unsigned int level = (unsigned int)GET_EL(read_spsr_el3()); 39068fe919SPali Rohár 40068fe919SPali Rohár /* 41068fe919SPali Rohár * Asynchronous External Abort with syndrome 0xbf000002 on Cortex A53 42068fe919SPali Rohár * core means SError interrupt caused by AXI SLVERR on external access. 43068fe919SPali Rohár * 44068fe919SPali Rohár * In most cases this indicates a bug in U-Boot or Linux kernel driver 45068fe919SPali Rohár * pci-aardvark.c which implements access to A3700 PCIe config space. 46068fe919SPali Rohár * Driver does not wait for PCIe PIO transfer completion and try to 47068fe919SPali Rohár * start a new PCIe PIO transfer while previous has not finished yet. 48068fe919SPali Rohár * A3700 PCIe controller in this case sends SLVERR via AXI which results 49068fe919SPali Rohár * in a fatal Asynchronous SError interrupt on Cortex A53 CPU. 50068fe919SPali Rohár * 51068fe919SPali Rohár * Following patches fix that bug in U-Boot and Linux kernel drivers: 52068fe919SPali Rohár * https://source.denx.de/u-boot/u-boot/-/commit/eccbd4ad8e4e182638eafbfb87ac139c04f24a01 53068fe919SPali Rohár * https://git.kernel.org/stable/c/f18139966d072dab8e4398c95ce955a9742e04f7 54068fe919SPali Rohár * 55068fe919SPali Rohár * As a hacky workaround for unpatched U-Boot and Linux kernel drivers 56068fe919SPali Rohár * ignore all asynchronous aborts with that syndrome value received on 57068fe919SPali Rohár * CPU from level lower than EL3. 58068fe919SPali Rohár * 59068fe919SPali Rohár * Because these aborts are delivered on CPU asynchronously, they are 60068fe919SPali Rohár * imprecise and we cannot check the real reason of abort and neither 61068fe919SPali Rohár * who and why sent this abort. We expect that on A3700 it is always 62068fe919SPali Rohár * PCIe controller. 63068fe919SPali Rohár * 64068fe919SPali Rohár * Hence ignoring all aborts with this syndrome value is just a giant 65068fe919SPali Rohár * hack that we need only because of bugs in old U-Boot and Linux kernel 66068fe919SPali Rohár * versions and because it was decided that TF-A would implement this 67068fe919SPali Rohár * hack for U-Boot and Linux kernel it in this way. New patched U-Boot 68068fe919SPali Rohár * and kernel versions do not need it anymore. 69068fe919SPali Rohár * 70068fe919SPali Rohár * Links to discussion about this workaround: 71068fe919SPali Rohár * https://lore.kernel.org/linux-pci/20190316161243.29517-1-repk@triplefau.lt/ 72068fe919SPali Rohár * https://lore.kernel.org/linux-pci/971be151d24312cc533989a64bd454b4@www.loen.fr/ 73068fe919SPali Rohár * https://review.trustedfirmware.org/c/TF-A/trusted-firmware-a/+/1541 74068fe919SPali Rohár */ 75068fe919SPali Rohár if (level < MODE_EL3 && ea_reason == ERROR_EA_ASYNC && 76068fe919SPali Rohár syndrome == A53_SERR_INT_AXI_SLVERR_ON_EXTERNAL_ACCESS) { 77068fe919SPali Rohár ERROR_NL(); 78068fe919SPali Rohár ERROR("Ignoring Asynchronous External Abort with" 79*4ce3e99aSScott Branden " syndrome 0x%" PRIx64 " received on 0x%lx from %s\n", 80068fe919SPali Rohár syndrome, read_mpidr_el1(), get_el_str(level)); 81068fe919SPali Rohár ERROR("SError interrupt: AXI SLVERR on external access\n"); 82068fe919SPali Rohár ERROR("This indicates a bug in pci-aardvark.c driver\n"); 83068fe919SPali Rohár ERROR("Please update U-Boot/Linux to the latest version\n"); 84068fe919SPali Rohár ERROR_NL(); 85068fe919SPali Rohár console_flush(); 8630e8fa7eSPali Rohár return; 87068fe919SPali Rohár } 8830e8fa7eSPali Rohár 8930e8fa7eSPali Rohár plat_default_ea_handler(ea_reason, syndrome, cookie, handle, flags); 90b5c850d4SMarcin Wojtas } 91