KVM: arm/arm64: Allow setting the timer IRQ numbers from userspace
authorChristoffer Dall <cdall@linaro.org>
Tue, 2 May 2017 18:19:15 +0000 (20:19 +0200)
committerChristoffer Dall <cdall@linaro.org>
Thu, 8 Jun 2017 14:59:57 +0000 (16:59 +0200)
First we define an ABI using the vcpu devices that lets userspace set
the interrupt numbers for the various timers on both the 32-bit and
64-bit KVM/ARM implementations.

Second, we add the definitions for the groups and attributes introduced
by the above ABI.  (We add the PMU define on the 32-bit side as well for
symmetry and it may get used some day.)

Third, we set up the arch-specific vcpu device operation handlers to
call into the timer code for anything related to the
KVM_ARM_VCPU_TIMER_CTRL group.

Fourth, we implement support for getting and setting the timer interrupt
numbers using the above defined ABI in the arch timer code.

Fifth, we introduce error checking upon enabling the arch timer (which
is called when first running a VCPU) to check that all VCPUs are
configured to use the same PPI for the timer (as mandated by the
architecture) and that the virtual and physical timers are not
configured to use the same IRQ number.

Signed-off-by: Christoffer Dall <cdall@linaro.org>
Reviewed-by: Marc Zyngier <marc.zyngier@arm.com>
Documentation/virtual/kvm/devices/vcpu.txt
arch/arm/include/uapi/asm/kvm.h
arch/arm/kvm/guest.c
arch/arm64/include/uapi/asm/kvm.h
arch/arm64/kvm/guest.c
include/kvm/arm_arch_timer.h
virt/kvm/arm/arch_timer.c

index d7236a3e01dc227f32c27a10c135ca09bc1bfbe1..2b5dab16c4f20552ecd17ff133c24649d43ef41a 100644 (file)
@@ -35,3 +35,28 @@ Returns: -ENODEV: PMUv3 not supported or GIC not initialized
 Request the initialization of the PMUv3.  If using the PMUv3 with an in-kernel
 virtual GIC implementation, this must be done after initializing the in-kernel
 irqchip.
