target/arm: Implement SCR_EL2.EEL2

This adds handling for the SCR_EL3.EEL2 bit.

Backports 926c1b97895879b78ca14bca2831c08740ed1c38
This commit is contained in:
Rémi Denis-Courmont 2021-03-04 15:03:05 -05:00 committed by Lioncash
parent 9690ed8236
commit e6d32dc2e0
4 changed files with 37 additions and 8 deletions

View file

@ -473,7 +473,7 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,
* masked from Secure state. The HCR and SCR settings * masked from Secure state. The HCR and SCR settings
* don't affect the masking logic, only the interrupt routing. * don't affect the masking logic, only the interrupt routing.
*/ */
if (target_el == 3 || !secure) { if (target_el == 3 || !secure || (env->cp15.scr_el3 & SCR_EEL2)) {
unmasked = true; unmasked = true;
} }
} else { } else {

View file

@ -2032,7 +2032,10 @@ static inline bool arm_is_secure(CPUARMState *env)
static inline bool arm_is_el2_enabled(CPUARMState *env) static inline bool arm_is_el2_enabled(CPUARMState *env)
{ {
if (arm_feature(env, ARM_FEATURE_EL2)) { if (arm_feature(env, ARM_FEATURE_EL2)) {
return !arm_is_secure_below_el3(env); if (arm_is_secure_below_el3(env)) {
return (env->cp15.scr_el3 & SCR_EEL2) != 0;
}
return true;
} }
return false; return false;
} }
@ -2079,7 +2082,8 @@ static inline bool arm_el_is_aa64(CPUARMState *env, int el)
return aa64; return aa64;
} }
if (arm_feature(env, ARM_FEATURE_EL3)) { if (arm_feature(env, ARM_FEATURE_EL3) &&
((env->cp15.scr_el3 & SCR_NS) || !(env->cp15.scr_el3 & SCR_EEL2))) {
aa64 = aa64 && (env->cp15.scr_el3 & SCR_RW); aa64 = aa64 && (env->cp15.scr_el3 & SCR_RW);
} }

View file

@ -272,6 +272,9 @@ static CPAccessResult access_trap_aa32s_el1(CPUARMState *env,
return CP_ACCESS_OK; return CP_ACCESS_OK;
} }
if (arm_is_secure_below_el3(env)) { if (arm_is_secure_below_el3(env)) {
if (env->cp15.scr_el3 & SCR_EEL2) {
return CP_ACCESS_TRAP_EL2;
}
return CP_ACCESS_TRAP_EL3; return CP_ACCESS_TRAP_EL3;
} }
/* This will be EL1 NS and EL2 NS, which just UNDEF */ /* This will be EL1 NS and EL2 NS, which just UNDEF */
@ -1723,6 +1726,9 @@ static void scr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
if (cpu_isar_feature(aa64_pauth, cpu)) { if (cpu_isar_feature(aa64_pauth, cpu)) {
valid_mask |= SCR_API | SCR_APK; valid_mask |= SCR_API | SCR_APK;
} }
if (cpu_isar_feature(aa64_sel2, cpu)) {
valid_mask |= SCR_EEL2;
}
if (cpu_isar_feature(aa64_mte, cpu)) { if (cpu_isar_feature(aa64_mte, cpu)) {
valid_mask |= SCR_ATA; valid_mask |= SCR_ATA;
} }
@ -3080,13 +3086,16 @@ static CPAccessResult ats_access(CPUARMState *env, const ARMCPRegInfo *ri,
bool isread) bool isread)
{ {
if (ri->opc2 & 4) { if (ri->opc2 & 4) {
/* The ATS12NSO* operations must trap to EL3 if executed in /* The ATS12NSO* operations must trap to EL3 or EL2 if executed in
* Secure EL1 (which can only happen if EL3 is AArch64). * Secure EL1 (which can only happen if EL3 is AArch64).
* They are simply UNDEF if executed from NS EL1. * They are simply UNDEF if executed from NS EL1.
* They function normally from EL2 or EL3. * They function normally from EL2 or EL3.
*/ */
if (arm_current_el(env) == 1) { if (arm_current_el(env) == 1) {
if (arm_is_secure_below_el3(env)) { if (arm_is_secure_below_el3(env)) {
if (env->cp15.scr_el3 & SCR_EEL2) {
return CP_ACCESS_TRAP_UNCATEGORIZED_EL2;
}
return CP_ACCESS_TRAP_UNCATEGORIZED_EL3; return CP_ACCESS_TRAP_UNCATEGORIZED_EL3;
} }
return CP_ACCESS_TRAP_UNCATEGORIZED; return CP_ACCESS_TRAP_UNCATEGORIZED;
@ -3349,7 +3358,8 @@ static void ats1h_write(CPUARMState *env, const ARMCPRegInfo *ri,
static CPAccessResult at_s1e2_access(CPUARMState *env, const ARMCPRegInfo *ri, static CPAccessResult at_s1e2_access(CPUARMState *env, const ARMCPRegInfo *ri,
bool isread) bool isread)
{ {
if (arm_current_el(env) == 3 && !(env->cp15.scr_el3 & SCR_NS)) { if (arm_current_el(env) == 3 &&
!(env->cp15.scr_el3 & (SCR_NS | SCR_EEL2))) {
return CP_ACCESS_TRAP; return CP_ACCESS_TRAP;
} }
return CP_ACCESS_OK; return CP_ACCESS_OK;
@ -5430,12 +5440,15 @@ static CPAccessResult nsacr_access(CPUARMState *env, const ARMCPRegInfo *ri,
bool isread) bool isread)
{ {
/* The NSACR is RW at EL3, and RO for NS EL1 and NS EL2. /* The NSACR is RW at EL3, and RO for NS EL1 and NS EL2.
* At Secure EL1 it traps to EL3. * At Secure EL1 it traps to EL3 or EL2.
*/ */
if (arm_current_el(env) == 3) { if (arm_current_el(env) == 3) {
return CP_ACCESS_OK; return CP_ACCESS_OK;
} }
if (arm_is_secure_below_el3(env)) { if (arm_is_secure_below_el3(env)) {
if (env->cp15.scr_el3 & SCR_EEL2) {
return CP_ACCESS_TRAP_EL2;
}
return CP_ACCESS_TRAP_EL3; return CP_ACCESS_TRAP_EL3;
} }
/* Accesses from EL1 NS and EL2 NS are UNDEF for write but allow reads. */ /* Accesses from EL1 NS and EL2 NS are UNDEF for write but allow reads. */

View file

@ -2796,6 +2796,7 @@ static bool msr_banked_access_decode(DisasContext *s, int r, int sysm, int rn,
int *tgtmode, int *regno) int *tgtmode, int *regno)
{ {
TCGContext *tcg_ctx = s->uc->tcg_ctx; TCGContext *tcg_ctx = s->uc->tcg_ctx;
/* Decode the r and sysm fields of MSR/MRS banked accesses into /* Decode the r and sysm fields of MSR/MRS banked accesses into
* the target mode and register number, and identify the various * the target mode and register number, and identify the various
* unpredictable cases. * unpredictable cases.
@ -2930,9 +2931,20 @@ static bool msr_banked_access_decode(DisasContext *s, int r, int sysm, int rn,
} }
if (s->current_el == 1) { if (s->current_el == 1) {
/* If we're in Secure EL1 (which implies that EL3 is AArch64) /* If we're in Secure EL1 (which implies that EL3 is AArch64)
* then accesses to Mon registers trap to EL3 * then accesses to Mon registers trap to Secure EL2, if it exists,
* otherwise EL3.
*/ */
TCGv_i32 tcg_el = tcg_const_i32(tcg_ctx, 3); TCGv_i32 tcg_el;
if (arm_dc_feature(s, ARM_FEATURE_AARCH64) &&
dc_isar_feature(aa64_sel2, s)) {
/* Target EL is EL<3 minus SCR_EL3.EEL2> */
tcg_el = load_cpu_field(s, cp15.scr_el3);
tcg_gen_sextract_i32(tcg_ctx, tcg_el, tcg_el, ctz32(SCR_EEL2), 1);
tcg_gen_addi_i32(tcg_ctx, tcg_el, tcg_el, 3);
} else {
tcg_el = tcg_const_i32(tcg_ctx, 3);
}
gen_exception_el(s, EXCP_UDEF, syn_uncategorized(), tcg_el); gen_exception_el(s, EXCP_UDEF, syn_uncategorized(), tcg_el);
tcg_temp_free_i32(tcg_ctx, tcg_el); tcg_temp_free_i32(tcg_ctx, tcg_el);