mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	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:
		
							parent
							
								
									982aa7c5f0
								
							
						
					
					
						commit
						787fd1d019
					
				| @ -350,6 +350,24 @@ alternative_endif | ||||
| #endif | ||||
| 	.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 | ||||
|  * [kaddr, kaddr + size) | ||||
|  | ||||
| @ -272,6 +272,8 @@ | ||||
| #define TCR_TG1_4K		(UL(2) << 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_TBI0		(UL(1) << 37) | ||||
| #define TCR_HA			(UL(1) << 39) | ||||
|  | ||||
| @ -471,6 +471,14 @@ | ||||
| #define ID_AA64MMFR0_TGRAN64_SUPPORTED	0x0 | ||||
| #define ID_AA64MMFR0_TGRAN16_NI		0x0 | ||||
| #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 */ | ||||
| #define ID_AA64MMFR1_PAN_SHIFT		20 | ||||
|  | ||||
| @ -90,11 +90,9 @@ __do_hyp_init: | ||||
| 	bfi	x4, x5, TCR_T0SZ_OFFSET, TCR_TxSZ_WIDTH | ||||
| #endif | ||||
| 	/* | ||||
| 	 * Read the PARange bits from ID_AA64MMFR0_EL1 and set the PS bits in | ||||
| 	 * TCR_EL2. | ||||
| 	 * Set the PS bits in TCR_EL2. | ||||
| 	 */ | ||||
| 	mrs	x5, ID_AA64MMFR0_EL1 | ||||
| 	bfi	x4, x5, #16, #3 | ||||
| 	tcr_compute_pa_size x4, #TCR_EL2_PS_SHIFT, x5, x6 | ||||
| 
 | ||||
| 	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... | ||||
| 	 */ | ||||
| 	parange = read_sysreg(id_aa64mmfr0_el1) & 7; | ||||
| 	if (parange > ID_AA64MMFR0_PARANGE_MAX) | ||||
| 		parange = ID_AA64MMFR0_PARANGE_MAX; | ||||
| 	val |= parange << 16; | ||||
| 
 | ||||
| 	/* Compute the actual PARange... */ | ||||
|  | ||||
| @ -228,11 +228,9 @@ ENTRY(__cpu_setup) | ||||
| 	tcr_set_idmap_t0sz	x10, x9 | ||||
| 
 | ||||
| 	/* | ||||
| 	 * Read the PARange bits from ID_AA64MMFR0_EL1 and set the IPS bits in | ||||
| 	 * TCR_EL1. | ||||
| 	 * Set the IPS bits in TCR_EL1. | ||||
| 	 */ | ||||
| 	mrs	x9, ID_AA64MMFR0_EL1 | ||||
| 	bfi	x10, x9, #32, #3 | ||||
| 	tcr_compute_pa_size x10, #TCR_IPS_SHIFT, x5, x6 | ||||
| #ifdef CONFIG_ARM64_HW_AFDBM | ||||
| 	/* | ||||
| 	 * Hardware update of the Access and Dirty bits. | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Kristina Martsenko
						Kristina Martsenko