target/arm: Correctly initialize MDCR_EL2.HPMN

When working with performance monitoring counters, we look at
MDCR_EL2.HPMN as part of the check whether a counter is enabled. This
check fails, because MDCR_EL2.HPMN is reset to 0, meaning that no
counters are "enabled" for < EL2.
That's in violation of the Arm specification, which states that

> On a Warm reset, this field [MDCR_EL2.HPMN] resets to the value in
> PMCR_EL0.N

That's also what a comment in the code acknowledges, but the necessary
adjustment seems to have been forgotten when support for more counters
was added.
This change fixes the issue by setting the reset value to PMCR.N, which
is four.

Backports d3c1183ffeb71ca3a783eae3d7e1c51e71e8a621
This commit is contained in:
Daniel Müller 2021-03-04 18:33:58 -05:00 committed by Lioncash
parent 2c926832bb
commit 642a683d7a

View file

@ -24,6 +24,7 @@
#endif #endif
#define ARM_CPU_FREQ 1000000000 /* FIXME: 1 GHz, should be configurable */ #define ARM_CPU_FREQ 1000000000 /* FIXME: 1 GHz, should be configurable */
#define PMCR_NUM_COUNTERS 4 /* QEMU IMPDEF choice */
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
@ -5409,13 +5410,11 @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
.writefn = gt_hyp_ctl_write, .raw_writefn = raw_write }, .writefn = gt_hyp_ctl_write, .raw_writefn = raw_write },
#endif #endif
/* The only field of MDCR_EL2 that has a defined architectural reset value /* The only field of MDCR_EL2 that has a defined architectural reset value
* is MDCR_EL2.HPMN which should reset to the value of PMCR_EL0.N; but we * is MDCR_EL2.HPMN which should reset to the value of PMCR_EL0.N.
* don't implement any PMU event counters, so using zero as a reset
* value for MDCR_EL2 is okay
*/ */
{ .name = "MDCR_EL2", .state = ARM_CP_STATE_BOTH, { .name = "MDCR_EL2", .state = ARM_CP_STATE_BOTH,
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 1, .opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 1,
.access = PL2_RW, .resetvalue = 0, .access = PL2_RW, .resetvalue = PMCR_NUM_COUNTERS,
.fieldoffset = offsetof(CPUARMState, cp15.mdcr_el2), }, .fieldoffset = offsetof(CPUARMState, cp15.mdcr_el2), },
{ .name = "HPFAR", .state = ARM_CP_STATE_AA32, { .name = "HPFAR", .state = ARM_CP_STATE_AA32,
.cp = 15, .opc1 = 4, .crn = 6, .crm = 0, .opc2 = 4, .cp = 15, .opc1 = 4, .crn = 6, .crm = 0, .opc2 = 4,
@ -6333,7 +6332,7 @@ static void define_pmu_regs(ARMCPU *cpu)
* field as main ID register, and we implement four counters in * field as main ID register, and we implement four counters in
* addition to the cycle count register. * addition to the cycle count register.
*/ */
unsigned int i, pmcrn = 4; unsigned int i, pmcrn = PMCR_NUM_COUNTERS;
ARMCPRegInfo pmcr = { ARMCPRegInfo pmcr = {
.name = "PMCR", .cp = 15, .crn = 9, .crm = 12, .opc1 = 0, .opc2 = 0, .name = "PMCR", .cp = 15, .crn = 9, .crm = 12, .opc1 = 0, .opc2 = 0,
.access = PL0_RW, .access = PL0_RW,
@ -9587,7 +9586,7 @@ static void arm_cpu_do_interrupt_aarch64_(CPUState *cs)
aarch64_save_sp(env, arm_current_el(env)); aarch64_save_sp(env, arm_current_el(env));
env->elr_el[new_el] = env->pc; env->elr_el[new_el] = env->pc;
} else { } else {
old_mode = cpsr_read(env); old_mode = cpsr_read_for_spsr_elx(env);
env->elr_el[new_el] = env->regs[15]; env->elr_el[new_el] = env->regs[15];
aarch64_sync_32_to_64(env); aarch64_sync_32_to_64(env);