armv7m: Fix reads of CONTROL register bit 1

The v7m CONTROL register bit 1 is SPSEL, which indicates
the stack being used. We were storing this information
not in v7m.control but in the separate v7m.other_sp
structure field. Unfortunately, the code handling reads
of the CONTROL register didn't take account of this, and
so if SPSEL was updated by an exception entry or exit then
a subsequent guest read of CONTROL would get the wrong value.

Using a separate structure field doesn't really gain us
anything in efficiency, so drop this unnecessary complexity
in favour of simply storing all the bits in v7m.control.

This is a migration compatibility break for M profile
CPUs only.

Backports commit abc24d86cc0364f402e438fae3acb14289b40734 from qemu
This commit is contained in:
Michael Davidsaver 2018-03-02 13:25:58 -05:00 committed by Lioncash
parent d8eb259032
commit 2769c6ada0
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7
4 changed files with 102 additions and 13 deletions

View file

@ -0,0 +1,73 @@
/*
* Register Definition API: field macros
*
* Copyright (c) 2016 Xilinx Inc.
* Copyright (c) 2013 Peter Crosthwaite <peter.crosthwaite@xilinx.com>
*
* This work is licensed under the terms of the GNU GPL, version 2. See
* the COPYING file in the top-level directory.
*/
#ifndef REGISTERFIELDS_H
#define REGISTERFIELDS_H
#include "qemu/bitops.h"
/* Define constants for a 32 bit register */
/* This macro will define A_FOO, for the byte address of a register
* as well as R_FOO for the uint32_t[] register number (A_FOO / 4).
*/
#define REG32(reg, addr) \
enum { A_ ## reg = (addr) }; \
enum { R_ ## reg = (addr) / 4 };
/* Define SHIFT, LENGTH and MASK constants for a field within a register */
/* This macro will define R_FOO_BAR_MASK, R_FOO_BAR_SHIFT and R_FOO_BAR_LENGTH
* constants for field BAR in register FOO.
*/
#define FIELD(reg, field, shift, length) \
enum { R_ ## reg ## _ ## field ## _SHIFT = (shift)}; \
enum { R_ ## reg ## _ ## field ## _LENGTH = (length)}; \
enum { R_ ## reg ## _ ## field ## _MASK = \
MAKE_64BIT_MASK(shift, length)};
/* Extract a field from a register */
#define FIELD_EX32(storage, reg, field) \
extract32((storage), R_ ## reg ## _ ## field ## _SHIFT, \
R_ ## reg ## _ ## field ## _LENGTH)
#define FIELD_EX64(storage, reg, field) \
extract64((storage), R_ ## reg ## _ ## field ## _SHIFT, \
R_ ## reg ## _ ## field ## _LENGTH)
/* Extract a field from an array of registers */
#define ARRAY_FIELD_EX32(regs, reg, field) \
FIELD_EX32((regs)[R_ ## reg], reg, field)
/* Deposit a register field.
* Assigning values larger then the target field will result in
* compilation warnings.
*/
#define FIELD_DP32(storage, reg, field, val) ({ \
struct { \
unsigned int v:R_ ## reg ## _ ## field ## _LENGTH; \
} v = { val }; \
uint32_t d; \
d = deposit32((storage), R_ ## reg ## _ ## field ## _SHIFT, \
R_ ## reg ## _ ## field ## _LENGTH, v.v); \
d; })
#define FIELD_DP64(storage, reg, field, val) ({ \
struct { \
unsigned int v:R_ ## reg ## _ ## field ## _LENGTH; \
} v = { val }; \
uint64_t d; \
d = deposit64((storage), R_ ## reg ## _ ## field ## _SHIFT, \
R_ ## reg ## _ ## field ## _LENGTH, v.v); \
d; })
/* Deposit a field to array of registers. */
#define ARRAY_FIELD_DP32(regs, reg, field, val) \
(regs)[R_ ## reg] = FIELD_DP32((regs)[R_ ## reg], reg, field, val);
#endif

View file

@ -414,7 +414,6 @@ typedef struct CPUARMState {
uint32_t vecbase; uint32_t vecbase;
uint32_t basepri; uint32_t basepri;
uint32_t control; uint32_t control;
int current_sp;
int exception; int exception;
uint32_t secure; /* Is CPU in Secure state? (not guest visible) */ uint32_t secure; /* Is CPU in Secure state? (not guest visible) */
} v7m; } v7m;

View file

@ -5318,14 +5318,18 @@ static uint32_t v7m_pop(CPUARMState *env)
} }
/* Switch to V7M main or process stack pointer. */ /* Switch to V7M main or process stack pointer. */
static void switch_v7m_sp(CPUARMState *env, int process) static void switch_v7m_sp(CPUARMState *env, bool new_spsel)
{ {
uint32_t tmp; uint32_t tmp;
if (env->v7m.current_sp != process) { bool old_spsel = env->v7m.control & R_V7M_CONTROL_SPSEL_MASK;
if (old_spsel != new_spsel) {
tmp = env->v7m.other_sp; tmp = env->v7m.other_sp;
env->v7m.other_sp = env->regs[13]; env->v7m.other_sp = env->regs[13];
env->regs[13] = tmp; env->regs[13] = tmp;
env->v7m.current_sp = process; env->v7m.control = deposit32(env->v7m.control,
R_V7M_CONTROL_SPSEL_SHIFT,
R_V7M_CONTROL_SPSEL_LENGTH, new_spsel);
} }
} }
@ -5395,8 +5399,9 @@ void arm_v7m_cpu_do_interrupt(CPUState *cs)
arm_log_exception(cs->exception_index); arm_log_exception(cs->exception_index);
lr = 0xfffffff1; lr = 0xfffffff1;
if (env->v7m.current_sp) if (env->v7m.control & R_V7M_CONTROL_SPSEL_MASK) {
lr |= 4; lr |= 4;
}
if (env->v7m.exception == 0) if (env->v7m.exception == 0)
lr |= 8; lr |= 8;
@ -7640,9 +7645,11 @@ uint32_t HELPER(v7m_mrs)(CPUARMState *env, uint32_t reg)
case 7: /* IEPSR */ case 7: /* IEPSR */
return xpsr_read(env) & 0x0700edff; return xpsr_read(env) & 0x0700edff;
case 8: /* MSP */ case 8: /* MSP */
return env->v7m.current_sp ? env->v7m.other_sp : env->regs[13]; return (env->v7m.control & R_V7M_CONTROL_SPSEL_MASK) ?
env->v7m.other_sp : env->regs[13];
case 9: /* PSP */ case 9: /* PSP */
return env->v7m.current_sp ? env->regs[13] : env->v7m.other_sp; return (env->v7m.control & R_V7M_CONTROL_SPSEL_MASK) ?
env->regs[13] : env->v7m.other_sp;
case 16: /* PRIMASK */ case 16: /* PRIMASK */
return (env->daif & PSTATE_I) != 0; return (env->daif & PSTATE_I) != 0;
case 17: /* BASEPRI */ case 17: /* BASEPRI */
@ -7686,16 +7693,18 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t reg, uint32_t val)
xpsr_write(env, val, 0x0600fc00); xpsr_write(env, val, 0x0600fc00);
break; break;
case 8: /* MSP */ case 8: /* MSP */
if (env->v7m.current_sp) if (env->v7m.control & R_V7M_CONTROL_SPSEL_MASK) {
env->v7m.other_sp = val; env->v7m.other_sp = val;
else } else {
env->regs[13] = val; env->regs[13] = val;
}
break; break;
case 9: /* PSP */ case 9: /* PSP */
if (env->v7m.current_sp) if (env->v7m.control & R_V7M_CONTROL_SPSEL_MASK) {
env->regs[13] = val; env->regs[13] = val;
else } else {
env->v7m.other_sp = val; env->v7m.other_sp = val;
}
break; break;
case 16: /* PRIMASK */ case 16: /* PRIMASK */
if (val & 1) { if (val & 1) {
@ -7720,8 +7729,9 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t reg, uint32_t val)
} }
break; break;
case 20: /* CONTROL */ case 20: /* CONTROL */
env->v7m.control = val & 3; switch_v7m_sp(env, (val & R_V7M_CONTROL_SPSEL_MASK) != 0);
switch_v7m_sp(env, (val & 2) != 0); env->v7m.control = val & (R_V7M_CONTROL_SPSEL_MASK |
R_V7M_CONTROL_NPRIV_MASK);
break; break;
default: default:
/* ??? For debugging only. */ /* ??? For debugging only. */

View file

@ -25,6 +25,8 @@
#ifndef TARGET_ARM_INTERNALS_H #ifndef TARGET_ARM_INTERNALS_H
#define TARGET_ARM_INTERNALS_H #define TARGET_ARM_INTERNALS_H
#include "hw/registerfields.h"
/* register banks for CPU modes */ /* register banks for CPU modes */
#define BANK_USRSYS 0 #define BANK_USRSYS 0
#define BANK_SVC 1 #define BANK_SVC 1
@ -77,6 +79,11 @@ static const char * const excnames[] = {
*/ */
#define GTIMER_SCALE 16 #define GTIMER_SCALE 16
/* Bit definitions for the v7M CONTROL register */
FIELD(V7M_CONTROL, NPRIV, 0, 1)
FIELD(V7M_CONTROL, SPSEL, 1, 1)
FIELD(V7M_CONTROL, FPCA, 2, 1)
/* /*
* For AArch64, map a given EL to an index in the banked_spsr array. * For AArch64, map a given EL to an index in the banked_spsr array.
* Note that this mapping and the AArch32 mapping defined in bank_number() * Note that this mapping and the AArch32 mapping defined in bank_number()