aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm
diff options
context:
space:
mode:
authorChristoffer Dall <christoffer.dall@linaro.org>2014-10-16 10:40:53 -0400
committerChristoffer Dall <christoffer.dall@linaro.org>2014-12-13 08:15:26 -0500
commitf7fa034dc8559c7d7326bfc8bd1a26175abd931a (patch)
treeaa0458365535fcc65662a40fef2dec95a43a47a0 /arch/arm
parentb856a59141b1066d3c896a0d0231f84dabd040af (diff)
arm/arm64: KVM: Clarify KVM_ARM_VCPU_INIT ABI
It is not clear that this ioctl can be called multiple times for a given vcpu. Userspace already does this, so clarify the ABI. Also specify that userspace is expected to always make secondary and subsequent calls to the ioctl with the same parameters for the VCPU as the initial call (which userspace also already does). Add code to check that userspace doesn't violate that ABI in the future, and move the kvm_vcpu_set_target() function which is currently duplicated between the 32-bit and 64-bit versions in guest.c to a common static function in arm.c, shared between both architectures. Acked-by: Marc Zyngier <marc.zyngier@arm.com> Signed-off-by: Christoffer Dall <christoffer.dall@linaro.org>
Diffstat (limited to 'arch/arm')
-rw-r--r--arch/arm/include/asm/kvm_host.h2
-rw-r--r--arch/arm/kvm/arm.c43
-rw-r--r--arch/arm/kvm/guest.c25
3 files changed, 43 insertions, 27 deletions
diff --git a/arch/arm/include/asm/kvm_host.h b/arch/arm/include/asm/kvm_host.h
index 53036e21756b..254e0650e48b 100644
--- a/arch/arm/include/asm/kvm_host.h
+++ b/arch/arm/include/asm/kvm_host.h
@@ -150,8 +150,6 @@ struct kvm_vcpu_stat {
150 u32 halt_wakeup; 150 u32 halt_wakeup;
151}; 151};
152 152
153int kvm_vcpu_set_target(struct kvm_vcpu *vcpu,
154 const struct kvm_vcpu_init *init);
155int kvm_vcpu_preferred_target(struct kvm_vcpu_init *init); 153int kvm_vcpu_preferred_target(struct kvm_vcpu_init *init);
156unsigned long kvm_arm_num_regs(struct kvm_vcpu *vcpu); 154unsigned long kvm_arm_num_regs(struct kvm_vcpu *vcpu);
157int kvm_arm_copy_reg_indices(struct kvm_vcpu *vcpu, u64 __user *indices); 155int kvm_arm_copy_reg_indices(struct kvm_vcpu *vcpu, u64 __user *indices);
diff --git a/arch/arm/kvm/arm.c b/arch/arm/kvm/arm.c
index 24c9ca4076b2..4043769583e7 100644
--- a/arch/arm/kvm/arm.c
+++ b/arch/arm/kvm/arm.c
@@ -263,6 +263,7 @@ int kvm_arch_vcpu_init(struct kvm_vcpu *vcpu)
263{ 263{
264 /* Force users to call KVM_ARM_VCPU_INIT */ 264 /* Force users to call KVM_ARM_VCPU_INIT */
265 vcpu->arch.target = -1; 265 vcpu->arch.target = -1;
266 bitmap_zero(vcpu->arch.features, KVM_VCPU_MAX_FEATURES);
266 267
267 /* Set up the timer */ 268 /* Set up the timer */
268 kvm_timer_vcpu_init(vcpu); 269 kvm_timer_vcpu_init(vcpu);
@@ -649,6 +650,48 @@ int kvm_vm_ioctl_irq_line(struct kvm *kvm, struct kvm_irq_level *irq_level,
649 return -EINVAL; 650 return -EINVAL;
650} 651}
651 652
653static int kvm_vcpu_set_target(struct kvm_vcpu *vcpu,
654 const struct kvm_vcpu_init *init)
655{
656 unsigned int i;
657 int phys_target = kvm_target_cpu();
658
659 if (init->target != phys_target)
660 return -EINVAL;
661
662 /*
663 * Secondary and subsequent calls to KVM_ARM_VCPU_INIT must
664 * use the same target.
665 */
666 if (vcpu->arch.target != -1 && vcpu->arch.target != init->target)
667 return -EINVAL;
668
669 /* -ENOENT for unknown features, -EINVAL for invalid combinations. */
670 for (i = 0; i < sizeof(init->features) * 8; i++) {
671 bool set = (init->features[i / 32] & (1 << (i % 32)));
672
673 if (set && i >= KVM_VCPU_MAX_FEATURES)
674 return -ENOENT;
675
676 /*
677 * Secondary and subsequent calls to KVM_ARM_VCPU_INIT must
678 * use the same feature set.
679 */
680 if (vcpu->arch.target != -1 && i < KVM_VCPU_MAX_FEATURES &&
681 test_bit(i, vcpu->arch.features) != set)
682 return -EINVAL;
683
684 if (set)
685 set_bit(i, vcpu->arch.features);
686 }
687
688 vcpu->arch.target = phys_target;
689
690 /* Now we know what it is, we can reset it. */
691 return kvm_reset_vcpu(vcpu);
692}
693
694
652static int kvm_arch_vcpu_ioctl_vcpu_init(struct kvm_vcpu *vcpu, 695static int kvm_arch_vcpu_ioctl_vcpu_init(struct kvm_vcpu *vcpu,
653 struct kvm_vcpu_init *init) 696 struct kvm_vcpu_init *init)
654{ 697{
diff --git a/arch/arm/kvm/guest.c b/arch/arm/kvm/guest.c
index 8c97208b9b97..384bab67c462 100644
--- a/arch/arm/kvm/guest.c
+++ b/arch/arm/kvm/guest.c
@@ -273,31 +273,6 @@ int __attribute_const__ kvm_target_cpu(void)
273 } 273 }
274} 274}
275 275
276int kvm_vcpu_set_target(struct kvm_vcpu *vcpu,
277 const struct kvm_vcpu_init *init)
278{
279 unsigned int i;
280
281 /* We can only cope with guest==host and only on A15/A7 (for now). */
282 if (init->target != kvm_target_cpu())
283 return -EINVAL;
284
285 vcpu->arch.target = init->target;
286 bitmap_zero(vcpu->arch.features, KVM_VCPU_MAX_FEATURES);
287
288 /* -ENOENT for unknown features, -EINVAL for invalid combinations. */
289 for (i = 0; i < sizeof(init->features) * 8; i++) {
290 if (test_bit(i, (void *)init->features)) {
291 if (i >= KVM_VCPU_MAX_FEATURES)
292 return -ENOENT;
293 set_bit(i, vcpu->arch.features);
294 }
295 }
296
297 /* Now we know what it is, we can reset it. */
298 return kvm_reset_vcpu(vcpu);
299}
300
301int kvm_vcpu_preferred_target(struct kvm_vcpu_init *init) 276int kvm_vcpu_preferred_target(struct kvm_vcpu_init *init)
302{ 277{
303 int target = kvm_target_cpu(); 278 int target = kvm_target_cpu();