提交 ac656b16 编写于 作者: P Peter Maydell

target/arm: Provide accessor functions for HCR_EL2.{IMO, FMO, AMO}

The IMO, FMO and AMO bits in HCR_EL2 are defined to "behave as
1 for all purposes other than direct reads" if HCR_EL2.TGE
is set and HCR_EL2.E2H is 0, and to "behave as 0 for all
purposes other than direct reads" if HCR_EL2.TGE is set
and HRC_EL2.E2H is 1.

To avoid having to check E2H and TGE everywhere where we test IMO and
FMO, provide accessors arm_hcr_el2_imo(), arm_hcr_el2_fmo()and
arm_hcr_el2_amo().  We don't implement ARMv8.1-VHE yet, so the E2H
case will never be true, but we include the logic to save effort when
we eventually do get to that.

(Note that in several of these callsites the change doesn't
actually make a difference as either the callsite is handling
TGE specially anyway, or the CPU can't get into that situation
with TGE set; we change everywhere for consistency.)
Signed-off-by: NPeter Maydell <peter.maydell@linaro.org>
Reviewed-by: NRichard Henderson <richard.henderson@linaro.org>
Message-id: 20180724115950.17316-5-peter.maydell@linaro.org
上级 7556edfb
...@@ -85,7 +85,10 @@ static bool icv_access(CPUARMState *env, int hcr_flags) ...@@ -85,7 +85,10 @@ static bool icv_access(CPUARMState *env, int hcr_flags)
* * access if NS EL1 and either IMO or FMO == 1: * * access if NS EL1 and either IMO or FMO == 1:
* CTLR, DIR, PMR, RPR * CTLR, DIR, PMR, RPR
*/ */
return (env->cp15.hcr_el2 & hcr_flags) && arm_current_el(env) == 1 bool flagmatch = ((hcr_flags & HCR_IMO) && arm_hcr_el2_imo(env)) ||
((hcr_flags & HCR_FMO) && arm_hcr_el2_fmo(env));
return flagmatch && arm_current_el(env) == 1
&& !arm_is_secure_below_el3(env); && !arm_is_secure_below_el3(env);
} }
...@@ -1549,8 +1552,8 @@ static void icc_dir_write(CPUARMState *env, const ARMCPRegInfo *ri, ...@@ -1549,8 +1552,8 @@ static void icc_dir_write(CPUARMState *env, const ARMCPRegInfo *ri,
/* No need to include !IsSecure in route_*_to_el2 as it's only /* No need to include !IsSecure in route_*_to_el2 as it's only
* tested in cases where we know !IsSecure is true. * tested in cases where we know !IsSecure is true.
*/ */
route_fiq_to_el2 = env->cp15.hcr_el2 & HCR_FMO; route_fiq_to_el2 = arm_hcr_el2_fmo(env);
route_irq_to_el2 = env->cp15.hcr_el2 & HCR_IMO; route_irq_to_el2 = arm_hcr_el2_imo(env);
switch (arm_current_el(env)) { switch (arm_current_el(env)) {
case 3: case 3:
...@@ -1893,7 +1896,7 @@ static CPAccessResult gicv3_irqfiq_access(CPUARMState *env, ...@@ -1893,7 +1896,7 @@ static CPAccessResult gicv3_irqfiq_access(CPUARMState *env,
switch (el) { switch (el) {
case 1: case 1:
if (arm_is_secure_below_el3(env) || if (arm_is_secure_below_el3(env) ||
((env->cp15.hcr_el2 & (HCR_IMO | HCR_FMO)) == 0)) { (arm_hcr_el2_imo(env) == 0 && arm_hcr_el2_fmo(env) == 0)) {
r = CP_ACCESS_TRAP_EL3; r = CP_ACCESS_TRAP_EL3;
} }
break; break;
...@@ -1933,7 +1936,7 @@ static CPAccessResult gicv3_dir_access(CPUARMState *env, ...@@ -1933,7 +1936,7 @@ static CPAccessResult gicv3_dir_access(CPUARMState *env,
static CPAccessResult gicv3_sgi_access(CPUARMState *env, static CPAccessResult gicv3_sgi_access(CPUARMState *env,
const ARMCPRegInfo *ri, bool isread) const ARMCPRegInfo *ri, bool isread)
{ {
if ((env->cp15.hcr_el2 & (HCR_IMO | HCR_FMO)) && if ((arm_hcr_el2_imo(env) || arm_hcr_el2_fmo(env)) &&
arm_current_el(env) == 1 && !arm_is_secure_below_el3(env)) { arm_current_el(env) == 1 && !arm_is_secure_below_el3(env)) {
/* Takes priority over a possible EL3 trap */ /* Takes priority over a possible EL3 trap */
return CP_ACCESS_TRAP_EL2; return CP_ACCESS_TRAP_EL2;
...@@ -1958,8 +1961,7 @@ static CPAccessResult gicv3_fiq_access(CPUARMState *env, ...@@ -1958,8 +1961,7 @@ static CPAccessResult gicv3_fiq_access(CPUARMState *env,
if (env->cp15.scr_el3 & SCR_FIQ) { if (env->cp15.scr_el3 & SCR_FIQ) {
switch (el) { switch (el) {
case 1: case 1:
if (arm_is_secure_below_el3(env) || if (arm_is_secure_below_el3(env) || !arm_hcr_el2_fmo(env)) {
((env->cp15.hcr_el2 & HCR_FMO) == 0)) {
r = CP_ACCESS_TRAP_EL3; r = CP_ACCESS_TRAP_EL3;
} }
break; break;
...@@ -1998,8 +2000,7 @@ static CPAccessResult gicv3_irq_access(CPUARMState *env, ...@@ -1998,8 +2000,7 @@ static CPAccessResult gicv3_irq_access(CPUARMState *env,
if (env->cp15.scr_el3 & SCR_IRQ) { if (env->cp15.scr_el3 & SCR_IRQ) {
switch (el) { switch (el) {
case 1: case 1:
if (arm_is_secure_below_el3(env) || if (arm_is_secure_below_el3(env) || !arm_hcr_el2_imo(env)) {
((env->cp15.hcr_el2 & HCR_IMO) == 0)) {
r = CP_ACCESS_TRAP_EL3; r = CP_ACCESS_TRAP_EL3;
} }
break; break;
......
...@@ -1229,6 +1229,12 @@ static inline void xpsr_write(CPUARMState *env, uint32_t val, uint32_t mask) ...@@ -1229,6 +1229,12 @@ static inline void xpsr_write(CPUARMState *env, uint32_t val, uint32_t mask)
#define HCR_RW (1ULL << 31) #define HCR_RW (1ULL << 31)
#define HCR_CD (1ULL << 32) #define HCR_CD (1ULL << 32)
#define HCR_ID (1ULL << 33) #define HCR_ID (1ULL << 33)
#define HCR_E2H (1ULL << 34)
/*
* When we actually implement ARMv8.1-VHE we should add HCR_E2H to
* HCR_MASK and then clear it again if the feature bit is not set in
* hcr_write().
*/
#define HCR_MASK ((1ULL << 34) - 1) #define HCR_MASK ((1ULL << 34) - 1)
#define SCR_NS (1U << 0) #define SCR_NS (1U << 0)
...@@ -2234,6 +2240,54 @@ bool write_cpustate_to_list(ARMCPU *cpu); ...@@ -2234,6 +2240,54 @@ bool write_cpustate_to_list(ARMCPU *cpu);
# define TARGET_VIRT_ADDR_SPACE_BITS 32 # define TARGET_VIRT_ADDR_SPACE_BITS 32
#endif #endif
/**
* arm_hcr_el2_imo(): Return the effective value of HCR_EL2.IMO.
* Depending on the values of HCR_EL2.E2H and TGE, this may be
* "behaves as 1 for all purposes other than direct read/write" or
* "behaves as 0 for all purposes other than direct read/write"
*/
static inline bool arm_hcr_el2_imo(CPUARMState *env)
{
switch (env->cp15.hcr_el2 & (HCR_TGE | HCR_E2H)) {
case HCR_TGE:
return true;
case HCR_TGE | HCR_E2H:
return false;
default:
return env->cp15.hcr_el2 & HCR_IMO;
}
}
/**
* arm_hcr_el2_fmo(): Return the effective value of HCR_EL2.FMO.
*/
static inline bool arm_hcr_el2_fmo(CPUARMState *env)
{
switch (env->cp15.hcr_el2 & (HCR_TGE | HCR_E2H)) {
case HCR_TGE:
return true;
case HCR_TGE | HCR_E2H:
return false;
default:
return env->cp15.hcr_el2 & HCR_FMO;
}
}
/**
* arm_hcr_el2_amo(): Return the effective value of HCR_EL2.AMO.
*/
static inline bool arm_hcr_el2_amo(CPUARMState *env)
{
switch (env->cp15.hcr_el2 & (HCR_TGE | HCR_E2H)) {
case HCR_TGE:
return true;
case HCR_TGE | HCR_E2H:
return false;
default:
return env->cp15.hcr_el2 & HCR_AMO;
}
}
static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx, static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,
unsigned int target_el) unsigned int target_el)
{ {
...@@ -2261,15 +2315,13 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx, ...@@ -2261,15 +2315,13 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,
break; break;
case EXCP_VFIQ: case EXCP_VFIQ:
if (secure || !(env->cp15.hcr_el2 & HCR_FMO) if (secure || !arm_hcr_el2_fmo(env) || (env->cp15.hcr_el2 & HCR_TGE)) {
|| (env->cp15.hcr_el2 & HCR_TGE)) {
/* VFIQs are only taken when hypervized and non-secure. */ /* VFIQs are only taken when hypervized and non-secure. */
return false; return false;
} }
return !(env->daif & PSTATE_F); return !(env->daif & PSTATE_F);
case EXCP_VIRQ: case EXCP_VIRQ:
if (secure || !(env->cp15.hcr_el2 & HCR_IMO) if (secure || !arm_hcr_el2_imo(env) || (env->cp15.hcr_el2 & HCR_TGE)) {
|| (env->cp15.hcr_el2 & HCR_TGE)) {
/* VIRQs are only taken when hypervized and non-secure. */ /* VIRQs are only taken when hypervized and non-secure. */
return false; return false;
} }
...@@ -2308,7 +2360,7 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx, ...@@ -2308,7 +2360,7 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,
* to the CPSR.F setting otherwise we further assess the state * to the CPSR.F setting otherwise we further assess the state
* below. * below.
*/ */
hcr = (env->cp15.hcr_el2 & HCR_FMO); hcr = arm_hcr_el2_fmo(env);
scr = (env->cp15.scr_el3 & SCR_FIQ); scr = (env->cp15.scr_el3 & SCR_FIQ);
/* When EL3 is 32-bit, the SCR.FW bit controls whether the /* When EL3 is 32-bit, the SCR.FW bit controls whether the
...@@ -2325,7 +2377,7 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx, ...@@ -2325,7 +2377,7 @@ static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,
* when setting the target EL, so it does not have a further * when setting the target EL, so it does not have a further
* affect here. * affect here.
*/ */
hcr = (env->cp15.hcr_el2 & HCR_IMO); hcr = arm_hcr_el2_imo(env);
scr = false; scr = false;
break; break;
default: default:
......
...@@ -6336,15 +6336,15 @@ uint32_t arm_phys_excp_target_el(CPUState *cs, uint32_t excp_idx, ...@@ -6336,15 +6336,15 @@ uint32_t arm_phys_excp_target_el(CPUState *cs, uint32_t excp_idx,
switch (excp_idx) { switch (excp_idx) {
case EXCP_IRQ: case EXCP_IRQ:
scr = ((env->cp15.scr_el3 & SCR_IRQ) == SCR_IRQ); scr = ((env->cp15.scr_el3 & SCR_IRQ) == SCR_IRQ);
hcr = ((env->cp15.hcr_el2 & HCR_IMO) == HCR_IMO); hcr = arm_hcr_el2_imo(env);
break; break;
case EXCP_FIQ: case EXCP_FIQ:
scr = ((env->cp15.scr_el3 & SCR_FIQ) == SCR_FIQ); scr = ((env->cp15.scr_el3 & SCR_FIQ) == SCR_FIQ);
hcr = ((env->cp15.hcr_el2 & HCR_FMO) == HCR_FMO); hcr = arm_hcr_el2_fmo(env);
break; break;
default: default:
scr = ((env->cp15.scr_el3 & SCR_EA) == SCR_EA); scr = ((env->cp15.scr_el3 & SCR_EA) == SCR_EA);
hcr = ((env->cp15.hcr_el2 & HCR_AMO) == HCR_AMO); hcr = arm_hcr_el2_amo(env);
break; break;
}; };
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册