Commit 377d0e5d authored by Oliver Upton's avatar Oliver Upton

Merge branch kvm-arm64/ctr-el0 into kvmarm/next

* kvm-arm64/ctr-el0:
  : Support for user changes to CTR_EL0, courtesy of Sebastian Ott
  :
  : Allow userspace to change the guest-visible value of CTR_EL0 for a VM,
  : so long as the requested value represents a subset of features supported
  : by hardware. In other words, prevent the VMM from over-promising the
  : capabilities of hardware.
  :
  : Make this happen by fitting CTR_EL0 into the existing infrastructure for
  : feature ID registers.
  KVM: selftests: Assert that MPIDR_EL1 is unchanged across vCPU reset
  KVM: arm64: nv: Unfudge ID_AA64PFR0_EL1 masking
  KVM: selftests: arm64: Test writes to CTR_EL0
  KVM: arm64: rename functions for invariant sys regs
  KVM: arm64: show writable masks for feature registers
  KVM: arm64: Treat CTR_EL0 as a VM feature ID register
  KVM: arm64: unify code to prepare traps
  KVM: arm64: nv: Use accessors for modifying ID registers
  KVM: arm64: Add helper for writing ID regs
  KVM: arm64: Use read-only helper for reading VM ID registers
  KVM: arm64: Make idregs debugfs iterator search sysreg table directly
  KVM: arm64: Get sys_reg encoding from descriptor in idregs_debug_show()
