diff options
| author | Michael S. Tsirkin <mst@redhat.com> | 2012-09-16 04:50:30 -0400 |
|---|---|---|
| committer | Marcelo Tosatti <mtosatti@redhat.com> | 2012-09-17 12:46:32 -0400 |
| commit | 9fc77441e5e1bf80b794cc546d2243ee9f4afb75 (patch) | |
| tree | 4661976f4fea7606a54b3f5276c0ecb45760979c /virt | |
| parent | 7454766f7bead388251aedee35a478356a7f4e72 (diff) | |
KVM: make processes waiting on vcpu mutex killable
vcpu mutex can be held for unlimited time so
taking it with mutex_lock on an ioctl is wrong:
one process could be passed a vcpu fd and
call this ioctl on the vcpu used by another process,
it will then be unkillable until the owner exits.
Call mutex_lock_killable instead and return status.
Note: mutex_lock_interruptible would be even nicer,
but I am not sure all users are prepared to handle EINTR
from these ioctls. They might misinterpret it as an error.
Cleanup paths expect a vcpu that can't be used by
any userspace so this will always succeed - catch bugs
by calling BUG_ON.
Catch callers that don't check return state by adding
__must_check.
Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com>
Diffstat (limited to 'virt')
| -rw-r--r-- | virt/kvm/kvm_main.c | 10 |
1 files changed, 7 insertions, 3 deletions
diff --git a/virt/kvm/kvm_main.c b/virt/kvm/kvm_main.c index 4fe02d900810..cc3f6dc506e4 100644 --- a/virt/kvm/kvm_main.c +++ b/virt/kvm/kvm_main.c | |||
| @@ -131,11 +131,12 @@ bool kvm_is_mmio_pfn(pfn_t pfn) | |||
| 131 | /* | 131 | /* |
| 132 | * Switches to specified vcpu, until a matching vcpu_put() | 132 | * Switches to specified vcpu, until a matching vcpu_put() |
| 133 | */ | 133 | */ |
| 134 | void vcpu_load(struct kvm_vcpu *vcpu) | 134 | int vcpu_load(struct kvm_vcpu *vcpu) |
| 135 | { | 135 | { |
| 136 | int cpu; | 136 | int cpu; |
| 137 | 137 | ||
| 138 | mutex_lock(&vcpu->mutex); | 138 | if (mutex_lock_killable(&vcpu->mutex)) |
| 139 | return -EINTR; | ||
| 139 | if (unlikely(vcpu->pid != current->pids[PIDTYPE_PID].pid)) { | 140 | if (unlikely(vcpu->pid != current->pids[PIDTYPE_PID].pid)) { |
| 140 | /* The thread running this VCPU changed. */ | 141 | /* The thread running this VCPU changed. */ |
| 141 | struct pid *oldpid = vcpu->pid; | 142 | struct pid *oldpid = vcpu->pid; |
| @@ -148,6 +149,7 @@ void vcpu_load(struct kvm_vcpu *vcpu) | |||
| 148 | preempt_notifier_register(&vcpu->preempt_notifier); | 149 | preempt_notifier_register(&vcpu->preempt_notifier); |
| 149 | kvm_arch_vcpu_load(vcpu, cpu); | 150 | kvm_arch_vcpu_load(vcpu, cpu); |
| 150 | put_cpu(); | 151 | put_cpu(); |
| 152 | return 0; | ||
| 151 | } | 153 | } |
| 152 | 154 | ||
| 153 | void vcpu_put(struct kvm_vcpu *vcpu) | 155 | void vcpu_put(struct kvm_vcpu *vcpu) |
| @@ -1891,7 +1893,9 @@ static long kvm_vcpu_ioctl(struct file *filp, | |||
| 1891 | #endif | 1893 | #endif |
| 1892 | 1894 | ||
| 1893 | 1895 | ||
| 1894 | vcpu_load(vcpu); | 1896 | r = vcpu_load(vcpu); |
| 1897 | if (r) | ||
| 1898 | return r; | ||
| 1895 | switch (ioctl) { | 1899 | switch (ioctl) { |
| 1896 | case KVM_RUN: | 1900 | case KVM_RUN: |
| 1897 | r = -EINVAL; | 1901 | r = -EINVAL; |
