]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
arm64: KVM: Add workaround for Cortex-A57 erratum 834220
authorMarc Zyngier <marc.zyngier@arm.com>
Mon, 16 Nov 2015 10:28:18 +0000 (10:28 +0000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sun, 31 Jan 2016 19:25:55 +0000 (11:25 -0800)
commit 498cd5c32be6e32bc0f8efcad48ab094bb2bfdf3 upstream.

Cortex-A57 parts up to r1p2 can misreport Stage 2 translation faults
when a Stage 1 permission fault or device alignment fault should
have been reported.

This patch implements the workaround (which is to validate that the
Stage-1 translation actually succeeds) by using code patching.

Reviewed-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
Signed-off-by: Christoffer Dall <christoffer.dall@linaro.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
arch/arm64/Kconfig
arch/arm64/include/asm/cpufeature.h
arch/arm64/kernel/cpu_errata.c
arch/arm64/kvm/hyp.S

index 07d1811aa03fcd1ecd5ee7c260688a59f7ab97e4..a92266e634cdf7b355b49916e2a834a58dfd6537 100644 (file)
@@ -311,6 +311,27 @@ config ARM64_ERRATUM_832075
 
          If unsure, say Y.
 
+config ARM64_ERRATUM_834220
+       bool "Cortex-A57: 834220: Stage 2 translation fault might be incorrectly reported in presence of a Stage 1 fault"
+       depends on KVM
+       default y
+       help
+         This option adds an alternative code sequence to work around ARM
+         erratum 834220 on Cortex-A57 parts up to r1p2.
+
+         Affected Cortex-A57 parts might report a Stage 2 translation
+         fault as a the result of a Stage 1 fault for a load crossing
+         a page boundary when there is a Stage 1 permission or device
+         memory alignment fault and a Stage 2 translation fault
+
+         The workaround is to verify that the Stage-1 translation
+         doesn't generate a fault before handling the Stage-2 fault.
+         Please note that this does not necessarily enable the workaround,
+         as it depends on the alternative framework, which will only patch
+         the kernel if an affected CPU is detected.
+
+         If unsure, say Y.
+
 config ARM64_ERRATUM_845719
        bool "Cortex-A53: 845719: a load might read incorrect data"
        depends on COMPAT
index 171570702bb801431f141f46238eb0d9e1895fe5..a1a5981526fe712f1c80b5832ed78336f21d4ceb 100644 (file)
@@ -27,8 +27,9 @@
 #define ARM64_HAS_SYSREG_GIC_CPUIF             3
 #define ARM64_HAS_PAN                          4
 #define ARM64_HAS_LSE_ATOMICS                  5
+#define ARM64_WORKAROUND_834220                        6
 
-#define ARM64_NCAPS                            6
+#define ARM64_NCAPS                            7
 
 #ifndef __ASSEMBLY__
 
index 6ffd914385609d0aab875813e4e9e8b690976421..dc0df822def344af91ee1257c9cc659b23eeaddc 100644 (file)
@@ -74,6 +74,15 @@ const struct arm64_cpu_capabilities arm64_errata[] = {
                           (1 << MIDR_VARIANT_SHIFT) | 2),
        },
 #endif
+#ifdef CONFIG_ARM64_ERRATUM_834220
+       {
+       /* Cortex-A57 r0p0 - r1p2 */
+               .desc = "ARM erratum 834220",
+               .capability = ARM64_WORKAROUND_834220,
+               MIDR_RANGE(MIDR_CORTEX_A57, 0x00,
+                          (1 << MIDR_VARIANT_SHIFT) | 2),
+       },
+#endif
 #ifdef CONFIG_ARM64_ERRATUM_845719
        {
        /* Cortex-A53 r0p[01234] */
index e5836138ec42a58841e7003bbb14f2b4a2126297..3e840649b133e0acb656d4412aac947ffd896f8b 100644 (file)
@@ -1007,9 +1007,15 @@ el1_trap:
        b.ne    1f              // Not an abort we care about
 
        /* This is an abort. Check for permission fault */
+alternative_if_not ARM64_WORKAROUND_834220
        and     x2, x1, #ESR_ELx_FSC_TYPE
        cmp     x2, #FSC_PERM
        b.ne    1f              // Not a permission fault
+alternative_else
+       nop                     // Force a Stage-1 translation to occur
+       nop                     // and return to the guest if it failed
+       nop
+alternative_endif
 
        /*
         * Check for Stage-1 page table walk, which is guaranteed