target-arm: Fix translation level on early translation faults

Qemu reports translation fault on 1st level instead of 0th level in case of
AArch64 address translation if the translation table walk is disabled or
the address is in the gap between the two regions.

Backports commit 1b4093ea6678ff79d3006db3d3abbf6990b4a59b from qemu
This commit is contained in:
Sergey Sorokin 2018-02-21 21:53:04 -05:00 committed by Lioncash
parent 8309945dcc
commit da6a9f331b
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7

View file

@ -6495,7 +6495,7 @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
CPUState *cs = CPU(cpu); CPUState *cs = CPU(cpu);
/* Read an LPAE long-descriptor translation table. */ /* Read an LPAE long-descriptor translation table. */
MMUFaultType fault_type = translation_fault; MMUFaultType fault_type = translation_fault;
uint32_t level = 1; uint32_t level;
uint32_t epd = 0; uint32_t epd = 0;
int32_t t0sz, t1sz; int32_t t0sz, t1sz;
uint32_t tg; uint32_t tg;
@ -6506,7 +6506,7 @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
target_ulong page_size; target_ulong page_size;
uint32_t attrs; uint32_t attrs;
int32_t stride = 9; int32_t stride = 9;
int32_t va_size = 32; int32_t va_size;
int inputsize; int inputsize;
int32_t tbi = 0; int32_t tbi = 0;
TCR *tcr = regime_tcr(env, mmu_idx); TCR *tcr = regime_tcr(env, mmu_idx);
@ -6522,6 +6522,7 @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
* support for those page table walks. * support for those page table walks.
*/ */
if (arm_el_is_aa64(env, el)) { if (arm_el_is_aa64(env, el)) {
level = 0;
va_size = 64; va_size = 64;
if (el > 1) { if (el > 1) {
if (mmu_idx != ARMMMUIdx_S2NS) { if (mmu_idx != ARMMMUIdx_S2NS) {
@ -6543,6 +6544,8 @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
ttbr1_valid = false; ttbr1_valid = false;
} }
} else { } else {
level = 1;
va_size = 32;
/* There is no TTBR1 for EL2 */ /* There is no TTBR1 for EL2 */
if (el == 2) { if (el == 2) {
ttbr1_valid = false; ttbr1_valid = false;
@ -6665,27 +6668,26 @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
/* For stage 2 translations the starting level is specified by the /* For stage 2 translations the starting level is specified by the
* VTCR_EL2.SL0 field (whose interpretation depends on the page size) * VTCR_EL2.SL0 field (whose interpretation depends on the page size)
*/ */
int startlevel = extract32(tcr->raw_tcr, 6, 2); uint32_t sl0 = extract32(tcr->raw_tcr, 6, 2);
uint32_t startlevel;
bool ok; bool ok;
if (va_size == 32 || stride == 9) { if (va_size == 32 || stride == 9) {
/* AArch32 or 4KB pages */ /* AArch32 or 4KB pages */
level = 2 - startlevel; startlevel = 2 - sl0;
} else { } else {
/* 16KB or 64KB pages */ /* 16KB or 64KB pages */
level = 3 - startlevel; startlevel = 3 - sl0;
} }
/* Check that the starting level is valid. */ /* Check that the starting level is valid. */
ok = check_s2_mmu_setup(cpu, va_size == 64, level, inputsize, stride); ok = check_s2_mmu_setup(cpu, va_size == 64, startlevel,
inputsize, stride);
if (!ok) { if (!ok) {
/* AArch64 reports these as level 0 faults.
* AArch32 reports these as level 1 faults.
*/
level = va_size == 64 ? 0 : 1;
fault_type = translation_fault; fault_type = translation_fault;
goto do_fault; goto do_fault;
} }
level = startlevel;
} }
/* Clear the vaddr bits which aren't part of the within-region address, /* Clear the vaddr bits which aren't part of the within-region address,