]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
arm64/mm: Override PARange for !LPA2 and use it consistently
authorArd Biesheuvel <ardb@kernel.org>
Thu, 12 Dec 2024 08:18:44 +0000 (09:18 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 17 Feb 2025 09:05:05 +0000 (10:05 +0100)
commit 62cffa496aac0c2c4eeca00d080058affd7a0172 upstream.

When FEAT_LPA{,2} are not implemented, the ID_AA64MMFR0_EL1.PARange and
TCR.IPS values corresponding with 52-bit physical addressing are
reserved.

Setting the TCR.IPS field to 0b110 (52-bit physical addressing) has side
effects, such as how the TTBRn_ELx.BADDR fields are interpreted, and so
it is important that disabling FEAT_LPA2 (by overriding the
ID_AA64MMFR0.TGran fields) also presents a PARange field consistent with
that.

So limit the field to 48 bits unless LPA2 is enabled, and update
existing references to use the override consistently.

Fixes: 352b0395b505 ("arm64: Enable 52-bit virtual addressing for 4k and 16k granule configs")
Cc: stable@vger.kernel.org
Signed-off-by: Ard Biesheuvel <ardb@kernel.org>
Acked-by: Marc Zyngier <maz@kernel.org>
Link: https://lore.kernel.org/r/20241212081841.2168124-10-ardb+git@google.com
Signed-off-by: Will Deacon <will@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
arch/arm64/include/asm/assembler.h
arch/arm64/kernel/cpufeature.c
arch/arm64/kernel/pi/idreg-override.c
arch/arm64/kernel/pi/map_kernel.c
arch/arm64/mm/init.c

index bc0b0d75acef7b46df814e0c49a1ac27ab7a75cc..c1f45fd6b3e9a93c64aa2ffb3dfd6cb2281385bc 100644 (file)
@@ -350,6 +350,11 @@ alternative_cb_end
        // Narrow PARange to fit the PS field in TCR_ELx
        ubfx    \tmp0, \tmp0, #ID_AA64MMFR0_EL1_PARANGE_SHIFT, #3
        mov     \tmp1, #ID_AA64MMFR0_EL1_PARANGE_MAX
+#ifdef CONFIG_ARM64_LPA2
+alternative_if_not ARM64_HAS_VA52
+       mov     \tmp1, #ID_AA64MMFR0_EL1_PARANGE_48
+alternative_else_nop_endif
+#endif
        cmp     \tmp0, \tmp1
        csel    \tmp0, \tmp1, \tmp0, hi
        bfi     \tcr, \tmp0, \pos, #3
index 1f8be4aafe69e7f099880d0801104b84672768e9..29e1e5eb7a0e060b7853857c1fc632c672fce6fa 100644 (file)
@@ -3390,7 +3390,7 @@ static void verify_hyp_capabilities(void)
                return;
 
        safe_mmfr1 = read_sanitised_ftr_reg(SYS_ID_AA64MMFR1_EL1);
-       mmfr0 = read_cpuid(ID_AA64MMFR0_EL1);
+       mmfr0 = read_sanitised_ftr_reg(SYS_ID_AA64MMFR0_EL1);
        mmfr1 = read_cpuid(ID_AA64MMFR1_EL1);
 
        /* Verify VMID bits */
index 29d4b6244a6f630515a01ba13cd9c64b26187f07..5c03f5e0d352dace5f61d3db65db0312bb80ba7c 100644 (file)
@@ -74,6 +74,15 @@ static bool __init mmfr2_varange_filter(u64 val)
                id_aa64mmfr0_override.val |=
                        (ID_AA64MMFR0_EL1_TGRAN_LPA2 - 1) << ID_AA64MMFR0_EL1_TGRAN_SHIFT;
                id_aa64mmfr0_override.mask |= 0xfU << ID_AA64MMFR0_EL1_TGRAN_SHIFT;
+
+               /*
+                * Override PARange to 48 bits - the override will just be
+                * ignored if the actual PARange is smaller, but this is
+                * unlikely to be the case for LPA2 capable silicon.
+                */
+               id_aa64mmfr0_override.val |=
+                       ID_AA64MMFR0_EL1_PARANGE_48 << ID_AA64MMFR0_EL1_PARANGE_SHIFT;
+               id_aa64mmfr0_override.mask |= 0xfU << ID_AA64MMFR0_EL1_PARANGE_SHIFT;
        }
 #endif
        return true;
index f374a3e5a5fe10b0a0491edaf1a31a7f62602606..e57b043f324b51b7db873daa61b079bfb07244e0 100644 (file)
@@ -136,6 +136,12 @@ static void noinline __section(".idmap.text") set_ttbr0_for_lpa2(u64 ttbr)
 {
        u64 sctlr = read_sysreg(sctlr_el1);
        u64 tcr = read_sysreg(tcr_el1) | TCR_DS;
+       u64 mmfr0 = read_sysreg(id_aa64mmfr0_el1);
+       u64 parange = cpuid_feature_extract_unsigned_field(mmfr0,
+                                                          ID_AA64MMFR0_EL1_PARANGE_SHIFT);
+
+       tcr &= ~TCR_IPS_MASK;
+       tcr |= parange << TCR_IPS_SHIFT;
 
        asm("   msr     sctlr_el1, %0           ;"
            "   isb                             ;"
index 93ba66de160ce4e59d77ef3b0ee2213ae0be0205..ea71ef2e343c2cb1943a0a61b011d21bf48f52ba 100644 (file)
@@ -278,7 +278,12 @@ void __init arm64_memblock_init(void)
 
        if (IS_ENABLED(CONFIG_RANDOMIZE_BASE)) {
                extern u16 memstart_offset_seed;
-               u64 mmfr0 = read_cpuid(ID_AA64MMFR0_EL1);
+
+               /*
+                * Use the sanitised version of id_aa64mmfr0_el1 so that linear
+                * map randomization can be enabled by shrinking the IPA space.
+                */
+               u64 mmfr0 = read_sanitised_ftr_reg(SYS_ID_AA64MMFR0_EL1);
                int parange = cpuid_feature_extract_unsigned_field(
                                        mmfr0, ID_AA64MMFR0_EL1_PARANGE_SHIFT);
                s64 range = linear_region_size -