2015-12-07 19:23:44 +03:00
|
|
|
#include "qemu/osdep.h"
|
2016-03-15 18:58:45 +03:00
|
|
|
#include "cpu.h"
|
2015-09-24 03:29:37 +03:00
|
|
|
#include "qemu/error-report.h"
|
2013-06-25 21:16:07 +04:00
|
|
|
#include "sysemu/kvm.h"
|
2023-02-17 23:11:27 +03:00
|
|
|
#include "sysemu/tcg.h"
|
2013-06-25 21:16:07 +04:00
|
|
|
#include "kvm_arm.h"
|
2014-09-12 17:06:49 +04:00
|
|
|
#include "internals.h"
|
2023-10-24 19:35:05 +03:00
|
|
|
#include "cpu-features.h"
|
2016-03-15 14:51:18 +03:00
|
|
|
#include "migration/cpu.h"
|
2024-01-18 23:06:40 +03:00
|
|
|
#include "target/arm/gtimer.h"
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static bool vfp_needed(void *opaque)
|
2008-05-04 17:11:44 +04:00
|
|
|
{
|
2013-04-19 15:24:19 +04:00
|
|
|
ARMCPU *cpu = opaque;
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2020-02-25 01:22:19 +03:00
|
|
|
return (arm_feature(&cpu->env, ARM_FEATURE_AARCH64)
|
|
|
|
? cpu_isar_feature(aa64_fp_simd, cpu)
|
|
|
|
: cpu_isar_feature(aa32_vfp_simd, cpu));
|
2013-04-19 15:24:19 +04:00
|
|
|
}
|
2008-05-04 17:11:44 +04:00
|
|
|
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
static bool vfp_fpcr_fpsr_needed(void *opaque)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If either the FPCR or the FPSR include set bits that are not
|
|
|
|
* visible in the AArch32 FPSCR view of floating point control/status
|
|
|
|
* then we must send the FPCR and FPSR as two separate fields in the
|
|
|
|
* cpu/vfp/fpcr_fpsr subsection, and we will send a 0 for the old
|
|
|
|
* FPSCR field in cpu/vfp.
|
|
|
|
*
|
|
|
|
* If all the set bits are representable in an AArch32 FPSCR then we
|
|
|
|
* send that value as the cpu/vfp FPSCR field, and don't send the
|
|
|
|
* cpu/vfp/fpcr_fpsr subsection.
|
|
|
|
*
|
|
|
|
* On incoming migration, if the cpu/vfp FPSCR field is non-zero we
|
|
|
|
* use it, and if the fpcr_fpsr subsection is present we use that.
|
|
|
|
* (The subsection will never be present with a non-zero FPSCR field,
|
|
|
|
* and if FPSCR is zero and the subsection is not present that means
|
|
|
|
* that FPSCR/FPSR/FPCR are zero.)
|
|
|
|
*
|
|
|
|
* This preserves migration compatibility with older QEMU versions,
|
|
|
|
* in both directions.
|
|
|
|
*/
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
2024-06-28 17:23:46 +03:00
|
|
|
return (vfp_get_fpcr(env) & ~FPSCR_FPCR_MASK) ||
|
|
|
|
(vfp_get_fpsr(env) & ~FPSCR_FPSR_MASK);
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
}
|
|
|
|
|
2017-01-19 22:00:50 +03:00
|
|
|
static int get_fpscr(QEMUFile *f, void *opaque, size_t size,
|
2018-11-14 16:29:30 +03:00
|
|
|
const VMStateField *field)
|
2013-04-19 15:24:19 +04:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
uint32_t val = qemu_get_be32(f);
|
|
|
|
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
if (val) {
|
|
|
|
/* 0 means we might have the data in the fpcr_fpsr subsection */
|
|
|
|
vfp_set_fpscr(env, val);
|
|
|
|
}
|
2013-04-19 15:24:19 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-01-19 22:00:50 +03:00
|
|
|
static int put_fpscr(QEMUFile *f, void *opaque, size_t size,
|
2020-12-11 20:11:48 +03:00
|
|
|
const VMStateField *field, JSONWriter *vmdesc)
|
2013-04-19 15:24:19 +04:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
uint32_t fpscr = vfp_fpcr_fpsr_needed(opaque) ? 0 : vfp_get_fpscr(env);
|
2013-04-19 15:24:19 +04:00
|
|
|
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
qemu_put_be32(f, fpscr);
|
2017-01-19 22:00:50 +03:00
|
|
|
return 0;
|
2013-04-19 15:24:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateInfo vmstate_fpscr = {
|
|
|
|
.name = "fpscr",
|
|
|
|
.get = get_fpscr,
|
|
|
|
.put = put_fpscr,
|
|
|
|
};
|
|
|
|
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
static int get_fpcr(QEMUFile *f, void *opaque, size_t size,
|
|
|
|
const VMStateField *field)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
uint64_t val = qemu_get_be64(f);
|
|
|
|
|
|
|
|
vfp_set_fpcr(env, val);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int put_fpcr(QEMUFile *f, void *opaque, size_t size,
|
|
|
|
const VMStateField *field, JSONWriter *vmdesc)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
qemu_put_be64(f, vfp_get_fpcr(env));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateInfo vmstate_fpcr = {
|
|
|
|
.name = "fpcr",
|
|
|
|
.get = get_fpcr,
|
|
|
|
.put = put_fpcr,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int get_fpsr(QEMUFile *f, void *opaque, size_t size,
|
|
|
|
const VMStateField *field)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
uint64_t val = qemu_get_be64(f);
|
|
|
|
|
|
|
|
vfp_set_fpsr(env, val);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int put_fpsr(QEMUFile *f, void *opaque, size_t size,
|
|
|
|
const VMStateField *field, JSONWriter *vmdesc)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
qemu_put_be64(f, vfp_get_fpsr(env));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateInfo vmstate_fpsr = {
|
|
|
|
.name = "fpsr",
|
|
|
|
.get = get_fpsr,
|
|
|
|
.put = put_fpsr,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_vfp_fpcr_fpsr = {
|
|
|
|
.name = "cpu/vfp/fpcr_fpsr",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = vfp_fpcr_fpsr_needed,
|
|
|
|
.fields = (const VMStateField[]) {
|
|
|
|
{
|
|
|
|
.name = "fpcr",
|
|
|
|
.version_id = 0,
|
|
|
|
.size = sizeof(uint64_t),
|
|
|
|
.info = &vmstate_fpcr,
|
|
|
|
.flags = VMS_SINGLE,
|
|
|
|
.offset = 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "fpsr",
|
|
|
|
.version_id = 0,
|
|
|
|
.size = sizeof(uint64_t),
|
|
|
|
.info = &vmstate_fpsr,
|
|
|
|
.flags = VMS_SINGLE,
|
|
|
|
.offset = 0,
|
|
|
|
},
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static const VMStateDescription vmstate_vfp = {
|
|
|
|
.name = "cpu/vfp",
|
2013-09-03 23:12:09 +04:00
|
|
|
.version_id = 3,
|
|
|
|
.minimum_version_id = 3,
|
2014-09-23 16:09:54 +04:00
|
|
|
.needed = vfp_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-02-09 13:40:31 +03:00
|
|
|
/* For compatibility, store Qn out of Zn here. */
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[0].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[1].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[2].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[3].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[4].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[5].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[6].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[7].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[8].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[9].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[10].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[11].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[12].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[13].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[14].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[15].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[16].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[17].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[18].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[19].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[20].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[21].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[22].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[23].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[24].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[25].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[26].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[27].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[28].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[29].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[30].d, ARMCPU, 0, 2),
|
|
|
|
VMSTATE_UINT64_SUB_ARRAY(env.vfp.zregs[31].d, ARMCPU, 0, 2),
|
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
/* The xregs array is a little awkward because element 1 (FPSCR)
|
|
|
|
* requires a specific accessor, so we have to split it up in
|
|
|
|
* the vmstate:
|
|
|
|
*/
|
|
|
|
VMSTATE_UINT32(env.vfp.xregs[0], ARMCPU),
|
|
|
|
VMSTATE_UINT32_SUB_ARRAY(env.vfp.xregs, ARMCPU, 2, 14),
|
|
|
|
{
|
|
|
|
.name = "fpscr",
|
|
|
|
.version_id = 0,
|
|
|
|
.size = sizeof(uint32_t),
|
|
|
|
.info = &vmstate_fpscr,
|
|
|
|
.flags = VMS_SINGLE,
|
|
|
|
.offset = 0,
|
|
|
|
},
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_END_OF_LIST()
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
},
|
|
|
|
.subsections = (const VMStateDescription * const []) {
|
|
|
|
&vmstate_vfp_fpcr_fpsr,
|
|
|
|
NULL
|
2008-05-04 17:11:44 +04:00
|
|
|
}
|
2013-04-19 15:24:19 +04:00
|
|
|
};
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static bool iwmmxt_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
return arm_feature(env, ARM_FEATURE_IWMMXT);
|
|
|
|
}
|
2009-07-30 16:33:47 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static const VMStateDescription vmstate_iwmmxt = {
|
|
|
|
.name = "cpu/iwmmxt",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2014-09-23 16:09:54 +04:00
|
|
|
.needed = iwmmxt_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_UINT64_ARRAY(env.iwmmxt.regs, ARMCPU, 16),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.iwmmxt.cregs, ARMCPU, 16),
|
|
|
|
VMSTATE_END_OF_LIST()
|
2009-07-30 16:33:47 +04:00
|
|
|
}
|
2013-04-19 15:24:19 +04:00
|
|
|
};
|
|
|
|
|
2018-01-23 06:53:47 +03:00
|
|
|
#ifdef TARGET_AARCH64
|
|
|
|
/* The expression ARM_MAX_VQ - 2 is 0 for pure AArch32 build,
|
|
|
|
* and ARMPredicateReg is actively empty. This triggers errors
|
|
|
|
* in the expansion of the VMSTATE macros.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static bool sve_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
2018-10-24 09:50:17 +03:00
|
|
|
return cpu_isar_feature(aa64_sve, cpu);
|
2018-01-23 06:53:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* The first two words of each Zreg is stored in VFP state. */
|
|
|
|
static const VMStateDescription vmstate_zreg_hi_reg = {
|
|
|
|
.name = "cpu/sve/zreg_hi",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-01-23 06:53:47 +03:00
|
|
|
VMSTATE_UINT64_SUB_ARRAY(d, ARMVectorReg, 2, ARM_MAX_VQ - 2),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_preg_reg = {
|
|
|
|
.name = "cpu/sve/preg",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-01-23 06:53:47 +03:00
|
|
|
VMSTATE_UINT64_ARRAY(p, ARMPredicateReg, 2 * ARM_MAX_VQ / 8),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_sve = {
|
|
|
|
.name = "cpu/sve",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = sve_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-01-23 06:53:47 +03:00
|
|
|
VMSTATE_STRUCT_ARRAY(env.vfp.zregs, ARMCPU, 32, 0,
|
|
|
|
vmstate_zreg_hi_reg, ARMVectorReg),
|
|
|
|
VMSTATE_STRUCT_ARRAY(env.vfp.pregs, ARMCPU, 17, 0,
|
|
|
|
vmstate_preg_reg, ARMPredicateReg),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2022-06-20 20:51:53 +03:00
|
|
|
|
|
|
|
static const VMStateDescription vmstate_vreg = {
|
|
|
|
.name = "vreg",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2022-06-20 20:51:53 +03:00
|
|
|
VMSTATE_UINT64_ARRAY(d, ARMVectorReg, ARM_MAX_VQ * 2),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool za_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When ZA storage is disabled, its contents are discarded.
|
|
|
|
* It will be zeroed when ZA storage is re-enabled.
|
|
|
|
*/
|
|
|
|
return FIELD_EX64(cpu->env.svcr, SVCR, ZA);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_za = {
|
|
|
|
.name = "cpu/sme",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = za_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2022-06-20 20:51:53 +03:00
|
|
|
VMSTATE_STRUCT_ARRAY(env.zarray, ARMCPU, ARM_MAX_VQ * 16, 0,
|
|
|
|
vmstate_vreg, ARMVectorReg),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2018-01-23 06:53:47 +03:00
|
|
|
#endif /* AARCH64 */
|
|
|
|
|
2018-10-24 09:50:16 +03:00
|
|
|
static bool serror_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return env->serror.pending != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_serror = {
|
|
|
|
.name = "cpu/serror",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = serror_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-10-24 09:50:16 +03:00
|
|
|
VMSTATE_UINT8(env.serror.pending, ARMCPU),
|
|
|
|
VMSTATE_UINT8(env.serror.has_esr, ARMCPU),
|
|
|
|
VMSTATE_UINT64(env.serror.esr, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-11-13 13:47:59 +03:00
|
|
|
static bool irq_line_state_needed(void *opaque)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_irq_line_state = {
|
|
|
|
.name = "cpu/irq-line-state",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = irq_line_state_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-11-13 13:47:59 +03:00
|
|
|
VMSTATE_UINT32(env.irq_line_state, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-04-30 17:00:35 +03:00
|
|
|
static bool wfxt_timer_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
/* We'll only have the timer object if FEAT_WFxT is implemented */
|
|
|
|
return cpu->wfxt_timer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_wfxt_timer = {
|
|
|
|
.name = "cpu/wfxt-timer",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = wfxt_timer_needed,
|
|
|
|
.fields = (const VMStateField[]) {
|
|
|
|
VMSTATE_TIMER_PTR(wfxt_timer, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static bool m_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return arm_feature(env, ARM_FEATURE_M);
|
2008-05-04 17:11:44 +04:00
|
|
|
}
|
|
|
|
|
2017-09-04 17:21:52 +03:00
|
|
|
static const VMStateDescription vmstate_m_faultmask_primask = {
|
|
|
|
.name = "cpu/m/faultmask-primask",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2018-08-06 15:34:43 +03:00
|
|
|
.needed = m_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.faultmask[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.primask[M_REG_NS], ARMCPU),
|
2017-09-04 17:21:52 +03:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-15 21:29:37 +03:00
|
|
|
/* CSSELR is in a subsection because we didn't implement it previously.
|
|
|
|
* Migration from an old implementation will leave it at zero, which
|
|
|
|
* is OK since the only CPUs in the old implementation make the
|
|
|
|
* register RAZ/WI.
|
|
|
|
* Since there was no version of QEMU which implemented the CSSELR for
|
|
|
|
* just non-secure, we transfer both banks here rather than putting
|
|
|
|
* the secure banked version in the m-security subsection.
|
|
|
|
*/
|
|
|
|
static bool csselr_vmstate_validate(void *opaque, int version_id)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
return cpu->env.v7m.csselr[M_REG_NS] <= R_V7M_CSSELR_INDEX_MASK
|
|
|
|
&& cpu->env.v7m.csselr[M_REG_S] <= R_V7M_CSSELR_INDEX_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool m_csselr_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
return !arm_v7m_csselr_razwi(cpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_m_csselr = {
|
|
|
|
.name = "cpu/m/csselr",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = m_csselr_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-02-15 21:29:37 +03:00
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.csselr, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_VALIDATE("CSSELR is valid", csselr_vmstate_validate),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-15 21:29:37 +03:00
|
|
|
static const VMStateDescription vmstate_m_scr = {
|
|
|
|
.name = "cpu/m/scr",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2018-08-06 15:34:43 +03:00
|
|
|
.needed = m_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-02-15 21:29:37 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.scr[M_REG_NS], ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-15 21:29:37 +03:00
|
|
|
static const VMStateDescription vmstate_m_other_sp = {
|
|
|
|
.name = "cpu/m/other-sp",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2018-08-06 15:34:43 +03:00
|
|
|
.needed = m_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-02-15 21:29:37 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.other_sp, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-15 21:29:38 +03:00
|
|
|
static bool m_v8m_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return arm_feature(env, ARM_FEATURE_M) && arm_feature(env, ARM_FEATURE_V8);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_m_v8m = {
|
|
|
|
.name = "cpu/m/v8m",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = m_v8m_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2018-02-15 21:29:38 +03:00
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.msplim, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.psplim, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-04-29 19:35:58 +03:00
|
|
|
static const VMStateDescription vmstate_m_fp = {
|
|
|
|
.name = "cpu/m/fp",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = vfp_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2019-04-29 19:35:58 +03:00
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.fpcar, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.fpccr, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.fpdscr, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.v7m.cpacr, ARMCPU, M_REG_NUM_BANKS),
|
|
|
|
VMSTATE_UINT32(env.v7m.nsacr, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-05-20 18:28:37 +03:00
|
|
|
static bool mve_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
return cpu_isar_feature(aa32_mve, cpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_m_mve = {
|
|
|
|
.name = "cpu/m/mve",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = mve_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2021-05-20 18:28:37 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.vpr, ARMCPU),
|
2021-05-20 18:28:38 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.ltpsize, ARMCPU),
|
2021-05-20 18:28:37 +03:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2014-03-16 22:07:55 +04:00
|
|
|
static const VMStateDescription vmstate_m = {
|
2013-04-19 15:24:19 +04:00
|
|
|
.name = "cpu/m",
|
2017-06-02 13:51:48 +03:00
|
|
|
.version_id = 4,
|
|
|
|
.minimum_version_id = 4,
|
2014-09-23 16:09:54 +04:00
|
|
|
.needed = m_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.vecbase[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.basepri[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.control[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.ccr[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.cfsr[M_REG_NS], ARMCPU),
|
armv7m: add state for v7M CCR, CFSR, HFSR, DFSR, MMFAR, BFAR
Add the structure fields, VMState fields, reset code and macros for
the v7M system control registers CCR, CFSR, HFSR, DFSR, MMFAR and
BFAR.
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
Message-id: 1485285380-10565-4-git-send-email-peter.maydell@linaro.org
2017-01-27 18:20:23 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.hfsr, ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.v7m.dfsr, ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.mmfar[M_REG_NS], ARMCPU),
|
armv7m: add state for v7M CCR, CFSR, HFSR, DFSR, MMFAR, BFAR
Add the structure fields, VMState fields, reset code and macros for
the v7M system control registers CCR, CFSR, HFSR, DFSR, MMFAR and
BFAR.
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
Message-id: 1485285380-10565-4-git-send-email-peter.maydell@linaro.org
2017-01-27 18:20:23 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.bfar, ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.mpu_ctrl[M_REG_NS], ARMCPU),
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_INT32(env.v7m.exception, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
2017-09-04 17:21:52 +03:00
|
|
|
},
|
2023-12-21 06:15:43 +03:00
|
|
|
.subsections = (const VMStateDescription * const []) {
|
2017-09-04 17:21:52 +03:00
|
|
|
&vmstate_m_faultmask_primask,
|
2018-02-15 21:29:37 +03:00
|
|
|
&vmstate_m_csselr,
|
2018-02-15 21:29:37 +03:00
|
|
|
&vmstate_m_scr,
|
2018-02-15 21:29:37 +03:00
|
|
|
&vmstate_m_other_sp,
|
2018-02-15 21:29:38 +03:00
|
|
|
&vmstate_m_v8m,
|
2019-04-29 19:35:58 +03:00
|
|
|
&vmstate_m_fp,
|
2021-05-20 18:28:37 +03:00
|
|
|
&vmstate_m_mve,
|
2017-09-04 17:21:52 +03:00
|
|
|
NULL
|
2013-04-19 15:24:19 +04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool thumb2ee_needed(void *opaque)
|
2008-05-04 17:11:44 +04:00
|
|
|
{
|
2013-04-19 15:24:19 +04:00
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
return arm_feature(env, ARM_FEATURE_THUMB2EE);
|
|
|
|
}
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static const VMStateDescription vmstate_thumb2ee = {
|
|
|
|
.name = "cpu/thumb2ee",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
2014-09-23 16:09:54 +04:00
|
|
|
.needed = thumb2ee_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_UINT32(env.teecr, ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.teehbr, ARMCPU),
|
|
|
|
VMSTATE_END_OF_LIST()
|
2008-05-04 17:11:44 +04:00
|
|
|
}
|
2013-04-19 15:24:19 +04:00
|
|
|
};
|
|
|
|
|
2015-06-19 16:17:44 +03:00
|
|
|
static bool pmsav7_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
2017-06-02 13:51:47 +03:00
|
|
|
return arm_feature(env, ARM_FEATURE_PMSA) &&
|
2017-09-07 15:54:51 +03:00
|
|
|
arm_feature(env, ARM_FEATURE_V7) &&
|
|
|
|
!arm_feature(env, ARM_FEATURE_V8);
|
2015-06-19 16:17:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool pmsav7_rgnr_vmstate_validate(void *opaque, int version_id)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
2017-09-07 15:54:53 +03:00
|
|
|
return cpu->env.pmsav7.rnr[M_REG_NS] < cpu->pmsav7_dregion;
|
2015-06-19 16:17:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_pmsav7 = {
|
|
|
|
.name = "cpu/pmsav7",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = pmsav7_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2015-06-19 16:17:44 +03:00
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav7.drbar, ARMCPU, pmsav7_dregion, 0,
|
|
|
|
vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav7.drsr, ARMCPU, pmsav7_dregion, 0,
|
|
|
|
vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav7.dracr, ARMCPU, pmsav7_dregion, 0,
|
|
|
|
vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VALIDATE("rgnr is valid", pmsav7_rgnr_vmstate_validate),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-07-27 13:59:10 +03:00
|
|
|
static bool pmsav7_rnr_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
/* For R profile cores pmsav7.rnr is migrated via the cpreg
|
|
|
|
* "RGNR" definition in helper.h. For M profile we have to
|
|
|
|
* migrate it separately.
|
|
|
|
*/
|
|
|
|
return arm_feature(env, ARM_FEATURE_M);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_pmsav7_rnr = {
|
|
|
|
.name = "cpu/pmsav7-rnr",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = pmsav7_rnr_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.pmsav7.rnr[M_REG_NS], ARMCPU),
|
2017-07-27 13:59:10 +03:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-09-07 15:54:51 +03:00
|
|
|
static bool pmsav8_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return arm_feature(env, ARM_FEATURE_PMSA) &&
|
|
|
|
arm_feature(env, ARM_FEATURE_V8);
|
|
|
|
}
|
|
|
|
|
2022-12-06 13:25:02 +03:00
|
|
|
static bool pmsav8r_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return arm_feature(env, ARM_FEATURE_PMSA) &&
|
|
|
|
arm_feature(env, ARM_FEATURE_V8) &&
|
|
|
|
!arm_feature(env, ARM_FEATURE_M);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_pmsav8r = {
|
|
|
|
.name = "cpu/pmsav8/pmsav8r",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = pmsav8r_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2022-12-06 13:25:02 +03:00
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.hprbar, ARMCPU,
|
|
|
|
pmsav8r_hdregion, 0, vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.hprlar, ARMCPU,
|
|
|
|
pmsav8r_hdregion, 0, vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2017-09-07 15:54:51 +03:00
|
|
|
static const VMStateDescription vmstate_pmsav8 = {
|
|
|
|
.name = "cpu/pmsav8",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = pmsav8_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.rbar[M_REG_NS], ARMCPU, pmsav7_dregion,
|
|
|
|
0, vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.rlar[M_REG_NS], ARMCPU, pmsav7_dregion,
|
|
|
|
0, vmstate_info_uint32, uint32_t),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.pmsav8.mair0[M_REG_NS], ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.pmsav8.mair1[M_REG_NS], ARMCPU),
|
2017-09-07 15:54:51 +03:00
|
|
|
VMSTATE_END_OF_LIST()
|
2022-12-06 13:25:02 +03:00
|
|
|
},
|
2023-12-21 06:15:43 +03:00
|
|
|
.subsections = (const VMStateDescription * const []) {
|
2022-12-06 13:25:02 +03:00
|
|
|
&vmstate_pmsav8r,
|
|
|
|
NULL
|
2017-09-07 15:54:51 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-09-07 15:54:53 +03:00
|
|
|
static bool s_rnr_vmstate_validate(void *opaque, int version_id)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
return cpu->env.pmsav7.rnr[M_REG_S] < cpu->pmsav7_dregion;
|
|
|
|
}
|
|
|
|
|
2017-10-06 18:46:49 +03:00
|
|
|
static bool sau_rnr_vmstate_validate(void *opaque, int version_id)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
return cpu->env.sau.rnr < cpu->sau_sregion;
|
|
|
|
}
|
|
|
|
|
2017-09-07 15:54:52 +03:00
|
|
|
static bool m_security_needed(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
|
|
|
return arm_feature(env, ARM_FEATURE_M_SECURITY);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription vmstate_m_security = {
|
|
|
|
.name = "cpu/m-security",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.needed = m_security_needed,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.secure, ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.other_ss_msp, ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.v7m.other_ss_psp, ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.basepri[M_REG_S], ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.primask[M_REG_S], ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.faultmask[M_REG_S], ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.control[M_REG_S], ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.vecbase[M_REG_S], ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.pmsav8.mair0[M_REG_S], ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.pmsav8.mair1[M_REG_S], ARMCPU),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.rbar[M_REG_S], ARMCPU, pmsav7_dregion,
|
|
|
|
0, vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.pmsav8.rlar[M_REG_S], ARMCPU, pmsav7_dregion,
|
|
|
|
0, vmstate_info_uint32, uint32_t),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.pmsav7.rnr[M_REG_S], ARMCPU),
|
|
|
|
VMSTATE_VALIDATE("secure MPU_RNR is valid", s_rnr_vmstate_validate),
|
2017-09-07 15:54:53 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.mpu_ctrl[M_REG_S], ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.ccr[M_REG_S], ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.mmfar[M_REG_S], ARMCPU),
|
2017-09-07 15:54:54 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.cfsr[M_REG_S], ARMCPU),
|
2017-10-06 18:46:48 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.sfsr, ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.v7m.sfar, ARMCPU),
|
2017-10-06 18:46:49 +03:00
|
|
|
VMSTATE_VARRAY_UINT32(env.sau.rbar, ARMCPU, sau_sregion, 0,
|
|
|
|
vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_VARRAY_UINT32(env.sau.rlar, ARMCPU, sau_sregion, 0,
|
|
|
|
vmstate_info_uint32, uint32_t),
|
|
|
|
VMSTATE_UINT32(env.sau.rnr, ARMCPU),
|
|
|
|
VMSTATE_VALIDATE("SAU_RNR is valid", sau_rnr_vmstate_validate),
|
|
|
|
VMSTATE_UINT32(env.sau.ctrl, ARMCPU),
|
2018-02-15 21:29:37 +03:00
|
|
|
VMSTATE_UINT32(env.v7m.scr[M_REG_S], ARMCPU),
|
2018-02-15 21:29:37 +03:00
|
|
|
/* AIRCR is not secure-only, but our implementation is R/O if the
|
|
|
|
* security extension is unimplemented, so we migrate it here.
|
|
|
|
*/
|
|
|
|
VMSTATE_UINT32(env.v7m.aircr, ARMCPU),
|
2017-09-07 15:54:52 +03:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-01-19 22:00:50 +03:00
|
|
|
static int get_cpsr(QEMUFile *f, void *opaque, size_t size,
|
2018-11-14 16:29:30 +03:00
|
|
|
const VMStateField *field)
|
2013-04-19 15:24:19 +04:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
uint32_t val = qemu_get_be32(f);
|
|
|
|
|
2017-09-04 17:21:52 +03:00
|
|
|
if (arm_feature(env, ARM_FEATURE_M)) {
|
2017-09-04 17:21:52 +03:00
|
|
|
if (val & XPSR_EXCP) {
|
|
|
|
/* This is a CPSR format value from an older QEMU. (We can tell
|
|
|
|
* because values transferred in XPSR format always have zero
|
|
|
|
* for the EXCP field, and CPSR format will always have bit 4
|
|
|
|
* set in CPSR_M.) Rearrange it into XPSR format. The significant
|
|
|
|
* differences are that the T bit is not in the same place, the
|
|
|
|
* primask/faultmask info may be in the CPSR I and F bits, and
|
|
|
|
* we do not want the mode bits.
|
2017-09-07 15:54:52 +03:00
|
|
|
* We know that this cleanup happened before v8M, so there
|
|
|
|
* is no complication with banked primask/faultmask.
|
2017-09-04 17:21:52 +03:00
|
|
|
*/
|
|
|
|
uint32_t newval = val;
|
|
|
|
|
2017-09-07 15:54:52 +03:00
|
|
|
assert(!arm_feature(env, ARM_FEATURE_M_SECURITY));
|
|
|
|
|
2017-09-04 17:21:52 +03:00
|
|
|
newval &= (CPSR_NZCV | CPSR_Q | CPSR_IT | CPSR_GE);
|
|
|
|
if (val & CPSR_T) {
|
|
|
|
newval |= XPSR_T;
|
|
|
|
}
|
|
|
|
/* If the I or F bits are set then this is a migration from
|
|
|
|
* an old QEMU which still stored the M profile FAULTMASK
|
|
|
|
* and PRIMASK in env->daif. For a new QEMU, the data is
|
|
|
|
* transferred using the vmstate_m_faultmask_primask subsection.
|
|
|
|
*/
|
|
|
|
if (val & CPSR_F) {
|
2017-09-07 15:54:52 +03:00
|
|
|
env->v7m.faultmask[M_REG_NS] = 1;
|
2017-09-04 17:21:52 +03:00
|
|
|
}
|
|
|
|
if (val & CPSR_I) {
|
2017-09-07 15:54:52 +03:00
|
|
|
env->v7m.primask[M_REG_NS] = 1;
|
2017-09-04 17:21:52 +03:00
|
|
|
}
|
|
|
|
val = newval;
|
2017-09-04 17:21:52 +03:00
|
|
|
}
|
2017-09-04 17:21:52 +03:00
|
|
|
/* Ignore the low bits, they are handled by vmstate_m. */
|
|
|
|
xpsr_write(env, val, ~XPSR_EXCP);
|
|
|
|
return 0;
|
2017-09-04 17:21:52 +03:00
|
|
|
}
|
|
|
|
|
2014-12-11 15:07:53 +03:00
|
|
|
env->aarch64 = ((val & PSTATE_nRW) == 0);
|
|
|
|
|
|
|
|
if (is_a64(env)) {
|
|
|
|
pstate_write(env, val);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-02-23 18:36:43 +03:00
|
|
|
cpsr_write(env, val, 0xffffffff, CPSRWriteRaw);
|
2013-04-19 15:24:19 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2017-01-19 22:00:50 +03:00
|
|
|
static int put_cpsr(QEMUFile *f, void *opaque, size_t size,
|
2020-12-11 20:11:48 +03:00
|
|
|
const VMStateField *field, JSONWriter *vmdesc)
|
2013-04-19 15:24:19 +04:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
2014-12-11 15:07:53 +03:00
|
|
|
uint32_t val;
|
|
|
|
|
2017-09-04 17:21:52 +03:00
|
|
|
if (arm_feature(env, ARM_FEATURE_M)) {
|
|
|
|
/* The low 9 bits are v7m.exception, which is handled by vmstate_m. */
|
|
|
|
val = xpsr_read(env) & ~XPSR_EXCP;
|
|
|
|
} else if (is_a64(env)) {
|
2014-12-11 15:07:53 +03:00
|
|
|
val = pstate_read(env);
|
|
|
|
} else {
|
|
|
|
val = cpsr_read(env);
|
|
|
|
}
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2014-12-11 15:07:53 +03:00
|
|
|
qemu_put_be32(f, val);
|
2017-01-19 22:00:50 +03:00
|
|
|
return 0;
|
2013-04-19 15:24:19 +04:00
|
|
|
}
|
2008-05-04 17:11:44 +04:00
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
static const VMStateInfo vmstate_cpsr = {
|
|
|
|
.name = "cpsr",
|
|
|
|
.get = get_cpsr,
|
|
|
|
.put = put_cpsr,
|
|
|
|
};
|
|
|
|
|
2017-02-23 21:29:23 +03:00
|
|
|
static int get_power(QEMUFile *f, void *opaque, size_t size,
|
2018-11-14 16:29:30 +03:00
|
|
|
const VMStateField *field)
|
2017-02-23 21:29:23 +03:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
bool powered_off = qemu_get_byte(f);
|
|
|
|
cpu->power_state = powered_off ? PSCI_OFF : PSCI_ON;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int put_power(QEMUFile *f, void *opaque, size_t size,
|
2020-12-11 20:11:48 +03:00
|
|
|
const VMStateField *field, JSONWriter *vmdesc)
|
2017-02-23 21:29:23 +03:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
/* Migration should never happen while we transition power states */
|
|
|
|
|
|
|
|
if (cpu->power_state == PSCI_ON ||
|
|
|
|
cpu->power_state == PSCI_OFF) {
|
|
|
|
bool powered_off = (cpu->power_state == PSCI_OFF) ? true : false;
|
|
|
|
qemu_put_byte(f, powered_off);
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateInfo vmstate_powered_off = {
|
|
|
|
.name = "powered_off",
|
|
|
|
.get = get_power,
|
|
|
|
.put = put_power,
|
|
|
|
};
|
|
|
|
|
2017-09-25 14:29:12 +03:00
|
|
|
static int cpu_pre_save(void *opaque)
|
2013-06-25 21:16:07 +04:00
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
2019-01-21 13:23:14 +03:00
|
|
|
if (!kvm_enabled()) {
|
|
|
|
pmu_op_start(&cpu->env);
|
|
|
|
}
|
|
|
|
|
2013-06-25 21:16:07 +04:00
|
|
|
if (kvm_enabled()) {
|
|
|
|
if (!write_kvmstate_to_list(cpu)) {
|
|
|
|
/* This should never fail */
|
2022-05-01 08:49:48 +03:00
|
|
|
g_assert_not_reached();
|
2013-06-25 21:16:07 +04:00
|
|
|
}
|
2020-01-30 19:02:06 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* kvm_arm_cpu_pre_save() must be called after
|
|
|
|
* write_kvmstate_to_list()
|
|
|
|
*/
|
|
|
|
kvm_arm_cpu_pre_save(cpu);
|
2013-06-25 21:16:07 +04:00
|
|
|
} else {
|
2019-05-07 14:55:02 +03:00
|
|
|
if (!write_cpustate_to_list(cpu, false)) {
|
2013-06-25 21:16:07 +04:00
|
|
|
/* This should never fail. */
|
2022-05-01 08:49:48 +03:00
|
|
|
g_assert_not_reached();
|
2013-06-25 21:16:07 +04:00
|
|
|
}
|
2013-06-25 21:16:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
cpu->cpreg_vmstate_array_len = cpu->cpreg_array_len;
|
|
|
|
memcpy(cpu->cpreg_vmstate_indexes, cpu->cpreg_indexes,
|
|
|
|
cpu->cpreg_array_len * sizeof(uint64_t));
|
|
|
|
memcpy(cpu->cpreg_vmstate_values, cpu->cpreg_values,
|
|
|
|
cpu->cpreg_array_len * sizeof(uint64_t));
|
2017-09-25 14:29:12 +03:00
|
|
|
|
|
|
|
return 0;
|
2013-06-25 21:16:07 +04:00
|
|
|
}
|
|
|
|
|
2019-01-21 13:23:14 +03:00
|
|
|
static int cpu_post_save(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
|
|
|
|
if (!kvm_enabled()) {
|
|
|
|
pmu_op_finish(&cpu->env);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-11-13 13:47:59 +03:00
|
|
|
static int cpu_pre_load(void *opaque)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
|
|
|
CPUARMState *env = &cpu->env;
|
|
|
|
|
target/arm: Support migration when FPSR/FPCR won't fit in the FPSCR
To support FPSR and FPCR bits that don't exist in the AArch32 FPSCR
view of floating point control and status (such as the FEAT_AFP ones),
we need to make sure those bits can be migrated. This commit allows
that, whilst maintaining backwards and forwards migration compatibility
for CPUs where there are no such bits:
On sending:
* If either the FPCR or the FPSR include set bits that are not
visible in the AArch32 FPSCR view of floating point control/status
then we send the FPCR and FPSR as two separate fields in a new
cpu/vfp/fpcr_fpsr subsection, and we send a 0 for the old
FPSCR field in cpu/vfp
* Otherwise, we don't send the fpcr_fpsr subsection, and we send
an FPSCR-format value in cpu/vfp as we did previously
On receiving:
* if we see a non-zero FPSCR field, that is the right information
* if we see a fpcr_fpsr subsection then that has the information
* if we see neither, then FPSCR/FPCR/FPSR are all zero on the source;
cpu_pre_load() ensures the CPU state defaults to that
* if we see both, then the migration source is buggy or malicious;
either the fpcr_fpsr or the FPSCR will "win" depending which
is first in the migration stream; we don't care which that is
We make the new FPCR and FPSR on-the-wire data be 64 bits, because
architecturally these registers are that wide, and this avoids the
need to engage in further migration-compatibility contortions in
future if some new architecture revision defines bits in the high
half of either register.
(We won't ever send the new migration subsection until we add support
for a CPU feature which enables setting overlapping FPCR bits, like
FEAT_AFP.)
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-id: 20240628142347.1283015-5-peter.maydell@linaro.org
2024-06-28 17:23:42 +03:00
|
|
|
/*
|
|
|
|
* In an inbound migration where on the source FPSCR/FPSR/FPCR are 0,
|
|
|
|
* there will be no fpcr_fpsr subsection so we won't call vfp_set_fpcr()
|
|
|
|
* and vfp_set_fpsr() from get_fpcr() and get_fpsr(); also the get_fpscr()
|
|
|
|
* function will not call vfp_set_fpscr() because it will see a 0 in the
|
|
|
|
* inbound data. Ensure that in this case we have a correctly set up
|
|
|
|
* zero FPSCR/FPCR/FPSR.
|
|
|
|
*
|
|
|
|
* This is not strictly needed because FPSCR is zero out of reset, but
|
|
|
|
* it avoids the possibility of future confusing migration bugs if some
|
|
|
|
* future architecture change makes the reset value non-zero.
|
|
|
|
*/
|
|
|
|
vfp_set_fpscr(env, 0);
|
|
|
|
|
2018-11-13 13:47:59 +03:00
|
|
|
/*
|
|
|
|
* Pre-initialize irq_line_state to a value that's never valid as
|
|
|
|
* real data, so cpu_post_load() can tell whether we've seen the
|
|
|
|
* irq-line-state subsection in the incoming migration state.
|
|
|
|
*/
|
|
|
|
env->irq_line_state = UINT32_MAX;
|
|
|
|
|
2019-01-21 13:23:14 +03:00
|
|
|
if (!kvm_enabled()) {
|
2024-01-29 19:44:43 +03:00
|
|
|
pmu_op_start(env);
|
2019-01-21 13:23:14 +03:00
|
|
|
}
|
|
|
|
|
2018-11-13 13:47:59 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-06-25 21:16:07 +04:00
|
|
|
static int cpu_post_load(void *opaque, int version_id)
|
|
|
|
{
|
|
|
|
ARMCPU *cpu = opaque;
|
2018-11-13 13:47:59 +03:00
|
|
|
CPUARMState *env = &cpu->env;
|
2013-06-25 21:16:07 +04:00
|
|
|
int i, v;
|
|
|
|
|
2018-11-13 13:47:59 +03:00
|
|
|
/*
|
|
|
|
* Handle migration compatibility from old QEMU which didn't
|
|
|
|
* send the irq-line-state subsection. A QEMU without it did not
|
|
|
|
* implement the HCR_EL2.{VI,VF} bits as generating interrupts,
|
|
|
|
* so for TCG the line state matches the bits set in cs->interrupt_request.
|
|
|
|
* For KVM the line state is not stored in cs->interrupt_request
|
|
|
|
* and so this will leave irq_line_state as 0, but this is OK because
|
|
|
|
* we only need to care about it for TCG.
|
|
|
|
*/
|
|
|
|
if (env->irq_line_state == UINT32_MAX) {
|
|
|
|
CPUState *cs = CPU(cpu);
|
|
|
|
|
|
|
|
env->irq_line_state = cs->interrupt_request &
|
|
|
|
(CPU_INTERRUPT_HARD | CPU_INTERRUPT_FIQ |
|
|
|
|
CPU_INTERRUPT_VIRQ | CPU_INTERRUPT_VFIQ);
|
|
|
|
}
|
|
|
|
|
2013-06-25 21:16:07 +04:00
|
|
|
/* Update the values list from the incoming migration data.
|
|
|
|
* Anything in the incoming data which we don't know about is
|
|
|
|
* a migration failure; anything we know about but the incoming
|
|
|
|
* data doesn't specify retains its current (reset) value.
|
|
|
|
* The indexes list remains untouched -- we only inspect the
|
|
|
|
* incoming migration index list so we can match the values array
|
|
|
|
* entries with the right slots in our own values array.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (i = 0, v = 0; i < cpu->cpreg_array_len
|
|
|
|
&& v < cpu->cpreg_vmstate_array_len; i++) {
|
|
|
|
if (cpu->cpreg_vmstate_indexes[v] > cpu->cpreg_indexes[i]) {
|
|
|
|
/* register in our list but not incoming : skip it */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (cpu->cpreg_vmstate_indexes[v] < cpu->cpreg_indexes[i]) {
|
|
|
|
/* register in their list but not ours: fail migration */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* matching register, copy the value over */
|
|
|
|
cpu->cpreg_values[i] = cpu->cpreg_vmstate_values[v];
|
|
|
|
v++;
|
|
|
|
}
|
|
|
|
|
2013-06-25 21:16:07 +04:00
|
|
|
if (kvm_enabled()) {
|
2015-07-21 13:18:45 +03:00
|
|
|
if (!write_list_to_kvmstate(cpu, KVM_PUT_FULL_STATE)) {
|
2013-06-25 21:16:07 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* Note that it's OK for the TCG side not to know about
|
|
|
|
* every register in the list; KVM is authoritative if
|
|
|
|
* we're using it.
|
|
|
|
*/
|
|
|
|
write_list_to_cpustate(cpu);
|
2020-01-30 19:02:06 +03:00
|
|
|
kvm_arm_cpu_post_load(cpu);
|
2013-06-25 21:16:07 +04:00
|
|
|
} else {
|
|
|
|
if (!write_list_to_cpustate(cpu)) {
|
|
|
|
return -1;
|
|
|
|
}
|
2013-06-25 21:16:07 +04:00
|
|
|
}
|
|
|
|
|
2023-02-13 23:29:03 +03:00
|
|
|
/*
|
|
|
|
* Misaligned thumb pc is architecturally impossible. Fail the
|
|
|
|
* incoming migration. For TCG it would trigger the assert in
|
|
|
|
* thumb_tr_translate_insn().
|
|
|
|
*/
|
|
|
|
if (!is_a64(env) && env->thumb && (env->regs[15] & 1)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-02-17 23:11:27 +03:00
|
|
|
if (tcg_enabled()) {
|
|
|
|
hw_breakpoint_update_all(cpu);
|
|
|
|
hw_watchpoint_update_all(cpu);
|
|
|
|
}
|
2014-09-12 17:06:49 +04:00
|
|
|
|
2021-09-13 12:54:30 +03:00
|
|
|
/*
|
|
|
|
* TCG gen_update_fp_context() relies on the invariant that
|
|
|
|
* FPDSCR.LTPSIZE is constant 4 for M-profile with the LOB extension;
|
|
|
|
* forbid bogus incoming data with some other value.
|
|
|
|
*/
|
|
|
|
if (arm_feature(env, ARM_FEATURE_M) && cpu_isar_feature(aa32_lob, cpu)) {
|
|
|
|
if (extract32(env->v7m.fpdscr[M_REG_NS],
|
|
|
|
FPCR_LTPSIZE_SHIFT, FPCR_LTPSIZE_LENGTH) != 4 ||
|
|
|
|
extract32(env->v7m.fpdscr[M_REG_S],
|
|
|
|
FPCR_LTPSIZE_SHIFT, FPCR_LTPSIZE_LENGTH) != 4) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2021-11-03 07:03:50 +03:00
|
|
|
|
2019-01-21 13:23:14 +03:00
|
|
|
if (!kvm_enabled()) {
|
2024-01-29 19:44:43 +03:00
|
|
|
pmu_op_finish(env);
|
2019-01-21 13:23:14 +03:00
|
|
|
}
|
2023-02-17 23:11:32 +03:00
|
|
|
|
|
|
|
if (tcg_enabled()) {
|
2024-01-29 19:44:43 +03:00
|
|
|
arm_rebuild_hflags(env);
|
2023-02-17 23:11:32 +03:00
|
|
|
}
|
2019-01-21 13:23:14 +03:00
|
|
|
|
2013-06-25 21:16:07 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-04-19 15:24:19 +04:00
|
|
|
const VMStateDescription vmstate_arm_cpu = {
|
|
|
|
.name = "cpu",
|
2014-12-11 15:07:53 +03:00
|
|
|
.version_id = 22,
|
|
|
|
.minimum_version_id = 22,
|
2013-06-25 21:16:07 +04:00
|
|
|
.pre_save = cpu_pre_save,
|
2019-01-21 13:23:14 +03:00
|
|
|
.post_save = cpu_post_save,
|
2018-11-13 13:47:59 +03:00
|
|
|
.pre_load = cpu_pre_load,
|
2013-06-25 21:16:07 +04:00
|
|
|
.post_load = cpu_post_load,
|
2023-12-21 06:15:43 +03:00
|
|
|
.fields = (const VMStateField[]) {
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_UINT32_ARRAY(env.regs, ARMCPU, 16),
|
2014-12-11 15:07:53 +03:00
|
|
|
VMSTATE_UINT64_ARRAY(env.xregs, ARMCPU, 32),
|
|
|
|
VMSTATE_UINT64(env.pc, ARMCPU),
|
2013-04-19 15:24:19 +04:00
|
|
|
{
|
|
|
|
.name = "cpsr",
|
|
|
|
.version_id = 0,
|
|
|
|
.size = sizeof(uint32_t),
|
|
|
|
.info = &vmstate_cpsr,
|
|
|
|
.flags = VMS_SINGLE,
|
|
|
|
.offset = 0,
|
|
|
|
},
|
|
|
|
VMSTATE_UINT32(env.spsr, ARMCPU),
|
2014-05-27 20:09:52 +04:00
|
|
|
VMSTATE_UINT64_ARRAY(env.banked_spsr, ARMCPU, 8),
|
2014-10-24 15:19:14 +04:00
|
|
|
VMSTATE_UINT32_ARRAY(env.banked_r13, ARMCPU, 8),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.banked_r14, ARMCPU, 8),
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_UINT32_ARRAY(env.usr_regs, ARMCPU, 5),
|
|
|
|
VMSTATE_UINT32_ARRAY(env.fiq_regs, ARMCPU, 5),
|
2014-05-27 20:09:52 +04:00
|
|
|
VMSTATE_UINT64_ARRAY(env.elr_el, ARMCPU, 4),
|
2014-05-27 20:09:52 +04:00
|
|
|
VMSTATE_UINT64_ARRAY(env.sp_el, ARMCPU, 4),
|
2013-06-25 21:16:07 +04:00
|
|
|
/* The length-check must come before the arrays to avoid
|
|
|
|
* incoming data possibly overflowing the array.
|
|
|
|
*/
|
2014-04-03 20:52:21 +04:00
|
|
|
VMSTATE_INT32_POSITIVE_LE(cpreg_vmstate_array_len, ARMCPU),
|
2013-06-25 21:16:07 +04:00
|
|
|
VMSTATE_VARRAY_INT32(cpreg_vmstate_indexes, ARMCPU,
|
|
|
|
cpreg_vmstate_array_len,
|
|
|
|
0, vmstate_info_uint64, uint64_t),
|
|
|
|
VMSTATE_VARRAY_INT32(cpreg_vmstate_values, ARMCPU,
|
|
|
|
cpreg_vmstate_array_len,
|
|
|
|
0, vmstate_info_uint64, uint64_t),
|
2014-01-05 02:15:47 +04:00
|
|
|
VMSTATE_UINT64(env.exclusive_addr, ARMCPU),
|
|
|
|
VMSTATE_UINT64(env.exclusive_val, ARMCPU),
|
|
|
|
VMSTATE_UINT64(env.exclusive_high, ARMCPU),
|
2021-02-03 19:13:40 +03:00
|
|
|
VMSTATE_UNUSED(sizeof(uint64_t)),
|
target-arm: Define exception record for AArch64 exceptions
For AArch32 exceptions, the only information provided about
the cause of an exception is the individual exception type (data
abort, undef, etc), which we store in cs->exception_index. For
AArch64, the CPU provides much more detail about the cause of
the exception, which can be found in the syndrome register.
Create a set of fields in CPUARMState which must be filled in
whenever an exception is raised, so that exception entry can
correctly fill in the syndrome register for the guest.
This includes the information which in AArch32 appears in
the DFAR and IFAR (fault address registers) and the DFSR
and IFSR (fault status registers) for data aborts and
prefetch aborts, since if we end up taking the MMU fault
to AArch64 rather than AArch32 this will need to end up
in different system registers.
This patch does a refactoring which moves the setting of the
AArch32 DFAR/DFSR/IFAR/IFSR from the point where the exception
is raised to the point where it is taken. (This is no change
for cores with an MMU, retains the existing clearly incorrect
behaviour for ARM946 of trashing the MP access permissions
registers which share the c5_data and c5_insn state fields,
and has no effect for v7M because we don't implement its
MPU fault status or address registers.)
As a side effect of the cleanup we fix a bug in the AArch64
linux-user mode code where we were passing a 64 bit fault
address through the 32 bit c6_data/c6_insn fields: it now
goes via the always-64-bit exception.vaddress.
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Peter Crosthwaite <peter.crosthwaite@xilinx.com>
2014-04-15 22:18:38 +04:00
|
|
|
VMSTATE_UINT32(env.exception.syndrome, ARMCPU),
|
|
|
|
VMSTATE_UINT32(env.exception.fsr, ARMCPU),
|
|
|
|
VMSTATE_UINT64(env.exception.vaddress, ARMCPU),
|
2015-01-08 12:18:59 +03:00
|
|
|
VMSTATE_TIMER_PTR(gt_timer[GTIMER_PHYS], ARMCPU),
|
|
|
|
VMSTATE_TIMER_PTR(gt_timer[GTIMER_VIRT], ARMCPU),
|
2017-02-23 21:29:23 +03:00
|
|
|
{
|
|
|
|
.name = "power_state",
|
|
|
|
.version_id = 0,
|
|
|
|
.size = sizeof(bool),
|
|
|
|
.info = &vmstate_powered_off,
|
|
|
|
.flags = VMS_SINGLE,
|
|
|
|
.offset = 0,
|
|
|
|
},
|
2013-04-19 15:24:19 +04:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
},
|
2023-12-21 06:15:43 +03:00
|
|
|
.subsections = (const VMStateDescription * const []) {
|
2014-09-23 16:09:54 +04:00
|
|
|
&vmstate_vfp,
|
|
|
|
&vmstate_iwmmxt,
|
|
|
|
&vmstate_m,
|
|
|
|
&vmstate_thumb2ee,
|
2017-07-27 13:59:10 +03:00
|
|
|
/* pmsav7_rnr must come before pmsav7 so that we have the
|
|
|
|
* region number before we test it in the VMSTATE_VALIDATE
|
|
|
|
* in vmstate_pmsav7.
|
|
|
|
*/
|
|
|
|
&vmstate_pmsav7_rnr,
|
2015-06-19 16:17:44 +03:00
|
|
|
&vmstate_pmsav7,
|
2017-09-07 15:54:51 +03:00
|
|
|
&vmstate_pmsav8,
|
2017-09-07 15:54:52 +03:00
|
|
|
&vmstate_m_security,
|
2018-01-23 06:53:47 +03:00
|
|
|
#ifdef TARGET_AARCH64
|
|
|
|
&vmstate_sve,
|
2022-06-20 20:51:53 +03:00
|
|
|
&vmstate_za,
|
2018-01-23 06:53:47 +03:00
|
|
|
#endif
|
2018-10-24 09:50:16 +03:00
|
|
|
&vmstate_serror,
|
2018-11-13 13:47:59 +03:00
|
|
|
&vmstate_irq_line_state,
|
2024-04-30 17:00:35 +03:00
|
|
|
&vmstate_wfxt_timer,
|
2014-09-23 16:09:54 +04:00
|
|
|
NULL
|
2009-07-30 16:33:47 +04:00
|
|
|
}
|
2013-04-19 15:24:19 +04:00
|
|
|
};
|