mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-04 00:03:54 -06:00
target/arm: expose CPUID registers to userspace
A number of CPUID registers are exposed to userspace by modern Linux kernels thanks to the "ARM64 CPU Feature Registers" ABI. For QEMU's user-mode emulation we don't need to emulate the kernels trap but just return the value the trap would have done. To avoid too much #ifdef hackery we process ARMCPRegInfo with a new helper (modify_arm_cp_regs) before defining the registers. The modify routine is driven by a simple data structure which describes which bits are exported and which are fixed. Signed-off-by: Alex Bennée <alex.bennee@linaro.org> Message-id: 20190205190224.2198-3-alex.bennee@linaro.org Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
b5bd744042
commit
6c5c0fec29
2 changed files with 80 additions and 0 deletions
|
@ -2464,6 +2464,27 @@ static inline void define_one_arm_cp_reg(ARMCPU *cpu, const ARMCPRegInfo *regs)
|
||||||
}
|
}
|
||||||
const ARMCPRegInfo *get_arm_cp_reginfo(GHashTable *cpregs, uint32_t encoded_cp);
|
const ARMCPRegInfo *get_arm_cp_reginfo(GHashTable *cpregs, uint32_t encoded_cp);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Definition of an ARM co-processor register as viewed from
|
||||||
|
* userspace. This is used for presenting sanitised versions of
|
||||||
|
* registers to userspace when emulating the Linux AArch64 CPU
|
||||||
|
* ID/feature ABI (advertised as HWCAP_CPUID).
|
||||||
|
*/
|
||||||
|
typedef struct ARMCPRegUserSpaceInfo {
|
||||||
|
/* Name of register */
|
||||||
|
const char *name;
|
||||||
|
|
||||||
|
/* Only some bits are exported to user space */
|
||||||
|
uint64_t exported_bits;
|
||||||
|
|
||||||
|
/* Fixed bits are applied after the mask */
|
||||||
|
uint64_t fixed_bits;
|
||||||
|
} ARMCPRegUserSpaceInfo;
|
||||||
|
|
||||||
|
#define REGUSERINFO_SENTINEL { .name = NULL }
|
||||||
|
|
||||||
|
void modify_arm_cp_regs(ARMCPRegInfo *regs, const ARMCPRegUserSpaceInfo *mods);
|
||||||
|
|
||||||
/* CPWriteFn that can be used to implement writes-ignored behaviour */
|
/* CPWriteFn that can be used to implement writes-ignored behaviour */
|
||||||
void arm_cp_write_ignore(CPUARMState *env, const ARMCPRegInfo *ri,
|
void arm_cp_write_ignore(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
uint64_t value);
|
uint64_t value);
|
||||||
|
|
|
@ -6109,6 +6109,30 @@ void register_cp_regs_for_features(ARMCPU *cpu)
|
||||||
.resetvalue = cpu->pmceid1 },
|
.resetvalue = cpu->pmceid1 },
|
||||||
REGINFO_SENTINEL
|
REGINFO_SENTINEL
|
||||||
};
|
};
|
||||||
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
ARMCPRegUserSpaceInfo v8_user_idregs[] = {
|
||||||
|
{ .name = "ID_AA64PFR0_EL1",
|
||||||
|
.exported_bits = 0x000f000f00ff0000,
|
||||||
|
.fixed_bits = 0x0000000000000011 },
|
||||||
|
{ .name = "ID_AA64PFR1_EL1",
|
||||||
|
.exported_bits = 0x00000000000000f0 },
|
||||||
|
{ .name = "ID_AA64ZFR0_EL1" },
|
||||||
|
{ .name = "ID_AA64MMFR0_EL1",
|
||||||
|
.fixed_bits = 0x00000000ff000000 },
|
||||||
|
{ .name = "ID_AA64MMFR1_EL1" },
|
||||||
|
{ .name = "ID_AA64DFR0_EL1",
|
||||||
|
.fixed_bits = 0x0000000000000006 },
|
||||||
|
{ .name = "ID_AA64DFR1_EL1" },
|
||||||
|
{ .name = "ID_AA64AFR0_EL1" },
|
||||||
|
{ .name = "ID_AA64AFR1_EL1" },
|
||||||
|
{ .name = "ID_AA64ISAR0_EL1",
|
||||||
|
.exported_bits = 0x00fffffff0fffff0 },
|
||||||
|
{ .name = "ID_AA64ISAR1_EL1",
|
||||||
|
.exported_bits = 0x000000f0ffffffff },
|
||||||
|
REGUSERINFO_SENTINEL
|
||||||
|
};
|
||||||
|
modify_arm_cp_regs(v8_idregs, v8_user_idregs);
|
||||||
|
#endif
|
||||||
/* RVBAR_EL1 is only implemented if EL1 is the highest EL */
|
/* RVBAR_EL1 is only implemented if EL1 is the highest EL */
|
||||||
if (!arm_feature(env, ARM_FEATURE_EL3) &&
|
if (!arm_feature(env, ARM_FEATURE_EL3) &&
|
||||||
!arm_feature(env, ARM_FEATURE_EL2)) {
|
!arm_feature(env, ARM_FEATURE_EL2)) {
|
||||||
|
@ -6385,6 +6409,15 @@ void register_cp_regs_for_features(ARMCPU *cpu)
|
||||||
.opc1 = CP_ANY, .opc2 = CP_ANY, .access = PL1_W,
|
.opc1 = CP_ANY, .opc2 = CP_ANY, .access = PL1_W,
|
||||||
.type = ARM_CP_NOP | ARM_CP_OVERRIDE
|
.type = ARM_CP_NOP | ARM_CP_OVERRIDE
|
||||||
};
|
};
|
||||||
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
ARMCPRegUserSpaceInfo id_v8_user_midr_cp_reginfo[] = {
|
||||||
|
{ .name = "MIDR_EL1",
|
||||||
|
.exported_bits = 0x00000000ffffffff },
|
||||||
|
{ .name = "REVIDR_EL1" },
|
||||||
|
REGUSERINFO_SENTINEL
|
||||||
|
};
|
||||||
|
modify_arm_cp_regs(id_v8_midr_cp_reginfo, id_v8_user_midr_cp_reginfo);
|
||||||
|
#endif
|
||||||
if (arm_feature(env, ARM_FEATURE_OMAPCP) ||
|
if (arm_feature(env, ARM_FEATURE_OMAPCP) ||
|
||||||
arm_feature(env, ARM_FEATURE_STRONGARM)) {
|
arm_feature(env, ARM_FEATURE_STRONGARM)) {
|
||||||
ARMCPRegInfo *r;
|
ARMCPRegInfo *r;
|
||||||
|
@ -6966,6 +6999,32 @@ void define_arm_cp_regs_with_opaque(ARMCPU *cpu,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Modify ARMCPRegInfo for access from userspace.
|
||||||
|
*
|
||||||
|
* This is a data driven modification directed by
|
||||||
|
* ARMCPRegUserSpaceInfo. All registers become ARM_CP_CONST as
|
||||||
|
* user-space cannot alter any values and dynamic values pertaining to
|
||||||
|
* execution state are hidden from user space view anyway.
|
||||||
|
*/
|
||||||
|
void modify_arm_cp_regs(ARMCPRegInfo *regs, const ARMCPRegUserSpaceInfo *mods)
|
||||||
|
{
|
||||||
|
const ARMCPRegUserSpaceInfo *m;
|
||||||
|
ARMCPRegInfo *r;
|
||||||
|
|
||||||
|
for (m = mods; m->name; m++) {
|
||||||
|
for (r = regs; r->type != ARM_CP_SENTINEL; r++) {
|
||||||
|
if (strcmp(r->name, m->name) == 0) {
|
||||||
|
r->type = ARM_CP_CONST;
|
||||||
|
r->access = PL0U_R;
|
||||||
|
r->resetvalue &= m->exported_bits;
|
||||||
|
r->resetvalue |= m->fixed_bits;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const ARMCPRegInfo *get_arm_cp_reginfo(GHashTable *cpregs, uint32_t encoded_cp)
|
const ARMCPRegInfo *get_arm_cp_reginfo(GHashTable *cpregs, uint32_t encoded_cp)
|
||||||
{
|
{
|
||||||
return g_hash_table_lookup(cpregs, &encoded_cp);
|
return g_hash_table_lookup(cpregs, &encoded_cp);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue