target/arm: set HPFAR_EL2.NS on secure stage 2 faults

Backport 9861248f637ecf11113b04b0b5c7b13c9aa06f09
This commit is contained in:
Rémi Denis-Courmont 2021-03-04 14:54:30 -05:00 committed by Lioncash
parent b49531cfef
commit ce8872709f
4 changed files with 13 additions and 0 deletions

View file

@ -1359,6 +1359,8 @@ static inline void xpsr_write(CPUARMState *env, uint32_t val, uint32_t mask)
#define HCR_TWEDEN (1ULL << 59) #define HCR_TWEDEN (1ULL << 59)
#define HCR_TWEDEL MAKE_64BIT_MASK(60, 4) #define HCR_TWEDEL MAKE_64BIT_MASK(60, 4)
#define HPFAR_NS (1ULL << 63)
#define SCR_NS (1U << 0) #define SCR_NS (1U << 0)
#define SCR_IRQ (1U << 1) #define SCR_IRQ (1U << 1)
#define SCR_FIQ (1U << 2) #define SCR_FIQ (1U << 2)

View file

@ -3137,6 +3137,9 @@ static uint64_t do_ats_write(CPUARMState *env, uint64_t value,
target_el = 3; target_el = 3;
} else { } else {
env->cp15.hpfar_el2 = extract64(fi.s2addr, 12, 47) << 4; env->cp15.hpfar_el2 = extract64(fi.s2addr, 12, 47) << 4;
if (arm_is_secure_below_el3(env) && fi.s1ns) {
env->cp15.hpfar_el2 |= HPFAR_NS;
}
target_el = 2; target_el = 2;
} }
take_exc = true; take_exc = true;
@ -10151,6 +10154,7 @@ static hwaddr S1_ptw_translate(CPUARMState *env, ARMMMUIdx mmu_idx,
fi->s2addr = addr; fi->s2addr = addr;
fi->stage2 = true; fi->stage2 = true;
fi->s1ptw = true; fi->s1ptw = true;
fi->s1ns = !*is_secure;
return ~0; return ~0;
} }
if ((arm_hcr_el2_eff(env) & HCR_PTW) && if ((arm_hcr_el2_eff(env) & HCR_PTW) &&
@ -10163,6 +10167,7 @@ static hwaddr S1_ptw_translate(CPUARMState *env, ARMMMUIdx mmu_idx,
fi->s2addr = addr; fi->s2addr = addr;
fi->stage2 = true; fi->stage2 = true;
fi->s1ptw = true; fi->s1ptw = true;
fi->s1ns = !*is_secure;
return ~0; return ~0;
} }
@ -11096,6 +11101,7 @@ do_fault:
/* Tag the error as S2 for failed S1 PTW at S2 or ordinary S2. */ /* Tag the error as S2 for failed S1 PTW at S2 or ordinary S2. */
fi->stage2 = fi->s1ptw || (mmu_idx == ARMMMUIdx_Stage2 || fi->stage2 = fi->s1ptw || (mmu_idx == ARMMMUIdx_Stage2 ||
mmu_idx == ARMMMUIdx_Stage2_S); mmu_idx == ARMMMUIdx_Stage2_S);
fi->s1ns = mmu_idx == ARMMMUIdx_Stage2;
return true; return true;
} }

View file

@ -595,6 +595,7 @@ typedef enum ARMFaultType {
* @s2addr: Address that caused a fault at stage 2 * @s2addr: Address that caused a fault at stage 2
* @stage2: True if we faulted at stage 2 * @stage2: True if we faulted at stage 2
* @s1ptw: True if we faulted at stage 2 while doing a stage 1 page-table walk * @s1ptw: True if we faulted at stage 2 while doing a stage 1 page-table walk
* @s1ns: True if we faulted on a non-secure IPA while in secure state
* @ea: True if we should set the EA (external abort type) bit in syndrome * @ea: True if we should set the EA (external abort type) bit in syndrome
*/ */
typedef struct ARMMMUFaultInfo ARMMMUFaultInfo; typedef struct ARMMMUFaultInfo ARMMMUFaultInfo;
@ -605,6 +606,7 @@ struct ARMMMUFaultInfo {
int domain; int domain;
bool stage2; bool stage2;
bool s1ptw; bool s1ptw;
bool s1ns;
bool ea; bool ea;
}; };

View file

@ -63,6 +63,9 @@ static void QEMU_NORETURN arm_deliver_fault(ARMCPU *cpu, vaddr addr,
if (fi->stage2) { if (fi->stage2) {
target_el = 2; target_el = 2;
env->cp15.hpfar_el2 = extract64(fi->s2addr, 12, 47) << 4; env->cp15.hpfar_el2 = extract64(fi->s2addr, 12, 47) << 4;
if (arm_is_secure_below_el3(env) && fi->s1ns) {
env->cp15.hpfar_el2 |= HPFAR_NS;
}
} }
same_el = (arm_current_el(env) == target_el); same_el = (arm_current_el(env) == target_el);