mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-08 02:03:56 -06:00
target-arm: Minimal implementation of performance counters
Newer Linux kernels assume the existence of the performance counter cp15 registers. Provide a minimal implementation of these registers. We support no events. This should be compliant with the ARM ARM, except that we don't implement the cycle counter. Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
b501b5e461
commit
74594c9d81
4 changed files with 184 additions and 17 deletions
|
@ -2472,12 +2472,28 @@ static int disas_cp_insn(CPUState *env, DisasContext *s, uint32_t insn)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int cp15_user_ok(uint32_t insn)
|
||||
static int cp15_user_ok(CPUState *env, uint32_t insn)
|
||||
{
|
||||
int cpn = (insn >> 16) & 0xf;
|
||||
int cpm = insn & 0xf;
|
||||
int op = ((insn >> 5) & 7) | ((insn >> 18) & 0x38);
|
||||
|
||||
if (arm_feature(env, ARM_FEATURE_V7) && cpn == 9) {
|
||||
/* Performance monitor registers fall into three categories:
|
||||
* (a) always UNDEF in usermode
|
||||
* (b) UNDEF only if PMUSERENR.EN is 0
|
||||
* (c) always read OK and UNDEF on write (PMUSERENR only)
|
||||
*/
|
||||
if ((cpm == 12 && (op < 6)) ||
|
||||
(cpm == 13 && (op < 3))) {
|
||||
return env->cp15.c9_pmuserenr;
|
||||
} else if (cpm == 14 && op == 0 && (insn & ARM_CP_RW_BIT)) {
|
||||
/* PMUSERENR, read only */
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (cpn == 13 && cpm == 0) {
|
||||
/* TLS register. */
|
||||
if (op == 2 || (op == 3 && (insn & ARM_CP_RW_BIT)))
|
||||
|
@ -2564,7 +2580,7 @@ static int disas_cp15_insn(CPUState *env, DisasContext *s, uint32_t insn)
|
|||
/* cdp */
|
||||
return 1;
|
||||
}
|
||||
if (IS_USER(s) && !cp15_user_ok(insn)) {
|
||||
if (IS_USER(s) && !cp15_user_ok(env, insn)) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue