target/arm: Enable trapping of ERET for FEAT_NV
When FEAT_NV is turned on via the HCR_EL2.NV bit, ERET instructions are trapped, with the same syndrome information as for the existing FEAT_FGT fine-grained trap (in the pseudocode this is handled in AArch64.CheckForEretTrap()). Rename the DisasContext and tbflag bits to reflect that they are no longer exclusively for FGT traps, and set the tbflag bit when FEAT_NV is enabled as well as when the FGT is enabled. Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Reviewed-by: Richard Henderson <richard.henderson@linaro.org> Tested-by: Miguel Luis <miguel.luis@oracle.com>
This commit is contained in:
parent
5725977915
commit
e37e98b7f9
@ -3232,7 +3232,7 @@ FIELD(TBFLAG_A64, PSTATE_ZA, 23, 1)
|
|||||||
FIELD(TBFLAG_A64, SVL, 24, 4)
|
FIELD(TBFLAG_A64, SVL, 24, 4)
|
||||||
/* Indicates that SME Streaming mode is active, and SMCR_ELx.FA64 is not. */
|
/* Indicates that SME Streaming mode is active, and SMCR_ELx.FA64 is not. */
|
||||||
FIELD(TBFLAG_A64, SME_TRAP_NONSTREAMING, 28, 1)
|
FIELD(TBFLAG_A64, SME_TRAP_NONSTREAMING, 28, 1)
|
||||||
FIELD(TBFLAG_A64, FGT_ERET, 29, 1)
|
FIELD(TBFLAG_A64, TRAP_ERET, 29, 1)
|
||||||
FIELD(TBFLAG_A64, NAA, 30, 1)
|
FIELD(TBFLAG_A64, NAA, 30, 1)
|
||||||
FIELD(TBFLAG_A64, ATA0, 31, 1)
|
FIELD(TBFLAG_A64, ATA0, 31, 1)
|
||||||
|
|
||||||
|
@ -169,6 +169,7 @@ static CPUARMTBFlags rebuild_hflags_a64(CPUARMState *env, int el, int fp_el,
|
|||||||
CPUARMTBFlags flags = {};
|
CPUARMTBFlags flags = {};
|
||||||
ARMMMUIdx stage1 = stage_1_mmu_idx(mmu_idx);
|
ARMMMUIdx stage1 = stage_1_mmu_idx(mmu_idx);
|
||||||
uint64_t tcr = regime_tcr(env, mmu_idx);
|
uint64_t tcr = regime_tcr(env, mmu_idx);
|
||||||
|
uint64_t hcr = arm_hcr_el2_eff(env);
|
||||||
uint64_t sctlr;
|
uint64_t sctlr;
|
||||||
int tbii, tbid;
|
int tbii, tbid;
|
||||||
|
|
||||||
@ -285,13 +286,21 @@ static CPUARMTBFlags rebuild_hflags_a64(CPUARMState *env, int el, int fp_el,
|
|||||||
if (arm_fgt_active(env, el)) {
|
if (arm_fgt_active(env, el)) {
|
||||||
DP_TBFLAG_ANY(flags, FGT_ACTIVE, 1);
|
DP_TBFLAG_ANY(flags, FGT_ACTIVE, 1);
|
||||||
if (FIELD_EX64(env->cp15.fgt_exec[FGTREG_HFGITR], HFGITR_EL2, ERET)) {
|
if (FIELD_EX64(env->cp15.fgt_exec[FGTREG_HFGITR], HFGITR_EL2, ERET)) {
|
||||||
DP_TBFLAG_A64(flags, FGT_ERET, 1);
|
DP_TBFLAG_A64(flags, TRAP_ERET, 1);
|
||||||
}
|
}
|
||||||
if (fgt_svc(env, el)) {
|
if (fgt_svc(env, el)) {
|
||||||
DP_TBFLAG_ANY(flags, FGT_SVC, 1);
|
DP_TBFLAG_ANY(flags, FGT_SVC, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ERET can also be trapped for FEAT_NV. arm_hcr_el2_eff() takes care
|
||||||
|
* of "is EL2 enabled" and the NV bit can only be set if FEAT_NV is present.
|
||||||
|
*/
|
||||||
|
if (el == 1 && (hcr & HCR_NV)) {
|
||||||
|
DP_TBFLAG_A64(flags, TRAP_ERET, 1);
|
||||||
|
}
|
||||||
|
|
||||||
if (cpu_isar_feature(aa64_mte, env_archcpu(env))) {
|
if (cpu_isar_feature(aa64_mte, env_archcpu(env))) {
|
||||||
/*
|
/*
|
||||||
* Set MTE_ACTIVE if any access may be Checked, and leave clear
|
* Set MTE_ACTIVE if any access may be Checked, and leave clear
|
||||||
|
@ -1606,7 +1606,7 @@ static bool trans_ERET(DisasContext *s, arg_ERET *a)
|
|||||||
if (s->current_el == 0) {
|
if (s->current_el == 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (s->fgt_eret) {
|
if (s->trap_eret) {
|
||||||
gen_exception_insn_el(s, 0, EXCP_UDEF, syn_erettrap(0), 2);
|
gen_exception_insn_el(s, 0, EXCP_UDEF, syn_erettrap(0), 2);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -1633,7 +1633,7 @@ static bool trans_ERETA(DisasContext *s, arg_reta *a)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
/* The FGT trap takes precedence over an auth trap. */
|
/* The FGT trap takes precedence over an auth trap. */
|
||||||
if (s->fgt_eret) {
|
if (s->trap_eret) {
|
||||||
gen_exception_insn_el(s, 0, EXCP_UDEF, syn_erettrap(a->m ? 3 : 2), 2);
|
gen_exception_insn_el(s, 0, EXCP_UDEF, syn_erettrap(a->m ? 3 : 2), 2);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -13980,7 +13980,7 @@ static void aarch64_tr_init_disas_context(DisasContextBase *dcbase,
|
|||||||
dc->pstate_il = EX_TBFLAG_ANY(tb_flags, PSTATE__IL);
|
dc->pstate_il = EX_TBFLAG_ANY(tb_flags, PSTATE__IL);
|
||||||
dc->fgt_active = EX_TBFLAG_ANY(tb_flags, FGT_ACTIVE);
|
dc->fgt_active = EX_TBFLAG_ANY(tb_flags, FGT_ACTIVE);
|
||||||
dc->fgt_svc = EX_TBFLAG_ANY(tb_flags, FGT_SVC);
|
dc->fgt_svc = EX_TBFLAG_ANY(tb_flags, FGT_SVC);
|
||||||
dc->fgt_eret = EX_TBFLAG_A64(tb_flags, FGT_ERET);
|
dc->trap_eret = EX_TBFLAG_A64(tb_flags, TRAP_ERET);
|
||||||
dc->sve_excp_el = EX_TBFLAG_A64(tb_flags, SVEEXC_EL);
|
dc->sve_excp_el = EX_TBFLAG_A64(tb_flags, SVEEXC_EL);
|
||||||
dc->sme_excp_el = EX_TBFLAG_A64(tb_flags, SMEEXC_EL);
|
dc->sme_excp_el = EX_TBFLAG_A64(tb_flags, SMEEXC_EL);
|
||||||
dc->vl = (EX_TBFLAG_A64(tb_flags, VL) + 1) * 16;
|
dc->vl = (EX_TBFLAG_A64(tb_flags, VL) + 1) * 16;
|
||||||
|
@ -138,10 +138,10 @@ typedef struct DisasContext {
|
|||||||
bool mve_no_pred;
|
bool mve_no_pred;
|
||||||
/* True if fine-grained traps are active */
|
/* True if fine-grained traps are active */
|
||||||
bool fgt_active;
|
bool fgt_active;
|
||||||
/* True if fine-grained trap on ERET is enabled */
|
|
||||||
bool fgt_eret;
|
|
||||||
/* True if fine-grained trap on SVC is enabled */
|
/* True if fine-grained trap on SVC is enabled */
|
||||||
bool fgt_svc;
|
bool fgt_svc;
|
||||||
|
/* True if a trap on ERET is enabled (FGT or NV) */
|
||||||
|
bool trap_eret;
|
||||||
/* True if FEAT_LSE2 SCTLR_ELx.nAA is set */
|
/* True if FEAT_LSE2 SCTLR_ELx.nAA is set */
|
||||||
bool naa;
|
bool naa;
|
||||||
/*
|
/*
|
||||||
|
Loading…
Reference in New Issue
Block a user