Commit 5c0aea0e authored by David Hildenbrand's avatar David Hildenbrand Committed by Paolo Bonzini

KVM: x86: don't hold kvm->lock in KVM_SET_GSI_ROUTING

We needed the lock to avoid racing with creation of the irqchip on x86. As
kvm_set_irq_routing() calls srcu_synchronize_expedited(), this lock
might be held for a longer time.

Let's introduce an arch specific callback to check if we can actually
add irq routes. For x86, all we have to do is check if we have an
irqchip in the kernel. We don't need kvm->lock at that point as the
irqchip is marked as inititalized only when actually fully created.
Reported-by: default avatarSteve Rutherford <srutherford@google.com>
Reviewed-by: default avatarRadim Krčmář <rkrcmar@redhat.com>
Fixes: 1df6dded ("KVM: x86: race between KVM_SET_GSI_ROUTING and KVM_CREATE_IRQCHIP")
Signed-off-by: default avatarDavid Hildenbrand <david@redhat.com>
Signed-off-by: default avatarPaolo Bonzini <pbonzini@redhat.com>
parent bcb85c88
...@@ -728,7 +728,6 @@ struct kvm_hv { ...@@ -728,7 +728,6 @@ struct kvm_hv {
enum kvm_irqchip_mode { enum kvm_irqchip_mode {
KVM_IRQCHIP_NONE, KVM_IRQCHIP_NONE,
KVM_IRQCHIP_INIT_IN_PROGRESS, /* temporarily set during creation */
KVM_IRQCHIP_KERNEL, /* created with KVM_CREATE_IRQCHIP */ KVM_IRQCHIP_KERNEL, /* created with KVM_CREATE_IRQCHIP */
KVM_IRQCHIP_SPLIT, /* created with KVM_CAP_SPLIT_IRQCHIP */ KVM_IRQCHIP_SPLIT, /* created with KVM_CAP_SPLIT_IRQCHIP */
}; };
......
...@@ -111,7 +111,7 @@ static inline int irqchip_in_kernel(struct kvm *kvm) ...@@ -111,7 +111,7 @@ static inline int irqchip_in_kernel(struct kvm *kvm)
/* Matches smp_wmb() when setting irqchip_mode */ /* Matches smp_wmb() when setting irqchip_mode */
smp_rmb(); smp_rmb();
return mode > KVM_IRQCHIP_INIT_IN_PROGRESS; return mode != KVM_IRQCHIP_NONE;
} }
void kvm_inject_pending_timer_irqs(struct kvm_vcpu *vcpu); void kvm_inject_pending_timer_irqs(struct kvm_vcpu *vcpu);
......
...@@ -274,16 +274,19 @@ void kvm_fire_mask_notifiers(struct kvm *kvm, unsigned irqchip, unsigned pin, ...@@ -274,16 +274,19 @@ void kvm_fire_mask_notifiers(struct kvm *kvm, unsigned irqchip, unsigned pin,
srcu_read_unlock(&kvm->irq_srcu, idx); srcu_read_unlock(&kvm->irq_srcu, idx);
} }
bool kvm_arch_can_set_irq_routing(struct kvm *kvm)
{
return irqchip_in_kernel(kvm);
}
int kvm_set_routing_entry(struct kvm *kvm, int kvm_set_routing_entry(struct kvm *kvm,
struct kvm_kernel_irq_routing_entry *e, struct kvm_kernel_irq_routing_entry *e,
const struct kvm_irq_routing_entry *ue) const struct kvm_irq_routing_entry *ue)
{ {
/* also allow creation of routes during KVM_IRQCHIP_INIT_IN_PROGRESS */ /* We can't check irqchip_in_kernel() here as some callers are
if (kvm->arch.irqchip_mode == KVM_IRQCHIP_NONE) * currently inititalizing the irqchip. Other callers should therefore
return -EINVAL; * check kvm_arch_can_set_irq_routing() before calling this function.
*/
/* Matches smp_wmb() when setting irqchip_mode */
smp_rmb();
switch (ue->type) { switch (ue->type) {
case KVM_IRQ_ROUTING_IRQCHIP: case KVM_IRQ_ROUTING_IRQCHIP:
if (irqchip_split(kvm)) if (irqchip_split(kvm))
......
...@@ -3919,14 +3919,9 @@ static int kvm_vm_ioctl_enable_cap(struct kvm *kvm, ...@@ -3919,14 +3919,9 @@ static int kvm_vm_ioctl_enable_cap(struct kvm *kvm,
goto split_irqchip_unlock; goto split_irqchip_unlock;
if (kvm->created_vcpus) if (kvm->created_vcpus)
goto split_irqchip_unlock; goto split_irqchip_unlock;
kvm->arch.irqchip_mode = KVM_IRQCHIP_INIT_IN_PROGRESS;
r = kvm_setup_empty_irq_routing(kvm); r = kvm_setup_empty_irq_routing(kvm);
if (r) { if (r)
kvm->arch.irqchip_mode = KVM_IRQCHIP_NONE;
/* Pairs with smp_rmb() when reading irqchip_mode */
smp_wmb();
goto split_irqchip_unlock; goto split_irqchip_unlock;
}
/* Pairs with irqchip_in_kernel. */ /* Pairs with irqchip_in_kernel. */
smp_wmb(); smp_wmb();
kvm->arch.irqchip_mode = KVM_IRQCHIP_SPLIT; kvm->arch.irqchip_mode = KVM_IRQCHIP_SPLIT;
...@@ -4012,12 +4007,8 @@ long kvm_arch_vm_ioctl(struct file *filp, ...@@ -4012,12 +4007,8 @@ long kvm_arch_vm_ioctl(struct file *filp,
goto create_irqchip_unlock; goto create_irqchip_unlock;
} }
kvm->arch.irqchip_mode = KVM_IRQCHIP_INIT_IN_PROGRESS;
r = kvm_setup_default_irq_routing(kvm); r = kvm_setup_default_irq_routing(kvm);
if (r) { if (r) {
kvm->arch.irqchip_mode = KVM_IRQCHIP_NONE;
/* Pairs with smp_rmb() when reading irqchip_mode */
smp_wmb();
kvm_ioapic_destroy(kvm); kvm_ioapic_destroy(kvm);
kvm_pic_destroy(kvm); kvm_pic_destroy(kvm);
goto create_irqchip_unlock; goto create_irqchip_unlock;
......
...@@ -1018,6 +1018,7 @@ static inline int mmu_notifier_retry(struct kvm *kvm, unsigned long mmu_seq) ...@@ -1018,6 +1018,7 @@ static inline int mmu_notifier_retry(struct kvm *kvm, unsigned long mmu_seq)
#define KVM_MAX_IRQ_ROUTES 1024 #define KVM_MAX_IRQ_ROUTES 1024
#endif #endif
bool kvm_arch_can_set_irq_routing(struct kvm *kvm);
int kvm_set_irq_routing(struct kvm *kvm, int kvm_set_irq_routing(struct kvm *kvm,
const struct kvm_irq_routing_entry *entries, const struct kvm_irq_routing_entry *entries,
unsigned nr, unsigned nr,
......
...@@ -172,6 +172,11 @@ void __attribute__((weak)) kvm_arch_irq_routing_update(struct kvm *kvm) ...@@ -172,6 +172,11 @@ void __attribute__((weak)) kvm_arch_irq_routing_update(struct kvm *kvm)
{ {
} }
bool __weak kvm_arch_can_set_irq_routing(struct kvm *kvm)
{
return true;
}
int kvm_set_irq_routing(struct kvm *kvm, int kvm_set_irq_routing(struct kvm *kvm,
const struct kvm_irq_routing_entry *ue, const struct kvm_irq_routing_entry *ue,
unsigned nr, unsigned nr,
......
...@@ -3075,6 +3075,8 @@ static long kvm_vm_ioctl(struct file *filp, ...@@ -3075,6 +3075,8 @@ static long kvm_vm_ioctl(struct file *filp,
if (copy_from_user(&routing, argp, sizeof(routing))) if (copy_from_user(&routing, argp, sizeof(routing)))
goto out; goto out;
r = -EINVAL; r = -EINVAL;
if (!kvm_arch_can_set_irq_routing(kvm))
goto out;
if (routing.nr > KVM_MAX_IRQ_ROUTES) if (routing.nr > KVM_MAX_IRQ_ROUTES)
goto out; goto out;
if (routing.flags) if (routing.flags)
...@@ -3090,11 +3092,8 @@ static long kvm_vm_ioctl(struct file *filp, ...@@ -3090,11 +3092,8 @@ static long kvm_vm_ioctl(struct file *filp,
routing.nr * sizeof(*entries))) routing.nr * sizeof(*entries)))
goto out_free_irq_routing; goto out_free_irq_routing;
} }
/* avoid races with KVM_CREATE_IRQCHIP on x86 */
mutex_lock(&kvm->lock);
r = kvm_set_irq_routing(kvm, entries, routing.nr, r = kvm_set_irq_routing(kvm, entries, routing.nr,
routing.flags); routing.flags);
mutex_unlock(&kvm->lock);
out_free_irq_routing: out_free_irq_routing:
vfree(entries); vfree(entries);
break; break;
......
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