mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2024-12-23 01:05:36 +00:00
target/arm: Make number of counters in PMCR follow the CPU
Currently we give all the v7-and-up CPUs a PMU with 4 counters. This means that we don't provide the 6 counters that are required by the Arm BSA (Base System Architecture) specification if the CPU supports the Virtualization extensions. Instead of having a single PMCR_NUM_COUNTERS, make each CPU type specify the PMCR reset value (obtained from the appropriate TRM), and use the 'N' field of that value to define the number of counters provided. This means that we now supply 6 counters for Cortex-A53, A57, A72, A15 and A9 as well as '-cpu max'; Cortex-A7 and A8 stay at 4; and Cortex-R5 goes down to 3. Note that because we now use the PMCR reset value of the specific implementation, we no longer set the LC bit out of reset. This has an UNKNOWN value out of reset for all cores with any AArch32 support, so guest software should be setting it anyway if it wants it. Backports f7fb73b8cdd3f77e26f9fcff8cf24ff1b58d200f
This commit is contained in:
parent
289eed9550
commit
82ce9221a0
|
@ -1565,6 +1565,7 @@ static void cortex_r5_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->isar.id_isar6 = 0x0;
|
cpu->isar.id_isar6 = 0x0;
|
||||||
cpu->mp_is_up = true;
|
cpu->mp_is_up = true;
|
||||||
cpu->pmsav7_dregion = 16;
|
cpu->pmsav7_dregion = 16;
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41151800;
|
||||||
define_arm_cp_regs(cpu, cortexr5_cp_reginfo);
|
define_arm_cp_regs(cpu, cortexr5_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1620,6 +1621,7 @@ static void cortex_a8_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[1] = 0x2007e01a; /* 16k L1 icache. */
|
cpu->ccsidr[1] = 0x2007e01a; /* 16k L1 icache. */
|
||||||
cpu->ccsidr[2] = 0xf0000000; /* No L2 icache. */
|
cpu->ccsidr[2] = 0xf0000000; /* No L2 icache. */
|
||||||
cpu->reset_auxcr = 2;
|
cpu->reset_auxcr = 2;
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41002000;
|
||||||
define_arm_cp_regs(cpu, cortexa8_cp_reginfo);
|
define_arm_cp_regs(cpu, cortexa8_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1692,6 +1694,7 @@ static void cortex_a9_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->clidr = (1 << 27) | (1 << 24) | 3;
|
cpu->clidr = (1 << 27) | (1 << 24) | 3;
|
||||||
cpu->ccsidr[0] = 0xe00fe019; /* 16k L1 dcache. */
|
cpu->ccsidr[0] = 0xe00fe019; /* 16k L1 dcache. */
|
||||||
cpu->ccsidr[1] = 0x200fe019; /* 16k L1 icache. */
|
cpu->ccsidr[1] = 0x200fe019; /* 16k L1 icache. */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41093000;
|
||||||
define_arm_cp_regs(cpu, cortexa9_cp_reginfo);
|
define_arm_cp_regs(cpu, cortexa9_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1760,6 +1763,7 @@ static void cortex_a7_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[0] = 0x701fe00a; /* 32K L1 dcache */
|
cpu->ccsidr[0] = 0x701fe00a; /* 32K L1 dcache */
|
||||||
cpu->ccsidr[1] = 0x201fe00a; /* 32K L1 icache */
|
cpu->ccsidr[1] = 0x201fe00a; /* 32K L1 icache */
|
||||||
cpu->ccsidr[2] = 0x711fe07a; /* 4096K L2 unified cache */
|
cpu->ccsidr[2] = 0x711fe07a; /* 4096K L2 unified cache */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41072000;
|
||||||
define_arm_cp_regs(cpu, cortexa15_cp_reginfo); /* Same as A15 */
|
define_arm_cp_regs(cpu, cortexa15_cp_reginfo); /* Same as A15 */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1802,6 +1806,7 @@ static void cortex_a15_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[0] = 0x701fe00a; /* 32K L1 dcache */
|
cpu->ccsidr[0] = 0x701fe00a; /* 32K L1 dcache */
|
||||||
cpu->ccsidr[1] = 0x201fe00a; /* 32K L1 icache */
|
cpu->ccsidr[1] = 0x201fe00a; /* 32K L1 icache */
|
||||||
cpu->ccsidr[2] = 0x711fe07a; /* 4096K L2 unified cache */
|
cpu->ccsidr[2] = 0x711fe07a; /* 4096K L2 unified cache */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x410F3000;
|
||||||
define_arm_cp_regs(cpu, cortexa15_cp_reginfo);
|
define_arm_cp_regs(cpu, cortexa15_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -888,6 +888,7 @@ struct ARMCPU {
|
||||||
uint64_t id_aa64mmfr2;
|
uint64_t id_aa64mmfr2;
|
||||||
uint64_t id_aa64dfr0;
|
uint64_t id_aa64dfr0;
|
||||||
uint64_t id_aa64dfr1;
|
uint64_t id_aa64dfr1;
|
||||||
|
uint64_t reset_pmcr_el0;
|
||||||
} isar;
|
} isar;
|
||||||
uint64_t midr;
|
uint64_t midr;
|
||||||
uint32_t revidr;
|
uint32_t revidr;
|
||||||
|
|
|
@ -128,6 +128,7 @@ static void aarch64_a57_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[1] = 0x201fe012; /* 48KB L1 icache */
|
cpu->ccsidr[1] = 0x201fe012; /* 48KB L1 icache */
|
||||||
cpu->ccsidr[2] = 0x70ffe07a; /* 2048KB L2 cache */
|
cpu->ccsidr[2] = 0x70ffe07a; /* 2048KB L2 cache */
|
||||||
cpu->dcz_blocksize = 4; /* 64 bytes */
|
cpu->dcz_blocksize = 4; /* 64 bytes */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41013000;
|
||||||
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -178,6 +179,7 @@ static void aarch64_a53_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[1] = 0x201fe00a; /* 32KB L1 icache */
|
cpu->ccsidr[1] = 0x201fe00a; /* 32KB L1 icache */
|
||||||
cpu->ccsidr[2] = 0x707fe07a; /* 1024KB L2 cache */
|
cpu->ccsidr[2] = 0x707fe07a; /* 1024KB L2 cache */
|
||||||
cpu->dcz_blocksize = 4; /* 64 bytes */
|
cpu->dcz_blocksize = 4; /* 64 bytes */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41033000;
|
||||||
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -226,6 +228,7 @@ static void aarch64_a72_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->ccsidr[1] = 0x201fe012; /* 48KB L1 icache */
|
cpu->ccsidr[1] = 0x201fe012; /* 48KB L1 icache */
|
||||||
cpu->ccsidr[2] = 0x707fe07a; /* 1MB L2 cache */
|
cpu->ccsidr[2] = 0x707fe07a; /* 1MB L2 cache */
|
||||||
cpu->dcz_blocksize = 4; /* 64 bytes */
|
cpu->dcz_blocksize = 4; /* 64 bytes */
|
||||||
|
cpu->isar.reset_pmcr_el0 = 0x41023000;
|
||||||
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
define_arm_cp_regs(cpu, cortex_a72_a57_a53_cp_reginfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,6 @@
|
||||||
#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
|
||||||
|
|
||||||
|
@ -898,7 +897,9 @@ static const ARMCPRegInfo v6_cp_reginfo[] = {
|
||||||
|
|
||||||
static inline uint32_t pmu_num_counters(CPUARMState *env)
|
static inline uint32_t pmu_num_counters(CPUARMState *env)
|
||||||
{
|
{
|
||||||
return (env->cp15.c9_pmcr & PMCRN_MASK) >> PMCRN_SHIFT;
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
|
||||||
|
return (cpu->isar.reset_pmcr_el0 & PMCRN_MASK) >> PMCRN_SHIFT;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bits allowed to be set/cleared for PMCNTEN* and PMINTEN* */
|
/* Bits allowed to be set/cleared for PMCNTEN* and PMINTEN* */
|
||||||
|
@ -5427,13 +5428,6 @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
|
||||||
.resetvalue = 0,
|
.resetvalue = 0,
|
||||||
.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
|
|
||||||
* is MDCR_EL2.HPMN which should reset to the value of PMCR_EL0.N.
|
|
||||||
*/
|
|
||||||
{ .name = "MDCR_EL2", .state = ARM_CP_STATE_BOTH,
|
|
||||||
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 1,
|
|
||||||
.access = PL2_RW, .resetvalue = PMCR_NUM_COUNTERS,
|
|
||||||
.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,
|
||||||
.access = PL2_RW, .accessfn = access_el3_aa32ns,
|
.access = PL2_RW, .accessfn = access_el3_aa32ns,
|
||||||
|
@ -6350,7 +6344,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 = PMCR_NUM_COUNTERS;
|
unsigned int i, pmcrn = pmu_num_counters(&cpu->env);
|
||||||
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,
|
||||||
|
@ -6365,10 +6359,10 @@ static void define_pmu_regs(ARMCPU *cpu)
|
||||||
.access = PL0_RW, .accessfn = pmreg_access,
|
.access = PL0_RW, .accessfn = pmreg_access,
|
||||||
.type = ARM_CP_IO,
|
.type = ARM_CP_IO,
|
||||||
.fieldoffset = offsetof(CPUARMState, cp15.c9_pmcr),
|
.fieldoffset = offsetof(CPUARMState, cp15.c9_pmcr),
|
||||||
.resetvalue = (cpu->midr & 0xff000000) | (pmcrn << PMCRN_SHIFT) |
|
.resetvalue = cpu->isar.reset_pmcr_el0,
|
||||||
PMCRLC,
|
|
||||||
.writefn = pmcr_write, .raw_writefn = raw_write,
|
.writefn = pmcr_write, .raw_writefn = raw_write,
|
||||||
};
|
};
|
||||||
|
|
||||||
define_one_arm_cp_reg(cpu, &pmcr);
|
define_one_arm_cp_reg(cpu, &pmcr);
|
||||||
define_one_arm_cp_reg(cpu, &pmcr64);
|
define_one_arm_cp_reg(cpu, &pmcr64);
|
||||||
for (i = 0; i < pmcrn; i++) {
|
for (i = 0; i < pmcrn; i++) {
|
||||||
|
@ -7490,6 +7484,17 @@ void register_cp_regs_for_features(ARMCPU *cpu)
|
||||||
.fieldoffset = offsetof(CPUARMState, cp15.vmpidr_el2) },
|
.fieldoffset = offsetof(CPUARMState, cp15.vmpidr_el2) },
|
||||||
REGINFO_SENTINEL
|
REGINFO_SENTINEL
|
||||||
};
|
};
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
ARMCPRegInfo mdcr_el2 = {
|
||||||
|
.name = "MDCR_EL2", .state = ARM_CP_STATE_BOTH,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 1,
|
||||||
|
.access = PL2_RW, .resetvalue = pmu_num_counters(env),
|
||||||
|
.fieldoffset = offsetof(CPUARMState, cp15.mdcr_el2),
|
||||||
|
};
|
||||||
|
define_one_arm_cp_reg(cpu, &mdcr_el2);
|
||||||
define_arm_cp_regs(cpu, vpidr_regs);
|
define_arm_cp_regs(cpu, vpidr_regs);
|
||||||
define_arm_cp_regs(cpu, el2_cp_reginfo);
|
define_arm_cp_regs(cpu, el2_cp_reginfo);
|
||||||
if (arm_feature(env, ARM_FEATURE_V8)) {
|
if (arm_feature(env, ARM_FEATURE_V8)) {
|
||||||
|
|
Loading…
Reference in a new issue