Signed-off-by: default avatarOliver Upton <oliver.upton@linux.dev>
parents 435a9f60 b0539664
...@@ -69,39 +69,17 @@ static __always_inline bool vcpu_el1_is_32bit(struct kvm_vcpu *vcpu) ...@@ -69,39 +69,17 @@ static __always_inline bool vcpu_el1_is_32bit(struct kvm_vcpu *vcpu)
static inline void vcpu_reset_hcr(struct kvm_vcpu *vcpu) static inline void vcpu_reset_hcr(struct kvm_vcpu *vcpu)
{ {
if (!vcpu_has_run_once(vcpu))
vcpu->arch.hcr_el2 = HCR_GUEST_FLAGS; vcpu->arch.hcr_el2 = HCR_GUEST_FLAGS;
if (has_vhe() || has_hvhe())
vcpu->arch.hcr_el2 |= HCR_E2H;
if (cpus_have_final_cap(ARM64_HAS_RAS_EXTN)) {
/* route synchronous external abort exceptions to EL2 */
vcpu->arch.hcr_el2 |= HCR_TEA;
/* trap error record accesses */
vcpu->arch.hcr_el2 |= HCR_TERR;
}
if (cpus_have_final_cap(ARM64_HAS_STAGE2_FWB)) {
vcpu->arch.hcr_el2 |= HCR_FWB;
} else {
/* /*
* For non-FWB CPUs, we trap VM ops (HCR_EL2.TVM) until M+C * For non-FWB CPUs, we trap VM ops (HCR_EL2.TVM) until M+C
* get set in SCTLR_EL1 such that we can detect when the guest * get set in SCTLR_EL1 such that we can detect when the guest
* MMU gets turned on and do the necessary cache maintenance * MMU gets turned on and do the necessary cache maintenance
* then. * then.
*/ */
if (!cpus_have_final_cap(ARM64_HAS_STAGE2_FWB))
vcpu->arch.hcr_el2 |= HCR_TVM; vcpu->arch.hcr_el2 |= HCR_TVM;
}
if (cpus_have_final_cap(ARM64_HAS_EVT) &&
!cpus_have_final_cap(ARM64_MISMATCHED_CACHE_TYPE))
vcpu->arch.hcr_el2 |= HCR_TID4;
else
vcpu->arch.hcr_el2 |= HCR_TID2;
if (vcpu_el1_is_32bit(vcpu))
vcpu->arch.hcr_el2 &= ~HCR_RW;
if (kvm_has_mte(vcpu->kvm))
vcpu->arch.hcr_el2 |= HCR_ATA;
} }
static inline unsigned long *vcpu_hcr(struct kvm_vcpu *vcpu) static inline unsigned long *vcpu_hcr(struct kvm_vcpu *vcpu)
......
...@@ -362,11 +362,11 @@ struct kvm_arch { ...@@ -362,11 +362,11 @@ struct kvm_arch {
* Atomic access to multiple idregs are guarded by kvm_arch.config_lock. * Atomic access to multiple idregs are guarded by kvm_arch.config_lock.
*/ */
#define IDREG_IDX(id) (((sys_reg_CRm(id) - 1) << 3) | sys_reg_Op2(id)) #define IDREG_IDX(id) (((sys_reg_CRm(id) - 1) << 3) | sys_reg_Op2(id))
#define IDX_IDREG(idx) sys_reg(3, 0, 0, ((idx) >> 3) + 1, (idx) & Op2_mask)
#define IDREG(kvm, id) ((kvm)->arch.id_regs[IDREG_IDX(id)])
#define KVM_ARM_ID_REG_NUM (IDREG_IDX(sys_reg(3, 0, 0, 7, 7)) + 1) #define KVM_ARM_ID_REG_NUM (IDREG_IDX(sys_reg(3, 0, 0, 7, 7)) + 1)
u64 id_regs[KVM_ARM_ID_REG_NUM]; u64 id_regs[KVM_ARM_ID_REG_NUM];
u64 ctr_el0;
/* Masks for VNCR-baked sysregs */ /* Masks for VNCR-baked sysregs */
struct kvm_sysreg_masks *sysreg_masks; struct kvm_sysreg_masks *sysreg_masks;
...@@ -1180,7 +1180,7 @@ int __init populate_nv_trap_config(void); ...@@ -1180,7 +1180,7 @@ int __init populate_nv_trap_config(void);
bool lock_all_vcpus(struct kvm *kvm); bool lock_all_vcpus(struct kvm *kvm);
void unlock_all_vcpus(struct kvm *kvm); void unlock_all_vcpus(struct kvm *kvm);
void kvm_init_sysreg(struct kvm_vcpu *); void kvm_calculate_traps(struct kvm_vcpu *vcpu);
/* MMIO helpers */ /* MMIO helpers */
void kvm_mmio_write_buf(void *buf, unsigned int len, unsigned long data); void kvm_mmio_write_buf(void *buf, unsigned int len, unsigned long data);
...@@ -1391,6 +1391,24 @@ static inline void kvm_hyp_reserve(void) { } ...@@ -1391,6 +1391,24 @@ static inline void kvm_hyp_reserve(void) { }
void kvm_arm_vcpu_power_off(struct kvm_vcpu *vcpu); void kvm_arm_vcpu_power_off(struct kvm_vcpu *vcpu);
bool kvm_arm_vcpu_stopped(struct kvm_vcpu *vcpu); bool kvm_arm_vcpu_stopped(struct kvm_vcpu *vcpu);
static inline u64 *__vm_id_reg(struct kvm_arch *ka, u32 reg)
{
switch (reg) {
case sys_reg(3, 0, 0, 1, 0) ... sys_reg(3, 0, 0, 7, 7):
return &ka->id_regs[IDREG_IDX(reg)];
case SYS_CTR_EL0:
return &ka->ctr_el0;
default:
WARN_ON_ONCE(1);
return NULL;
}
}
#define kvm_read_vm_id_reg(kvm, reg) \
({ u64 __val = *__vm_id_reg(&(kvm)->arch, reg); __val; })
void kvm_set_vm_id_reg(struct kvm *kvm, u32 reg, u64 val);
#define __expand_field_sign_unsigned(id, fld, val) \ #define __expand_field_sign_unsigned(id, fld, val) \
((u64)SYS_FIELD_VALUE(id, fld, val)) ((u64)SYS_FIELD_VALUE(id, fld, val))
...@@ -1407,7 +1425,7 @@ bool kvm_arm_vcpu_stopped(struct kvm_vcpu *vcpu); ...@@ -1407,7 +1425,7 @@ bool kvm_arm_vcpu_stopped(struct kvm_vcpu *vcpu);
#define get_idreg_field_unsigned(kvm, id, fld) \ #define get_idreg_field_unsigned(kvm, id, fld) \
({ \ ({ \
u64 __val = IDREG((kvm), SYS_##id); \ u64 __val = kvm_read_vm_id_reg((kvm), SYS_##id); \
FIELD_GET(id##_##fld##_MASK, __val); \ FIELD_GET(id##_##fld##_MASK, __val); \
}) })
......
...@@ -836,7 +836,7 @@ int kvm_arch_vcpu_run_pid_change(struct kvm_vcpu *vcpu) ...@@ -836,7 +836,7 @@ int kvm_arch_vcpu_run_pid_change(struct kvm_vcpu *vcpu)
* This needs to happen after NV has imposed its own restrictions on * This needs to happen after NV has imposed its own restrictions on
* the feature set * the feature set
*/ */
kvm_init_sysreg(vcpu); kvm_calculate_traps(vcpu);
ret = kvm_timer_enable(vcpu); ret = kvm_timer_enable(vcpu);
if (ret) if (ret)
......
...@@ -799,24 +799,23 @@ void kvm_arch_flush_shadow_all(struct kvm *kvm) ...@@ -799,24 +799,23 @@ void kvm_arch_flush_shadow_all(struct kvm *kvm)
* This list should get updated as new features get added to the NV * This list should get updated as new features get added to the NV
* support, and new extension to the architecture. * support, and new extension to the architecture.
*/ */
static u64 limit_nv_id_reg(u32 id, u64 val) static void limit_nv_id_regs(struct kvm *kvm)
{ {
u64 tmp; u64 val, tmp;
switch (id) {
case SYS_ID_AA64ISAR0_EL1:
/* Support everything but TME */ /* Support everything but TME */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64ISAR0_EL1);
val &= ~NV_FTR(ISAR0, TME); val &= ~NV_FTR(ISAR0, TME);
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64ISAR0_EL1, val);
case SYS_ID_AA64ISAR1_EL1:
/* Support everything but Spec Invalidation and LS64 */ /* Support everything but Spec Invalidation and LS64 */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64ISAR1_EL1);
val &= ~(NV_FTR(ISAR1, LS64) | val &= ~(NV_FTR(ISAR1, LS64) |
NV_FTR(ISAR1, SPECRES)); NV_FTR(ISAR1, SPECRES));
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64ISAR1_EL1, val);
case SYS_ID_AA64PFR0_EL1:
/* No AMU, MPAM, S-EL2, RAS or SVE */ /* No AMU, MPAM, S-EL2, RAS or SVE */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64PFR0_EL1);
val &= ~(GENMASK_ULL(55, 52) | val &= ~(GENMASK_ULL(55, 52) |
NV_FTR(PFR0, AMU) | NV_FTR(PFR0, AMU) |
NV_FTR(PFR0, MPAM) | NV_FTR(PFR0, MPAM) |
...@@ -830,17 +829,17 @@ static u64 limit_nv_id_reg(u32 id, u64 val) ...@@ -830,17 +829,17 @@ static u64 limit_nv_id_reg(u32 id, u64 val)
val |= FIELD_PREP(NV_FTR(PFR0, EL1), 0b0001); val |= FIELD_PREP(NV_FTR(PFR0, EL1), 0b0001);
val |= FIELD_PREP(NV_FTR(PFR0, EL2), 0b0001); val |= FIELD_PREP(NV_FTR(PFR0, EL2), 0b0001);
val |= FIELD_PREP(NV_FTR(PFR0, EL3), 0b0001); val |= FIELD_PREP(NV_FTR(PFR0, EL3), 0b0001);
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64PFR0_EL1, val);
case SYS_ID_AA64PFR1_EL1:
/* Only support BTI, SSBS, CSV2_frac */ /* Only support BTI, SSBS, CSV2_frac */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64PFR1_EL1);
val &= (NV_FTR(PFR1, BT) | val &= (NV_FTR(PFR1, BT) |
NV_FTR(PFR1, SSBS) | NV_FTR(PFR1, SSBS) |
NV_FTR(PFR1, CSV2_frac)); NV_FTR(PFR1, CSV2_frac));
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64PFR1_EL1, val);
case SYS_ID_AA64MMFR0_EL1:
/* Hide ECV, ExS, Secure Memory */ /* Hide ECV, ExS, Secure Memory */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64MMFR0_EL1);
val &= ~(NV_FTR(MMFR0, ECV) | val &= ~(NV_FTR(MMFR0, ECV) |
NV_FTR(MMFR0, EXS) | NV_FTR(MMFR0, EXS) |
NV_FTR(MMFR0, TGRAN4_2) | NV_FTR(MMFR0, TGRAN4_2) |
...@@ -883,18 +882,18 @@ static u64 limit_nv_id_reg(u32 id, u64 val) ...@@ -883,18 +882,18 @@ static u64 limit_nv_id_reg(u32 id, u64 val)
val &= ~NV_FTR(MMFR0, PARANGE); val &= ~NV_FTR(MMFR0, PARANGE);
val |= FIELD_PREP(NV_FTR(MMFR0, PARANGE), 0b0101); val |= FIELD_PREP(NV_FTR(MMFR0, PARANGE), 0b0101);
} }
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64MMFR0_EL1, val);
case SYS_ID_AA64MMFR1_EL1: val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64MMFR1_EL1);
val &= (NV_FTR(MMFR1, HCX) | val &= (NV_FTR(MMFR1, HCX) |
NV_FTR(MMFR1, PAN) | NV_FTR(MMFR1, PAN) |
NV_FTR(MMFR1, LO) | NV_FTR(MMFR1, LO) |
NV_FTR(MMFR1, HPDS) | NV_FTR(MMFR1, HPDS) |
NV_FTR(MMFR1, VH) | NV_FTR(MMFR1, VH) |
NV_FTR(MMFR1, VMIDBits)); NV_FTR(MMFR1, VMIDBits));
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64MMFR1_EL1, val);
case SYS_ID_AA64MMFR2_EL1: val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64MMFR2_EL1);
val &= ~(NV_FTR(MMFR2, BBM) | val &= ~(NV_FTR(MMFR2, BBM) |
NV_FTR(MMFR2, TTL) | NV_FTR(MMFR2, TTL) |
GENMASK_ULL(47, 44) | GENMASK_ULL(47, 44) |
...@@ -904,17 +903,16 @@ static u64 limit_nv_id_reg(u32 id, u64 val) ...@@ -904,17 +903,16 @@ static u64 limit_nv_id_reg(u32 id, u64 val)
/* Force TTL support */ /* Force TTL support */
val |= FIELD_PREP(NV_FTR(MMFR2, TTL), 0b0001); val |= FIELD_PREP(NV_FTR(MMFR2, TTL), 0b0001);
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64MMFR2_EL1, val);
case SYS_ID_AA64MMFR4_EL1:
val = 0; val = 0;
if (!cpus_have_final_cap(ARM64_HAS_HCR_NV1)) if (!cpus_have_final_cap(ARM64_HAS_HCR_NV1))
val |= FIELD_PREP(NV_FTR(MMFR4, E2H0), val |= FIELD_PREP(NV_FTR(MMFR4, E2H0),
ID_AA64MMFR4_EL1_E2H0_NI_NV1); ID_AA64MMFR4_EL1_E2H0_NI_NV1);
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64MMFR4_EL1, val);
case SYS_ID_AA64DFR0_EL1:
/* Only limited support for PMU, Debug, BPs and WPs */ /* Only limited support for PMU, Debug, BPs and WPs */
val = kvm_read_vm_id_reg(kvm, SYS_ID_AA64DFR0_EL1);
val &= (NV_FTR(DFR0, PMUVer) | val &= (NV_FTR(DFR0, PMUVer) |
NV_FTR(DFR0, WRPs) | NV_FTR(DFR0, WRPs) |
NV_FTR(DFR0, BRPs) | NV_FTR(DFR0, BRPs) |
...@@ -926,15 +924,7 @@ static u64 limit_nv_id_reg(u32 id, u64 val) ...@@ -926,15 +924,7 @@ static u64 limit_nv_id_reg(u32 id, u64 val)
val &= ~NV_FTR(DFR0, DebugVer); val &= ~NV_FTR(DFR0, DebugVer);
val |= FIELD_PREP(NV_FTR(DFR0, DebugVer), 0b0111); val |= FIELD_PREP(NV_FTR(DFR0, DebugVer), 0b0111);
} }
break; kvm_set_vm_id_reg(kvm, SYS_ID_AA64DFR0_EL1, val);
default:
/* Unknown register, just wipe it clean */
val = 0;
break;
}
return val;
} }
u64 kvm_vcpu_sanitise_vncr_reg(const struct kvm_vcpu *vcpu, enum vcpu_sysreg sr) u64 kvm_vcpu_sanitise_vncr_reg(const struct kvm_vcpu *vcpu, enum vcpu_sysreg sr)
...@@ -979,9 +969,7 @@ int kvm_init_nv_sysregs(struct kvm *kvm) ...@@ -979,9 +969,7 @@ int kvm_init_nv_sysregs(struct kvm *kvm)
goto out; goto out;
} }
for (int i = 0; i < KVM_ARM_ID_REG_NUM; i++) limit_nv_id_regs(kvm);
kvm->arch.id_regs[i] = limit_nv_id_reg(IDX_IDREG(i),
kvm->arch.id_regs[i]);
/* VTTBR_EL2 */ /* VTTBR_EL2 */
res0 = res1 = 0; res0 = res1 = 0;
......
...@@ -54,7 +54,7 @@ static u32 __kvm_pmu_event_mask(unsigned int pmuver) ...@@ -54,7 +54,7 @@ static u32 __kvm_pmu_event_mask(unsigned int pmuver)
static u32 kvm_pmu_event_mask(struct kvm *kvm) static u32 kvm_pmu_event_mask(struct kvm *kvm)
{ {
u64 dfr0 = IDREG(kvm, SYS_ID_AA64DFR0_EL1); u64 dfr0 = kvm_read_vm_id_reg(kvm, SYS_ID_AA64DFR0_EL1);
u8 pmuver = SYS_FIELD_GET(ID_AA64DFR0_EL1, PMUVer, dfr0); u8 pmuver = SYS_FIELD_GET(ID_AA64DFR0_EL1, PMUVer, dfr0);
return __kvm_pmu_event_mask(pmuver); return __kvm_pmu_event_mask(pmuver);
......
...@@ -1565,7 +1565,7 @@ static u64 kvm_read_sanitised_id_reg(struct kvm_vcpu *vcpu, ...@@ -1565,7 +1565,7 @@ static u64 kvm_read_sanitised_id_reg(struct kvm_vcpu *vcpu,
static u64 read_id_reg(const struct kvm_vcpu *vcpu, const struct sys_reg_desc *r) static u64 read_id_reg(const struct kvm_vcpu *vcpu, const struct sys_reg_desc *r)
{ {
return IDREG(vcpu->kvm, reg_to_encoding(r)); return kvm_read_vm_id_reg(vcpu->kvm, reg_to_encoding(r));
} }
static bool is_feature_id_reg(u32 encoding) static bool is_feature_id_reg(u32 encoding)
...@@ -1583,6 +1583,9 @@ static bool is_feature_id_reg(u32 encoding) ...@@ -1583,6 +1583,9 @@ static bool is_feature_id_reg(u32 encoding)
*/ */
static inline bool is_vm_ftr_id_reg(u32 id) static inline bool is_vm_ftr_id_reg(u32 id)
{ {
if (id == SYS_CTR_EL0)
return true;
return (sys_reg_Op0(id) == 3 && sys_reg_Op1(id) == 0 && return (sys_reg_Op0(id) == 3 && sys_reg_Op1(id) == 0 &&
sys_reg_CRn(id) == 0 && sys_reg_CRm(id) >= 1 && sys_reg_CRn(id) == 0 && sys_reg_CRm(id) >= 1 &&
sys_reg_CRm(id) < 8); sys_reg_CRm(id) < 8);
...@@ -1851,7 +1854,7 @@ static int set_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd, ...@@ -1851,7 +1854,7 @@ static int set_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd,
ret = arm64_check_features(vcpu, rd, val); ret = arm64_check_features(vcpu, rd, val);
if (!ret) if (!ret)
IDREG(vcpu->kvm, id) = val; kvm_set_vm_id_reg(vcpu->kvm, id, val);
mutex_unlock(&vcpu->kvm->arch.config_lock); mutex_unlock(&vcpu->kvm->arch.config_lock);
...@@ -1867,6 +1870,18 @@ static int set_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd, ...@@ -1867,6 +1870,18 @@ static int set_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd,
return ret; return ret;
} }
void kvm_set_vm_id_reg(struct kvm *kvm, u32 reg, u64 val)
{
u64 *p = __vm_id_reg(&kvm->arch, reg);
lockdep_assert_held(&kvm->arch.config_lock);
if (KVM_BUG_ON(kvm_vm_has_ran_once(kvm) || !p, kvm))
return;
*p = val;
}
static int get_raz_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd, static int get_raz_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc *rd,
u64 *val) u64 *val)
{ {
...@@ -1886,7 +1901,7 @@ static bool access_ctr(struct kvm_vcpu *vcpu, struct sys_reg_params *p, ...@@ -1886,7 +1901,7 @@ static bool access_ctr(struct kvm_vcpu *vcpu, struct sys_reg_params *p,
if (p->is_write) if (p->is_write)
return write_to_read_only(vcpu, p, r); return write_to_read_only(vcpu, p, r);
p->regval = read_sanitised_ftr_reg(SYS_CTR_EL0); p->regval = kvm_read_vm_id_reg(vcpu->kvm, SYS_CTR_EL0);
return true; return true;
} }
...@@ -2471,11 +2486,14 @@ static const struct sys_reg_desc sys_reg_descs[] = { ...@@ -2471,11 +2486,14 @@ static const struct sys_reg_desc sys_reg_descs[] = {
{ SYS_DESC(SYS_CCSIDR_EL1), access_ccsidr }, { SYS_DESC(SYS_CCSIDR_EL1), access_ccsidr },
{ SYS_DESC(SYS_CLIDR_EL1), access_clidr, reset_clidr, CLIDR_EL1, { SYS_DESC(SYS_CLIDR_EL1), access_clidr, reset_clidr, CLIDR_EL1,
.set_user = set_clidr }, .set_user = set_clidr, .val = ~CLIDR_EL1_RES0 },
{ SYS_DESC(SYS_CCSIDR2_EL1), undef_access }, { SYS_DESC(SYS_CCSIDR2_EL1), undef_access },
{ SYS_DESC(SYS_SMIDR_EL1), undef_access }, { SYS_DESC(SYS_SMIDR_EL1), undef_access },
{ SYS_DESC(SYS_CSSELR_EL1), access_csselr, reset_unknown, CSSELR_EL1 }, { SYS_DESC(SYS_CSSELR_EL1), access_csselr, reset_unknown, CSSELR_EL1 },
{ SYS_DESC(SYS_CTR_EL0), access_ctr }, ID_WRITABLE(CTR_EL0, CTR_EL0_DIC_MASK |
CTR_EL0_IDC_MASK |
CTR_EL0_DminLine_MASK |
CTR_EL0_IminLine_MASK),
{ SYS_DESC(SYS_SVCR), undef_access }, { SYS_DESC(SYS_SVCR), undef_access },
{ PMU_SYS_REG(PMCR_EL0), .access = access_pmcr, .reset = reset_pmcr, { PMU_SYS_REG(PMCR_EL0), .access = access_pmcr, .reset = reset_pmcr,
...@@ -3151,8 +3169,6 @@ static struct sys_reg_desc sys_insn_descs[] = { ...@@ -3151,8 +3169,6 @@ static struct sys_reg_desc sys_insn_descs[] = {
SYS_INSN(TLBI_VMALLS12E1NXS, handle_vmalls12e1is), SYS_INSN(TLBI_VMALLS12E1NXS, handle_vmalls12e1is),
}; };
static const struct sys_reg_desc *first_idreg;
static bool trap_dbgdidr(struct kvm_vcpu *vcpu, static bool trap_dbgdidr(struct kvm_vcpu *vcpu,
struct sys_reg_params *p, struct sys_reg_params *p,
const struct sys_reg_desc *r) const struct sys_reg_desc *r)
...@@ -3160,7 +3176,7 @@ static bool trap_dbgdidr(struct kvm_vcpu *vcpu, ...@@ -3160,7 +3176,7 @@ static bool trap_dbgdidr(struct kvm_vcpu *vcpu,
if (p->is_write) { if (p->is_write) {
return ignore_write(vcpu, p); return ignore_write(vcpu, p);
} else { } else {
u64 dfr = IDREG(vcpu->kvm, SYS_ID_AA64DFR0_EL1); u64 dfr = kvm_read_vm_id_reg(vcpu->kvm, SYS_ID_AA64DFR0_EL1);
u32 el3 = kvm_has_feat(vcpu->kvm, ID_AA64PFR0_EL1, EL3, IMP); u32 el3 = kvm_has_feat(vcpu->kvm, ID_AA64PFR0_EL1, EL3, IMP);
p->regval = ((SYS_FIELD_GET(ID_AA64DFR0_EL1, WRPs, dfr) << 28) | p->regval = ((SYS_FIELD_GET(ID_AA64DFR0_EL1, WRPs, dfr) << 28) |
...@@ -3838,6 +3854,25 @@ static bool emulate_sys_reg(struct kvm_vcpu *vcpu, ...@@ -3838,6 +3854,25 @@ static bool emulate_sys_reg(struct kvm_vcpu *vcpu,
return false; return false;
} }
static const struct sys_reg_desc *idregs_debug_find(struct kvm *kvm, u8 pos)
{
unsigned long i, idreg_idx = 0;
for (i = 0; i < ARRAY_SIZE(sys_reg_descs); i++) {
const struct sys_reg_desc *r = &sys_reg_descs[i];
if (!is_vm_ftr_id_reg(reg_to_encoding(r)))
continue;
if (idreg_idx == pos)
return r;
idreg_idx++;
}
return NULL;
}
static void *idregs_debug_start(struct seq_file *s, loff_t *pos) static void *idregs_debug_start(struct seq_file *s, loff_t *pos)
{ {
struct kvm *kvm = s->private; struct kvm *kvm = s->private;
...@@ -3849,7 +3884,7 @@ static void *idregs_debug_start(struct seq_file *s, loff_t *pos) ...@@ -3849,7 +3884,7 @@ static void *idregs_debug_start(struct seq_file *s, loff_t *pos)
if (test_bit(KVM_ARCH_FLAG_ID_REGS_INITIALIZED, &kvm->arch.flags) && if (test_bit(KVM_ARCH_FLAG_ID_REGS_INITIALIZED, &kvm->arch.flags) &&
*iter == (u8)~0) { *iter == (u8)~0) {
*iter = *pos; *iter = *pos;
if (*iter >= KVM_ARM_ID_REG_NUM) if (!idregs_debug_find(kvm, *iter))
iter = NULL; iter = NULL;
} else { } else {
iter = ERR_PTR(-EBUSY); iter = ERR_PTR(-EBUSY);
...@@ -3866,7 +3901,7 @@ static void *idregs_debug_next(struct seq_file *s, void *v, loff_t *pos) ...@@ -3866,7 +3901,7 @@ static void *idregs_debug_next(struct seq_file *s, void *v, loff_t *pos)
(*pos)++; (*pos)++;
if ((kvm->arch.idreg_debugfs_iter + 1) < KVM_ARM_ID_REG_NUM) { if (idregs_debug_find(kvm, kvm->arch.idreg_debugfs_iter + 1)) {
kvm->arch.idreg_debugfs_iter++; kvm->arch.idreg_debugfs_iter++;
return &kvm->arch.idreg_debugfs_iter; return &kvm->arch.idreg_debugfs_iter;
...@@ -3891,16 +3926,16 @@ static void idregs_debug_stop(struct seq_file *s, void *v) ...@@ -3891,16 +3926,16 @@ static void idregs_debug_stop(struct seq_file *s, void *v)
static int idregs_debug_show(struct seq_file *s, void *v) static int idregs_debug_show(struct seq_file *s, void *v)
{ {
struct kvm *kvm = s->private;
const struct sys_reg_desc *desc; const struct sys_reg_desc *desc;
struct kvm *kvm = s->private;
desc = first_idreg + kvm->arch.idreg_debugfs_iter; desc = idregs_debug_find(kvm, kvm->arch.idreg_debugfs_iter);
if (!desc->name) if (!desc->name)
return 0; return 0;
seq_printf(s, "%20s:\t%016llx\n", seq_printf(s, "%20s:\t%016llx\n",
desc->name, IDREG(kvm, IDX_IDREG(kvm->arch.idreg_debugfs_iter))); desc->name, kvm_read_vm_id_reg(kvm, reg_to_encoding(desc)));
return 0; return 0;
} }
...@@ -3930,8 +3965,7 @@ static void reset_vm_ftr_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc ...@@ -3930,8 +3965,7 @@ static void reset_vm_ftr_id_reg(struct kvm_vcpu *vcpu, const struct sys_reg_desc
if (test_bit(KVM_ARCH_FLAG_ID_REGS_INITIALIZED, &kvm->arch.flags)) if (test_bit(KVM_ARCH_FLAG_ID_REGS_INITIALIZED, &kvm->arch.flags))
return; return;
lockdep_assert_held(&kvm->arch.config_lock); kvm_set_vm_id_reg(kvm, id, reg->reset(vcpu, reg));
IDREG(kvm, id) = reg->reset(vcpu, reg);
} }
static void reset_vcpu_ftr_id_reg(struct kvm_vcpu *vcpu, static void reset_vcpu_ftr_id_reg(struct kvm_vcpu *vcpu,
...@@ -4084,7 +4118,7 @@ id_to_sys_reg_desc(struct kvm_vcpu *vcpu, u64 id, ...@@ -4084,7 +4118,7 @@ id_to_sys_reg_desc(struct kvm_vcpu *vcpu, u64 id,
*/ */
#define FUNCTION_INVARIANT(reg) \ #define FUNCTION_INVARIANT(reg) \
static u64 get_##reg(struct kvm_vcpu *v, \ static u64 reset_##reg(struct kvm_vcpu *v, \
const struct sys_reg_desc *r) \ const struct sys_reg_desc *r) \
{ \ { \
((struct sys_reg_desc *)r)->val = read_sysreg(reg); \ ((struct sys_reg_desc *)r)->val = read_sysreg(reg); \
...@@ -4095,18 +4129,11 @@ FUNCTION_INVARIANT(midr_el1) ...@@ -4095,18 +4129,11 @@ FUNCTION_INVARIANT(midr_el1)
FUNCTION_INVARIANT(revidr_el1) FUNCTION_INVARIANT(revidr_el1)
FUNCTION_INVARIANT(aidr_el1) FUNCTION_INVARIANT(aidr_el1)
static u64 get_ctr_el0(struct kvm_vcpu *v, const struct sys_reg_desc *r)
{
((struct sys_reg_desc *)r)->val = read_sanitised_ftr_reg(SYS_CTR_EL0);
return ((struct sys_reg_desc *)r)->val;
}
/* ->val is filled in by kvm_sys_reg_table_init() */ /* ->val is filled in by kvm_sys_reg_table_init() */
static struct sys_reg_desc invariant_sys_regs[] __ro_after_init = { static struct sys_reg_desc invariant_sys_regs[] __ro_after_init = {
{ SYS_DESC(SYS_MIDR_EL1), NULL, get_midr_el1 }, { SYS_DESC(SYS_MIDR_EL1), NULL, reset_midr_el1 },
{ SYS_DESC(SYS_REVIDR_EL1), NULL, get_revidr_el1 }, { SYS_DESC(SYS_REVIDR_EL1), NULL, reset_revidr_el1 },
{ SYS_DESC(SYS_AIDR_EL1), NULL, get_aidr_el1 }, { SYS_DESC(SYS_AIDR_EL1), NULL, reset_aidr_el1 },
{ SYS_DESC(SYS_CTR_EL0), NULL, get_ctr_el0 },
}; };
static int get_invariant_sys_reg(u64 id, u64 __user *uaddr) static int get_invariant_sys_reg(u64 id, u64 __user *uaddr)
...@@ -4417,20 +4444,11 @@ int kvm_vm_ioctl_get_reg_writable_masks(struct kvm *kvm, struct reg_mask_range * ...@@ -4417,20 +4444,11 @@ int kvm_vm_ioctl_get_reg_writable_masks(struct kvm *kvm, struct reg_mask_range *
if (!is_feature_id_reg(encoding) || !reg->set_user) if (!is_feature_id_reg(encoding) || !reg->set_user)
continue; continue;
/*
* For ID registers, we return the writable mask. Other feature
* registers return a full 64bit mask. That's not necessary
* compliant with a given revision of the architecture, but the
* RES0/RES1 definitions allow us to do that.
*/
if (is_vm_ftr_id_reg(encoding)) {
if (!reg->val || if (!reg->val ||
(is_aa32_id_reg(encoding) && !kvm_supports_32bit_el0())) (is_aa32_id_reg(encoding) && !kvm_supports_32bit_el0())) {
continue; continue;
val = reg->val;
} else {
val = ~0UL;
} }
val = reg->val;
if (put_user(val, (masks + KVM_ARM_FEATURE_ID_RANGE_INDEX(encoding)))) if (put_user(val, (masks + KVM_ARM_FEATURE_ID_RANGE_INDEX(encoding))))
return -EFAULT; return -EFAULT;
...@@ -4439,11 +4457,34 @@ int kvm_vm_ioctl_get_reg_writable_masks(struct kvm *kvm, struct reg_mask_range * ...@@ -4439,11 +4457,34 @@ int kvm_vm_ioctl_get_reg_writable_masks(struct kvm *kvm, struct reg_mask_range *
return 0; return 0;
} }
void kvm_init_sysreg(struct kvm_vcpu *vcpu) static void vcpu_set_hcr(struct kvm_vcpu *vcpu)
{ {
struct kvm *kvm = vcpu->kvm; struct kvm *kvm = vcpu->kvm;
mutex_lock(&kvm->arch.config_lock); if (has_vhe() || has_hvhe())
vcpu->arch.hcr_el2 |= HCR_E2H;
if (cpus_have_final_cap(ARM64_HAS_RAS_EXTN)) {
/* route synchronous external abort exceptions to EL2 */
vcpu->arch.hcr_el2 |= HCR_TEA;
/* trap error record accesses */
vcpu->arch.hcr_el2 |= HCR_TERR;
}
if (cpus_have_final_cap(ARM64_HAS_STAGE2_FWB))
vcpu->arch.hcr_el2 |= HCR_FWB;
if (cpus_have_final_cap(ARM64_HAS_EVT) &&
!cpus_have_final_cap(ARM64_MISMATCHED_CACHE_TYPE) &&
kvm_read_vm_id_reg(kvm, SYS_CTR_EL0) == read_sanitised_ftr_reg(SYS_CTR_EL0))
vcpu->arch.hcr_el2 |= HCR_TID4;
else
vcpu->arch.hcr_el2 |= HCR_TID2;
if (vcpu_el1_is_32bit(vcpu))
vcpu->arch.hcr_el2 &= ~HCR_RW;
if (kvm_has_mte(vcpu->kvm))
vcpu->arch.hcr_el2 |= HCR_ATA;
/* /*
* In the absence of FGT, we cannot independently trap TLBI * In the absence of FGT, we cannot independently trap TLBI
...@@ -4452,6 +4493,14 @@ void kvm_init_sysreg(struct kvm_vcpu *vcpu) ...@@ -4452,6 +4493,14 @@ void kvm_init_sysreg(struct kvm_vcpu *vcpu)
*/ */
if (!kvm_has_feat(kvm, ID_AA64ISAR0_EL1, TLB, OS)) if (!kvm_has_feat(kvm, ID_AA64ISAR0_EL1, TLB, OS))
vcpu->arch.hcr_el2 |= HCR_TTLBOS; vcpu->arch.hcr_el2 |= HCR_TTLBOS;
}
void kvm_calculate_traps(struct kvm_vcpu *vcpu)
{
struct kvm *kvm = vcpu->kvm;
mutex_lock(&kvm->arch.config_lock);
vcpu_set_hcr(vcpu);
if (cpus_have_final_cap(ARM64_HAS_HCX)) { if (cpus_have_final_cap(ARM64_HAS_HCX)) {
vcpu->arch.hcrx_el2 = HCRX_GUEST_FLAGS; vcpu->arch.hcrx_el2 = HCRX_GUEST_FLAGS;
...@@ -4513,7 +4562,6 @@ void kvm_init_sysreg(struct kvm_vcpu *vcpu) ...@@ -4513,7 +4562,6 @@ void kvm_init_sysreg(struct kvm_vcpu *vcpu)
int __init kvm_sys_reg_table_init(void) int __init kvm_sys_reg_table_init(void)
{ {
struct sys_reg_params params;
bool valid = true; bool valid = true;
unsigned int i; unsigned int i;
int ret = 0; int ret = 0;
...@@ -4534,12 +4582,6 @@ int __init kvm_sys_reg_table_init(void) ...@@ -4534,12 +4582,6 @@ int __init kvm_sys_reg_table_init(void)
for (i = 0; i < ARRAY_SIZE(invariant_sys_regs); i++) for (i = 0; i < ARRAY_SIZE(invariant_sys_regs); i++)
invariant_sys_regs[i].reset(NULL, &invariant_sys_regs[i]); invariant_sys_regs[i].reset(NULL, &invariant_sys_regs[i]);
/* Find the first idreg (SYS_ID_PFR0_EL1) in sys_reg_descs. */
params = encoding_to_params(SYS_ID_PFR0_EL1);
first_idreg = find_reg(&params, sys_reg_descs, ARRAY_SIZE(sys_reg_descs));
if (!first_idreg)
return -EINVAL;
ret = populate_nv_trap_config(); ret = populate_nv_trap_config();
for (i = 0; !ret && i < ARRAY_SIZE(sys_reg_descs); i++) for (i = 0; !ret && i < ARRAY_SIZE(sys_reg_descs); i++)
......
...@@ -219,6 +219,7 @@ static void guest_code(void) ...@@ -219,6 +219,7 @@ static void guest_code(void)
GUEST_REG_SYNC(SYS_ID_AA64MMFR1_EL1); GUEST_REG_SYNC(SYS_ID_AA64MMFR1_EL1);
GUEST_REG_SYNC(SYS_ID_AA64MMFR2_EL1); GUEST_REG_SYNC(SYS_ID_AA64MMFR2_EL1);
GUEST_REG_SYNC(SYS_ID_AA64ZFR0_EL1); GUEST_REG_SYNC(SYS_ID_AA64ZFR0_EL1);
GUEST_REG_SYNC(SYS_CTR_EL0);
GUEST_DONE(); GUEST_DONE();
} }
...@@ -490,11 +491,25 @@ static void test_clidr(struct kvm_vcpu *vcpu) ...@@ -490,11 +491,25 @@ static void test_clidr(struct kvm_vcpu *vcpu)
test_reg_vals[encoding_to_range_idx(SYS_CLIDR_EL1)] = clidr; test_reg_vals[encoding_to_range_idx(SYS_CLIDR_EL1)] = clidr;
} }
static void test_ctr(struct kvm_vcpu *vcpu)
{
u64 ctr;
vcpu_get_reg(vcpu, KVM_ARM64_SYS_REG(SYS_CTR_EL0), &ctr);
ctr &= ~CTR_EL0_DIC_MASK;
if (ctr & CTR_EL0_IminLine_MASK)
ctr--;
vcpu_set_reg(vcpu, KVM_ARM64_SYS_REG(SYS_CTR_EL0), ctr);
test_reg_vals[encoding_to_range_idx(SYS_CTR_EL0)] = ctr;
}
static void test_vcpu_ftr_id_regs(struct kvm_vcpu *vcpu) static void test_vcpu_ftr_id_regs(struct kvm_vcpu *vcpu)
{ {
u64 val; u64 val;
test_clidr(vcpu); test_clidr(vcpu);
test_ctr(vcpu);
vcpu_get_reg(vcpu, KVM_ARM64_SYS_REG(SYS_MPIDR_EL1), &val); vcpu_get_reg(vcpu, KVM_ARM64_SYS_REG(SYS_MPIDR_EL1), &val);
val++; val++;
...@@ -524,7 +539,9 @@ static void test_reset_preserves_id_regs(struct kvm_vcpu *vcpu) ...@@ -524,7 +539,9 @@ static void test_reset_preserves_id_regs(struct kvm_vcpu *vcpu)
for (int i = 0; i < ARRAY_SIZE(test_regs); i++) for (int i = 0; i < ARRAY_SIZE(test_regs); i++)
test_assert_id_reg_unchanged(vcpu, test_regs[i].reg); test_assert_id_reg_unchanged(vcpu, test_regs[i].reg);
test_assert_id_reg_unchanged(vcpu, SYS_MPIDR_EL1);
test_assert_id_reg_unchanged(vcpu, SYS_CLIDR_EL1); test_assert_id_reg_unchanged(vcpu, SYS_CLIDR_EL1);
test_assert_id_reg_unchanged(vcpu, SYS_CTR_EL0);
ksft_test_result_pass("%s\n", __func__); ksft_test_result_pass("%s\n", __func__);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment