target/arm: Do hflags rebuild in cpsr_write()
Currently we rely on all the callsites of cpsr_write() to rebuild the cached hflags if they change one of the CPSR bits which we use as a TB flag and cache in hflags. This is a bit awkward when we want to change the set of CPSR bits that we cache, because it means we need to re-audit all the cpsr_write() callsites to see which flags they are writing and whether they now need to rebuild the hflags. Switch instead to making cpsr_write() call arm_rebuild_hflags() itself if one of the bits being changed is a cached bit. We don't do the rebuild for the CPSRWriteRaw write type, because that kind of write is generally doing something special anyway. For the CPSRWriteRaw callsites in the KVM code and inbound migration we definitely don't want to recalculate the hflags; the callsites in boot.c and arm-powerctl.c have to do a rebuild-hflags call themselves anyway because of other CPU state changes they make. This allows us to drop explicit arm_rebuild_hflags() calls in a couple of places where the only reason we needed to call it was the CPSR write. This fixes a bug where we were incorrectly failing to rebuild hflags in the code path for a gdbstub write to CPSR, which meant that you could make QEMU assert by breaking into a running guest, altering the CPSR to change the value of, for example, CPSR.E, and then continuing. Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Reviewed-by: Richard Henderson <richard.henderson@linaro.org> Message-id: 20210817201843.3829-1-peter.maydell@linaro.org
This commit is contained in:
parent
8e228c9e4b
commit
e784807cd2
@ -289,7 +289,6 @@ setup_return(CPUARMState *env, struct target_sigaction *ka,
|
|||||||
env->regs[14] = retcode;
|
env->regs[14] = retcode;
|
||||||
env->regs[15] = handler & (thumb ? ~1 : ~3);
|
env->regs[15] = handler & (thumb ? ~1 : ~3);
|
||||||
cpsr_write(env, cpsr, CPSR_IT | CPSR_T | CPSR_E, CPSRWriteByInstr);
|
cpsr_write(env, cpsr, CPSR_IT | CPSR_T | CPSR_E, CPSRWriteByInstr);
|
||||||
arm_rebuild_hflags(env);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -547,7 +546,6 @@ restore_sigcontext(CPUARMState *env, struct target_sigcontext *sc)
|
|||||||
__get_user(env->regs[15], &sc->arm_pc);
|
__get_user(env->regs[15], &sc->arm_pc);
|
||||||
__get_user(cpsr, &sc->arm_cpsr);
|
__get_user(cpsr, &sc->arm_cpsr);
|
||||||
cpsr_write(env, cpsr, CPSR_USER | CPSR_EXEC, CPSRWriteByInstr);
|
cpsr_write(env, cpsr, CPSR_USER | CPSR_EXEC, CPSRWriteByInstr);
|
||||||
arm_rebuild_hflags(env);
|
|
||||||
|
|
||||||
err |= !valid_user_regs(env);
|
err |= !valid_user_regs(env);
|
||||||
|
|
||||||
|
@ -1398,11 +1398,17 @@ uint32_t cpsr_read(CPUARMState *env);
|
|||||||
typedef enum CPSRWriteType {
|
typedef enum CPSRWriteType {
|
||||||
CPSRWriteByInstr = 0, /* from guest MSR or CPS */
|
CPSRWriteByInstr = 0, /* from guest MSR or CPS */
|
||||||
CPSRWriteExceptionReturn = 1, /* from guest exception return insn */
|
CPSRWriteExceptionReturn = 1, /* from guest exception return insn */
|
||||||
CPSRWriteRaw = 2, /* trust values, do not switch reg banks */
|
CPSRWriteRaw = 2,
|
||||||
|
/* trust values, no reg bank switch, no hflags rebuild */
|
||||||
CPSRWriteByGDBStub = 3, /* from the GDB stub */
|
CPSRWriteByGDBStub = 3, /* from the GDB stub */
|
||||||
} CPSRWriteType;
|
} CPSRWriteType;
|
||||||
|
|
||||||
/* Set the CPSR. Note that some bits of mask must be all-set or all-clear.*/
|
/*
|
||||||
|
* Set the CPSR. Note that some bits of mask must be all-set or all-clear.
|
||||||
|
* This will do an arm_rebuild_hflags() if any of the bits in @mask
|
||||||
|
* correspond to TB flags bits cached in the hflags, unless @write_type
|
||||||
|
* is CPSRWriteRaw.
|
||||||
|
*/
|
||||||
void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
|
void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
|
||||||
CPSRWriteType write_type);
|
CPSRWriteType write_type);
|
||||||
|
|
||||||
|
@ -9246,6 +9246,8 @@ void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
|
|||||||
CPSRWriteType write_type)
|
CPSRWriteType write_type)
|
||||||
{
|
{
|
||||||
uint32_t changed_daif;
|
uint32_t changed_daif;
|
||||||
|
bool rebuild_hflags = (write_type != CPSRWriteRaw) &&
|
||||||
|
(mask & (CPSR_M | CPSR_E | CPSR_IL));
|
||||||
|
|
||||||
if (mask & CPSR_NZCV) {
|
if (mask & CPSR_NZCV) {
|
||||||
env->ZF = (~val) & CPSR_Z;
|
env->ZF = (~val) & CPSR_Z;
|
||||||
@ -9365,6 +9367,9 @@ void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
|
|||||||
}
|
}
|
||||||
mask &= ~CACHED_CPSR_BITS;
|
mask &= ~CACHED_CPSR_BITS;
|
||||||
env->uncached_cpsr = (env->uncached_cpsr & ~mask) | (val & mask);
|
env->uncached_cpsr = (env->uncached_cpsr & ~mask) | (val & mask);
|
||||||
|
if (rebuild_hflags) {
|
||||||
|
arm_rebuild_hflags(env);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Sign/zero extend */
|
/* Sign/zero extend */
|
||||||
|
Loading…
Reference in New Issue
Block a user