target/arm: Correctly implement handling of HCR_EL2.{VI, VF}

In commit 8a0fc3a29fc2315325400 we tried to implement HCR_EL2.{VI,VF},
but we got it wrong and had to revert it.

In that commit we implemented them as simply tracking whether there
is a pending virtual IRQ or virtual FIQ. This is not correct -- these
bits cause a software-generated VIRQ/VFIQ, which is distinct from
whether there is a hardware-generated VIRQ/VFIQ caused by the
external interrupt controller. So we need to track separately
the HCR_EL2 bit state and the external virq/vfiq line state, and
OR the two together to get the actual pending VIRQ/VFIQ state.

Fixes: 8a0fc3a29fc2315325400c738f807d0d4ae0ab7f

Backports commit 89430fc6f80a5aef1d4cbd6fc26b40c30793786c from qemu
This commit is contained in:
Peter Maydell 2018-11-16 21:53:21 -05:00 committed by Lioncash
parent a9d07a19c0
commit 92bf8ee620
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7
17 changed files with 107 additions and 3 deletions

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_aarch64
#define arm_cpu_register_types arm_cpu_register_types_aarch64
#define arm_cpu_set_pc arm_cpu_set_pc_aarch64
#define arm_cpu_update_virq arm_cpu_update_virq_aarch64
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_aarch64
#define arm_cpus arm_cpus_aarch64
#define arm_current_el arm_current_el_aarch64
#define arm_dc_feature arm_dc_feature_aarch64

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_aarch64eb
#define arm_cpu_register_types arm_cpu_register_types_aarch64eb
#define arm_cpu_set_pc arm_cpu_set_pc_aarch64eb
#define arm_cpu_update_virq arm_cpu_update_virq_aarch64eb
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_aarch64eb
#define arm_cpus arm_cpus_aarch64eb
#define arm_current_el arm_current_el_aarch64eb
#define arm_dc_feature arm_dc_feature_aarch64eb

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_arm
#define arm_cpu_register_types arm_cpu_register_types_arm
#define arm_cpu_set_pc arm_cpu_set_pc_arm
#define arm_cpu_update_virq arm_cpu_update_virq_arm
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_arm
#define arm_cpus arm_cpus_arm
#define arm_current_el arm_current_el_arm
#define arm_dc_feature arm_dc_feature_arm

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_armeb
#define arm_cpu_register_types arm_cpu_register_types_armeb
#define arm_cpu_set_pc arm_cpu_set_pc_armeb
#define arm_cpu_update_virq arm_cpu_update_virq_armeb
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_armeb
#define arm_cpus arm_cpus_armeb
#define arm_current_el arm_current_el_armeb
#define arm_dc_feature arm_dc_feature_armeb

View file