+
+
+2. GROUP: KVM_ARM_VCPU_TIMER_CTRL
+Architectures: ARM,ARM64
+
+2.1. ATTRIBUTE: KVM_ARM_VCPU_TIMER_IRQ_VTIMER
+2.2. ATTRIBUTE: KVM_ARM_VCPU_TIMER_IRQ_PTIMER
+Parameters: in kvm_device_attr.addr the address for the timer interrupt is a
+            pointer to an int
+Returns: -EINVAL: Invalid timer interrupt number
+         -EBUSY:  One or more VCPUs has already run
+
+A value describing the architected timer interrupt number when connected to an
+in-kernel virtual GIC.  These must be a PPI (16 <= intid < 32).  Setting the
+attribute overrides the default values (see below).
+
+KVM_ARM_VCPU_TIMER_IRQ_VTIMER: The EL1 virtual timer intid (default: 27)
+KVM_ARM_VCPU_TIMER_IRQ_PTIMER: The EL1 physical timer intid (default: 30)
+
+Setting the same PPI for different timers will prevent the VCPUs from running.
+Setting the interrupt number on a VCPU configures all VCPUs created at that
+time to use the number provided for a given timer, overwriting any previously
+configured values on other VCPUs.  Userspace should configure the interrupt
+numbers on at least one VCPU after creating all VCPUs and before running any
+VCPUs.
index 5e3c673fa3f4435b027fbc634815be10c26cf39f..5db2d4c6a55faf7940be63b8b03ee2a7383fe988 100644 (file)
@@ -203,6 +203,14 @@ struct kvm_arch_memory_slot {
 #define KVM_DEV_ARM_VGIC_LINE_LEVEL_INTID_MASK 0x3ff
 #define VGIC_LEVEL_INFO_LINE_LEVEL     0
 
+/* Device Control API on vcpu fd */
+#define KVM_ARM_VCPU_PMU_V3_CTRL       0
+#define   KVM_ARM_VCPU_PMU_V3_IRQ      0
+#define   KVM_ARM_VCPU_PMU_V3_INIT     1
+#define KVM_ARM_VCPU_TIMER_CTRL                1
+#define   KVM_ARM_VCPU_TIMER_IRQ_VTIMER                0
+#define   KVM_ARM_VCPU_TIMER_IRQ_PTIMER                1
+
 #define   KVM_DEV_ARM_VGIC_CTRL_INIT           0
 #define   KVM_DEV_ARM_ITS_SAVE_TABLES          1
 #define   KVM_DEV_ARM_ITS_RESTORE_TABLES       2
index acea05e9db4ef978628bb5bc7fc114dffdeb59e6..1e0784ebbfd6db77595652961ad3c48c41fceabe 100644 (file)
@@ -308,6 +308,9 @@ int kvm_arm_vcpu_arch_set_attr(struct kvm_vcpu *vcpu,
        int ret;
 
        switch (attr->group) {
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_set_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
@@ -322,6 +325,9 @@ int kvm_arm_vcpu_arch_get_attr(struct kvm_vcpu *vcpu,
        int ret;
 
        switch (attr->group) {
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_get_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
@@ -336,6 +342,9 @@ int kvm_arm_vcpu_arch_has_attr(struct kvm_vcpu *vcpu,
        int ret;
 
        switch (attr->group) {
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_has_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
index 70eea2ecc6631cab3d718c3958cd0513a7f7e6a3..9f3ca24bbcc6c28da8e1fbaf0d97998c11e93261 100644 (file)
@@ -232,6 +232,9 @@ struct kvm_arch_memory_slot {
 #define KVM_ARM_VCPU_PMU_V3_CTRL       0
 #define   KVM_ARM_VCPU_PMU_V3_IRQ      0
 #define   KVM_ARM_VCPU_PMU_V3_INIT     1
+#define KVM_ARM_VCPU_TIMER_CTRL                1
+#define   KVM_ARM_VCPU_TIMER_IRQ_VTIMER                0
+#define   KVM_ARM_VCPU_TIMER_IRQ_PTIMER                1
 
 /* KVM_IRQ_LINE irq field index values */
 #define KVM_ARM_IRQ_TYPE_SHIFT         24
index b37446a8ffdb89de0c7072ef576f7de9fa354fa6..5c7f657dd20740c1b21e75d13cf711f4b7bfbdf8 100644 (file)
@@ -390,6 +390,9 @@ int kvm_arm_vcpu_arch_set_attr(struct kvm_vcpu *vcpu,
        case KVM_ARM_VCPU_PMU_V3_CTRL:
                ret = kvm_arm_pmu_v3_set_attr(vcpu, attr);
                break;
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_set_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
@@ -407,6 +410,9 @@ int kvm_arm_vcpu_arch_get_attr(struct kvm_vcpu *vcpu,
        case KVM_ARM_VCPU_PMU_V3_CTRL:
                ret = kvm_arm_pmu_v3_get_attr(vcpu, attr);
                break;
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_get_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
@@ -424,6 +430,9 @@ int kvm_arm_vcpu_arch_has_attr(struct kvm_vcpu *vcpu,
        case KVM_ARM_VCPU_PMU_V3_CTRL:
                ret = kvm_arm_pmu_v3_has_attr(vcpu, attr);
                break;
+       case KVM_ARM_VCPU_TIMER_CTRL:
+               ret = kvm_arm_timer_has_attr(vcpu, attr);
+               break;
        default:
                ret = -ENXIO;
                break;
index f1c967a4f603ae7982d43f5887331d2a38a97570..f0053f884b4a2abd860be1a131fadf6d2b02df52 100644 (file)
@@ -68,6 +68,10 @@ void kvm_timer_vcpu_terminate(struct kvm_vcpu *vcpu);
 u64 kvm_arm_timer_get_reg(struct kvm_vcpu *, u64 regid);
 int kvm_arm_timer_set_reg(struct kvm_vcpu *, u64 regid, u64 value);
 
+int kvm_arm_timer_set_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr);
+int kvm_arm_timer_get_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr);
+int kvm_arm_timer_has_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr);
+
 bool kvm_timer_should_fire(struct arch_timer_context *timer_ctx);
 void kvm_timer_schedule(struct kvm_vcpu *vcpu);
 void kvm_timer_unschedule(struct kvm_vcpu *vcpu);
index 72d5aa7d4c64e3c051f660088a18f5b77808734e..e03da1abd11f5a72f7708a8dc502e615b2d11632 100644 (file)
@@ -21,6 +21,7 @@
 #include <linux/kvm_host.h>
 #include <linux/interrupt.h>
 #include <linux/irq.h>
+#include <linux/uaccess.h>
 
 #include <clocksource/arm_arch_timer.h>
 #include <asm/arch_timer.h>
@@ -617,6 +618,28 @@ void kvm_timer_vcpu_terminate(struct kvm_vcpu *vcpu)
        kvm_vgic_unmap_phys_irq(vcpu, vtimer->irq.irq);
 }
 
+static bool timer_irqs_are_valid(struct kvm *kvm)
+{
+       struct kvm_vcpu *vcpu;
+       int vtimer_irq, ptimer_irq;
+       int i;
+
+       vcpu = kvm_get_vcpu(kvm, 0);
+       vtimer_irq = vcpu_vtimer(vcpu)->irq.irq;
+       ptimer_irq = vcpu_ptimer(vcpu)->irq.irq;
+
+       if (vtimer_irq == ptimer_irq)
+               return false;
+
+       kvm_for_each_vcpu(i, vcpu, kvm) {
+               if (vcpu_vtimer(vcpu)->irq.irq != vtimer_irq ||
+                   vcpu_ptimer(vcpu)->irq.irq != ptimer_irq)
+                       return false;
+       }
+
+       return true;
+}
+
 int kvm_timer_enable(struct kvm_vcpu *vcpu)
 {
        struct arch_timer_cpu *timer = &vcpu->arch.timer_cpu;
@@ -636,6 +659,11 @@ int kvm_timer_enable(struct kvm_vcpu *vcpu)
        if (!vgic_initialized(vcpu->kvm))
                return -ENODEV;
 
+       if (!timer_irqs_are_valid(vcpu->kvm)) {
+               kvm_debug("incorrectly configured timer irqs\n");
+               return -EINVAL;
+       }
+
        /*
         * Find the physical IRQ number corresponding to the host_vtimer_irq
         */
@@ -685,3 +713,79 @@ void kvm_timer_init_vhe(void)
        val |= (CNTHCTL_EL1PCTEN << cnthctl_shift);
        write_sysreg(val, cnthctl_el2);
 }
+
+static void set_timer_irqs(struct kvm *kvm, int vtimer_irq, int ptimer_irq)
+{
+       struct kvm_vcpu *vcpu;
+       int i;
+
+       kvm_for_each_vcpu(i, vcpu, kvm) {
+               vcpu_vtimer(vcpu)->irq.irq = vtimer_irq;
+               vcpu_ptimer(vcpu)->irq.irq = ptimer_irq;
+       }
+}
+
+int kvm_arm_timer_set_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr)
+{
+       int __user *uaddr = (int __user *)(long)attr->addr;
+       struct arch_timer_context *vtimer = vcpu_vtimer(vcpu);
+       struct arch_timer_context *ptimer = vcpu_ptimer(vcpu);
+       int irq;
+
+       if (!irqchip_in_kernel(vcpu->kvm))
+               return -EINVAL;
+
+       if (get_user(irq, uaddr))
+               return -EFAULT;
+
+       if (!(irq_is_ppi(irq)))
+               return -EINVAL;
+
+       if (vcpu->arch.timer_cpu.enabled)
+               return -EBUSY;
+
+       switch (attr->attr) {
+       case KVM_ARM_VCPU_TIMER_IRQ_VTIMER:
+               set_timer_irqs(vcpu->kvm, irq, ptimer->irq.irq);
+               break;
+       case KVM_ARM_VCPU_TIMER_IRQ_PTIMER:
+               set_timer_irqs(vcpu->kvm, vtimer->irq.irq, irq);
+               break;
+       default:
+               return -ENXIO;
+       }
+
+       return 0;
+}
+
+int kvm_arm_timer_get_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr)
+{
+       int __user *uaddr = (int __user *)(long)attr->addr;
+       struct arch_timer_context *timer;
+       int irq;
+
+       switch (attr->attr) {
+       case KVM_ARM_VCPU_TIMER_IRQ_VTIMER:
+               timer = vcpu_vtimer(vcpu);
+               break;
+       case KVM_ARM_VCPU_TIMER_IRQ_PTIMER:
+               timer = vcpu_ptimer(vcpu);
+               break;
+       default:
+               return -ENXIO;
+       }
+
+       irq = timer->irq.irq;
+       return put_user(irq, uaddr);
+}
+
+int kvm_arm_timer_has_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr)
+{
+       switch (attr->attr) {
+       case KVM_ARM_VCPU_TIMER_IRQ_VTIMER:
+       case KVM_ARM_VCPU_TIMER_IRQ_PTIMER:
+               return 0;
+       }
+
+       return -ENXIO;
+}