mirror of https://gitee.com/openkylin/qemu.git
target-arm queue:
* aspeed: code cleanup to use unimplemented_device * preparatory work for 'raspi3' RaspberryPi 3 machine model * more SVE prep work * v8M: add minor missing registers * v7M: fix bug where we weren't migrating v7m.other_sp * v7M: fix bugs in handling of interrupt registers for external interrupts beyond 32 -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIcBAABCAAGBQJahdLAAAoJEDwlJe0UNgzezlsQAIe4csOMrr4BTXlZF6tCJLz/ iF4j6ABmsLIAj8kn3KmLWLJ8AZ/hYHcRc9YygRKwJty+cXu2AVAHWXPlfPpv5lne zCbbXAw9QGwCVmPhBMFTX5qntbt8sMk+2o6A4QE/PqtmO4u8a9O7belgKrHuiCCm mn+AIH4Eo1cv58Tiijv+mxWk60mnooEwKFl7cqMYznnV0XQ4oLCq3jaW7YZL7hFF nOSo9lz5NCPwbAWNRvlr/dtY7ZhjkL5CimdzrPVR5Qn2FvnD7IgsoIyOb/y5rycf Bd2WijN1L6AE+h952A4qu0zDNxnyDCW18F54eB7IUrZqC7BRfXpPXs663UOeyQzX LtlRPQWtXIc6lDy4JznDpB6YXh7nfrQA5DrWF2CwNVXB2gEadna2gHpqyz60FoVZ 2I8QDX7hqTiE8//z81ykp6591Z1xo1OlQbLCUtJUOoZsZ8Pn/4p3amVbOCCPAbe6 IayrfLVltpPhrzfJXxd9ZpxsHqluwgo3ZLkTfv7hywIsirhMLWtztEUm2IRV/g6D O/NktDNmlfgpOVFPw60HpzkTe4CM10CVBMyTwsbQlsuw+3TFiE3Q1XRbxplG8CAs WhRTMsxA1V/x0Mj4zbMDbjKDsCEDQqj9f7KQgFnZnDhM2x+2jY98UGQHq0aoMlZI K4lCXS0x6MiJDJThjEm+ =U8z2 -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/pmaydell/tags/pull-target-arm-20180215-1' into staging target-arm queue: * aspeed: code cleanup to use unimplemented_device * preparatory work for 'raspi3' RaspberryPi 3 machine model * more SVE prep work * v8M: add minor missing registers * v7M: fix bug where we weren't migrating v7m.other_sp * v7M: fix bugs in handling of interrupt registers for external interrupts beyond 32 # gpg: Signature made Thu 15 Feb 2018 18:34:40 GMT # gpg: using RSA key 3C2525ED14360CDE # gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>" # gpg: aka "Peter Maydell <pmaydell@gmail.com>" # gpg: aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>" # Primary key fingerprint: E1A5 C593 CD41 9DE2 8E83 15CF 3C25 25ED 1436 0CDE * remotes/pmaydell/tags/pull-target-arm-20180215-1: raspi: Raspberry Pi 3 support bcm2836: Make CPU type configurable target/arm: Implement v8M MSPLIM and PSPLIM registers target/arm: Migrate v7m.other_sp target/arm: Add AIRCR to vmstate struct hw/intc/armv7m_nvic: Fix byte-to-interrupt number conversions target/arm: Implement writing to CONTROL_NS for v8M hw/intc/armv7m_nvic: Implement SCR hw/intc/armv7m_nvic: Implement cache ID registers hw/intc/armv7m_nvic: Implement v8M CPPWR register hw/intc/armv7m_nvic: Implement M profile cache maintenance ops hw/intc/armv7m_nvic: Fix ICSR PENDNMISET/CLR handling hw/intc/armv7m_nvic: Don't hardcode M profile ID registers in NVIC target/arm: Handle SVE registers when using clear_vec_high target/arm: Enforce access to ZCR_EL at translation target/arm: Suppress TB end for FPCR/FPSR target/arm: Enforce FP access to FPCR/FPSR target/arm: Remove ARM_CP_64BIT from ZCR_EL registers hw/arm/aspeed: simplify using the 'unimplemented device' for aspeed_soc.io hw/arm/aspeed: directly map the serial device to the system address space Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
cc5a0ae03e
|
@ -15,6 +15,7 @@
|
|||
#include "qemu-common.h"
|
||||
#include "cpu.h"
|
||||
#include "exec/address-spaces.h"
|
||||
#include "hw/misc/unimp.h"
|
||||
#include "hw/arm/aspeed_soc.h"
|
||||
#include "hw/char/serial.h"
|
||||
#include "qemu/log.h"
|
||||
|
@ -99,31 +100,6 @@ static const AspeedSoCInfo aspeed_socs[] = {
|
|||
},
|
||||
};
|
||||
|
||||
/*
|
||||
* IO handlers: simply catch any reads/writes to IO addresses that aren't
|
||||
* handled by a device mapping.
|
||||
*/
|
||||
|
||||
static uint64_t aspeed_soc_io_read(void *p, hwaddr offset, unsigned size)
|
||||
{
|
||||
qemu_log_mask(LOG_UNIMP, "%s: 0x%" HWADDR_PRIx " [%u]\n",
|
||||
__func__, offset, size);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void aspeed_soc_io_write(void *opaque, hwaddr offset, uint64_t value,
|
||||
unsigned size)
|
||||
{
|
||||
qemu_log_mask(LOG_UNIMP, "%s: 0x%" HWADDR_PRIx " <- 0x%" PRIx64 " [%u]\n",
|
||||
__func__, offset, value, size);
|
||||
}
|
||||
|
||||
static const MemoryRegionOps aspeed_soc_io_ops = {
|
||||
.read = aspeed_soc_io_read,
|
||||
.write = aspeed_soc_io_write,
|
||||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||||
};
|
||||
|
||||
static void aspeed_soc_init(Object *obj)
|
||||
{
|
||||
AspeedSoCState *s = ASPEED_SOC(obj);
|
||||
|
@ -199,10 +175,8 @@ static void aspeed_soc_realize(DeviceState *dev, Error **errp)
|
|||
Error *err = NULL, *local_err = NULL;
|
||||
|
||||
/* IO space */
|
||||
memory_region_init_io(&s->iomem, NULL, &aspeed_soc_io_ops, NULL,
|
||||
"aspeed_soc.io", ASPEED_SOC_IOMEM_SIZE);
|
||||
memory_region_add_subregion_overlap(get_system_memory(),
|
||||
ASPEED_SOC_IOMEM_BASE, &s->iomem, -1);
|
||||
create_unimplemented_device("aspeed_soc.io",
|
||||
ASPEED_SOC_IOMEM_BASE, ASPEED_SOC_IOMEM_SIZE);
|
||||
|
||||
/* CPU */
|
||||
object_property_set_bool(OBJECT(&s->cpu), true, "realized", &err);
|
||||
|
@ -257,7 +231,8 @@ static void aspeed_soc_realize(DeviceState *dev, Error **errp)
|
|||
/* UART - attach an 8250 to the IO space as our UART5 */
|
||||
if (serial_hds[0]) {
|
||||
qemu_irq uart5 = qdev_get_gpio_in(DEVICE(&s->vic), uart_irqs[4]);
|
||||
serial_mm_init(&s->iomem, ASPEED_SOC_UART_5_BASE, 2,
|
||||
serial_mm_init(get_system_memory(),
|
||||
ASPEED_SOC_IOMEM_BASE + ASPEED_SOC_UART_5_BASE, 2,
|
||||
uart5, 38400, serial_hds[0], DEVICE_LITTLE_ENDIAN);
|
||||
}
|
||||
|
||||
|
|
|
@ -26,14 +26,6 @@
|
|||
static void bcm2836_init(Object *obj)
|
||||
{
|
||||
BCM2836State *s = BCM2836(obj);
|
||||
int n;
|
||||
|
||||
for (n = 0; n < BCM2836_NCPUS; n++) {
|
||||
object_initialize(&s->cpus[n], sizeof(s->cpus[n]),
|
||||
"cortex-a15-" TYPE_ARM_CPU);
|
||||
object_property_add_child(obj, "cpu[*]", OBJECT(&s->cpus[n]),
|
||||
&error_abort);
|
||||
}
|
||||
|
||||
object_initialize(&s->control, sizeof(s->control), TYPE_BCM2836_CONTROL);
|
||||
object_property_add_child(obj, "control", OBJECT(&s->control), NULL);
|
||||
|
@ -59,6 +51,14 @@ static void bcm2836_realize(DeviceState *dev, Error **errp)
|
|||
|
||||
/* common peripherals from bcm2835 */
|
||||
|
||||
obj = OBJECT(dev);
|
||||
for (n = 0; n < BCM2836_NCPUS; n++) {
|
||||
object_initialize(&s->cpus[n], sizeof(s->cpus[n]),
|
||||
s->cpu_type);
|
||||
object_property_add_child(obj, "cpu[*]", OBJECT(&s->cpus[n]),
|
||||
&error_abort);
|
||||
}
|
||||
|
||||
obj = object_property_get_link(OBJECT(dev), "ram", &err);
|
||||
if (obj == NULL) {
|
||||
error_setg(errp, "%s: required ram link not found: %s",
|
||||
|
@ -150,6 +150,7 @@ static void bcm2836_realize(DeviceState *dev, Error **errp)
|
|||
}
|
||||
|
||||
static Property bcm2836_props[] = {
|
||||
DEFINE_PROP_STRING("cpu-type", BCM2836State, cpu_type),
|
||||
DEFINE_PROP_UINT32("enabled-cpus", BCM2836State, enabled_cpus, BCM2836_NCPUS),
|
||||
DEFINE_PROP_END_OF_LIST()
|
||||
};
|
||||
|
|
|
@ -5,6 +5,9 @@
|
|||
* Rasperry Pi 2 emulation Copyright (c) 2015, Microsoft
|
||||
* Written by Andrew Baumann
|
||||
*
|
||||
* Raspberry Pi 3 emulation Copyright (c) 2018 Zoltán Baldaszti
|
||||
* Upstream code cleanup (c) 2018 Pekka Enberg
|
||||
*
|
||||
* This code is licensed under the GNU GPLv2 and later.
|
||||
*/
|
||||
|
||||
|
@ -22,10 +25,11 @@
|
|||
#define SMPBOOT_ADDR 0x300 /* this should leave enough space for ATAGS */
|
||||
#define MVBAR_ADDR 0x400 /* secure vectors */
|
||||
#define BOARDSETUP_ADDR (MVBAR_ADDR + 0x20) /* board setup code */
|
||||
#define FIRMWARE_ADDR 0x8000 /* Pi loads kernel.img here by default */
|
||||
#define FIRMWARE_ADDR_2 0x8000 /* Pi 2 loads kernel.img here by default */
|
||||
#define FIRMWARE_ADDR_3 0x80000 /* Pi 3 loads kernel.img here by default */
|
||||
|
||||
/* Table of Linux board IDs for different Pi versions */
|
||||
static const int raspi_boardid[] = {[1] = 0xc42, [2] = 0xc43};
|
||||
static const int raspi_boardid[] = {[1] = 0xc42, [2] = 0xc43, [3] = 0xc44};
|
||||
|
||||
typedef struct RasPiState {
|
||||
BCM2836State soc;
|
||||
|
@ -83,8 +87,8 @@ static void setup_boot(MachineState *machine, int version, size_t ram_size)
|
|||
binfo.secure_board_setup = true;
|
||||
binfo.secure_boot = true;
|
||||
|
||||
/* Pi2 requires SMP setup */
|
||||
if (version == 2) {
|
||||
/* Pi2 and Pi3 requires SMP setup */
|
||||
if (version >= 2) {
|
||||
binfo.smp_loader_start = SMPBOOT_ADDR;
|
||||
binfo.write_secondary_boot = write_smpboot;
|
||||
binfo.secondary_cpu_reset_hook = reset_secondary;
|
||||
|
@ -94,15 +98,16 @@ static void setup_boot(MachineState *machine, int version, size_t ram_size)
|
|||
* the normal Linux boot process
|
||||
*/
|
||||
if (machine->firmware) {
|
||||
hwaddr firmware_addr = version == 3 ? FIRMWARE_ADDR_3 : FIRMWARE_ADDR_2;
|
||||
/* load the firmware image (typically kernel.img) */
|
||||
r = load_image_targphys(machine->firmware, FIRMWARE_ADDR,
|
||||
ram_size - FIRMWARE_ADDR);
|
||||
r = load_image_targphys(machine->firmware, firmware_addr,
|
||||
ram_size - firmware_addr);
|
||||
if (r < 0) {
|
||||
error_report("Failed to load firmware from %s", machine->firmware);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
binfo.entry = FIRMWARE_ADDR;
|
||||
binfo.entry = firmware_addr;
|
||||
binfo.firmware_loaded = true;
|
||||
} else {
|
||||
binfo.kernel_filename = machine->kernel_filename;
|
||||
|
@ -113,7 +118,7 @@ static void setup_boot(MachineState *machine, int version, size_t ram_size)
|
|||
arm_load_kernel(ARM_CPU(first_cpu), &binfo);
|
||||
}
|
||||
|
||||
static void raspi2_init(MachineState *machine)
|
||||
static void raspi_init(MachineState *machine, int version)
|
||||
{
|
||||
RasPiState *s = g_new0(RasPiState, 1);
|
||||
uint32_t vcram_size;
|
||||
|
@ -135,9 +140,12 @@ static void raspi2_init(MachineState *machine)
|
|||
/* Setup the SOC */
|
||||
object_property_add_const_link(OBJECT(&s->soc), "ram", OBJECT(&s->ram),
|
||||
&error_abort);
|
||||
object_property_set_str(OBJECT(&s->soc), machine->cpu_type, "cpu-type",
|
||||
&error_abort);
|
||||
object_property_set_int(OBJECT(&s->soc), smp_cpus, "enabled-cpus",
|
||||
&error_abort);
|
||||
object_property_set_int(OBJECT(&s->soc), 0xa21041, "board-rev",
|
||||
int board_rev = version == 3 ? 0xa02082 : 0xa21041;
|
||||
object_property_set_int(OBJECT(&s->soc), board_rev, "board-rev",
|
||||
&error_abort);
|
||||
object_property_set_bool(OBJECT(&s->soc), true, "realized", &error_abort);
|
||||
|
||||
|
@ -155,7 +163,12 @@ static void raspi2_init(MachineState *machine)
|
|||
|
||||
vcram_size = object_property_get_uint(OBJECT(&s->soc), "vcram-size",
|
||||
&error_abort);
|
||||
setup_boot(machine, 2, machine->ram_size - vcram_size);
|
||||
setup_boot(machine, version, machine->ram_size - vcram_size);
|
||||
}
|
||||
|
||||
static void raspi2_init(MachineState *machine)
|
||||
{
|
||||
raspi_init(machine, 2);
|
||||
}
|
||||
|
||||
static void raspi2_machine_init(MachineClass *mc)
|
||||
|
@ -166,6 +179,7 @@ static void raspi2_machine_init(MachineClass *mc)
|
|||
mc->no_parallel = 1;
|
||||
mc->no_floppy = 1;
|
||||
mc->no_cdrom = 1;
|
||||
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-a15");
|
||||
mc->max_cpus = BCM2836_NCPUS;
|
||||
mc->min_cpus = BCM2836_NCPUS;
|
||||
mc->default_cpus = BCM2836_NCPUS;
|
||||
|
|
|
@ -776,6 +776,14 @@ static uint32_t nvic_readl(NVICState *s, uint32_t offset, MemTxAttrs attrs)
|
|||
switch (offset) {
|
||||
case 4: /* Interrupt Control Type. */
|
||||
return ((s->num_irq - NVIC_FIRST_IRQ) / 32) - 1;
|
||||
case 0xc: /* CPPWR */
|
||||
if (!arm_feature(&cpu->env, ARM_FEATURE_V8)) {
|
||||
goto bad_offset;
|
||||
}
|
||||
/* We make the IMPDEF choice that nothing can ever go into a
|
||||
* non-retentive power state, which allows us to RAZ/WI this.
|
||||
*/
|
||||
return 0;
|
||||
case 0x380 ... 0x3bf: /* NVIC_ITNS<n> */
|
||||
{
|
||||
int startvec = 8 * (offset - 0x380) + NVIC_FIRST_IRQ;
|
||||
|
@ -830,8 +838,8 @@ static uint32_t nvic_readl(NVICState *s, uint32_t offset, MemTxAttrs attrs)
|
|||
}
|
||||
}
|
||||
/* NMIPENDSET */
|
||||
if ((cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK) &&
|
||||
s->vectors[ARMV7M_EXCP_NMI].pending) {
|
||||
if ((attrs.secure || (cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK))
|
||||
&& s->vectors[ARMV7M_EXCP_NMI].pending) {
|
||||
val |= (1 << 31);
|
||||
}
|
||||
/* ISRPREEMPT: RES0 when halting debug not implemented */
|
||||
|
@ -855,8 +863,7 @@ static uint32_t nvic_readl(NVICState *s, uint32_t offset, MemTxAttrs attrs)
|
|||
}
|
||||
return val;
|
||||
case 0xd10: /* System Control. */
|
||||
/* TODO: Implement SLEEPONEXIT. */
|
||||
return 0;
|
||||
return cpu->env.v7m.scr[attrs.secure];
|
||||
case 0xd14: /* Configuration Control. */
|
||||
/* The BFHFNMIGN bit is the only non-banked bit; we
|
||||
* keep it in the non-secure copy of the register.
|
||||
|
@ -990,31 +997,44 @@ static uint32_t nvic_readl(NVICState *s, uint32_t offset, MemTxAttrs attrs)
|
|||
"Aux Fault status registers unimplemented\n");
|
||||
return 0;
|
||||
case 0xd40: /* PFR0. */
|
||||
return 0x00000030;
|
||||
case 0xd44: /* PRF1. */
|
||||
return 0x00000200;
|
||||
return cpu->id_pfr0;
|
||||
case 0xd44: /* PFR1. */
|
||||
return cpu->id_pfr1;
|
||||
case 0xd48: /* DFR0. */
|
||||
return 0x00100000;
|
||||
return cpu->id_dfr0;
|
||||
case 0xd4c: /* AFR0. */
|
||||
return 0x00000000;
|
||||
return cpu->id_afr0;
|
||||
case 0xd50: /* MMFR0. */
|
||||
return 0x00000030;
|
||||
return cpu->id_mmfr0;
|
||||
case 0xd54: /* MMFR1. */
|
||||
return 0x00000000;
|
||||
return cpu->id_mmfr1;
|
||||
case 0xd58: /* MMFR2. */
|
||||
return 0x00000000;
|
||||
return cpu->id_mmfr2;
|
||||
case 0xd5c: /* MMFR3. */
|
||||
return 0x00000000;
|
||||
return cpu->id_mmfr3;
|
||||
case 0xd60: /* ISAR0. */
|
||||
return 0x01141110;
|
||||
return cpu->id_isar0;
|
||||
case 0xd64: /* ISAR1. */
|
||||
return 0x02111000;
|
||||
return cpu->id_isar1;
|
||||
case 0xd68: /* ISAR2. */
|
||||
return 0x21112231;
|
||||
return cpu->id_isar2;
|
||||
case 0xd6c: /* ISAR3. */
|
||||
return 0x01111110;
|
||||
return cpu->id_isar3;
|
||||
case 0xd70: /* ISAR4. */
|
||||
return 0x01310102;
|
||||
return cpu->id_isar4;
|
||||
case 0xd74: /* ISAR5. */
|
||||
return cpu->id_isar5;
|
||||
case 0xd78: /* CLIDR */
|
||||
return cpu->clidr;
|
||||
case 0xd7c: /* CTR */
|
||||
return cpu->ctr;
|
||||
case 0xd80: /* CSSIDR */
|
||||
{
|
||||
int idx = cpu->env.v7m.csselr[attrs.secure] & R_V7M_CSSELR_INDEX_MASK;
|
||||
return cpu->ccsidr[idx];
|
||||
}
|
||||
case 0xd84: /* CSSELR */
|
||||
return cpu->env.v7m.csselr[attrs.secure];
|
||||
/* TODO: Implement debug registers. */
|
||||
case 0xd90: /* MPU_TYPE */
|
||||
/* Unified MPU; if the MPU is not present this value is zero */
|
||||
|
@ -1173,6 +1193,12 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
|
|||
ARMCPU *cpu = s->cpu;
|
||||
|
||||
switch (offset) {
|
||||
case 0xc: /* CPPWR */
|
||||
if (!arm_feature(&cpu->env, ARM_FEATURE_V8)) {
|
||||
goto bad_offset;
|
||||
}
|
||||
/* Make the IMPDEF choice to RAZ/WI this. */
|
||||
break;
|
||||
case 0x380 ... 0x3bf: /* NVIC_ITNS<n> */
|
||||
{
|
||||
int startvec = 8 * (offset - 0x380) + NVIC_FIRST_IRQ;
|
||||
|
@ -1191,7 +1217,7 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
|
|||
break;
|
||||
}
|
||||
case 0xd04: /* Interrupt Control State (ICSR) */
|
||||
if (cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK) {
|
||||
if (attrs.secure || cpu->env.v7m.aircr & R_V7M_AIRCR_BFHFNMINS_MASK) {
|
||||
if (value & (1 << 31)) {
|
||||
armv7m_nvic_set_pending(s, ARMV7M_EXCP_NMI, false);
|
||||
} else if (value & (1 << 30) &&
|
||||
|
@ -1258,8 +1284,13 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
|
|||
}
|
||||
break;
|
||||
case 0xd10: /* System Control. */
|
||||
/* TODO: Implement control registers. */
|
||||
qemu_log_mask(LOG_UNIMP, "NVIC: SCR unimplemented\n");
|
||||
/* We don't implement deep-sleep so these bits are RAZ/WI.
|
||||
* The other bits in the register are banked.
|
||||
* QEMU's implementation ignores SEVONPEND and SLEEPONEXIT, which
|
||||
* is architecturally permitted.
|
||||
*/
|
||||
value &= ~(R_V7M_SCR_SLEEPDEEP_MASK | R_V7M_SCR_SLEEPDEEPS_MASK);
|
||||
cpu->env.v7m.scr[attrs.secure] = value;
|
||||
break;
|
||||
case 0xd14: /* Configuration Control. */
|
||||
/* Enforce RAZ/WI on reserved and must-RAZ/WI bits */
|
||||
|
@ -1369,6 +1400,11 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
|
|||
qemu_log_mask(LOG_UNIMP,
|
||||
"NVIC: Aux fault status registers unimplemented\n");
|
||||
break;
|
||||
case 0xd84: /* CSSELR */
|
||||
if (!arm_v7m_csselr_razwi(cpu)) {
|
||||
cpu->env.v7m.csselr[attrs.secure] = value & R_V7M_CSSELR_INDEX_MASK;
|
||||
}
|
||||
break;
|
||||
case 0xd90: /* MPU_TYPE */
|
||||
return; /* RO */
|
||||
case 0xd94: /* MPU_CTRL */
|
||||
|
@ -1592,6 +1628,18 @@ static void nvic_writel(NVICState *s, uint32_t offset, uint32_t value,
|
|||
}
|
||||
break;
|
||||
}
|
||||
case 0xf50: /* ICIALLU */
|
||||
case 0xf58: /* ICIMVAU */
|
||||
case 0xf5c: /* DCIMVAC */
|
||||
case 0xf60: /* DCISW */
|
||||
case 0xf64: /* DCCMVAU */
|
||||
case 0xf68: /* DCCMVAC */
|
||||
case 0xf6c: /* DCCSW */
|
||||
case 0xf70: /* DCCIMVAC */
|
||||
case 0xf74: /* DCCISW */
|
||||
case 0xf78: /* BPIALL */
|
||||
/* Cache and branch predictor maintenance: for QEMU these always NOP */
|
||||
break;
|
||||
default:
|
||||
bad_offset:
|
||||
qemu_log_mask(LOG_GUEST_ERROR,
|
||||
|
@ -1676,7 +1724,7 @@ static MemTxResult nvic_sysreg_read(void *opaque, hwaddr addr,
|
|||
/* fall through */
|
||||
case 0x180 ... 0x1bf: /* NVIC Clear enable */
|
||||
val = 0;
|
||||
startvec = offset - 0x180 + NVIC_FIRST_IRQ; /* vector # */
|
||||
startvec = 8 * (offset - 0x180) + NVIC_FIRST_IRQ; /* vector # */
|
||||
|
||||
for (i = 0, end = size * 8; i < end && startvec + i < s->num_irq; i++) {
|
||||
if (s->vectors[startvec + i].enabled &&
|
||||
|
@ -1690,7 +1738,7 @@ static MemTxResult nvic_sysreg_read(void *opaque, hwaddr addr,
|
|||
/* fall through */
|
||||
case 0x280 ... 0x2bf: /* NVIC Clear pend */
|
||||
val = 0;
|
||||
startvec = offset - 0x280 + NVIC_FIRST_IRQ; /* vector # */
|
||||
startvec = 8 * (offset - 0x280) + NVIC_FIRST_IRQ; /* vector # */
|
||||
for (i = 0, end = size * 8; i < end && startvec + i < s->num_irq; i++) {
|
||||
if (s->vectors[startvec + i].pending &&
|
||||
(attrs.secure || s->itns[startvec + i])) {
|
||||
|
@ -1700,7 +1748,7 @@ static MemTxResult nvic_sysreg_read(void *opaque, hwaddr addr,
|
|||
break;
|
||||
case 0x300 ... 0x33f: /* NVIC Active */
|
||||
val = 0;
|
||||
startvec = offset - 0x300 + NVIC_FIRST_IRQ; /* vector # */
|
||||
startvec = 8 * (offset - 0x300) + NVIC_FIRST_IRQ; /* vector # */
|
||||
|
||||
for (i = 0, end = size * 8; i < end && startvec + i < s->num_irq; i++) {
|
||||
if (s->vectors[startvec + i].active &&
|
||||
|
@ -1815,7 +1863,7 @@ static MemTxResult nvic_sysreg_write(void *opaque, hwaddr addr,
|
|||
case 0x300 ... 0x33f: /* NVIC Active */
|
||||
return MEMTX_OK; /* R/O */
|
||||
case 0x400 ... 0x5ef: /* NVIC Priority */
|
||||
startvec = 8 * (offset - 0x400) + NVIC_FIRST_IRQ; /* vector # */
|
||||
startvec = (offset - 0x400) + NVIC_FIRST_IRQ; /* vector # */
|
||||
|
||||
for (i = 0; i < size && startvec + i < s->num_irq; i++) {
|
||||
if (attrs.secure || s->itns[startvec + i]) {
|
||||
|
|
|
@ -31,7 +31,6 @@ typedef struct AspeedSoCState {
|
|||
|
||||
/*< public >*/
|
||||
ARMCPU cpu;
|
||||
MemoryRegion iomem;
|
||||
MemoryRegion sram;
|
||||
AspeedVICState vic;
|
||||
AspeedTimerCtrlState timerctrl;
|
||||
|
|
|
@ -25,6 +25,7 @@ typedef struct BCM2836State {
|
|||
DeviceState parent_obj;
|
||||
/*< public >*/
|
||||
|
||||
char *cpu_type;
|
||||
uint32_t enabled_cpus;
|
||||
|
||||
ARMCPU cpus[BCM2836_NCPUS];
|
||||
|
|
|
@ -1146,6 +1146,20 @@ static void cortex_m3_initfn(Object *obj)
|
|||
set_feature(&cpu->env, ARM_FEATURE_M);
|
||||
cpu->midr = 0x410fc231;
|
||||
cpu->pmsav7_dregion = 8;
|
||||
cpu->id_pfr0 = 0x00000030;
|
||||
cpu->id_pfr1 = 0x00000200;
|
||||
cpu->id_dfr0 = 0x00100000;
|
||||
cpu->id_afr0 = 0x00000000;
|
||||
cpu->id_mmfr0 = 0x00000030;
|
||||
cpu->id_mmfr1 = 0x00000000;
|
||||
cpu->id_mmfr2 = 0x00000000;
|
||||
cpu->id_mmfr3 = 0x00000000;
|
||||
cpu->id_isar0 = 0x01141110;
|
||||
cpu->id_isar1 = 0x02111000;
|
||||
cpu->id_isar2 = 0x21112231;
|
||||
cpu->id_isar3 = 0x01111110;
|
||||
cpu->id_isar4 = 0x01310102;
|
||||
cpu->id_isar5 = 0x00000000;
|
||||
}
|
||||
|
||||
static void cortex_m4_initfn(Object *obj)
|
||||
|
@ -1157,6 +1171,20 @@ static void cortex_m4_initfn(Object *obj)
|
|||
set_feature(&cpu->env, ARM_FEATURE_THUMB_DSP);
|
||||
cpu->midr = 0x410fc240; /* r0p0 */
|
||||
cpu->pmsav7_dregion = 8;
|
||||
cpu->id_pfr0 = 0x00000030;
|
||||
cpu->id_pfr1 = 0x00000200;
|
||||
cpu->id_dfr0 = 0x00100000;
|
||||
cpu->id_afr0 = 0x00000000;
|
||||
cpu->id_mmfr0 = 0x00000030;
|
||||
cpu->id_mmfr1 = 0x00000000;
|
||||
cpu->id_mmfr2 = 0x00000000;
|
||||
cpu->id_mmfr3 = 0x00000000;
|
||||
cpu->id_isar0 = 0x01141110;
|
||||
cpu->id_isar1 = 0x02111000;
|
||||
cpu->id_isar2 = 0x21112231;
|
||||
cpu->id_isar3 = 0x01111110;
|
||||
cpu->id_isar4 = 0x01310102;
|
||||
cpu->id_isar5 = 0x00000000;
|
||||
}
|
||||
|
||||
static void arm_v7m_class_init(ObjectClass *oc, void *data)
|
||||
|
|
|
@ -496,6 +496,10 @@ typedef struct CPUARMState {
|
|||
uint32_t faultmask[M_REG_NUM_BANKS];
|
||||
uint32_t aircr; /* only holds r/w state if security extn implemented */
|
||||
uint32_t secure; /* Is CPU in Secure state? (not guest visible) */
|
||||
uint32_t csselr[M_REG_NUM_BANKS];
|
||||
uint32_t scr[M_REG_NUM_BANKS];
|
||||
uint32_t msplim[M_REG_NUM_BANKS];
|
||||
uint32_t psplim[M_REG_NUM_BANKS];
|
||||
} v7m;
|
||||
|
||||
/* Information associated with an exception about to be taken:
|
||||
|
@ -1257,6 +1261,12 @@ FIELD(V7M_CCR, STKALIGN, 9, 1)
|
|||
FIELD(V7M_CCR, DC, 16, 1)
|
||||
FIELD(V7M_CCR, IC, 17, 1)
|
||||
|
||||
/* V7M SCR bits */
|
||||
FIELD(V7M_SCR, SLEEPONEXIT, 1, 1)
|
||||
FIELD(V7M_SCR, SLEEPDEEP, 2, 1)
|
||||
FIELD(V7M_SCR, SLEEPDEEPS, 3, 1)
|
||||
FIELD(V7M_SCR, SEVONPEND, 4, 1)
|
||||
|
||||
/* V7M AIRCR bits */
|
||||
FIELD(V7M_AIRCR, VECTRESET, 0, 1)
|
||||
FIELD(V7M_AIRCR, VECTCLRACTIVE, 1, 1)
|
||||
|
@ -1325,6 +1335,23 @@ FIELD(V7M_MPU_CTRL, ENABLE, 0, 1)
|
|||
FIELD(V7M_MPU_CTRL, HFNMIENA, 1, 1)
|
||||
FIELD(V7M_MPU_CTRL, PRIVDEFENA, 2, 1)
|
||||
|
||||
/* v7M CLIDR bits */
|
||||
FIELD(V7M_CLIDR, CTYPE_ALL, 0, 21)
|
||||
FIELD(V7M_CLIDR, LOUIS, 21, 3)
|
||||
FIELD(V7M_CLIDR, LOC, 24, 3)
|
||||
FIELD(V7M_CLIDR, LOUU, 27, 3)
|
||||
FIELD(V7M_CLIDR, ICB, 30, 2)
|
||||
|
||||
FIELD(V7M_CSSELR, IND, 0, 1)
|
||||
FIELD(V7M_CSSELR, LEVEL, 1, 3)
|
||||
/* We use the combination of InD and Level to index into cpu->ccsidr[];
|
||||
* define a mask for this and check that it doesn't permit running off
|
||||
* the end of the array.
|
||||
*/
|
||||
FIELD(V7M_CSSELR, INDEX, 0, 4)
|
||||
|
||||
QEMU_BUILD_BUG_ON(ARRAY_SIZE(((ARMCPU *)0)->ccsidr) <= R_V7M_CSSELR_INDEX_MASK);
|
||||
|
||||
/* If adding a feature bit which corresponds to a Linux ELF
|
||||
* HWCAP bit, remember to update the feature-bit-to-hwcap
|
||||
* mapping in linux-user/elfload.c:get_elf_hwcap().
|
||||
|
@ -1714,7 +1741,7 @@ static inline uint64_t cpreg_to_kvm_id(uint32_t cpregid)
|
|||
}
|
||||
|
||||
/* ARMCPRegInfo type field bits. If the SPECIAL bit is set this is a
|
||||
* special-behaviour cp reg and bits [15..8] indicate what behaviour
|
||||
* special-behaviour cp reg and bits [11..8] indicate what behaviour
|
||||
* it has. Otherwise it is a simple cp reg, where CONST indicates that
|
||||
* TCG can assume the value to be constant (ie load at translate time)
|
||||
* and 64BIT indicates a 64 bit wide coprocessor register. SUPPRESS_TB_END
|
||||
|
@ -1735,24 +1762,26 @@ static inline uint64_t cpreg_to_kvm_id(uint32_t cpregid)
|
|||
* need to be surrounded by gen_io_start()/gen_io_end(). In particular,
|
||||
* registers which implement clocks or timers require this.
|
||||
*/
|
||||
#define ARM_CP_SPECIAL 1
|
||||
#define ARM_CP_CONST 2
|
||||
#define ARM_CP_64BIT 4
|
||||
#define ARM_CP_SUPPRESS_TB_END 8
|
||||
#define ARM_CP_OVERRIDE 16
|
||||
#define ARM_CP_ALIAS 32
|
||||
#define ARM_CP_IO 64
|
||||
#define ARM_CP_NO_RAW 128
|
||||
#define ARM_CP_NOP (ARM_CP_SPECIAL | (1 << 8))
|
||||
#define ARM_CP_WFI (ARM_CP_SPECIAL | (2 << 8))
|
||||
#define ARM_CP_NZCV (ARM_CP_SPECIAL | (3 << 8))
|
||||
#define ARM_CP_CURRENTEL (ARM_CP_SPECIAL | (4 << 8))
|
||||
#define ARM_CP_DC_ZVA (ARM_CP_SPECIAL | (5 << 8))
|
||||
#define ARM_LAST_SPECIAL ARM_CP_DC_ZVA
|
||||
#define ARM_CP_SPECIAL 0x0001
|
||||
#define ARM_CP_CONST 0x0002
|
||||
#define ARM_CP_64BIT 0x0004
|
||||
#define ARM_CP_SUPPRESS_TB_END 0x0008
|
||||
#define ARM_CP_OVERRIDE 0x0010
|
||||
#define ARM_CP_ALIAS 0x0020
|
||||
#define ARM_CP_IO 0x0040
|
||||
#define ARM_CP_NO_RAW 0x0080
|
||||
#define ARM_CP_NOP (ARM_CP_SPECIAL | 0x0100)
|
||||
#define ARM_CP_WFI (ARM_CP_SPECIAL | 0x0200)
|
||||
#define ARM_CP_NZCV (ARM_CP_SPECIAL | 0x0300)
|
||||
#define ARM_CP_CURRENTEL (ARM_CP_SPECIAL | 0x0400)
|
||||
#define ARM_CP_DC_ZVA (ARM_CP_SPECIAL | 0x0500)
|
||||
#define ARM_LAST_SPECIAL ARM_CP_DC_ZVA
|
||||
#define ARM_CP_FPU 0x1000
|
||||
#define ARM_CP_SVE 0x2000
|
||||
/* Used only as a terminator for ARMCPRegInfo lists */
|
||||
#define ARM_CP_SENTINEL 0xffff
|
||||
#define ARM_CP_SENTINEL 0xffff
|
||||
/* Mask of only the flag bits in a type field */
|
||||
#define ARM_CP_FLAG_MASK 0xff
|
||||
#define ARM_CP_FLAG_MASK 0x30ff
|
||||
|
||||
/* Valid values for ARMCPRegInfo state field, indicating which of
|
||||
* the AArch32 and AArch64 execution states this register is visible in.
|
||||
|
@ -2485,6 +2514,14 @@ static inline int arm_debug_target_el(CPUARMState *env)
|
|||
}
|
||||
}
|
||||
|
||||
static inline bool arm_v7m_csselr_razwi(ARMCPU *cpu)
|
||||
{
|
||||
/* If all the CLIDR.Ctypem bits are 0 there are no caches, and
|
||||
* CSSELR is RAZ/WI.
|
||||
*/
|
||||
return (cpu->clidr & R_V7M_CLIDR_CTYPE_ALL_MASK) != 0;
|
||||
}
|
||||
|
||||
static inline bool aa64_generate_debug_exceptions(CPUARMState *env)
|
||||
{
|
||||
if (arm_is_secure(env)) {
|
||||
|
|
|
@ -3356,10 +3356,12 @@ static const ARMCPRegInfo v8_cp_reginfo[] = {
|
|||
.writefn = aa64_daif_write, .resetfn = arm_cp_reset_ignore },
|
||||
{ .name = "FPCR", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 3, .opc2 = 0, .crn = 4, .crm = 4,
|
||||
.access = PL0_RW, .readfn = aa64_fpcr_read, .writefn = aa64_fpcr_write },
|
||||
.access = PL0_RW, .type = ARM_CP_FPU | ARM_CP_SUPPRESS_TB_END,
|
||||
.readfn = aa64_fpcr_read, .writefn = aa64_fpcr_write },
|
||||
{ .name = "FPSR", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 3, .opc2 = 1, .crn = 4, .crm = 4,
|
||||
.access = PL0_RW, .readfn = aa64_fpsr_read, .writefn = aa64_fpsr_write },
|
||||
.access = PL0_RW, .type = ARM_CP_FPU | ARM_CP_SUPPRESS_TB_END,
|
||||
.readfn = aa64_fpsr_read, .writefn = aa64_fpsr_write },
|
||||
{ .name = "DCZID_EL0", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 3, .opc2 = 7, .crn = 0, .crm = 0,
|
||||
.access = PL0_R, .type = ARM_CP_NO_RAW,
|
||||
|
@ -4333,20 +4335,6 @@ static int sve_exception_el(CPUARMState *env)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static CPAccessResult zcr_access(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||
bool isread)
|
||||
{
|
||||
switch (sve_exception_el(env)) {
|
||||
case 3:
|
||||
return CP_ACCESS_TRAP_EL3;
|
||||
case 2:
|
||||
return CP_ACCESS_TRAP_EL2;
|
||||
case 1:
|
||||
return CP_ACCESS_TRAP;
|
||||
}
|
||||
return CP_ACCESS_OK;
|
||||
}
|
||||
|
||||
static void zcr_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||
uint64_t value)
|
||||
{
|
||||
|
@ -4357,7 +4345,7 @@ static void zcr_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
|||
static const ARMCPRegInfo zcr_el1_reginfo = {
|
||||
.name = "ZCR_EL1", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 0, .crn = 1, .crm = 2, .opc2 = 0,
|
||||
.access = PL1_RW, .accessfn = zcr_access, .type = ARM_CP_64BIT,
|
||||
.access = PL1_RW, .type = ARM_CP_SVE | ARM_CP_FPU,
|
||||
.fieldoffset = offsetof(CPUARMState, vfp.zcr_el[1]),
|
||||
.writefn = zcr_write, .raw_writefn = raw_write
|
||||
};
|
||||
|
@ -4365,7 +4353,7 @@ static const ARMCPRegInfo zcr_el1_reginfo = {
|
|||
static const ARMCPRegInfo zcr_el2_reginfo = {
|
||||
.name = "ZCR_EL2", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 2, .opc2 = 0,
|
||||
.access = PL2_RW, .accessfn = zcr_access, .type = ARM_CP_64BIT,
|
||||
.access = PL2_RW, .type = ARM_CP_SVE | ARM_CP_FPU,
|
||||
.fieldoffset = offsetof(CPUARMState, vfp.zcr_el[2]),
|
||||
.writefn = zcr_write, .raw_writefn = raw_write
|
||||
};
|
||||
|
@ -4373,14 +4361,14 @@ static const ARMCPRegInfo zcr_el2_reginfo = {
|
|||
static const ARMCPRegInfo zcr_no_el2_reginfo = {
|
||||
.name = "ZCR_EL2", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 2, .opc2 = 0,
|
||||
.access = PL2_RW, .type = ARM_CP_64BIT,
|
||||
.access = PL2_RW, .type = ARM_CP_SVE | ARM_CP_FPU,
|
||||
.readfn = arm_cp_read_zero, .writefn = arm_cp_write_ignore
|
||||
};
|
||||
|
||||
static const ARMCPRegInfo zcr_el3_reginfo = {
|
||||
.name = "ZCR_EL3", .state = ARM_CP_STATE_AA64,
|
||||
.opc0 = 3, .opc1 = 6, .crn = 1, .crm = 2, .opc2 = 0,
|
||||
.access = PL3_RW, .accessfn = zcr_access, .type = ARM_CP_64BIT,
|
||||
.access = PL3_RW, .type = ARM_CP_SVE | ARM_CP_FPU,
|
||||
.fieldoffset = offsetof(CPUARMState, vfp.zcr_el[3]),
|
||||
.writefn = zcr_write, .raw_writefn = raw_write
|
||||
};
|
||||
|
@ -10415,6 +10403,16 @@ uint32_t HELPER(v7m_mrs)(CPUARMState *env, uint32_t reg)
|
|||
return 0;
|
||||
}
|
||||
return env->v7m.other_ss_psp;
|
||||
case 0x8a: /* MSPLIM_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return 0;
|
||||
}
|
||||
return env->v7m.msplim[M_REG_NS];
|
||||
case 0x8b: /* PSPLIM_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return 0;
|
||||
}
|
||||
return env->v7m.psplim[M_REG_NS];
|
||||
case 0x90: /* PRIMASK_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return 0;
|
||||
|
@ -10456,6 +10454,16 @@ uint32_t HELPER(v7m_mrs)(CPUARMState *env, uint32_t reg)
|
|||
return v7m_using_psp(env) ? env->v7m.other_sp : env->regs[13];
|
||||
case 9: /* PSP */
|
||||
return v7m_using_psp(env) ? env->regs[13] : env->v7m.other_sp;
|
||||
case 10: /* MSPLIM */
|
||||
if (!arm_feature(env, ARM_FEATURE_V8)) {
|
||||
goto bad_reg;
|
||||
}
|
||||
return env->v7m.msplim[env->v7m.secure];
|
||||
case 11: /* PSPLIM */
|
||||
if (!arm_feature(env, ARM_FEATURE_V8)) {
|
||||
goto bad_reg;
|
||||
}
|
||||
return env->v7m.psplim[env->v7m.secure];
|
||||
case 16: /* PRIMASK */
|
||||
return env->v7m.primask[env->v7m.secure];
|
||||
case 17: /* BASEPRI */
|
||||
|
@ -10464,6 +10472,7 @@ uint32_t HELPER(v7m_mrs)(CPUARMState *env, uint32_t reg)
|
|||
case 19: /* FAULTMASK */
|
||||
return env->v7m.faultmask[env->v7m.secure];
|
||||
default:
|
||||
bad_reg:
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "Attempt to read unknown special"
|
||||
" register %d\n", reg);
|
||||
return 0;
|
||||
|
@ -10501,6 +10510,18 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t maskreg, uint32_t val)
|
|||
}
|
||||
env->v7m.other_ss_psp = val;
|
||||
return;
|
||||
case 0x8a: /* MSPLIM_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return;
|
||||
}
|
||||
env->v7m.msplim[M_REG_NS] = val & ~7;
|
||||
return;
|
||||
case 0x8b: /* PSPLIM_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return;
|
||||
}
|
||||
env->v7m.psplim[M_REG_NS] = val & ~7;
|
||||
return;
|
||||
case 0x90: /* PRIMASK_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return;
|
||||
|
@ -10519,6 +10540,16 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t maskreg, uint32_t val)
|
|||
}
|
||||
env->v7m.faultmask[M_REG_NS] = val & 1;
|
||||
return;
|
||||
case 0x94: /* CONTROL_NS */
|
||||
if (!env->v7m.secure) {
|
||||
return;
|
||||
}
|
||||
write_v7m_control_spsel_for_secstate(env,
|
||||
val & R_V7M_CONTROL_SPSEL_MASK,
|
||||
M_REG_NS);
|
||||
env->v7m.control[M_REG_NS] &= ~R_V7M_CONTROL_NPRIV_MASK;
|
||||
env->v7m.control[M_REG_NS] |= val & R_V7M_CONTROL_NPRIV_MASK;
|
||||
return;
|
||||
case 0x98: /* SP_NS */
|
||||
{
|
||||
/* This gives the non-secure SP selected based on whether we're
|
||||
|
@ -10570,6 +10601,18 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t maskreg, uint32_t val)
|
|||
env->v7m.other_sp = val;
|
||||
}
|
||||
break;
|
||||
case 10: /* MSPLIM */
|
||||
if (!arm_feature(env, ARM_FEATURE_V8)) {
|
||||
goto bad_reg;
|
||||
}
|
||||
env->v7m.msplim[env->v7m.secure] = val & ~7;
|
||||
break;
|
||||
case 11: /* PSPLIM */
|
||||
if (!arm_feature(env, ARM_FEATURE_V8)) {
|
||||
goto bad_reg;
|
||||
}
|
||||
env->v7m.psplim[env->v7m.secure] = val & ~7;
|
||||
break;
|
||||
case 16: /* PRIMASK */
|
||||
env->v7m.primask[env->v7m.secure] = val & 1;
|
||||
break;
|
||||
|
@ -10602,6 +10645,7 @@ void HELPER(v7m_msr)(CPUARMState *env, uint32_t maskreg, uint32_t val)
|
|||
env->v7m.control[env->v7m.secure] |= val & R_V7M_CONTROL_NPRIV_MASK;
|
||||
break;
|
||||
default:
|
||||
bad_reg:
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "Attempt to write unknown special"
|
||||
" register %d\n", reg);
|
||||
return;
|
||||
|
|
|
@ -243,6 +243,7 @@ enum arm_exception_class {
|
|||
EC_AA64_HVC = 0x16,
|
||||
EC_AA64_SMC = 0x17,
|
||||
EC_SYSTEMREGISTERTRAP = 0x18,
|
||||
EC_SVEACCESSTRAP = 0x19,
|
||||
EC_INSNABORT = 0x20,
|
||||
EC_INSNABORT_SAME_EL = 0x21,
|
||||
EC_PCALIGNMENT = 0x22,
|
||||
|
@ -381,6 +382,11 @@ static inline uint32_t syn_fp_access_trap(int cv, int cond, bool is_16bit)
|
|||
| (cv << 24) | (cond << 20);
|
||||
}
|
||||
|
||||
static inline uint32_t syn_sve_access_trap(void)
|
||||
{
|
||||
return EC_SVEACCESSTRAP << ARM_EL_EC_SHIFT;
|
||||
}
|
||||
|
||||
static inline uint32_t syn_insn_abort(int same_el, int ea, int s1ptw, int fsc)
|
||||
{
|
||||
return (EC_INSNABORT << ARM_EL_EC_SHIFT) | (same_el << ARM_EL_EC_SHIFT)
|
||||
|
|
|
@ -191,6 +191,81 @@ static const VMStateDescription vmstate_m_faultmask_primask = {
|
|||
}
|
||||
};
|
||||
|
||||
/* 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,
|
||||
.fields = (VMStateField[]) {
|
||||
VMSTATE_UINT32_ARRAY(env.v7m.csselr, ARMCPU, M_REG_NUM_BANKS),
|
||||
VMSTATE_VALIDATE("CSSELR is valid", csselr_vmstate_validate),
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
||||
static const VMStateDescription vmstate_m_scr = {
|
||||
.name = "cpu/m/scr",
|
||||
.version_id = 1,
|
||||
.minimum_version_id = 1,
|
||||
.fields = (VMStateField[]) {
|
||||
VMSTATE_UINT32(env.v7m.scr[M_REG_NS], ARMCPU),
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
||||
static const VMStateDescription vmstate_m_other_sp = {
|
||||
.name = "cpu/m/other-sp",
|
||||
.version_id = 1,
|
||||
.minimum_version_id = 1,
|
||||
.fields = (VMStateField[]) {
|
||||
VMSTATE_UINT32(env.v7m.other_sp, ARMCPU),
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
||||
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,
|
||||
.fields = (VMStateField[]) {
|
||||
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()
|
||||
}
|
||||
};
|
||||
|
||||
static const VMStateDescription vmstate_m = {
|
||||
.name = "cpu/m",
|
||||
.version_id = 4,
|
||||
|
@ -212,6 +287,10 @@ static const VMStateDescription vmstate_m = {
|
|||
},
|
||||
.subsections = (const VMStateDescription*[]) {
|
||||
&vmstate_m_faultmask_primask,
|
||||
&vmstate_m_csselr,
|
||||
&vmstate_m_scr,
|
||||
&vmstate_m_other_sp,
|
||||
&vmstate_m_v8m,
|
||||
NULL
|
||||
}
|
||||
};
|
||||
|
@ -375,6 +454,11 @@ static const VMStateDescription vmstate_m_security = {
|
|||
VMSTATE_UINT32(env.sau.rnr, ARMCPU),
|
||||
VMSTATE_VALIDATE("SAU_RNR is valid", sau_rnr_vmstate_validate),
|
||||
VMSTATE_UINT32(env.sau.ctrl, ARMCPU),
|
||||
VMSTATE_UINT32(env.v7m.scr[M_REG_S], ARMCPU),
|
||||
/* 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),
|
||||
VMSTATE_END_OF_LIST()
|
||||
}
|
||||
};
|
||||
|
|
|
@ -602,13 +602,30 @@ static TCGv_i32 read_fp_sreg(DisasContext *s, int reg)
|
|||
return v;
|
||||
}
|
||||
|
||||
/* Clear the bits above an N-bit vector, for N = (is_q ? 128 : 64).
|
||||
* If SVE is not enabled, then there are only 128 bits in the vector.
|
||||
*/
|
||||
static void clear_vec_high(DisasContext *s, bool is_q, int rd)
|
||||
{
|
||||
unsigned ofs = fp_reg_offset(s, rd, MO_64);
|
||||
unsigned vsz = vec_full_reg_size(s);
|
||||
|
||||
if (!is_q) {
|
||||
TCGv_i64 tcg_zero = tcg_const_i64(0);
|
||||
tcg_gen_st_i64(tcg_zero, cpu_env, ofs + 8);
|
||||
tcg_temp_free_i64(tcg_zero);
|
||||
}
|
||||
if (vsz > 16) {
|
||||
tcg_gen_gvec_dup8i(ofs + 16, vsz - 16, vsz - 16, 0);
|
||||
}
|
||||
}
|
||||
|
||||
static void write_fp_dreg(DisasContext *s, int reg, TCGv_i64 v)
|
||||
{
|
||||
TCGv_i64 tcg_zero = tcg_const_i64(0);
|
||||
unsigned ofs = fp_reg_offset(s, reg, MO_64);
|
||||
|
||||
tcg_gen_st_i64(v, cpu_env, fp_reg_offset(s, reg, MO_64));
|
||||
tcg_gen_st_i64(tcg_zero, cpu_env, fp_reg_hi_offset(s, reg));
|
||||
tcg_temp_free_i64(tcg_zero);
|
||||
tcg_gen_st_i64(v, cpu_env, ofs);
|
||||
clear_vec_high(s, false, reg);
|
||||
}
|
||||
|
||||
static void write_fp_sreg(DisasContext *s, int reg, TCGv_i32 v)
|
||||
|
@ -1009,6 +1026,8 @@ static void do_fp_ld(DisasContext *s, int destidx, TCGv_i64 tcg_addr, int size)
|
|||
|
||||
tcg_temp_free_i64(tmplo);
|
||||
tcg_temp_free_i64(tmphi);
|
||||
|
||||
clear_vec_high(s, true, destidx);
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -1124,17 +1143,6 @@ static void write_vec_element_i32(DisasContext *s, TCGv_i32 tcg_src,
|
|||
}
|
||||
}
|
||||
|
||||
/* Clear the high 64 bits of a 128 bit vector (in general non-quad
|
||||
* vector ops all need to do this).
|
||||
*/
|
||||
static void clear_vec_high(DisasContext *s, int rd)
|
||||
{
|
||||
TCGv_i64 tcg_zero = tcg_const_i64(0);
|
||||
|
||||
write_vec_element(s, tcg_zero, rd, 1, MO_64);
|
||||
tcg_temp_free_i64(tcg_zero);
|
||||
}
|
||||
|
||||
/* Store from vector register to memory */
|
||||
static void do_vec_st(DisasContext *s, int srcidx, int element,
|
||||
TCGv_i64 tcg_addr, int size)
|
||||
|
@ -1182,6 +1190,19 @@ static inline bool fp_access_check(DisasContext *s)
|
|||
return false;
|
||||
}
|
||||
|
||||
/* Check that SVE access is enabled. If it is, return true.
|
||||
* If not, emit code to generate an appropriate exception and return false.
|
||||
*/
|
||||
static inline bool sve_access_check(DisasContext *s)
|
||||
{
|
||||
if (s->sve_excp_el) {
|
||||
gen_exception_insn(s, 4, EXCP_UDEF, syn_sve_access_trap(),
|
||||
s->sve_excp_el);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/*
|
||||
* This utility function is for doing register extension with an
|
||||
* optional shift. You will likely want to pass a temporary for the
|
||||
|
@ -1631,6 +1652,12 @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
|
|||
default:
|
||||
break;
|
||||
}
|
||||
if ((ri->type & ARM_CP_SVE) && !sve_access_check(s)) {
|
||||
return;
|
||||
}
|
||||
if ((ri->type & ARM_CP_FPU) && !fp_access_check(s)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ((tb_cflags(s->base.tb) & CF_USE_ICOUNT) && (ri->type & ARM_CP_IO)) {
|
||||
gen_io_start();
|
||||
|
@ -2775,12 +2802,13 @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
|
|||
/* For non-quad operations, setting a slice of the low
|
||||
* 64 bits of the register clears the high 64 bits (in
|
||||
* the ARM ARM pseudocode this is implicit in the fact
|
||||
* that 'rval' is a 64 bit wide variable). We optimize
|
||||
* by noticing that we only need to do this the first
|
||||
* time we touch a register.
|
||||
* that 'rval' is a 64 bit wide variable).
|
||||
* For quad operations, we might still need to zero the
|
||||
* high bits of SVE. We optimize by noticing that we only
|
||||
* need to do this the first time we touch a register.
|
||||
*/
|
||||
if (!is_q && e == 0 && (r == 0 || xs == selem - 1)) {
|
||||
clear_vec_high(s, tt);
|
||||
if (e == 0 && (r == 0 || xs == selem - 1)) {
|
||||
clear_vec_high(s, is_q, tt);
|
||||
}
|
||||
}
|
||||
tcg_gen_addi_i64(tcg_addr, tcg_addr, ebytes);
|
||||
|
@ -2923,10 +2951,9 @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
|
|||
write_vec_element(s, tcg_tmp, rt, 0, MO_64);
|
||||
if (is_q) {
|
||||
write_vec_element(s, tcg_tmp, rt, 1, MO_64);
|
||||
} else {
|
||||
clear_vec_high(s, rt);
|
||||
}
|
||||
tcg_temp_free_i64(tcg_tmp);
|
||||
clear_vec_high(s, is_q, rt);
|
||||
} else {
|
||||
/* Load/store one element per register */
|
||||
if (is_load) {
|
||||
|
@ -6699,7 +6726,6 @@ static void handle_vec_simd_sqshrn(DisasContext *s, bool is_scalar, bool is_q,
|
|||
}
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
write_vec_element(s, tcg_final, rd, 0, MO_64);
|
||||
} else {
|
||||
write_vec_element(s, tcg_final, rd, 1, MO_64);
|
||||
|
@ -6712,7 +6738,8 @@ static void handle_vec_simd_sqshrn(DisasContext *s, bool is_scalar, bool is_q,
|
|||
tcg_temp_free_i64(tcg_rd);
|
||||
tcg_temp_free_i32(tcg_rd_narrowed);
|
||||
tcg_temp_free_i64(tcg_final);
|
||||
return;
|
||||
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
/* SQSHLU, UQSHL, SQSHL: saturating left shifts */
|
||||
|
@ -6776,10 +6803,7 @@ static void handle_simd_qshl(DisasContext *s, bool scalar, bool is_q,
|
|||
tcg_temp_free_i64(tcg_op);
|
||||
}
|
||||
tcg_temp_free_i64(tcg_shift);
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
} else {
|
||||
TCGv_i32 tcg_shift = tcg_const_i32(shift);
|
||||
static NeonGenTwoOpEnvFn * const fns[2][2][3] = {
|
||||
|
@ -6828,8 +6852,8 @@ static void handle_simd_qshl(DisasContext *s, bool scalar, bool is_q,
|
|||
}
|
||||
tcg_temp_free_i32(tcg_shift);
|
||||
|
||||
if (!is_q && !scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
if (!scalar) {
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -6882,13 +6906,11 @@ static void handle_simd_intfp_conv(DisasContext *s, int rd, int rn,
|
|||
}
|
||||
}
|
||||
|
||||
if (!is_double && elements == 2) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i64(tcg_int);
|
||||
tcg_temp_free_ptr(tcg_fpst);
|
||||
tcg_temp_free_i32(tcg_shift);
|
||||
|
||||
clear_vec_high(s, elements << size == 16, rd);
|
||||
}
|
||||
|
||||
/* UCVTF/SCVTF - Integer to FP conversion */
|
||||
|
@ -6976,9 +6998,7 @@ static void handle_simd_shift_fpint_conv(DisasContext *s, bool is_scalar,
|
|||
write_vec_element(s, tcg_op, rd, pass, MO_64);
|
||||
tcg_temp_free_i64(tcg_op);
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
} else {
|
||||
int maxpass = is_scalar ? 1 : is_q ? 4 : 2;
|
||||
for (pass = 0; pass < maxpass; pass++) {
|
||||
|
@ -6997,8 +7017,8 @@ static void handle_simd_shift_fpint_conv(DisasContext *s, bool is_scalar,
|
|||
}
|
||||
tcg_temp_free_i32(tcg_op);
|
||||
}
|
||||
if (!is_q && !is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
if (!is_scalar) {
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -7483,10 +7503,7 @@ static void handle_3same_float(DisasContext *s, int size, int elements,
|
|||
|
||||
tcg_temp_free_ptr(fpst);
|
||||
|
||||
if ((elements << size) < 4) {
|
||||
/* scalar, or non-quad vector op */
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, elements * (size ? 8 : 4) > 8, rd);
|
||||
}
|
||||
|
||||
/* AdvSIMD scalar three same
|
||||
|
@ -7812,13 +7829,11 @@ static void handle_2misc_fcmp_zero(DisasContext *s, int opcode,
|
|||
}
|
||||
write_vec_element(s, tcg_res, rd, pass, MO_64);
|
||||
}
|
||||
if (is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i64(tcg_res);
|
||||
tcg_temp_free_i64(tcg_zero);
|
||||
tcg_temp_free_i64(tcg_op);
|
||||
|
||||
clear_vec_high(s, !is_scalar, rd);
|
||||
} else {
|
||||
TCGv_i32 tcg_op = tcg_temp_new_i32();
|
||||
TCGv_i32 tcg_zero = tcg_const_i32(0);
|
||||
|
@ -7869,8 +7884,8 @@ static void handle_2misc_fcmp_zero(DisasContext *s, int opcode,
|
|||
tcg_temp_free_i32(tcg_res);
|
||||
tcg_temp_free_i32(tcg_zero);
|
||||
tcg_temp_free_i32(tcg_op);
|
||||
if (!is_q && !is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
if (!is_scalar) {
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -7906,12 +7921,9 @@ static void handle_2misc_reciprocal(DisasContext *s, int opcode,
|
|||
}
|
||||
write_vec_element(s, tcg_res, rd, pass, MO_64);
|
||||
}
|
||||
if (is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i64(tcg_res);
|
||||
tcg_temp_free_i64(tcg_op);
|
||||
clear_vec_high(s, !is_scalar, rd);
|
||||
} else {
|
||||
TCGv_i32 tcg_op = tcg_temp_new_i32();
|
||||
TCGv_i32 tcg_res = tcg_temp_new_i32();
|
||||
|
@ -7951,8 +7963,8 @@ static void handle_2misc_reciprocal(DisasContext *s, int opcode,
|
|||
}
|
||||
tcg_temp_free_i32(tcg_res);
|
||||
tcg_temp_free_i32(tcg_op);
|
||||
if (!is_q && !is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
if (!is_scalar) {
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
}
|
||||
tcg_temp_free_ptr(fpst);
|
||||
|
@ -8058,9 +8070,7 @@ static void handle_2misc_narrow(DisasContext *s, bool scalar,
|
|||
write_vec_element_i32(s, tcg_res[pass], rd, destelt + pass, MO_32);
|
||||
tcg_temp_free_i32(tcg_res[pass]);
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
/* Remaining saturating accumulating ops */
|
||||
|
@ -8085,12 +8095,9 @@ static void handle_2misc_satacc(DisasContext *s, bool is_scalar, bool is_u,
|
|||
}
|
||||
write_vec_element(s, tcg_rd, rd, pass, MO_64);
|
||||
}
|
||||
if (is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i64(tcg_rd);
|
||||
tcg_temp_free_i64(tcg_rn);
|
||||
clear_vec_high(s, !is_scalar, rd);
|
||||
} else {
|
||||
TCGv_i32 tcg_rn = tcg_temp_new_i32();
|
||||
TCGv_i32 tcg_rd = tcg_temp_new_i32();
|
||||
|
@ -8148,13 +8155,9 @@ static void handle_2misc_satacc(DisasContext *s, bool is_scalar, bool is_u,
|
|||
}
|
||||
write_vec_element_i32(s, tcg_rd, rd, pass, MO_32);
|
||||
}
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i32(tcg_rd);
|
||||
tcg_temp_free_i32(tcg_rn);
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -8645,9 +8648,7 @@ static void handle_vec_simd_shri(DisasContext *s, bool is_q, bool is_u,
|
|||
tcg_temp_free_i64(tcg_round);
|
||||
|
||||
done:
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
static void gen_shl8_ins_i64(TCGv_i64 d, TCGv_i64 a, int64_t shift)
|
||||
|
@ -8836,19 +8837,18 @@ static void handle_vec_simd_shrn(DisasContext *s, bool is_q,
|
|||
}
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
write_vec_element(s, tcg_final, rd, 0, MO_64);
|
||||
} else {
|
||||
write_vec_element(s, tcg_final, rd, 1, MO_64);
|
||||
}
|
||||
|
||||
if (round) {
|
||||
tcg_temp_free_i64(tcg_round);
|
||||
}
|
||||
tcg_temp_free_i64(tcg_rn);
|
||||
tcg_temp_free_i64(tcg_rd);
|
||||
tcg_temp_free_i64(tcg_final);
|
||||
return;
|
||||
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
|
||||
|
@ -9242,9 +9242,7 @@ static void handle_3rd_narrowing(DisasContext *s, int is_q, int is_u, int size,
|
|||
write_vec_element_i32(s, tcg_res[pass], rd, pass + part, MO_32);
|
||||
tcg_temp_free_i32(tcg_res[pass]);
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
static void handle_pmull_64(DisasContext *s, int is_q, int rd, int rn, int rm)
|
||||
|
@ -9652,9 +9650,7 @@ static void handle_simd_3same_pair(DisasContext *s, int is_q, int u, int opcode,
|
|||
write_vec_element_i32(s, tcg_res[pass], rd, pass, MO_32);
|
||||
tcg_temp_free_i32(tcg_res[pass]);
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
if (fpst) {
|
||||
|
@ -10142,10 +10138,7 @@ static void disas_simd_3same_int(DisasContext *s, uint32_t insn)
|
|||
tcg_temp_free_i32(tcg_op2);
|
||||
}
|
||||
}
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
}
|
||||
|
||||
/* AdvSIMD three same
|
||||
|
@ -10284,9 +10277,7 @@ static void handle_rev(DisasContext *s, int opcode, bool u,
|
|||
write_vec_element(s, tcg_tmp, rd, i, grp_size);
|
||||
tcg_temp_free_i64(tcg_tmp);
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
} else {
|
||||
int revmask = (1 << grp_size) - 1;
|
||||
int esize = 8 << size;
|
||||
|
@ -10930,9 +10921,7 @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
|
|||
tcg_temp_free_i32(tcg_op);
|
||||
}
|
||||
}
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
|
||||
if (need_rmode) {
|
||||
gen_helper_set_rmode(tcg_rmode, tcg_rmode, cpu_env);
|
||||
|
@ -11111,11 +11100,8 @@ static void disas_simd_indexed(DisasContext *s, uint32_t insn)
|
|||
tcg_temp_free_i64(tcg_res);
|
||||
}
|
||||
|
||||
if (is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
|
||||
tcg_temp_free_i64(tcg_idx);
|
||||
clear_vec_high(s, !is_scalar, rd);
|
||||
} else if (!is_long) {
|
||||
/* 32 bit floating point, or 16 or 32 bit integer.
|
||||
* For the 16 bit scalar case we use the usual Neon helpers and
|
||||
|
@ -11219,10 +11205,7 @@ static void disas_simd_indexed(DisasContext *s, uint32_t insn)
|
|||
}
|
||||
|
||||
tcg_temp_free_i32(tcg_idx);
|
||||
|
||||
if (!is_q) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
} else {
|
||||
/* long ops: 16x16->32 or 32x32->64 */
|
||||
TCGv_i64 tcg_res[2];
|
||||
|
@ -11299,9 +11282,7 @@ static void disas_simd_indexed(DisasContext *s, uint32_t insn)
|
|||
}
|
||||
tcg_temp_free_i64(tcg_idx);
|
||||
|
||||
if (is_scalar) {
|
||||
clear_vec_high(s, rd);
|
||||
}
|
||||
clear_vec_high(s, !is_scalar, rd);
|
||||
} else {
|
||||
TCGv_i32 tcg_idx = tcg_temp_new_i32();
|
||||
|
||||
|
|
Loading…
Reference in New Issue