@ -157,6 +157,8 @@ symbols = (
'arm_cpu_register_gdb_regs_for_features',
'arm_cpu_register_types',
'arm_cpu_set_pc',
'arm_cpu_update_virq',
'arm_cpu_update_vfiq',
'arm_cpus',
'arm_current_el',
'arm_dc_feature',

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_m68k
#define arm_cpu_register_types arm_cpu_register_types_m68k
#define arm_cpu_set_pc arm_cpu_set_pc_m68k
#define arm_cpu_update_virq arm_cpu_update_virq_m68k
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_m68k
#define arm_cpus arm_cpus_m68k
#define arm_current_el arm_current_el_m68k
#define arm_dc_feature arm_dc_feature_m68k

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_mips
#define arm_cpu_register_types arm_cpu_register_types_mips
#define arm_cpu_set_pc arm_cpu_set_pc_mips
#define arm_cpu_update_virq arm_cpu_update_virq_mips
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_mips
#define arm_cpus arm_cpus_mips
#define arm_current_el arm_current_el_mips
#define arm_dc_feature arm_dc_feature_mips

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_mips64
#define arm_cpu_register_types arm_cpu_register_types_mips64
#define arm_cpu_set_pc arm_cpu_set_pc_mips64
#define arm_cpu_update_virq arm_cpu_update_virq_mips64
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_mips64
#define arm_cpus arm_cpus_mips64
#define arm_current_el arm_current_el_mips64
#define arm_dc_feature arm_dc_feature_mips64

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_mips64el
#define arm_cpu_register_types arm_cpu_register_types_mips64el
#define arm_cpu_set_pc arm_cpu_set_pc_mips64el
#define arm_cpu_update_virq arm_cpu_update_virq_mips64el
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_mips64el
#define arm_cpus arm_cpus_mips64el
#define arm_current_el arm_current_el_mips64el
#define arm_dc_feature arm_dc_feature_mips64el

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_mipsel
#define arm_cpu_register_types arm_cpu_register_types_mipsel
#define arm_cpu_set_pc arm_cpu_set_pc_mipsel
#define arm_cpu_update_virq arm_cpu_update_virq_mipsel
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_mipsel
#define arm_cpus arm_cpus_mipsel
#define arm_current_el arm_current_el_mipsel
#define arm_dc_feature arm_dc_feature_mipsel

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_powerpc
#define arm_cpu_register_types arm_cpu_register_types_powerpc
#define arm_cpu_set_pc arm_cpu_set_pc_powerpc
#define arm_cpu_update_virq arm_cpu_update_virq_powerpc
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_powerpc
#define arm_cpus arm_cpus_powerpc
#define arm_current_el arm_current_el_powerpc
#define arm_dc_feature arm_dc_feature_powerpc

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_sparc
#define arm_cpu_register_types arm_cpu_register_types_sparc
#define arm_cpu_set_pc arm_cpu_set_pc_sparc
#define arm_cpu_update_virq arm_cpu_update_virq_sparc
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_sparc
#define arm_cpus arm_cpus_sparc
#define arm_current_el arm_current_el_sparc
#define arm_dc_feature arm_dc_feature_sparc

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_sparc64
#define arm_cpu_register_types arm_cpu_register_types_sparc64
#define arm_cpu_set_pc arm_cpu_set_pc_sparc64
#define arm_cpu_update_virq arm_cpu_update_virq_sparc64
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_sparc64
#define arm_cpus arm_cpus_sparc64
#define arm_current_el arm_current_el_sparc64
#define arm_dc_feature arm_dc_feature_sparc64

View file

@ -428,6 +428,48 @@ static bool arm_v7m_cpu_exec_interrupt(CPUState *cs, int interrupt_request)
}
#endif
void arm_cpu_update_virq(ARMCPU *cpu)
{
/*
* Update the interrupt level for VIRQ, which is the logical OR of
* the HCR_EL2.VI bit and the input line level from the GIC.
*/
CPUARMState *env = &cpu->env;
CPUState *cs = CPU(cpu);
bool new_state = (env->cp15.hcr_el2 & HCR_VI) ||
(env->irq_line_state & CPU_INTERRUPT_VIRQ);
if (new_state != ((cs->interrupt_request & CPU_INTERRUPT_VIRQ) != 0)) {
if (new_state) {
cpu_interrupt(cs, CPU_INTERRUPT_VIRQ);
} else {
cpu_reset_interrupt(cs, CPU_INTERRUPT_VIRQ);
}
}
}
void arm_cpu_update_vfiq(ARMCPU *cpu)
{
/*
* Update the interrupt level for VFIQ, which is the logical OR of
* the HCR_EL2.VF bit and the input line level from the GIC.
*/
CPUARMState *env = &cpu->env;
CPUState *cs = CPU(cpu);
bool new_state = (env->cp15.hcr_el2 & HCR_VF) ||
(env->irq_line_state & CPU_INTERRUPT_VFIQ);
if (new_state != ((cs->interrupt_request & CPU_INTERRUPT_VFIQ) != 0)) {
if (new_state) {
cpu_interrupt(cs, CPU_INTERRUPT_VFIQ);
} else {
cpu_reset_interrupt(cs, CPU_INTERRUPT_VFIQ);
}
}
}
static QEMU_UNUSED_FUNC bool arm_cpu_is_big_endian(CPUState *cs)
{
ARMCPU *cpu = ARM_CPU(NULL, cs);

View file

@ -3512,6 +3512,22 @@ static void hcr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
tlb_flush(CPU(cpu));
}
env->cp15.hcr_el2 = value;
/*
* Updates to VI and VF require us to update the status of
* virtual interrupts, which are the logical OR of these bits
* and the state of the input lines from the GIC. (This requires
* that we have the iothread lock, which is done by marking the
* reginfo structs as ARM_CP_IO.)
* Note that if a write to HCR pends a VIRQ or VFIQ it is never
* possible for it to be taken immediately, because VIRQ and
* VFIQ are masked unless running at EL0 or EL1, and HCR
* can only be written at EL2.
*/
// Unicorn: Commented out
//g_assert(qemu_mutex_iothread_locked());
arm_cpu_update_virq(cpu);
arm_cpu_update_vfiq(cpu);
}
static void hcr_writehigh(CPUARMState *env, const ARMCPRegInfo *ri,
@ -3532,10 +3548,10 @@ static void hcr_writelow(CPUARMState *env, const ARMCPRegInfo *ri,
static const ARMCPRegInfo el2_cp_reginfo[] = {
{ "HCR_EL2", 0,1,1, 3,4,0, ARM_CP_STATE_AA64,
0, PL2_RW, 0, NULL, 0, offsetof(CPUARMState, cp15.hcr_el2), {0, 0},
ARM_CP_IO, PL2_RW, 0, NULL, 0, offsetof(CPUARMState, cp15.hcr_el2), {0, 0},
NULL, NULL, hcr_write },
{ "HCR", 15,1,1, 0,4,0, ARM_CP_STATE_AA32,
ARM_CP_ALIAS, PL2_RW, 0, NULL, 0, offsetof(CPUARMState, cp15.hcr_el2),
ARM_CP_ALIAS | ARM_CP_IO, PL2_RW, 0, NULL, 0, offsetof(CPUARMState, cp15.hcr_el2),
{0, 0}, NULL, NULL, hcr_writelow },
{ "ELR_EL2", 0,4,0, 3,4,1, ARM_CP_STATE_AA64,
ARM_CP_ALIAS, PL2_RW, 0, NULL, 0, offsetof(CPUARMState, elr_el[2]) },
@ -3700,7 +3716,7 @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
static const ARMCPRegInfo el2_v8_cp_reginfo[] = {
{ "HCR2", 15,1,1, 0,4,4, ARM_CP_STATE_AA32,
ARM_CP_ALIAS, PL2_RW, 0, NULL, 0, offsetofhigh32(CPUARMState, cp15.hcr_el2),
ARM_CP_ALIAS | ARM_CP_IO, PL2_RW, 0, NULL, 0, offsetofhigh32(CPUARMState, cp15.hcr_el2),
{0, 0}, NULL, NULL, hcr_writehigh },
REGINFO_SENTINEL
};

View file

@ -873,4 +873,22 @@ static inline const char *aarch32_mode_name(uint32_t psr)
return cpu_mode_names[psr & 0xf];
}
/**
* arm_cpu_update_virq: Update CPU_INTERRUPT_VIRQ bit in cs->interrupt_request
*
* Update the CPU_INTERRUPT_VIRQ bit in cs->interrupt_request, following
* a change to either the input VIRQ line from the GIC or the HCR_EL2.VI bit.
* Must be called with the iothread lock held.
*/
void arm_cpu_update_virq(ARMCPU *cpu);
/**
* arm_cpu_update_vfiq: Update CPU_INTERRUPT_VFIQ bit in cs->interrupt_request
*
* Update the CPU_INTERRUPT_VFIQ bit in cs->interrupt_request, following
* a change to either the input VFIQ line from the GIC or the HCR_EL2.VF bit.
* Must be called with the iothread lock held.
*/
void arm_cpu_update_vfiq(ARMCPU *cpu);
#endif

View file

@ -151,6 +151,8 @@
#define arm_cpu_register_gdb_regs_for_features arm_cpu_register_gdb_regs_for_features_x86_64
#define arm_cpu_register_types arm_cpu_register_types_x86_64
#define arm_cpu_set_pc arm_cpu_set_pc_x86_64
#define arm_cpu_update_virq arm_cpu_update_virq_x86_64
#define arm_cpu_update_vfiq arm_cpu_update_vfiq_x86_64
#define arm_cpus arm_cpus_x86_64
#define arm_current_el arm_current_el_x86_64
#define arm_dc_feature arm_dc_feature_x86_64