target-arm queue:

* hw/arm: Move raspberrypi-fw-defs.h to the include/hw/arm/ folder
  * hw/arm/exynos4210: Get arm_boot_info declaration from 'hw/arm/boot'
  * xlnx devices: remove deprecated device reset
  * xlnx-bbram: hw/nvram: Use dot in device type name
  * elf2dmp: fix coverity issues
  * elf2dmp: convert to g_malloc, g_new and g_free
  * target/arm: Fix CNTPCT_EL0 trapping from EL0 when HCR_EL2.E2H is 0
  * hw/arm: refactor virt PPI logic
  * arm/kvm: convert to kvm_set_one_reg, kvm_get_one_reg
  * target/arm: Permit T32 LDM with single register
  * smmuv3: Advertise SMMUv3.1-XNX
  * target/arm: Implement FEAT_HPMN0
  * Remove some unnecessary include lines
  * target/arm/arm-powerctl: Correctly init CPUs when powered on to lower EL
  * hw/timer/npcm7xx_timer: Prevent timer from counting down past zero
 -----BEGIN PGP SIGNATURE-----
 
 iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmUxMF4ZHHBldGVyLm1h
 eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3oJND/4p64q0Wxq8x8yXCDUZAHME
 lZe2liBPBkqZusGfK0O4CpClwGbM5+8tMeLaRgSOUgJ/WGFiLCGAKEKB0S7EiCa5
 1bNvVn+a7cdDj7FdYf+Dvp5fNZZIus4w+CUlUaiRyDhIfYquz53J1RD1wN5+SQ/I
 g6JQRp2gONeqGM5hT+0v2J/wGMmhuI5XO+PtQ1QNGoUnAA4QNof1thYjqdTJxzfz
 V2CUSOKnAT/PDcUWoy8BVPDDE+wYTnjTO1j/ZsQvnNQm7r18OiMUn85teLq1JtB+
 T3vyVZ2f2gc8lAgkKy5n3NH5fmLVgbO0WXgpWLHNkcp+shZMM6J5J/u/P6B/wk95
 DMzQy4slu/UfWMvsaxq+OjejhAtbdiIOeNfF6dAMy2NAyZplEAjlP8dsFrqAdACL
 9m/DA4ODAV6OJ3E0zQ0dI4o6kr+/wbPVseLklqn3Ss0dndjU1K9XR0qpC8OruUJq
 4h6kl5q6V3BHAoELvBtAqb0yHYdqhLqznpO8HsrUEmU5eTjDaOyyI4HW+AY5GG1R
 dtvrCLSiPe0EMartMMtezaB2GxQb9O7e+OI3XL2zVxb1F+QQ+vRZE3zVIdXm+Ev4
 oBztF1peZC3c8zurjr7/MxnDSnzynpkSR1zOY8+WJnAqpQ+C1YvdF6/Llwn7IMHw
 ZHh6sGzQsaAu7u/DW9yY5w==
 =WreO
 -----END PGP SIGNATURE-----

Merge tag 'pull-target-arm-20231019' of https://git.linaro.org/people/pmaydell/qemu-arm into staging

target-arm queue:
 * hw/arm: Move raspberrypi-fw-defs.h to the include/hw/arm/ folder
 * hw/arm/exynos4210: Get arm_boot_info declaration from 'hw/arm/boot'
 * xlnx devices: remove deprecated device reset
 * xlnx-bbram: hw/nvram: Use dot in device type name
 * elf2dmp: fix coverity issues
 * elf2dmp: convert to g_malloc, g_new and g_free
 * target/arm: Fix CNTPCT_EL0 trapping from EL0 when HCR_EL2.E2H is 0
 * hw/arm: refactor virt PPI logic
 * arm/kvm: convert to kvm_set_one_reg, kvm_get_one_reg
 * target/arm: Permit T32 LDM with single register
 * smmuv3: Advertise SMMUv3.1-XNX
 * target/arm: Implement FEAT_HPMN0
 * Remove some unnecessary include lines
 * target/arm/arm-powerctl: Correctly init CPUs when powered on to lower EL
 * hw/timer/npcm7xx_timer: Prevent timer from counting down past zero

# -----BEGIN PGP SIGNATURE-----
#
# iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmUxMF4ZHHBldGVyLm1h
# eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3oJND/4p64q0Wxq8x8yXCDUZAHME
# lZe2liBPBkqZusGfK0O4CpClwGbM5+8tMeLaRgSOUgJ/WGFiLCGAKEKB0S7EiCa5
# 1bNvVn+a7cdDj7FdYf+Dvp5fNZZIus4w+CUlUaiRyDhIfYquz53J1RD1wN5+SQ/I
# g6JQRp2gONeqGM5hT+0v2J/wGMmhuI5XO+PtQ1QNGoUnAA4QNof1thYjqdTJxzfz
# V2CUSOKnAT/PDcUWoy8BVPDDE+wYTnjTO1j/ZsQvnNQm7r18OiMUn85teLq1JtB+
# T3vyVZ2f2gc8lAgkKy5n3NH5fmLVgbO0WXgpWLHNkcp+shZMM6J5J/u/P6B/wk95
# DMzQy4slu/UfWMvsaxq+OjejhAtbdiIOeNfF6dAMy2NAyZplEAjlP8dsFrqAdACL
# 9m/DA4ODAV6OJ3E0zQ0dI4o6kr+/wbPVseLklqn3Ss0dndjU1K9XR0qpC8OruUJq
# 4h6kl5q6V3BHAoELvBtAqb0yHYdqhLqznpO8HsrUEmU5eTjDaOyyI4HW+AY5GG1R
# dtvrCLSiPe0EMartMMtezaB2GxQb9O7e+OI3XL2zVxb1F+QQ+vRZE3zVIdXm+Ev4
# oBztF1peZC3c8zurjr7/MxnDSnzynpkSR1zOY8+WJnAqpQ+C1YvdF6/Llwn7IMHw
# ZHh6sGzQsaAu7u/DW9yY5w==
# =WreO
# -----END PGP SIGNATURE-----
# gpg: Signature made Thu 19 Oct 2023 06:34:22 PDT
# gpg:                using RSA key E1A5C593CD419DE28E8315CF3C2525ED14360CDE
# gpg:                issuer "peter.maydell@linaro.org"
# gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>" [full]
# gpg:                 aka "Peter Maydell <pmaydell@gmail.com>" [full]
# gpg:                 aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>" [full]
# gpg:                 aka "Peter Maydell <peter@archaic.org.uk>" [unknown]
# Primary key fingerprint: E1A5 C593 CD41 9DE2 8E83  15CF 3C25 25ED 1436 0CDE

* tag 'pull-target-arm-20231019' of https://git.linaro.org/people/pmaydell/qemu-arm: (24 commits)
  contrib/elf2dmp: Use g_malloc(), g_new() and g_free()
  hw/timer/npcm7xx_timer: Prevent timer from counting down past zero
  target/arm/arm-powerctl: Correctly init CPUs when powered on to lower EL
  target/arm/common-semi-target.h: Remove unnecessary boot.h include
  target/arm/kvm64.c: Remove unused include
  target/arm: Implement FEAT_HPMN0
  hw/arm/smmuv3: Advertise SMMUv3.1-XNX feature
  hw/arm/smmuv3: Sort ID register setting into field order
  hw/arm/smmuv3: Update ID register bit field definitions
  target/arm: Permit T32 LDM with single register
  arm/kvm: convert to kvm_get_one_reg
  arm/kvm: convert to kvm_set_one_reg
  hw/arm/sbsa-ref: use bsa.h for PPI definitions
  include/hw/arm: move BSA definitions to bsa.h
  {include/}hw/arm: refactor virt PPI logic
  target/arm: Fix CNTPCT_EL0 trapping from EL0 when HCR_EL2.E2H is 0
  elf2dmp: check array bounds in pdb_get_file_size
  elf2dmp: limit print length for sign_rsds
  xlnx-bbram: hw/nvram: Use dot in device type name
  xlnx-versal-efuse: hw/nvram: Remove deprecated device reset
  ...

Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
This commit is contained in:
Stefan Hajnoczi 2023-10-20 06:45:37 -07:00
commit caa75cc56e
33 changed files with 368 additions and 359 deletions

View File

@ -885,7 +885,7 @@ S: Odd Fixes
F: hw/arm/raspi.c F: hw/arm/raspi.c
F: hw/arm/raspi_platform.h F: hw/arm/raspi_platform.h
F: hw/*/bcm283* F: hw/*/bcm283*
F: include/hw/arm/raspi* F: include/hw/arm/rasp*
F: include/hw/*/bcm283* F: include/hw/*/bcm283*
F: docs/system/arm/raspi.rst F: docs/system/arm/raspi.rst

View File

@ -72,10 +72,7 @@ int pa_space_create(struct pa_space *ps, QEMU_Elf *qemu_elf)
} }
} }
ps->block = malloc(sizeof(*ps->block) * ps->block_nr); ps->block = g_new(struct pa_block, ps->block_nr);
if (!ps->block) {
return 1;
}
for (i = 0; i < phdr_nr; i++) { for (i = 0; i < phdr_nr; i++) {
if (phdr[i].p_type == PT_LOAD) { if (phdr[i].p_type == PT_LOAD) {
@ -97,7 +94,7 @@ int pa_space_create(struct pa_space *ps, QEMU_Elf *qemu_elf)
void pa_space_destroy(struct pa_space *ps) void pa_space_destroy(struct pa_space *ps)
{ {
ps->block_nr = 0; ps->block_nr = 0;
free(ps->block); g_free(ps->block);
} }
void va_space_set_dtb(struct va_space *vs, uint64_t dtb) void va_space_set_dtb(struct va_space *vs, uint64_t dtb)

View File

@ -120,14 +120,11 @@ static KDDEBUGGER_DATA64 *get_kdbg(uint64_t KernBase, struct pdb_reader *pdb,
} }
} }
kdbg = malloc(kdbg_hdr.Size); kdbg = g_malloc(kdbg_hdr.Size);
if (!kdbg) {
return NULL;
}
if (va_space_rw(vs, KdDebuggerDataBlock, kdbg, kdbg_hdr.Size, 0)) { if (va_space_rw(vs, KdDebuggerDataBlock, kdbg, kdbg_hdr.Size, 0)) {
eprintf("Failed to extract entire KDBG\n"); eprintf("Failed to extract entire KDBG\n");
free(kdbg); g_free(kdbg);
return NULL; return NULL;
} }
@ -478,7 +475,7 @@ static bool pe_check_pdb_name(uint64_t base, void *start_addr,
} }
if (memcmp(&rsds->Signature, sign_rsds, sizeof(sign_rsds))) { if (memcmp(&rsds->Signature, sign_rsds, sizeof(sign_rsds))) {
eprintf("CodeView signature is \'%.4s\', \'%s\' expected\n", eprintf("CodeView signature is \'%.4s\', \'%.4s\' expected\n",
rsds->Signature, sign_rsds); rsds->Signature, sign_rsds);
return false; return false;
} }
@ -643,7 +640,7 @@ int main(int argc, char *argv[])
} }
out_kdbg: out_kdbg:
free(kdbg); g_free(kdbg);
out_pdb: out_pdb:
pdb_exit(&pdb); pdb_exit(&pdb);
out_pdb_file: out_pdb_file:

View File

@ -25,6 +25,10 @@
static uint32_t pdb_get_file_size(const struct pdb_reader *r, unsigned idx) static uint32_t pdb_get_file_size(const struct pdb_reader *r, unsigned idx)
{ {
if (idx >= r->ds.toc->num_files) {
return 0;
}
return r->ds.toc->file_size[idx]; return r->ds.toc->file_size[idx];
} }
@ -90,18 +94,18 @@ uint64_t pdb_resolve(uint64_t img_base, struct pdb_reader *r, const char *name)
static void pdb_reader_ds_exit(struct pdb_reader *r) static void pdb_reader_ds_exit(struct pdb_reader *r)
{ {
free(r->ds.toc); g_free(r->ds.toc);
} }
static void pdb_exit_symbols(struct pdb_reader *r) static void pdb_exit_symbols(struct pdb_reader *r)
{ {
free(r->modimage); g_free(r->modimage);
free(r->symbols); g_free(r->symbols);
} }
static void pdb_exit_segments(struct pdb_reader *r) static void pdb_exit_segments(struct pdb_reader *r)
{ {
free(r->segs); g_free(r->segs);
} }
static void *pdb_ds_read(const PDB_DS_HEADER *header, static void *pdb_ds_read(const PDB_DS_HEADER *header,
@ -116,10 +120,7 @@ static void *pdb_ds_read(const PDB_DS_HEADER *header,
nBlocks = (size + header->block_size - 1) / header->block_size; nBlocks = (size + header->block_size - 1) / header->block_size;
buffer = malloc(nBlocks * header->block_size); buffer = g_malloc(nBlocks * header->block_size);
if (!buffer) {
return NULL;
}
for (i = 0; i < nBlocks; i++) { for (i = 0; i < nBlocks; i++) {
memcpy(buffer + i * header->block_size, (const char *)header + memcpy(buffer + i * header->block_size, (const char *)header +
@ -159,16 +160,17 @@ static void *pdb_ds_read_file(struct pdb_reader* r, uint32_t file_number)
static int pdb_init_segments(struct pdb_reader *r) static int pdb_init_segments(struct pdb_reader *r)
{ {
char *segs;
unsigned stream_idx = r->segments; unsigned stream_idx = r->segments;
segs = pdb_ds_read_file(r, stream_idx); r->segs = pdb_ds_read_file(r, stream_idx);
if (!segs) { if (!r->segs) {
return 1; return 1;
} }
r->segs = segs;
r->segs_size = pdb_get_file_size(r, stream_idx); r->segs_size = pdb_get_file_size(r, stream_idx);
if (!r->segs_size) {
return 1;
}
return 0; return 0;
} }
@ -201,7 +203,7 @@ static int pdb_init_symbols(struct pdb_reader *r)
return 0; return 0;
out_symbols: out_symbols:
free(symbols); g_free(symbols);
return err; return err;
} }
@ -258,7 +260,7 @@ static int pdb_reader_init(struct pdb_reader *r, void *data)
out_sym: out_sym:
pdb_exit_symbols(r); pdb_exit_symbols(r);
out_root: out_root:
free(r->ds.root); g_free(r->ds.root);
out_ds: out_ds:
pdb_reader_ds_exit(r); pdb_reader_ds_exit(r);
@ -269,7 +271,7 @@ static void pdb_reader_exit(struct pdb_reader *r)
{ {
pdb_exit_segments(r); pdb_exit_segments(r);
pdb_exit_symbols(r); pdb_exit_symbols(r);
free(r->ds.root); g_free(r->ds.root);
pdb_reader_ds_exit(r); pdb_reader_ds_exit(r);
} }

View File

@ -94,10 +94,7 @@ static int init_states(QEMU_Elf *qe)
printf("%zu CPU states has been found\n", cpu_nr); printf("%zu CPU states has been found\n", cpu_nr);
qe->state = malloc(sizeof(*qe->state) * cpu_nr); qe->state = g_new(QEMUCPUState*, cpu_nr);
if (!qe->state) {
return 1;
}
cpu_nr = 0; cpu_nr = 0;
@ -115,7 +112,7 @@ static int init_states(QEMU_Elf *qe)
static void exit_states(QEMU_Elf *qe) static void exit_states(QEMU_Elf *qe)
{ {
free(qe->state); g_free(qe->state);
} }
static bool check_ehdr(QEMU_Elf *qe) static bool check_ehdr(QEMU_Elf *qe)

View File

@ -46,6 +46,7 @@ the following architecture extensions:
- FEAT_HCX (Support for the HCRX_EL2 register) - FEAT_HCX (Support for the HCRX_EL2 register)
- FEAT_HPDS (Hierarchical permission disables) - FEAT_HPDS (Hierarchical permission disables)
- FEAT_HPDS2 (Translation table page-based hardware attributes) - FEAT_HPDS2 (Translation table page-based hardware attributes)
- FEAT_HPMN0 (Setting of MDCR_EL2.HPMN to zero)
- FEAT_I8MM (AArch64 Int8 matrix multiplication instructions) - FEAT_I8MM (AArch64 Int8 matrix multiplication instructions)
- FEAT_IDST (ID space trap handling) - FEAT_IDST (ID space trap handling)
- FEAT_IESB (Implicit error synchronization event) - FEAT_IESB (Implicit error synchronization event)

View File

@ -722,83 +722,34 @@ static void do_cpu_reset(void *opaque)
cpu_set_pc(cs, entry); cpu_set_pc(cs, entry);
} else { } else {
/* If we are booting Linux then we need to check whether we are
* booting into secure or non-secure state and adjust the state
* accordingly. Out of reset, ARM is defined to be in secure state
* (SCR.NS = 0), we change that here if non-secure boot has been
* requested.
*/
if (arm_feature(env, ARM_FEATURE_EL3)) {
/* AArch64 is defined to come out of reset into EL3 if enabled.
* If we are booting Linux then we need to adjust our EL as
* Linux expects us to be in EL2 or EL1. AArch32 resets into
* SVC, which Linux expects, so no privilege/exception level to
* adjust.
*/
if (env->aarch64) {
env->cp15.scr_el3 |= SCR_RW;
if (arm_feature(env, ARM_FEATURE_EL2)) {
env->cp15.hcr_el2 |= HCR_RW;
env->pstate = PSTATE_MODE_EL2h;
} else {
env->pstate = PSTATE_MODE_EL1h;
}
if (cpu_isar_feature(aa64_pauth, cpu)) {
env->cp15.scr_el3 |= SCR_API | SCR_APK;
}
if (cpu_isar_feature(aa64_mte, cpu)) {
env->cp15.scr_el3 |= SCR_ATA;
}
if (cpu_isar_feature(aa64_sve, cpu)) {
env->cp15.cptr_el[3] |= R_CPTR_EL3_EZ_MASK;
env->vfp.zcr_el[3] = 0xf;
}
if (cpu_isar_feature(aa64_sme, cpu)) {
env->cp15.cptr_el[3] |= R_CPTR_EL3_ESM_MASK;
env->cp15.scr_el3 |= SCR_ENTP2;
env->vfp.smcr_el[3] = 0xf;
}
if (cpu_isar_feature(aa64_hcx, cpu)) {
env->cp15.scr_el3 |= SCR_HXEN;
}
if (cpu_isar_feature(aa64_fgt, cpu)) {
env->cp15.scr_el3 |= SCR_FGTEN;
}
/* AArch64 kernels never boot in secure mode */
assert(!info->secure_boot);
/* This hook is only supported for AArch32 currently:
* bootloader_aarch64[] will not call the hook, and
* the code above has already dropped us into EL2 or EL1.
*/
assert(!info->secure_board_setup);
}
if (arm_feature(env, ARM_FEATURE_EL2)) {
/* If we have EL2 then Linux expects the HVC insn to work */
env->cp15.scr_el3 |= SCR_HCE;
}
/* Set to non-secure if not a secure boot */
if (!info->secure_boot &&
(cs != first_cpu || !info->secure_board_setup)) {
/* Linux expects non-secure state */
env->cp15.scr_el3 |= SCR_NS;
/* Set NSACR.{CP11,CP10} so NS can access the FPU */
env->cp15.nsacr |= 3 << 10;
}
}
if (!env->aarch64 && !info->secure_boot &&
arm_feature(env, ARM_FEATURE_EL2)) {
/* /*
* This is an AArch32 boot not to Secure state, and * If we are booting Linux then we might need to do so at:
* we have Hyp mode available, so boot the kernel into * - AArch64 NS EL2 or NS EL1
* Hyp mode. This is not how the CPU comes out of reset, * - AArch32 Secure SVC (EL3)
* so we need to manually put it there. * - AArch32 NS Hyp (EL2)
* - AArch32 NS SVC (EL1)
* Configure the CPU in the way boot firmware would do to
* drop us down to the appropriate level.
*/ */
cpsr_write(env, ARM_CPU_MODE_HYP, CPSR_M, CPSRWriteRaw); int target_el = arm_feature(env, ARM_FEATURE_EL2) ? 2 : 1;
if (env->aarch64) {
/*
* AArch64 kernels never boot in secure mode, and we don't
* support the secure_board_setup hook for AArch64.
*/
assert(!info->secure_boot);
assert(!info->secure_board_setup);
} else {
if (arm_feature(env, ARM_FEATURE_EL3) &&
(info->secure_boot ||
(info->secure_board_setup && cs == first_cpu))) {
/* Start this CPU in Secure SVC */
target_el = 3;
} }
}
arm_emulate_firmware_reset(cs, target_el);
if (cs == first_cpu) { if (cs == first_cpu) {
AddressSpace *as = arm_boot_address_space(cpu, info); AddressSpace *as = arm_boot_address_space(cpu, info);

View File

@ -2,6 +2,7 @@
* ARM SBSA Reference Platform emulation * ARM SBSA Reference Platform emulation
* *
* Copyright (c) 2018 Linaro Limited * Copyright (c) 2018 Linaro Limited
* Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
* Written by Hongbo Zhang <hongbo.zhang@linaro.org> * Written by Hongbo Zhang <hongbo.zhang@linaro.org>
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
@ -30,6 +31,7 @@
#include "exec/hwaddr.h" #include "exec/hwaddr.h"
#include "kvm_arm.h" #include "kvm_arm.h"
#include "hw/arm/boot.h" #include "hw/arm/boot.h"
#include "hw/arm/bsa.h"
#include "hw/arm/fdt.h" #include "hw/arm/fdt.h"
#include "hw/arm/smmuv3.h" #include "hw/arm/smmuv3.h"
#include "hw/block/flash.h" #include "hw/block/flash.h"
@ -55,14 +57,6 @@
#define NUM_SMMU_IRQS 4 #define NUM_SMMU_IRQS 4
#define NUM_SATA_PORTS 6 #define NUM_SATA_PORTS 6
#define VIRTUAL_PMU_IRQ 7
#define ARCH_GIC_MAINT_IRQ 9
#define ARCH_TIMER_VIRT_IRQ 11
#define ARCH_TIMER_S_EL1_IRQ 13
#define ARCH_TIMER_NS_EL1_IRQ 14
#define ARCH_TIMER_NS_EL2_IRQ 10
#define ARCH_TIMER_NS_EL2_VIRT_IRQ 12
enum { enum {
SBSA_FLASH, SBSA_FLASH,
SBSA_MEM, SBSA_MEM,
@ -479,7 +473,7 @@ static void create_gic(SBSAMachineState *sms, MemoryRegion *mem)
*/ */
for (i = 0; i < smp_cpus; i++) { for (i = 0; i < smp_cpus; i++) {
DeviceState *cpudev = DEVICE(qemu_get_cpu(i)); DeviceState *cpudev = DEVICE(qemu_get_cpu(i));
int ppibase = NUM_IRQS + i * GIC_INTERNAL + GIC_NR_SGIS; int intidbase = NUM_IRQS + i * GIC_INTERNAL;
int irq; int irq;
/* /*
* Mapping from the output timer irq lines from the CPU to the * Mapping from the output timer irq lines from the CPU to the
@ -496,14 +490,17 @@ static void create_gic(SBSAMachineState *sms, MemoryRegion *mem)
for (irq = 0; irq < ARRAY_SIZE(timer_irq); irq++) { for (irq = 0; irq < ARRAY_SIZE(timer_irq); irq++) {
qdev_connect_gpio_out(cpudev, irq, qdev_connect_gpio_out(cpudev, irq,
qdev_get_gpio_in(sms->gic, qdev_get_gpio_in(sms->gic,
ppibase + timer_irq[irq])); intidbase + timer_irq[irq]));
} }
qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt", 0, qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt", 0,
qdev_get_gpio_in(sms->gic, ppibase qdev_get_gpio_in(sms->gic,
intidbase
+ ARCH_GIC_MAINT_IRQ)); + ARCH_GIC_MAINT_IRQ));
qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0, qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0,
qdev_get_gpio_in(sms->gic, ppibase qdev_get_gpio_in(sms->gic,
intidbase
+ VIRTUAL_PMU_IRQ)); + VIRTUAL_PMU_IRQ));
sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ)); sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));

View File

@ -38,33 +38,71 @@ REG32(IDR0, 0x0)
FIELD(IDR0, S1P, 1 , 1) FIELD(IDR0, S1P, 1 , 1)
FIELD(IDR0, TTF, 2 , 2) FIELD(IDR0, TTF, 2 , 2)
FIELD(IDR0, COHACC, 4 , 1) FIELD(IDR0, COHACC, 4 , 1)
FIELD(IDR0, BTM, 5 , 1)
FIELD(IDR0, HTTU, 6 , 2)
FIELD(IDR0, DORMHINT, 8 , 1)
FIELD(IDR0, HYP, 9 , 1)
FIELD(IDR0, ATS, 10, 1)
FIELD(IDR0, NS1ATS, 11, 1)
FIELD(IDR0, ASID16, 12, 1) FIELD(IDR0, ASID16, 12, 1)
FIELD(IDR0, MSI, 13, 1)
FIELD(IDR0, SEV, 14, 1)
FIELD(IDR0, ATOS, 15, 1)
FIELD(IDR0, PRI, 16, 1)
FIELD(IDR0, VMW, 17, 1)
FIELD(IDR0, VMID16, 18, 1) FIELD(IDR0, VMID16, 18, 1)
FIELD(IDR0, CD2L, 19, 1)
FIELD(IDR0, VATOS, 20, 1)
FIELD(IDR0, TTENDIAN, 21, 2) FIELD(IDR0, TTENDIAN, 21, 2)
FIELD(IDR0, ATSRECERR, 23, 1)
FIELD(IDR0, STALL_MODEL, 24, 2) FIELD(IDR0, STALL_MODEL, 24, 2)
FIELD(IDR0, TERM_MODEL, 26, 1) FIELD(IDR0, TERM_MODEL, 26, 1)
FIELD(IDR0, STLEVEL, 27, 2) FIELD(IDR0, STLEVEL, 27, 2)
FIELD(IDR0, RME_IMPL, 30, 1)
REG32(IDR1, 0x4) REG32(IDR1, 0x4)
FIELD(IDR1, SIDSIZE, 0 , 6) FIELD(IDR1, SIDSIZE, 0 , 6)
FIELD(IDR1, SSIDSIZE, 6 , 5)
FIELD(IDR1, PRIQS, 11, 5)
FIELD(IDR1, EVENTQS, 16, 5) FIELD(IDR1, EVENTQS, 16, 5)
FIELD(IDR1, CMDQS, 21, 5) FIELD(IDR1, CMDQS, 21, 5)
FIELD(IDR1, ATTR_PERMS_OVR, 26, 1)
FIELD(IDR1, ATTR_TYPES_OVR, 27, 1)
FIELD(IDR1, REL, 28, 1)
FIELD(IDR1, QUEUES_PRESET, 29, 1)
FIELD(IDR1, TABLES_PRESET, 30, 1)
FIELD(IDR1, ECMDQ, 31, 1)
#define SMMU_IDR1_SIDSIZE 16 #define SMMU_IDR1_SIDSIZE 16
#define SMMU_CMDQS 19 #define SMMU_CMDQS 19
#define SMMU_EVENTQS 19 #define SMMU_EVENTQS 19
REG32(IDR2, 0x8) REG32(IDR2, 0x8)
FIELD(IDR2, BA_VATOS, 0, 10)
REG32(IDR3, 0xc) REG32(IDR3, 0xc)
FIELD(IDR3, HAD, 2, 1); FIELD(IDR3, HAD, 2, 1);
FIELD(IDR3, PBHA, 3, 1);
FIELD(IDR3, XNX, 4, 1);
FIELD(IDR3, PPS, 5, 1);
FIELD(IDR3, MPAM, 7, 1);
FIELD(IDR3, FWB, 8, 1);
FIELD(IDR3, STT, 9, 1);
FIELD(IDR3, RIL, 10, 1); FIELD(IDR3, RIL, 10, 1);
FIELD(IDR3, BBML, 11, 2); FIELD(IDR3, BBML, 11, 2);
FIELD(IDR3, E0PD, 13, 1);
FIELD(IDR3, PTWNNC, 14, 1);
FIELD(IDR3, DPT, 15, 1);
REG32(IDR4, 0x10) REG32(IDR4, 0x10)
REG32(IDR5, 0x14) REG32(IDR5, 0x14)
FIELD(IDR5, OAS, 0, 3); FIELD(IDR5, OAS, 0, 3);
FIELD(IDR5, GRAN4K, 4, 1); FIELD(IDR5, GRAN4K, 4, 1);
FIELD(IDR5, GRAN16K, 5, 1); FIELD(IDR5, GRAN16K, 5, 1);
FIELD(IDR5, GRAN64K, 6, 1); FIELD(IDR5, GRAN64K, 6, 1);
FIELD(IDR5, VAX, 10, 2);
FIELD(IDR5, STALL_MAX, 16, 16);
#define SMMU_IDR5_OAS 4 #define SMMU_IDR5_OAS 4

View File

@ -278,15 +278,19 @@ static void smmuv3_init_regs(SMMUv3State *s)
s->idr[1] = FIELD_DP32(s->idr[1], IDR1, EVENTQS, SMMU_EVENTQS); s->idr[1] = FIELD_DP32(s->idr[1], IDR1, EVENTQS, SMMU_EVENTQS);
s->idr[1] = FIELD_DP32(s->idr[1], IDR1, CMDQS, SMMU_CMDQS); s->idr[1] = FIELD_DP32(s->idr[1], IDR1, CMDQS, SMMU_CMDQS);
s->idr[3] = FIELD_DP32(s->idr[3], IDR3, RIL, 1);
s->idr[3] = FIELD_DP32(s->idr[3], IDR3, HAD, 1); s->idr[3] = FIELD_DP32(s->idr[3], IDR3, HAD, 1);
if (FIELD_EX32(s->idr[0], IDR0, S2P)) {
/* XNX is a stage-2-specific feature */
s->idr[3] = FIELD_DP32(s->idr[3], IDR3, XNX, 1);
}
s->idr[3] = FIELD_DP32(s->idr[3], IDR3, RIL, 1);
s->idr[3] = FIELD_DP32(s->idr[3], IDR3, BBML, 2); s->idr[3] = FIELD_DP32(s->idr[3], IDR3, BBML, 2);
s->idr[5] = FIELD_DP32(s->idr[5], IDR5, OAS, SMMU_IDR5_OAS); /* 44 bits */
/* 4K, 16K and 64K granule support */ /* 4K, 16K and 64K granule support */
s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN4K, 1); s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN4K, 1);
s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN16K, 1); s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN16K, 1);
s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN64K, 1); s->idr[5] = FIELD_DP32(s->idr[5], IDR5, GRAN64K, 1);
s->idr[5] = FIELD_DP32(s->idr[5], IDR5, OAS, SMMU_IDR5_OAS); /* 44 bits */
s->cmdq.base = deposit64(s->cmdq.base, 0, 5, SMMU_CMDQS); s->cmdq.base = deposit64(s->cmdq.base, 0, 5, SMMU_CMDQS);
s->cmdq.prod = 0; s->cmdq.prod = 0;

View File

@ -601,21 +601,21 @@ build_gtdt(GArray *table_data, BIOSLinker *linker, VirtMachineState *vms)
* The interrupt values are the same with the device tree when adding 16 * The interrupt values are the same with the device tree when adding 16
*/ */
/* Secure EL1 timer GSIV */ /* Secure EL1 timer GSIV */
build_append_int_noprefix(table_data, ARCH_TIMER_S_EL1_IRQ + 16, 4); build_append_int_noprefix(table_data, ARCH_TIMER_S_EL1_IRQ, 4);
/* Secure EL1 timer Flags */ /* Secure EL1 timer Flags */
build_append_int_noprefix(table_data, irqflags, 4); build_append_int_noprefix(table_data, irqflags, 4);
/* Non-Secure EL1 timer GSIV */ /* Non-Secure EL1 timer GSIV */
build_append_int_noprefix(table_data, ARCH_TIMER_NS_EL1_IRQ + 16, 4); build_append_int_noprefix(table_data, ARCH_TIMER_NS_EL1_IRQ, 4);
/* Non-Secure EL1 timer Flags */ /* Non-Secure EL1 timer Flags */
build_append_int_noprefix(table_data, irqflags | build_append_int_noprefix(table_data, irqflags |
1UL << 2, /* Always-on Capability */ 1UL << 2, /* Always-on Capability */
4); 4);
/* Virtual timer GSIV */ /* Virtual timer GSIV */
build_append_int_noprefix(table_data, ARCH_TIMER_VIRT_IRQ + 16, 4); build_append_int_noprefix(table_data, ARCH_TIMER_VIRT_IRQ, 4);
/* Virtual Timer Flags */ /* Virtual Timer Flags */
build_append_int_noprefix(table_data, irqflags, 4); build_append_int_noprefix(table_data, irqflags, 4);
/* Non-Secure EL2 timer GSIV */ /* Non-Secure EL2 timer GSIV */
build_append_int_noprefix(table_data, ARCH_TIMER_NS_EL2_IRQ + 16, 4); build_append_int_noprefix(table_data, ARCH_TIMER_NS_EL2_IRQ, 4);
/* Non-Secure EL2 timer Flags */ /* Non-Secure EL2 timer Flags */
build_append_int_noprefix(table_data, irqflags, 4); build_append_int_noprefix(table_data, irqflags, 4);
/* CntReadBase Physical address */ /* CntReadBase Physical address */
@ -729,9 +729,9 @@ build_madt(GArray *table_data, BIOSLinker *linker, VirtMachineState *vms)
for (i = 0; i < MACHINE(vms)->smp.cpus; i++) { for (i = 0; i < MACHINE(vms)->smp.cpus; i++) {
ARMCPU *armcpu = ARM_CPU(qemu_get_cpu(i)); ARMCPU *armcpu = ARM_CPU(qemu_get_cpu(i));
uint64_t physical_base_address = 0, gich = 0, gicv = 0; uint64_t physical_base_address = 0, gich = 0, gicv = 0;
uint32_t vgic_interrupt = vms->virt ? PPI(ARCH_GIC_MAINT_IRQ) : 0; uint32_t vgic_interrupt = vms->virt ? ARCH_GIC_MAINT_IRQ : 0;
uint32_t pmu_interrupt = arm_feature(&armcpu->env, ARM_FEATURE_PMU) ? uint32_t pmu_interrupt = arm_feature(&armcpu->env, ARM_FEATURE_PMU) ?
PPI(VIRTUAL_PMU_IRQ) : 0; VIRTUAL_PMU_IRQ : 0;
if (vms->gic_version == VIRT_GIC_VERSION_2) { if (vms->gic_version == VIRT_GIC_VERSION_2) {
physical_base_address = memmap[VIRT_GIC_CPU].base; physical_base_address = memmap[VIRT_GIC_CPU].base;

View File

@ -366,10 +366,14 @@ static void fdt_add_timer_nodes(const VirtMachineState *vms)
} }
qemu_fdt_setprop(ms->fdt, "/timer", "always-on", NULL, 0); qemu_fdt_setprop(ms->fdt, "/timer", "always-on", NULL, 0);
qemu_fdt_setprop_cells(ms->fdt, "/timer", "interrupts", qemu_fdt_setprop_cells(ms->fdt, "/timer", "interrupts",
GIC_FDT_IRQ_TYPE_PPI, ARCH_TIMER_S_EL1_IRQ, irqflags, GIC_FDT_IRQ_TYPE_PPI,
GIC_FDT_IRQ_TYPE_PPI, ARCH_TIMER_NS_EL1_IRQ, irqflags, INTID_TO_PPI(ARCH_TIMER_S_EL1_IRQ), irqflags,
GIC_FDT_IRQ_TYPE_PPI, ARCH_TIMER_VIRT_IRQ, irqflags, GIC_FDT_IRQ_TYPE_PPI,
GIC_FDT_IRQ_TYPE_PPI, ARCH_TIMER_NS_EL2_IRQ, irqflags); INTID_TO_PPI(ARCH_TIMER_NS_EL1_IRQ), irqflags,
GIC_FDT_IRQ_TYPE_PPI,
INTID_TO_PPI(ARCH_TIMER_VIRT_IRQ), irqflags,
GIC_FDT_IRQ_TYPE_PPI,
INTID_TO_PPI(ARCH_TIMER_NS_EL2_IRQ), irqflags);
} }
static void fdt_add_cpu_nodes(const VirtMachineState *vms) static void fdt_add_cpu_nodes(const VirtMachineState *vms)
@ -800,7 +804,7 @@ static void create_gic(VirtMachineState *vms, MemoryRegion *mem)
*/ */
for (i = 0; i < smp_cpus; i++) { for (i = 0; i < smp_cpus; i++) {
DeviceState *cpudev = DEVICE(qemu_get_cpu(i)); DeviceState *cpudev = DEVICE(qemu_get_cpu(i));
int ppibase = NUM_IRQS + i * GIC_INTERNAL + GIC_NR_SGIS; int intidbase = NUM_IRQS + i * GIC_INTERNAL;
/* Mapping from the output timer irq lines from the CPU to the /* Mapping from the output timer irq lines from the CPU to the
* GIC PPI inputs we use for the virt board. * GIC PPI inputs we use for the virt board.
*/ */
@ -814,22 +818,22 @@ static void create_gic(VirtMachineState *vms, MemoryRegion *mem)
for (unsigned irq = 0; irq < ARRAY_SIZE(timer_irq); irq++) { for (unsigned irq = 0; irq < ARRAY_SIZE(timer_irq); irq++) {
qdev_connect_gpio_out(cpudev, irq, qdev_connect_gpio_out(cpudev, irq,
qdev_get_gpio_in(vms->gic, qdev_get_gpio_in(vms->gic,
ppibase + timer_irq[irq])); intidbase + timer_irq[irq]));
} }
if (vms->gic_version != VIRT_GIC_VERSION_2) { if (vms->gic_version != VIRT_GIC_VERSION_2) {
qemu_irq irq = qdev_get_gpio_in(vms->gic, qemu_irq irq = qdev_get_gpio_in(vms->gic,
ppibase + ARCH_GIC_MAINT_IRQ); intidbase + ARCH_GIC_MAINT_IRQ);
qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt", qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt",
0, irq); 0, irq);
} else if (vms->virt) { } else if (vms->virt) {
qemu_irq irq = qdev_get_gpio_in(vms->gic, qemu_irq irq = qdev_get_gpio_in(vms->gic,
ppibase + ARCH_GIC_MAINT_IRQ); intidbase + ARCH_GIC_MAINT_IRQ);
sysbus_connect_irq(gicbusdev, i + 4 * smp_cpus, irq); sysbus_connect_irq(gicbusdev, i + 4 * smp_cpus, irq);
} }
qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0, qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0,
qdev_get_gpio_in(vms->gic, ppibase qdev_get_gpio_in(vms->gic, intidbase
+ VIRTUAL_PMU_IRQ)); + VIRTUAL_PMU_IRQ));
sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ)); sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
@ -1989,7 +1993,7 @@ static void virt_cpu_post_init(VirtMachineState *vms, MemoryRegion *sysmem)
if (pmu) { if (pmu) {
assert(arm_feature(&ARM_CPU(cpu)->env, ARM_FEATURE_PMU)); assert(arm_feature(&ARM_CPU(cpu)->env, ARM_FEATURE_PMU));
if (kvm_irqchip_in_kernel()) { if (kvm_irqchip_in_kernel()) {
kvm_arm_pmu_set_irq(cpu, PPI(VIRTUAL_PMU_IRQ)); kvm_arm_pmu_set_irq(cpu, VIRTUAL_PMU_IRQ);
} }
kvm_arm_pmu_init(cpu); kvm_arm_pmu_init(cpu);
} }

View File

@ -12,7 +12,7 @@
#include "migration/vmstate.h" #include "migration/vmstate.h"
#include "hw/irq.h" #include "hw/irq.h"
#include "hw/misc/bcm2835_mbox_defs.h" #include "hw/misc/bcm2835_mbox_defs.h"
#include "hw/misc/raspberrypi-fw-defs.h" #include "hw/arm/raspberrypi-fw-defs.h"
#include "sysemu/dma.h" #include "sysemu/dma.h"
#include "qemu/log.h" #include "qemu/log.h"
#include "qemu/module.h" #include "qemu/module.h"

View File

@ -2,6 +2,7 @@
* QEMU model of the Xilinx BBRAM Battery Backed RAM * QEMU model of the Xilinx BBRAM Battery Backed RAM
* *
* Copyright (c) 2014-2021 Xilinx Inc. * Copyright (c) 2014-2021 Xilinx Inc.
* Copyright (c) 2023 Advanced Micro Devices, Inc.
* *
* Permission is hereby granted, free of charge, to any person obtaining a copy * Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal * of this software and associated documentation files (the "Software"), to deal
@ -416,9 +417,9 @@ static RegisterAccessInfo bbram_ctrl_regs_info[] = {
} }
}; };
static void bbram_ctrl_reset(DeviceState *dev) static void bbram_ctrl_reset_hold(Object *obj)
{ {
XlnxBBRam *s = XLNX_BBRAM(dev); XlnxBBRam *s = XLNX_BBRAM(obj);
unsigned int i; unsigned int i;
for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) { for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) {
@ -522,8 +523,9 @@ static Property bbram_ctrl_props[] = {
static void bbram_ctrl_class_init(ObjectClass *klass, void *data) static void bbram_ctrl_class_init(ObjectClass *klass, void *data)
{ {
DeviceClass *dc = DEVICE_CLASS(klass); DeviceClass *dc = DEVICE_CLASS(klass);
ResettableClass *rc = RESETTABLE_CLASS(klass);
dc->reset = bbram_ctrl_reset; rc->phases.hold = bbram_ctrl_reset_hold;
dc->realize = bbram_ctrl_realize; dc->realize = bbram_ctrl_realize;
dc->vmsd = &vmstate_bbram_ctrl; dc->vmsd = &vmstate_bbram_ctrl;
device_class_set_props(dc, bbram_ctrl_props); device_class_set_props(dc, bbram_ctrl_props);

View File

@ -2,6 +2,7 @@
* QEMU model of the Versal eFuse controller * QEMU model of the Versal eFuse controller
* *
* Copyright (c) 2020 Xilinx Inc. * Copyright (c) 2020 Xilinx Inc.
* Copyright (c) 2023 Advanced Micro Devices, Inc.
* *
* Permission is hereby granted, free of charge, to any person obtaining a copy * Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal * of this software and associated documentation files (the "Software"), to deal
@ -657,9 +658,9 @@ static void efuse_ctrl_register_reset(RegisterInfo *reg)
register_reset(reg); register_reset(reg);
} }
static void efuse_ctrl_reset(DeviceState *dev) static void efuse_ctrl_reset_hold(Object *obj)
{ {
XlnxVersalEFuseCtrl *s = XLNX_VERSAL_EFUSE_CTRL(dev); XlnxVersalEFuseCtrl *s = XLNX_VERSAL_EFUSE_CTRL(obj);
unsigned int i; unsigned int i;
for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) { for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) {
@ -749,8 +750,9 @@ static Property efuse_ctrl_props[] = {
static void efuse_ctrl_class_init(ObjectClass *klass, void *data) static void efuse_ctrl_class_init(ObjectClass *klass, void *data)
{ {
DeviceClass *dc = DEVICE_CLASS(klass); DeviceClass *dc = DEVICE_CLASS(klass);
ResettableClass *rc = RESETTABLE_CLASS(klass);
dc->reset = efuse_ctrl_reset; rc->phases.hold = efuse_ctrl_reset_hold;
dc->realize = efuse_ctrl_realize; dc->realize = efuse_ctrl_realize;
dc->vmsd = &vmstate_efuse_ctrl; dc->vmsd = &vmstate_efuse_ctrl;
device_class_set_props(dc, efuse_ctrl_props); device_class_set_props(dc, efuse_ctrl_props);

View File

@ -2,6 +2,7 @@
* QEMU model of the ZynqMP eFuse * QEMU model of the ZynqMP eFuse
* *
* Copyright (c) 2015 Xilinx Inc. * Copyright (c) 2015 Xilinx Inc.
* Copyright (c) 2023 Advanced Micro Devices, Inc.
* *
* Written by Edgar E. Iglesias <edgari@xilinx.com> * Written by Edgar E. Iglesias <edgari@xilinx.com>
* *
@ -769,9 +770,9 @@ static void zynqmp_efuse_register_reset(RegisterInfo *reg)
register_reset(reg); register_reset(reg);
} }
static void zynqmp_efuse_reset(DeviceState *dev) static void zynqmp_efuse_reset_hold(Object *obj)
{ {
XlnxZynqMPEFuse *s = XLNX_ZYNQMP_EFUSE(dev); XlnxZynqMPEFuse *s = XLNX_ZYNQMP_EFUSE(obj);
unsigned int i; unsigned int i;
for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) { for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) {
@ -837,8 +838,9 @@ static Property zynqmp_efuse_props[] = {
static void zynqmp_efuse_class_init(ObjectClass *klass, void *data) static void zynqmp_efuse_class_init(ObjectClass *klass, void *data)
{ {
DeviceClass *dc = DEVICE_CLASS(klass); DeviceClass *dc = DEVICE_CLASS(klass);
ResettableClass *rc = RESETTABLE_CLASS(klass);
dc->reset = zynqmp_efuse_reset; rc->phases.hold = zynqmp_efuse_reset_hold;
dc->realize = zynqmp_efuse_realize; dc->realize = zynqmp_efuse_realize;
dc->vmsd = &vmstate_efuse; dc->vmsd = &vmstate_efuse;
device_class_set_props(dc, zynqmp_efuse_props); device_class_set_props(dc, zynqmp_efuse_props);

View File

@ -138,6 +138,9 @@ static int64_t npcm7xx_timer_count_to_ns(NPCM7xxTimer *t, uint32_t count)
/* Convert a time interval in nanoseconds to a timer cycle count. */ /* Convert a time interval in nanoseconds to a timer cycle count. */
static uint32_t npcm7xx_timer_ns_to_count(NPCM7xxTimer *t, int64_t ns) static uint32_t npcm7xx_timer_ns_to_count(NPCM7xxTimer *t, int64_t ns)
{ {
if (ns < 0) {
return 0;
}
return clock_ns_to_ticks(t->ctrl->clock, ns) / return clock_ns_to_ticks(t->ctrl->clock, ns) /
npcm7xx_tcsr_prescaler(t->tcsr); npcm7xx_tcsr_prescaler(t->tcsr);
} }

35
include/hw/arm/bsa.h Normal file
View File

@ -0,0 +1,35 @@
/*
* Common definitions for Arm Base System Architecture (BSA) platforms.
*
* Copyright (c) 2015 Linaro Limited
* Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2 or later, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along with
* this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
#ifndef QEMU_ARM_BSA_H
#define QEMU_ARM_BSA_H
/* These are architectural INTID values */
#define VIRTUAL_PMU_IRQ 23
#define ARCH_GIC_MAINT_IRQ 25
#define ARCH_TIMER_NS_EL2_IRQ 26
#define ARCH_TIMER_VIRT_IRQ 27
#define ARCH_TIMER_NS_EL2_VIRT_IRQ 28
#define ARCH_TIMER_S_EL1_IRQ 29
#define ARCH_TIMER_NS_EL1_IRQ 30
#define INTID_TO_PPI(irq) ((irq) - 16)
#endif /* QEMU_ARM_BSA_H */

View File

@ -30,7 +30,7 @@
#include "hw/intc/exynos4210_gic.h" #include "hw/intc/exynos4210_gic.h"
#include "hw/intc/exynos4210_combiner.h" #include "hw/intc/exynos4210_combiner.h"
#include "hw/core/split-irq.h" #include "hw/core/split-irq.h"
#include "target/arm/cpu-qom.h" #include "hw/arm/boot.h"
#include "qom/object.h" #include "qom/object.h"
#define EXYNOS4210_NCPUS 2 #define EXYNOS4210_NCPUS 2

View File

@ -34,6 +34,7 @@
#include "qemu/notify.h" #include "qemu/notify.h"
#include "hw/boards.h" #include "hw/boards.h"
#include "hw/arm/boot.h" #include "hw/arm/boot.h"
#include "hw/arm/bsa.h"
#include "hw/block/flash.h" #include "hw/block/flash.h"
#include "sysemu/kvm.h" #include "sysemu/kvm.h"
#include "hw/intc/arm_gicv3_common.h" #include "hw/intc/arm_gicv3_common.h"
@ -43,17 +44,6 @@
#define NUM_VIRTIO_TRANSPORTS 32 #define NUM_VIRTIO_TRANSPORTS 32
#define NUM_SMMU_IRQS 4 #define NUM_SMMU_IRQS 4
#define ARCH_GIC_MAINT_IRQ 9
#define ARCH_TIMER_VIRT_IRQ 11
#define ARCH_TIMER_S_EL1_IRQ 13
#define ARCH_TIMER_NS_EL1_IRQ 14
#define ARCH_TIMER_NS_EL2_IRQ 10
#define VIRTUAL_PMU_IRQ 7
#define PPI(irq) ((irq) + 16)
/* See Linux kernel arch/arm64/include/asm/pvclock-abi.h */ /* See Linux kernel arch/arm64/include/asm/pvclock-abi.h */
#define PVTIME_SIZE_PER_CPU 64 #define PVTIME_SIZE_PER_CPU 64

View File

@ -34,7 +34,7 @@
#define RMAX_XLNX_BBRAM ((0x4c / 4) + 1) #define RMAX_XLNX_BBRAM ((0x4c / 4) + 1)
#define TYPE_XLNX_BBRAM "xlnx,bbram-ctrl" #define TYPE_XLNX_BBRAM "xlnx.bbram-ctrl"
OBJECT_DECLARE_SIMPLE_TYPE(XlnxBBRam, XLNX_BBRAM); OBJECT_DECLARE_SIMPLE_TYPE(XlnxBBRam, XLNX_BBRAM);
struct XlnxBBRam { struct XlnxBBRam {

View File

@ -65,60 +65,9 @@ static void arm_set_cpu_on_async_work(CPUState *target_cpu_state,
/* Initialize the cpu we are turning on */ /* Initialize the cpu we are turning on */
cpu_reset(target_cpu_state); cpu_reset(target_cpu_state);
arm_emulate_firmware_reset(target_cpu_state, info->target_el);
target_cpu_state->halted = 0; target_cpu_state->halted = 0;
if (info->target_aa64) {
if ((info->target_el < 3) && arm_feature(&target_cpu->env,
ARM_FEATURE_EL3)) {
/*
* As target mode is AArch64, we need to set lower
* exception level (the requested level 2) to AArch64
*/
target_cpu->env.cp15.scr_el3 |= SCR_RW;
}
if ((info->target_el < 2) && arm_feature(&target_cpu->env,
ARM_FEATURE_EL2)) {
/*
* As target mode is AArch64, we need to set lower
* exception level (the requested level 1) to AArch64
*/
target_cpu->env.cp15.hcr_el2 |= HCR_RW;
}
target_cpu->env.pstate = aarch64_pstate_mode(info->target_el, true);
} else {
/* We are requested to boot in AArch32 mode */
static const uint32_t mode_for_el[] = { 0,
ARM_CPU_MODE_SVC,
ARM_CPU_MODE_HYP,
ARM_CPU_MODE_SVC };
cpsr_write(&target_cpu->env, mode_for_el[info->target_el], CPSR_M,
CPSRWriteRaw);
}
if (info->target_el == 3) {
/* Processor is in secure mode */
target_cpu->env.cp15.scr_el3 &= ~SCR_NS;
} else {
/* Processor is not in secure mode */
target_cpu->env.cp15.scr_el3 |= SCR_NS;
/* Set NSACR.{CP11,CP10} so NS can access the FPU */
target_cpu->env.cp15.nsacr |= 3 << 10;
/*
* If QEMU is providing the equivalent of EL3 firmware, then we need
* to make sure a CPU targeting EL2 comes out of reset with a
* functional HVC insn.
*/
if (arm_feature(&target_cpu->env, ARM_FEATURE_EL3)
&& info->target_el == 2) {
target_cpu->env.cp15.scr_el3 |= SCR_HCE;
}
}
/* We check if the started CPU is now at the correct level */ /* We check if the started CPU is now at the correct level */
assert(info->target_el == arm_current_el(&target_cpu->env)); assert(info->target_el == arm_current_el(&target_cpu->env));

View File

@ -10,9 +10,7 @@
#ifndef TARGET_ARM_COMMON_SEMI_TARGET_H #ifndef TARGET_ARM_COMMON_SEMI_TARGET_H
#define TARGET_ARM_COMMON_SEMI_TARGET_H #define TARGET_ARM_COMMON_SEMI_TARGET_H
#ifndef CONFIG_USER_ONLY #include "target/arm/cpu-qom.h"
#include "hw/arm/boot.h"
#endif
static inline target_ulong common_semi_arg(CPUState *cs, int argno) static inline target_ulong common_semi_arg(CPUState *cs, int argno)
{ {

View File

@ -23,8 +23,6 @@
#include "hw/core/cpu.h" #include "hw/core/cpu.h"
#include "qom/object.h" #include "qom/object.h"
struct arm_boot_info;
#define TYPE_ARM_CPU "arm-cpu" #define TYPE_ARM_CPU "arm-cpu"
OBJECT_DECLARE_CPU_TYPE(ARMCPU, ARMCPUClass, ARM_CPU) OBJECT_DECLARE_CPU_TYPE(ARMCPU, ARMCPUClass, ARM_CPU)

View File

@ -553,6 +553,101 @@ static void arm_cpu_reset_hold(Object *obj)
} }
} }
void arm_emulate_firmware_reset(CPUState *cpustate, int target_el)
{
ARMCPU *cpu = ARM_CPU(cpustate);
CPUARMState *env = &cpu->env;
bool have_el3 = arm_feature(env, ARM_FEATURE_EL3);
bool have_el2 = arm_feature(env, ARM_FEATURE_EL2);
/*
* Check we have the EL we're aiming for. If that is the
* highest implemented EL, then cpu_reset has already done
* all the work.
*/
switch (target_el) {
case 3:
assert(have_el3);
return;
case 2:
assert(have_el2);
if (!have_el3) {
return;
}
break;
case 1:
if (!have_el3 && !have_el2) {
return;
}
break;
default:
g_assert_not_reached();
}
if (have_el3) {
/*
* Set the EL3 state so code can run at EL2. This should match
* the requirements set by Linux in its booting spec.
*/
if (env->aarch64) {
env->cp15.scr_el3 |= SCR_RW;
if (cpu_isar_feature(aa64_pauth, cpu)) {
env->cp15.scr_el3 |= SCR_API | SCR_APK;
}
if (cpu_isar_feature(aa64_mte, cpu)) {
env->cp15.scr_el3 |= SCR_ATA;
}
if (cpu_isar_feature(aa64_sve, cpu)) {
env->cp15.cptr_el[3] |= R_CPTR_EL3_EZ_MASK;
env->vfp.zcr_el[3] = 0xf;
}
if (cpu_isar_feature(aa64_sme, cpu)) {
env->cp15.cptr_el[3] |= R_CPTR_EL3_ESM_MASK;
env->cp15.scr_el3 |= SCR_ENTP2;
env->vfp.smcr_el[3] = 0xf;
}
if (cpu_isar_feature(aa64_hcx, cpu)) {
env->cp15.scr_el3 |= SCR_HXEN;
}
if (cpu_isar_feature(aa64_fgt, cpu)) {
env->cp15.scr_el3 |= SCR_FGTEN;
}
}
if (target_el == 2) {
/* If the guest is at EL2 then Linux expects the HVC insn to work */
env->cp15.scr_el3 |= SCR_HCE;
}
/* Put CPU into non-secure state */
env->cp15.scr_el3 |= SCR_NS;
/* Set NSACR.{CP11,CP10} so NS can access the FPU */
env->cp15.nsacr |= 3 << 10;
}
if (have_el2 && target_el < 2) {
/* Set EL2 state so code can run at EL1. */
if (env->aarch64) {
env->cp15.hcr_el2 |= HCR_RW;
}
}
/* Set the CPU to the desired state */
if (env->aarch64) {
env->pstate = aarch64_pstate_mode(target_el, true);
} else {
static const uint32_t mode_for_el[] = {
0,
ARM_CPU_MODE_SVC,
ARM_CPU_MODE_HYP,
ARM_CPU_MODE_SVC,
};
cpsr_write(env, mode_for_el[target_el], CPSR_M, CPSRWriteRaw);
}
}
#if defined(CONFIG_TCG) && !defined(CONFIG_USER_ONLY) #if defined(CONFIG_TCG) && !defined(CONFIG_USER_ONLY)
static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx, static inline bool arm_excp_unmasked(CPUState *cs, unsigned int excp_idx,

View File

@ -1149,6 +1149,28 @@ int arm_cpu_write_elf64_note(WriteCoreDumpFunction f, CPUState *cs,
int arm_cpu_write_elf32_note(WriteCoreDumpFunction f, CPUState *cs, int arm_cpu_write_elf32_note(WriteCoreDumpFunction f, CPUState *cs,
int cpuid, DumpState *s); int cpuid, DumpState *s);
/**
* arm_emulate_firmware_reset: Emulate firmware CPU reset handling
* @cpu: CPU (which must have been freshly reset)
* @target_el: exception level to put the CPU into
* @secure: whether to put the CPU in secure state
*
* When QEMU is directly running a guest kernel at a lower level than
* EL3 it implicitly emulates some aspects of the guest firmware.
* This includes that on reset we need to configure the parts of the
* CPU corresponding to EL3 so that the real guest code can run at its
* lower exception level. This function does that post-reset CPU setup,
* for when we do direct boot of a guest kernel, and for when we
* emulate PSCI and similar firmware interfaces starting a CPU at a
* lower exception level.
*
* @target_el must be an EL implemented by the CPU between 1 and 3.
* We do not support dropping into a Secure EL other than 3.
*
* It is the responsibility of the caller to call arm_rebuild_hflags().
*/
void arm_emulate_firmware_reset(CPUState *cpustate, int target_el);
#ifdef TARGET_AARCH64 #ifdef TARGET_AARCH64
int aarch64_cpu_gdb_read_register(CPUState *cpu, GByteArray *buf, int reg); int aarch64_cpu_gdb_read_register(CPUState *cpu, GByteArray *buf, int reg);
int aarch64_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); int aarch64_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg);

View File

@ -1283,7 +1283,7 @@ static bool pmevcntr_is_64_bit(CPUARMState *env, int counter)
bool hlp = env->cp15.mdcr_el2 & MDCR_HLP; bool hlp = env->cp15.mdcr_el2 & MDCR_HLP;
int hpmn = env->cp15.mdcr_el2 & MDCR_HPMN; int hpmn = env->cp15.mdcr_el2 & MDCR_HPMN;
if (hpmn != 0 && counter >= hpmn) { if (counter >= hpmn) {
return hlp; return hlp;
} }
} }
@ -2475,22 +2475,7 @@ static CPAccessResult gt_counter_access(CPUARMState *env, int timeridx,
if (!extract32(env->cp15.c14_cntkctl, timeridx, 1)) { if (!extract32(env->cp15.c14_cntkctl, timeridx, 1)) {
return CP_ACCESS_TRAP; return CP_ACCESS_TRAP;
} }
/* fall through */
/* If HCR_EL2.<E2H,TGE> == '10': check CNTHCTL_EL2.EL1PCTEN. */
if (hcr & HCR_E2H) {
if (timeridx == GTIMER_PHYS &&
!extract32(env->cp15.cnthctl_el2, 10, 1)) {
return CP_ACCESS_TRAP_EL2;
}
} else {
/* If HCR_EL2.<E2H> == 0: check CNTHCTL_EL2.EL1PCEN. */
if (has_el2 && timeridx == GTIMER_PHYS &&
!extract32(env->cp15.cnthctl_el2, 1, 1)) {
return CP_ACCESS_TRAP_EL2;
}
}
break;
case 1: case 1:
/* Check CNTHCTL_EL2.EL1PCTEN, which changes location based on E2H. */ /* Check CNTHCTL_EL2.EL1PCTEN, which changes location based on E2H. */
if (has_el2 && timeridx == GTIMER_PHYS && if (has_el2 && timeridx == GTIMER_PHYS &&

View File

@ -553,24 +553,19 @@ bool write_kvmstate_to_list(ARMCPU *cpu)
bool ok = true; bool ok = true;
for (i = 0; i < cpu->cpreg_array_len; i++) { for (i = 0; i < cpu->cpreg_array_len; i++) {
struct kvm_one_reg r;
uint64_t regidx = cpu->cpreg_indexes[i]; uint64_t regidx = cpu->cpreg_indexes[i];
uint32_t v32; uint32_t v32;
int ret; int ret;
r.id = regidx;
switch (regidx & KVM_REG_SIZE_MASK) { switch (regidx & KVM_REG_SIZE_MASK) {
case KVM_REG_SIZE_U32: case KVM_REG_SIZE_U32:
r.addr = (uintptr_t)&v32; ret = kvm_get_one_reg(cs, regidx, &v32);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &r);
if (!ret) { if (!ret) {
cpu->cpreg_values[i] = v32; cpu->cpreg_values[i] = v32;
} }
break; break;
case KVM_REG_SIZE_U64: case KVM_REG_SIZE_U64:
r.addr = (uintptr_t)(cpu->cpreg_values + i); ret = kvm_get_one_reg(cs, regidx, cpu->cpreg_values + i);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &r);
break; break;
default: default:
g_assert_not_reached(); g_assert_not_reached();
@ -589,7 +584,6 @@ bool write_list_to_kvmstate(ARMCPU *cpu, int level)
bool ok = true; bool ok = true;
for (i = 0; i < cpu->cpreg_array_len; i++) { for (i = 0; i < cpu->cpreg_array_len; i++) {
struct kvm_one_reg r;
uint64_t regidx = cpu->cpreg_indexes[i]; uint64_t regidx = cpu->cpreg_indexes[i];
uint32_t v32; uint32_t v32;
int ret; int ret;
@ -598,19 +592,17 @@ bool write_list_to_kvmstate(ARMCPU *cpu, int level)
continue; continue;
} }
r.id = regidx;
switch (regidx & KVM_REG_SIZE_MASK) { switch (regidx & KVM_REG_SIZE_MASK) {
case KVM_REG_SIZE_U32: case KVM_REG_SIZE_U32:
v32 = cpu->cpreg_values[i]; v32 = cpu->cpreg_values[i];
r.addr = (uintptr_t)&v32; ret = kvm_set_one_reg(cs, regidx, &v32);
break; break;
case KVM_REG_SIZE_U64: case KVM_REG_SIZE_U64:
r.addr = (uintptr_t)(cpu->cpreg_values + i); ret = kvm_set_one_reg(cs, regidx, cpu->cpreg_values + i);
break; break;
default: default:
g_assert_not_reached(); g_assert_not_reached();
} }
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &r);
if (ret) { if (ret) {
/* We might fail for "unknown register" and also for /* We might fail for "unknown register" and also for
* "you tried to set a register which is constant with * "you tried to set a register which is constant with
@ -709,17 +701,13 @@ int kvm_arm_sync_mpstate_to_qemu(ARMCPU *cpu)
void kvm_arm_get_virtual_time(CPUState *cs) void kvm_arm_get_virtual_time(CPUState *cs)
{ {
ARMCPU *cpu = ARM_CPU(cs); ARMCPU *cpu = ARM_CPU(cs);
struct kvm_one_reg reg = {
.id = KVM_REG_ARM_TIMER_CNT,
.addr = (uintptr_t)&cpu->kvm_vtime,
};
int ret; int ret;
if (cpu->kvm_vtime_dirty) { if (cpu->kvm_vtime_dirty) {
return; return;
} }
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg); ret = kvm_get_one_reg(cs, KVM_REG_ARM_TIMER_CNT, &cpu->kvm_vtime);
if (ret) { if (ret) {
error_report("Failed to get KVM_REG_ARM_TIMER_CNT"); error_report("Failed to get KVM_REG_ARM_TIMER_CNT");
abort(); abort();
@ -731,17 +719,13 @@ void kvm_arm_get_virtual_time(CPUState *cs)
void kvm_arm_put_virtual_time(CPUState *cs) void kvm_arm_put_virtual_time(CPUState *cs)
{ {
ARMCPU *cpu = ARM_CPU(cs); ARMCPU *cpu = ARM_CPU(cs);
struct kvm_one_reg reg = {
.id = KVM_REG_ARM_TIMER_CNT,
.addr = (uintptr_t)&cpu->kvm_vtime,
};
int ret; int ret;
if (!cpu->kvm_vtime_dirty) { if (!cpu->kvm_vtime_dirty) {
return; return;
} }
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg); ret = kvm_set_one_reg(cs, KVM_REG_ARM_TIMER_CNT, &cpu->kvm_vtime);
if (ret) { if (ret) {
error_report("Failed to set KVM_REG_ARM_TIMER_CNT"); error_report("Failed to set KVM_REG_ARM_TIMER_CNT");
abort(); abort();

View File

@ -30,7 +30,6 @@
#include "internals.h" #include "internals.h"
#include "hw/acpi/acpi.h" #include "hw/acpi/acpi.h"
#include "hw/acpi/ghes.h" #include "hw/acpi/ghes.h"
#include "hw/arm/virt.h"
static bool have_guest_debug; static bool have_guest_debug;
@ -540,14 +539,10 @@ static int kvm_arm_sve_set_vls(CPUState *cs)
{ {
ARMCPU *cpu = ARM_CPU(cs); ARMCPU *cpu = ARM_CPU(cs);
uint64_t vls[KVM_ARM64_SVE_VLS_WORDS] = { cpu->sve_vq.map }; uint64_t vls[KVM_ARM64_SVE_VLS_WORDS] = { cpu->sve_vq.map };
struct kvm_one_reg reg = {
.id = KVM_REG_ARM64_SVE_VLS,
.addr = (uint64_t)&vls[0],
};
assert(cpu->sve_max_vq <= KVM_ARM64_SVE_VQ_MAX); assert(cpu->sve_max_vq <= KVM_ARM64_SVE_VQ_MAX);
return kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg); return kvm_set_one_reg(cs, KVM_REG_ARM64_SVE_VLS, &vls[0]);
} }
#define ARM_CPU_ID_MPIDR 3, 0, 0, 0, 5 #define ARM_CPU_ID_MPIDR 3, 0, 0, 0, 5
@ -726,19 +721,17 @@ static void kvm_inject_arm_sea(CPUState *c)
static int kvm_arch_put_fpsimd(CPUState *cs) static int kvm_arch_put_fpsimd(CPUState *cs)
{ {
CPUARMState *env = &ARM_CPU(cs)->env; CPUARMState *env = &ARM_CPU(cs)->env;
struct kvm_one_reg reg;
int i, ret; int i, ret;
for (i = 0; i < 32; i++) { for (i = 0; i < 32; i++) {
uint64_t *q = aa64_vfp_qreg(env, i); uint64_t *q = aa64_vfp_qreg(env, i);
#if HOST_BIG_ENDIAN #if HOST_BIG_ENDIAN
uint64_t fp_val[2] = { q[1], q[0] }; uint64_t fp_val[2] = { q[1], q[0] };
reg.addr = (uintptr_t)fp_val; ret = kvm_set_one_reg(cs, AARCH64_SIMD_CORE_REG(fp_regs.vregs[i]),
fp_val);
#else #else
reg.addr = (uintptr_t)q; ret = kvm_set_one_reg(cs, AARCH64_SIMD_CORE_REG(fp_regs.vregs[i]), q);
#endif #endif
reg.id = AARCH64_SIMD_CORE_REG(fp_regs.vregs[i]);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -759,14 +752,11 @@ static int kvm_arch_put_sve(CPUState *cs)
CPUARMState *env = &cpu->env; CPUARMState *env = &cpu->env;
uint64_t tmp[ARM_MAX_VQ * 2]; uint64_t tmp[ARM_MAX_VQ * 2];
uint64_t *r; uint64_t *r;
struct kvm_one_reg reg;
int n, ret; int n, ret;
for (n = 0; n < KVM_ARM64_SVE_NUM_ZREGS; ++n) { for (n = 0; n < KVM_ARM64_SVE_NUM_ZREGS; ++n) {
r = sve_bswap64(tmp, &env->vfp.zregs[n].d[0], cpu->sve_max_vq * 2); r = sve_bswap64(tmp, &env->vfp.zregs[n].d[0], cpu->sve_max_vq * 2);
reg.addr = (uintptr_t)r; ret = kvm_set_one_reg(cs, KVM_REG_ARM64_SVE_ZREG(n, 0), r);
reg.id = KVM_REG_ARM64_SVE_ZREG(n, 0);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -775,9 +765,7 @@ static int kvm_arch_put_sve(CPUState *cs)
for (n = 0; n < KVM_ARM64_SVE_NUM_PREGS; ++n) { for (n = 0; n < KVM_ARM64_SVE_NUM_PREGS; ++n) {
r = sve_bswap64(tmp, r = &env->vfp.pregs[n].p[0], r = sve_bswap64(tmp, r = &env->vfp.pregs[n].p[0],
DIV_ROUND_UP(cpu->sve_max_vq * 2, 8)); DIV_ROUND_UP(cpu->sve_max_vq * 2, 8));
reg.addr = (uintptr_t)r; ret = kvm_set_one_reg(cs, KVM_REG_ARM64_SVE_PREG(n, 0), r);
reg.id = KVM_REG_ARM64_SVE_PREG(n, 0);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -785,9 +773,7 @@ static int kvm_arch_put_sve(CPUState *cs)
r = sve_bswap64(tmp, &env->vfp.pregs[FFR_PRED_NUM].p[0], r = sve_bswap64(tmp, &env->vfp.pregs[FFR_PRED_NUM].p[0],
DIV_ROUND_UP(cpu->sve_max_vq * 2, 8)); DIV_ROUND_UP(cpu->sve_max_vq * 2, 8));
reg.addr = (uintptr_t)r; ret = kvm_set_one_reg(cs, KVM_REG_ARM64_SVE_FFR(0), r);
reg.id = KVM_REG_ARM64_SVE_FFR(0);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -797,7 +783,6 @@ static int kvm_arch_put_sve(CPUState *cs)
int kvm_arch_put_registers(CPUState *cs, int level) int kvm_arch_put_registers(CPUState *cs, int level)
{ {
struct kvm_one_reg reg;
uint64_t val; uint64_t val;
uint32_t fpr; uint32_t fpr;
int i, ret; int i, ret;
@ -814,9 +799,8 @@ int kvm_arch_put_registers(CPUState *cs, int level)
} }
for (i = 0; i < 31; i++) { for (i = 0; i < 31; i++) {
reg.id = AARCH64_CORE_REG(regs.regs[i]); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(regs.regs[i]),
reg.addr = (uintptr_t) &env->xregs[i]; &env->xregs[i]);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -827,16 +811,12 @@ int kvm_arch_put_registers(CPUState *cs, int level)
*/ */
aarch64_save_sp(env, 1); aarch64_save_sp(env, 1);
reg.id = AARCH64_CORE_REG(regs.sp); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(regs.sp), &env->sp_el[0]);
reg.addr = (uintptr_t) &env->sp_el[0];
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.id = AARCH64_CORE_REG(sp_el1); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(sp_el1), &env->sp_el[1]);
reg.addr = (uintptr_t) &env->sp_el[1];
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -847,23 +827,17 @@ int kvm_arch_put_registers(CPUState *cs, int level)
} else { } else {
val = cpsr_read(env); val = cpsr_read(env);
} }
reg.id = AARCH64_CORE_REG(regs.pstate); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(regs.pstate), &val);
reg.addr = (uintptr_t) &val;
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.id = AARCH64_CORE_REG(regs.pc); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(regs.pc), &env->pc);
reg.addr = (uintptr_t) &env->pc;
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.id = AARCH64_CORE_REG(elr_el1); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(elr_el1), &env->elr_el[1]);
reg.addr = (uintptr_t) &env->elr_el[1];
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -882,9 +856,8 @@ int kvm_arch_put_registers(CPUState *cs, int level)
/* KVM 0-4 map to QEMU banks 1-5 */ /* KVM 0-4 map to QEMU banks 1-5 */
for (i = 0; i < KVM_NR_SPSR; i++) { for (i = 0; i < KVM_NR_SPSR; i++) {
reg.id = AARCH64_CORE_REG(spsr[i]); ret = kvm_set_one_reg(cs, AARCH64_CORE_REG(spsr[i]),
reg.addr = (uintptr_t) &env->banked_spsr[i + 1]; &env->banked_spsr[i + 1]);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -899,18 +872,14 @@ int kvm_arch_put_registers(CPUState *cs, int level)
return ret; return ret;
} }
reg.addr = (uintptr_t)(&fpr);
fpr = vfp_get_fpsr(env); fpr = vfp_get_fpsr(env);
reg.id = AARCH64_SIMD_CTRL_REG(fp_regs.fpsr); ret = kvm_set_one_reg(cs, AARCH64_SIMD_CTRL_REG(fp_regs.fpsr), &fpr);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.addr = (uintptr_t)(&fpr);
fpr = vfp_get_fpcr(env); fpr = vfp_get_fpcr(env);
reg.id = AARCH64_SIMD_CTRL_REG(fp_regs.fpcr); ret = kvm_set_one_reg(cs, AARCH64_SIMD_CTRL_REG(fp_regs.fpcr), &fpr);
ret = kvm_vcpu_ioctl(cs, KVM_SET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -939,14 +908,11 @@ int kvm_arch_put_registers(CPUState *cs, int level)
static int kvm_arch_get_fpsimd(CPUState *cs) static int kvm_arch_get_fpsimd(CPUState *cs)
{ {
CPUARMState *env = &ARM_CPU(cs)->env; CPUARMState *env = &ARM_CPU(cs)->env;
struct kvm_one_reg reg;
int i, ret; int i, ret;
for (i = 0; i < 32; i++) { for (i = 0; i < 32; i++) {
uint64_t *q = aa64_vfp_qreg(env, i); uint64_t *q = aa64_vfp_qreg(env, i);
reg.id = AARCH64_SIMD_CORE_REG(fp_regs.vregs[i]); ret = kvm_get_one_reg(cs, AARCH64_SIMD_CORE_REG(fp_regs.vregs[i]), q);
reg.addr = (uintptr_t)q;
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} else { } else {
@ -970,15 +936,12 @@ static int kvm_arch_get_sve(CPUState *cs)
{ {
ARMCPU *cpu = ARM_CPU(cs); ARMCPU *cpu = ARM_CPU(cs);
CPUARMState *env = &cpu->env; CPUARMState *env = &cpu->env;
struct kvm_one_reg reg;
uint64_t *r; uint64_t *r;
int n, ret; int n, ret;
for (n = 0; n < KVM_ARM64_SVE_NUM_ZREGS; ++n) { for (n = 0; n < KVM_ARM64_SVE_NUM_ZREGS; ++n) {
r = &env->vfp.zregs[n].d[0]; r = &env->vfp.zregs[n].d[0];
reg.addr = (uintptr_t)r; ret = kvm_get_one_reg(cs, KVM_REG_ARM64_SVE_ZREG(n, 0), r);
reg.id = KVM_REG_ARM64_SVE_ZREG(n, 0);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -987,9 +950,7 @@ static int kvm_arch_get_sve(CPUState *cs)
for (n = 0; n < KVM_ARM64_SVE_NUM_PREGS; ++n) { for (n = 0; n < KVM_ARM64_SVE_NUM_PREGS; ++n) {
r = &env->vfp.pregs[n].p[0]; r = &env->vfp.pregs[n].p[0];
reg.addr = (uintptr_t)r; ret = kvm_get_one_reg(cs, KVM_REG_ARM64_SVE_PREG(n, 0), r);
reg.id = KVM_REG_ARM64_SVE_PREG(n, 0);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -997,9 +958,7 @@ static int kvm_arch_get_sve(CPUState *cs)
} }
r = &env->vfp.pregs[FFR_PRED_NUM].p[0]; r = &env->vfp.pregs[FFR_PRED_NUM].p[0];
reg.addr = (uintptr_t)r; ret = kvm_get_one_reg(cs, KVM_REG_ARM64_SVE_FFR(0), r);
reg.id = KVM_REG_ARM64_SVE_FFR(0);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -1010,7 +969,6 @@ static int kvm_arch_get_sve(CPUState *cs)
int kvm_arch_get_registers(CPUState *cs) int kvm_arch_get_registers(CPUState *cs)
{ {
struct kvm_one_reg reg;
uint64_t val; uint64_t val;
unsigned int el; unsigned int el;
uint32_t fpr; uint32_t fpr;
@ -1020,31 +978,24 @@ int kvm_arch_get_registers(CPUState *cs)
CPUARMState *env = &cpu->env; CPUARMState *env = &cpu->env;
for (i = 0; i < 31; i++) { for (i = 0; i < 31; i++) {
reg.id = AARCH64_CORE_REG(regs.regs[i]); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(regs.regs[i]),
reg.addr = (uintptr_t) &env->xregs[i]; &env->xregs[i]);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
} }
reg.id = AARCH64_CORE_REG(regs.sp); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(regs.sp), &env->sp_el[0]);
reg.addr = (uintptr_t) &env->sp_el[0];
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.id = AARCH64_CORE_REG(sp_el1); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(sp_el1), &env->sp_el[1]);
reg.addr = (uintptr_t) &env->sp_el[1];
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
reg.id = AARCH64_CORE_REG(regs.pstate); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(regs.pstate), &val);
reg.addr = (uintptr_t) &val;
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -1061,9 +1012,7 @@ int kvm_arch_get_registers(CPUState *cs)
*/ */
aarch64_restore_sp(env, 1); aarch64_restore_sp(env, 1);
reg.id = AARCH64_CORE_REG(regs.pc); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(regs.pc), &env->pc);
reg.addr = (uintptr_t) &env->pc;
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -1077,9 +1026,7 @@ int kvm_arch_get_registers(CPUState *cs)
aarch64_sync_64_to_32(env); aarch64_sync_64_to_32(env);
} }
reg.id = AARCH64_CORE_REG(elr_el1); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(elr_el1), &env->elr_el[1]);
reg.addr = (uintptr_t) &env->elr_el[1];
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -1089,9 +1036,8 @@ int kvm_arch_get_registers(CPUState *cs)
* KVM SPSRs 0-4 map to QEMU banks 1-5 * KVM SPSRs 0-4 map to QEMU banks 1-5
*/ */
for (i = 0; i < KVM_NR_SPSR; i++) { for (i = 0; i < KVM_NR_SPSR; i++) {
reg.id = AARCH64_CORE_REG(spsr[i]); ret = kvm_get_one_reg(cs, AARCH64_CORE_REG(spsr[i]),
reg.addr = (uintptr_t) &env->banked_spsr[i + 1]; &env->banked_spsr[i + 1]);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
@ -1112,17 +1058,13 @@ int kvm_arch_get_registers(CPUState *cs)
return ret; return ret;
} }
reg.addr = (uintptr_t)(&fpr); ret = kvm_get_one_reg(cs, AARCH64_SIMD_CTRL_REG(fp_regs.fpsr), &fpr);
reg.id = AARCH64_SIMD_CTRL_REG(fp_regs.fpsr);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }
vfp_set_fpsr(env, fpr); vfp_set_fpsr(env, fpr);
reg.addr = (uintptr_t)(&fpr); ret = kvm_get_one_reg(cs, AARCH64_SIMD_CTRL_REG(fp_regs.fpcr), &fpr);
reg.id = AARCH64_SIMD_CTRL_REG(fp_regs.fpcr);
ret = kvm_vcpu_ioctl(cs, KVM_GET_ONE_REG, &reg);
if (ret) { if (ret) {
return ret; return ret;
} }

View File

@ -89,6 +89,10 @@ void aa32_max_features(ARMCPU *cpu)
t = FIELD_DP32(t, ID_DFR0, COPSDBG, 9); /* FEAT_Debugv8p4 */ t = FIELD_DP32(t, ID_DFR0, COPSDBG, 9); /* FEAT_Debugv8p4 */
t = FIELD_DP32(t, ID_DFR0, PERFMON, 6); /* FEAT_PMUv3p5 */ t = FIELD_DP32(t, ID_DFR0, PERFMON, 6); /* FEAT_PMUv3p5 */
cpu->isar.id_dfr0 = t; cpu->isar.id_dfr0 = t;
t = cpu->isar.id_dfr1;
t = FIELD_DP32(t, ID_DFR1, HPMN0, 1); /* FEAT_HPMN0 */
cpu->isar.id_dfr1 = t;
} }
/* CPU models. These are not needed for the AArch64 linux-user build. */ /* CPU models. These are not needed for the AArch64 linux-user build. */

View File

@ -1109,6 +1109,7 @@ void aarch64_max_tcg_initfn(Object *obj)
t = cpu->isar.id_aa64dfr0; t = cpu->isar.id_aa64dfr0;
t = FIELD_DP64(t, ID_AA64DFR0, DEBUGVER, 9); /* FEAT_Debugv8p4 */ t = FIELD_DP64(t, ID_AA64DFR0, DEBUGVER, 9); /* FEAT_Debugv8p4 */
t = FIELD_DP64(t, ID_AA64DFR0, PMUVER, 6); /* FEAT_PMUv3p5 */ t = FIELD_DP64(t, ID_AA64DFR0, PMUVER, 6); /* FEAT_PMUv3p5 */
t = FIELD_DP64(t, ID_AA64DFR0, HPMN0, 1); /* FEAT_HPMN0 */
cpu->isar.id_aa64dfr0 = t; cpu->isar.id_aa64dfr0 = t;
t = cpu->isar.id_aa64smfr0; t = cpu->isar.id_aa64smfr0;

View File

@ -7882,7 +7882,7 @@ static void op_addr_block_post(DisasContext *s, arg_ldst_block *a,
} }
} }
static bool op_stm(DisasContext *s, arg_ldst_block *a, int min_n) static bool op_stm(DisasContext *s, arg_ldst_block *a)
{ {
int i, j, n, list, mem_idx; int i, j, n, list, mem_idx;
bool user = a->u; bool user = a->u;
@ -7899,7 +7899,14 @@ static bool op_stm(DisasContext *s, arg_ldst_block *a, int min_n)
list = a->list; list = a->list;
n = ctpop16(list); n = ctpop16(list);
if (n < min_n || a->rn == 15) { /*
* This is UNPREDICTABLE for n < 1 in all encodings, and we choose
* to UNDEF. In the T32 STM encoding n == 1 is also UNPREDICTABLE,
* but hardware treats it like the A32 version and implements the
* single-register-store, and some in-the-wild (buggy) software
* assumes that, so we don't UNDEF on that case.
*/
if (n < 1 || a->rn == 15) {
unallocated_encoding(s); unallocated_encoding(s);
return true; return true;
} }
@ -7935,8 +7942,7 @@ static bool op_stm(DisasContext *s, arg_ldst_block *a, int min_n)
static bool trans_STM(DisasContext *s, arg_ldst_block *a) static bool trans_STM(DisasContext *s, arg_ldst_block *a)
{ {
/* BitCount(list) < 1 is UNPREDICTABLE */ return op_stm(s, a);
return op_stm(s, a, 1);
} }
static bool trans_STM_t32(DisasContext *s, arg_ldst_block *a) static bool trans_STM_t32(DisasContext *s, arg_ldst_block *a)
@ -7946,11 +7952,10 @@ static bool trans_STM_t32(DisasContext *s, arg_ldst_block *a)
unallocated_encoding(s); unallocated_encoding(s);
return true; return true;
} }
/* BitCount(list) < 2 is UNPREDICTABLE */ return op_stm(s, a);
return op_stm(s, a, 2);
} }
static bool do_ldm(DisasContext *s, arg_ldst_block *a, int min_n) static bool do_ldm(DisasContext *s, arg_ldst_block *a)
{ {
int i, j, n, list, mem_idx; int i, j, n, list, mem_idx;
bool loaded_base; bool loaded_base;
@ -7979,7 +7984,14 @@ static bool do_ldm(DisasContext *s, arg_ldst_block *a, int min_n)
list = a->list; list = a->list;
n = ctpop16(list); n = ctpop16(list);
if (n < min_n || a->rn == 15) { /*
* This is UNPREDICTABLE for n < 1 in all encodings, and we choose
* to UNDEF. In the T32 LDM encoding n == 1 is also UNPREDICTABLE,
* but hardware treats it like the A32 version and implements the
* single-register-load, and some in-the-wild (buggy) software
* assumes that, so we don't UNDEF on that case.
*/
if (n < 1 || a->rn == 15) {
unallocated_encoding(s); unallocated_encoding(s);
return true; return true;
} }
@ -8045,8 +8057,7 @@ static bool trans_LDM_a32(DisasContext *s, arg_ldst_block *a)
unallocated_encoding(s); unallocated_encoding(s);
return true; return true;
} }
/* BitCount(list) < 1 is UNPREDICTABLE */ return do_ldm(s, a);
return do_ldm(s, a, 1);
} }
static bool trans_LDM_t32(DisasContext *s, arg_ldst_block *a) static bool trans_LDM_t32(DisasContext *s, arg_ldst_block *a)
@ -8056,16 +8067,14 @@ static bool trans_LDM_t32(DisasContext *s, arg_ldst_block *a)
unallocated_encoding(s); unallocated_encoding(s);
return true; return true;
} }
/* BitCount(list) < 2 is UNPREDICTABLE */ return do_ldm(s, a);
return do_ldm(s, a, 2);
} }
static bool trans_LDM_t16(DisasContext *s, arg_ldst_block *a) static bool trans_LDM_t16(DisasContext *s, arg_ldst_block *a)
{ {
/* Writeback is conditional on the base register not being loaded. */ /* Writeback is conditional on the base register not being loaded. */
a->w = !(a->list & (1 << a->rn)); a->w = !(a->list & (1 << a->rn));
/* BitCount(list) < 1 is UNPREDICTABLE */ return do_ldm(s, a);
return do_ldm(s, a, 1);
} }
static bool trans_CLRM(DisasContext *s, arg_CLRM *a) static bool trans_CLRM(DisasContext *s, arg_CLRM *a)