arm64: limit PA size to supported range
We currently copy the physical address size from ID_AA64MMFR0_EL1.PARange directly into TCR.(I)PS. This will not work for 4k and 16k granule kernels on systems that support 52-bit physical addresses, since 52-bit addresses are only permitted with the 64k granule. To fix this, fall back to 48 bits when configuring the PA size when the kernel does not support 52-bit PAs. When it does, fall back to 52, to avoid similar problems in the future if the PA size is ever increased above 52. Tested-by: Suzuki K Poulose <suzuki.poulose@arm.com> Reviewed-by: Suzuki K Poulose <suzuki.poulose@arm.com> Reviewed-by: Marc Zyngier <marc.zyngier@arm.com> Tested-by: Bob Picco <bob.picco@oracle.com> Reviewed-by: Bob Picco <bob.picco@oracle.com> Signed-off-by: Kristina Martsenko <kristina.martsenko@arm.com> [catalin.marinas@arm.com: tcr_set_pa_size macro renamed to tcr_compute_pa_size] [catalin.marinas@arm.com: comments added to tcr_compute_pa_size] [catalin.marinas@arm.com: definitions added for TCR_*PS_SHIFT] Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
This commit is contained in:
Родитель
982aa7c5f0
Коммит
787fd1d019
|
@ -350,6 +350,24 @@ alternative_endif
|
||||||
#endif
|
#endif
|
||||||
.endm
|
.endm
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tcr_compute_pa_size - set TCR.(I)PS to the highest supported
|
||||||
|
* ID_AA64MMFR0_EL1.PARange value
|
||||||
|
*
|
||||||
|
* tcr: register with the TCR_ELx value to be updated
|
||||||
|
* pos: PARange bitfield position
|
||||||
|
* tmp{0,1}: temporary registers
|
||||||
|
*/
|
||||||
|
.macro tcr_compute_pa_size, tcr, pos, tmp0, tmp1
|
||||||
|
mrs \tmp0, ID_AA64MMFR0_EL1
|
||||||
|
// Narrow PARange to fit the PS field in TCR_ELx
|
||||||
|
ubfx \tmp0, \tmp0, #ID_AA64MMFR0_PARANGE_SHIFT, #3
|
||||||
|
mov \tmp1, #ID_AA64MMFR0_PARANGE_MAX
|
||||||
|
cmp \tmp0, \tmp1
|
||||||
|
csel \tmp0, \tmp1, \tmp0, hi
|
||||||
|
bfi \tcr, \tmp0, \pos, #3
|
||||||
|
.endm
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Macro to perform a data cache maintenance for the interval
|
* Macro to perform a data cache maintenance for the interval
|
||||||
* [kaddr, kaddr + size)
|
* [kaddr, kaddr + size)
|
||||||
|
|
|
@ -272,6 +272,8 @@
|
||||||
#define TCR_TG1_4K (UL(2) << TCR_TG1_SHIFT)
|
#define TCR_TG1_4K (UL(2) << TCR_TG1_SHIFT)
|
||||||
#define TCR_TG1_64K (UL(3) << TCR_TG1_SHIFT)
|
#define TCR_TG1_64K (UL(3) << TCR_TG1_SHIFT)
|
||||||
|
|
||||||
|
#define TCR_IPS_SHIFT 32
|
||||||
|
#define TCR_IPS_MASK (UL(7) << TCR_IPS_SHIFT)
|
||||||
#define TCR_ASID16 (UL(1) << 36)
|
#define TCR_ASID16 (UL(1) << 36)
|
||||||
#define TCR_TBI0 (UL(1) << 37)
|
#define TCR_TBI0 (UL(1) << 37)
|
||||||
#define TCR_HA (UL(1) << 39)
|
#define TCR_HA (UL(1) << 39)
|
||||||
|
|
|
@ -471,6 +471,14 @@
|
||||||
#define ID_AA64MMFR0_TGRAN64_SUPPORTED 0x0
|
#define ID_AA64MMFR0_TGRAN64_SUPPORTED 0x0
|
||||||
#define ID_AA64MMFR0_TGRAN16_NI 0x0
|
#define ID_AA64MMFR0_TGRAN16_NI 0x0
|
||||||
#define ID_AA64MMFR0_TGRAN16_SUPPORTED 0x1
|
#define ID_AA64MMFR0_TGRAN16_SUPPORTED 0x1
|
||||||
|
#define ID_AA64MMFR0_PARANGE_48 0x5
|
||||||
|
#define ID_AA64MMFR0_PARANGE_52 0x6
|
||||||
|
|
||||||
|
#ifdef CONFIG_ARM64_PA_BITS_52
|
||||||
|
#define ID_AA64MMFR0_PARANGE_MAX ID_AA64MMFR0_PARANGE_52
|
||||||
|
#else
|
||||||
|
#define ID_AA64MMFR0_PARANGE_MAX ID_AA64MMFR0_PARANGE_48
|
||||||
|
#endif
|
||||||
|
|
||||||
/* id_aa64mmfr1 */
|
/* id_aa64mmfr1 */
|
||||||
#define ID_AA64MMFR1_PAN_SHIFT 20
|
#define ID_AA64MMFR1_PAN_SHIFT 20
|
||||||
|
|
|
@ -90,11 +90,9 @@ __do_hyp_init:
|
||||||
bfi x4, x5, TCR_T0SZ_OFFSET, TCR_TxSZ_WIDTH
|
bfi x4, x5, TCR_T0SZ_OFFSET, TCR_TxSZ_WIDTH
|
||||||
#endif
|
#endif
|
||||||
/*
|
/*
|
||||||
* Read the PARange bits from ID_AA64MMFR0_EL1 and set the PS bits in
|
* Set the PS bits in TCR_EL2.
|
||||||
* TCR_EL2.
|
|
||||||
*/
|
*/
|
||||||
mrs x5, ID_AA64MMFR0_EL1
|
tcr_compute_pa_size x4, #TCR_EL2_PS_SHIFT, x5, x6
|
||||||
bfi x4, x5, #16, #3
|
|
||||||
|
|
||||||
msr tcr_el2, x4
|
msr tcr_el2, x4
|
||||||
|
|
||||||
|
|
|
@ -32,6 +32,8 @@ u32 __hyp_text __init_stage2_translation(void)
|
||||||
* PS is only 3. Fortunately, bit 19 is RES0 in VTCR_EL2...
|
* PS is only 3. Fortunately, bit 19 is RES0 in VTCR_EL2...
|
||||||
*/
|
*/
|
||||||
parange = read_sysreg(id_aa64mmfr0_el1) & 7;
|
parange = read_sysreg(id_aa64mmfr0_el1) & 7;
|
||||||
|
if (parange > ID_AA64MMFR0_PARANGE_MAX)
|
||||||
|
parange = ID_AA64MMFR0_PARANGE_MAX;
|
||||||
val |= parange << 16;
|
val |= parange << 16;
|
||||||
|
|
||||||
/* Compute the actual PARange... */
|
/* Compute the actual PARange... */
|
||||||
|
|
|
@ -228,11 +228,9 @@ ENTRY(__cpu_setup)
|
||||||
tcr_set_idmap_t0sz x10, x9
|
tcr_set_idmap_t0sz x10, x9
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read the PARange bits from ID_AA64MMFR0_EL1 and set the IPS bits in
|
* Set the IPS bits in TCR_EL1.
|
||||||
* TCR_EL1.
|
|
||||||
*/
|
*/
|
||||||
mrs x9, ID_AA64MMFR0_EL1
|
tcr_compute_pa_size x10, #TCR_IPS_SHIFT, x5, x6
|
||||||
bfi x10, x9, #32, #3
|
|
||||||
#ifdef CONFIG_ARM64_HW_AFDBM
|
#ifdef CONFIG_ARM64_HW_AFDBM
|
||||||
/*
|
/*
|
||||||
* Hardware update of the Access and Dirty bits.
|
* Hardware update of the Access and Dirty bits.
|
||||||
|
|
Загрузка…
Ссылка в новой задаче