hw/intc/armv7m_nvic: NS BFAR and BFSR are RAZ/WI if BFHFNMINS == 0

The non-secure versions of the BFAR and BFSR registers are
supposed to be RAZ/WI if AICR.BFHFNMINS == 0; we were
incorrectly allowing NS code to access the real values.

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20190430131439.25251-3-peter.maydell@linaro.org
This commit is contained in:
Peter Maydell 2019-05-07 12:55:03 +01:00
parent b01e2f0284
commit 339327b6d4
1 changed files with 24 additions and 3 deletions

View File

@ -1167,6 +1167,10 @@ static uint32_t nvic_readl(NVICState *s, uint32_t offset, MemTxAttrs attrs)
if (!arm_feature(&cpu->env, ARM_FEATURE_M_MAIN)) { if (!arm_feature(&cpu->env, ARM_FEATURE_M_MAIN)) {
goto bad_offset; goto bad_offset;
} }
if (!attrs.secure &&
!(s->cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK)) {
return 0;
}
return cpu->env.v7m.bfar; return cpu->env.v7m.bfar;
case 0xd3c: /* Aux Fault Status. */ case 0xd3c: /* Aux Fault Status. */
/* TODO: Implement fault status registers. */ /* TODO: Implement fault status registers. */
@ -1646,6 +1650,10 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
if (!arm_feature(&cpu->env, ARM_FEATURE_M_MAIN)) { if (!arm_feature(&cpu->env, ARM_FEATURE_M_MAIN)) {
goto bad_offset; goto bad_offset;
} }
if (!attrs.secure &&
!(s->cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK)) {
return;
}
cpu->env.v7m.bfar = value; cpu->env.v7m.bfar = value;
return; return;
case 0xd3c: /* Aux Fault Status. */ case 0xd3c: /* Aux Fault Status. */
@ -2130,11 +2138,18 @@ static MemTxResult nvic_sysreg_read(void *opaque, hwaddr addr,
val = 0; val = 0;
break; break;
}; };
/* The BFSR bits [15:8] are shared between security states /*
* and we store them in the NS copy * The BFSR bits [15:8] are shared between security states
* and we store them in the NS copy. They are RAZ/WI for
* NS code if AIRCR.BFHFNMINS is 0.
*/ */
val = s->cpu->env.v7m.cfsr[attrs.secure]; val = s->cpu->env.v7m.cfsr[attrs.secure];
if (!attrs.secure &&
!(s->cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK)) {
val &= ~R_V7M_CFSR_BFSR_MASK;
} else {
val |= s->cpu->env.v7m.cfsr[M_REG_NS] & R_V7M_CFSR_BFSR_MASK; val |= s->cpu->env.v7m.cfsr[M_REG_NS] & R_V7M_CFSR_BFSR_MASK;
}
val = extract32(val, (offset - 0xd28) * 8, size * 8); val = extract32(val, (offset - 0xd28) * 8, size * 8);
break; break;
case 0xfe0 ... 0xfff: /* ID. */ case 0xfe0 ... 0xfff: /* ID. */
@ -2249,6 +2264,12 @@ static MemTxResult nvic_sysreg_write(void *opaque, hwaddr addr,
*/ */
value <<= ((offset - 0xd28) * 8); value <<= ((offset - 0xd28) * 8);
if (!attrs.secure &&
!(s->cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK)) {
/* BFSR bits are RAZ/WI for NS if BFHFNMINS is set */
value &= ~R_V7M_CFSR_BFSR_MASK;
}
s->cpu->env.v7m.cfsr[attrs.secure] &= ~value; s->cpu->env.v7m.cfsr[attrs.secure] &= ~value;
if (attrs.secure) { if (attrs.secure) {
/* The BFSR bits [15:8] are shared between security states /* The BFSR bits [15:8] are shared between security states