kvm: arm64: Only list the registers that can be touched
For partitions, this changes over the lifecycle and is always a subset.
diff --git a/arch/arm64/kvm/guest.c b/arch/arm64/kvm/guest.c
index fae63c3..c15942b 100644
--- a/arch/arm64/kvm/guest.c
+++ b/arch/arm64/kvm/guest.c
@@ -19,6 +19,7 @@
#include <linux/vmalloc.h>
#include <linux/fs.h>
#include <kvm/arm_psci.h>
+#include <kvm/arm_spci.h>
#include <asm/cputype.h>
#include <linux/uaccess.h>
#include <asm/fpsimd.h>
@@ -607,6 +608,10 @@ unsigned long kvm_arm_num_regs(struct kvm_vcpu *vcpu)
{
unsigned long res = 0;
+ /* XXX: This whole thing is pretty nasty */
+ if (kvm_spci_vcpu_reg_list_num(vcpu, &res))
+ return res;
+
res += num_core_regs(vcpu);
res += num_sve_regs(vcpu);
res += kvm_arm_num_sys_reg_descs(vcpu);
@@ -625,6 +630,11 @@ int kvm_arm_copy_reg_indices(struct kvm_vcpu *vcpu, u64 __user *uindices)
{
int ret;
+ /* XXX: This is _really_ nasty */
+ ret = kvm_spci_vcpu_reg_list(vcpu, uindices);
+ if (ret <= 0)
+ return ret;
+
ret = copy_core_reg_indices(vcpu, uindices);
if (ret < 0)
return ret;
diff --git a/arch/arm64/kvm/spci.c b/arch/arm64/kvm/spci.c
index 4a1d998..eb2123a 100644
--- a/arch/arm64/kvm/spci.c
+++ b/arch/arm64/kvm/spci.c
@@ -541,6 +541,46 @@ int kvm_spci_vcpu_first_run_init(struct kvm_vcpu *vcpu)
return 0;
}
+int kvm_spci_vcpu_reg_list_num(struct kvm_vcpu *vcpu, unsigned long *num)
+{
+ const struct kvm_spci_partition *part = part_get_linked(vcpu->kvm);
+
+ if (!part)
+ return 0;
+
+ if (likely(vcpu->arch.has_run_once))
+ *num = 0;
+ else
+ *num = 8;
+
+ return 1;
+}
+
+int kvm_spci_vcpu_reg_list(struct kvm_vcpu *vcpu, u64 __user *uindices)
+{
+ const struct kvm_spci_partition *part = part_get_linked(vcpu->kvm);
+ u64 i;
+
+ if (!part)
+ return 1;
+
+ if (likely(vcpu->arch.has_run_once))
+ return 1;
+
+ for (i = 0; i < 16; i += 2) {
+ u64 reg = KVM_REG_ARM64 | KVM_REG_ARM_CORE
+ | KVM_REG_SIZE_U64 | i;
+
+ if (uindices) {
+ if (put_user(reg, uindices))
+ return -EFAULT;
+ ++uindices;
+ }
+ }
+
+ return 0;
+}
+
/*
* Checks the registers can be accessed by user space. If the vCPU is part of an
* SPCI partition, the only registers that can be accessed are x0-7 and,
diff --git a/include/kvm/arm_spci.h b/include/kvm/arm_spci.h
index 5bd7071..8733901 100644
--- a/include/kvm/arm_spci.h
+++ b/include/kvm/arm_spci.h
@@ -45,6 +45,8 @@ void kvm_spci_destroy_vm(struct kvm *kvm);
int kvm_spci_check_vcpu_init_features(const struct kvm_vcpu *vcpu,
const struct kvm_vcpu_init *init);
int kvm_spci_vcpu_first_run_init(struct kvm_vcpu *vcpu);
+int kvm_spci_vcpu_reg_list_num(struct kvm_vcpu *vcpu, unsigned long *num);
+int kvm_spci_vcpu_reg_list(struct kvm_vcpu *vcpu, u64 __user *uindices);
int kvm_spci_check_vcpu_access_reg(struct kvm_vcpu *vcpu,
struct kvm_one_reg *reg);