[PATCH v5 07/14] KVM: ARM: World-switch implementation
Gleb Natapov
gleb at redhat.com
Wed Jan 16 10:52:12 EST 2013
On Wed, Jan 16, 2013 at 10:42:02AM -0500, Christoffer Dall wrote:
> [...]
>
> >
> >> read side RCU protects against is the memslots data structure as far
> >> as I can see, so the second patch pasted below fixes this for the code
> >> that actually accesses this data structure.
> > Many memory related functions that you call access memslots under the
> > hood and assume that locking is done by the caller. From the quick look
> > I found those that you've missed:
> > kvm_is_visible_gfn()
> > kvm_read_guest()
> > gfn_to_hva()
> > gfn_to_pfn_prot()
> > kvm_memslots()
> >
> > May be there are more. Can you enable RCU debugging in your kernel config
> > and check? This does not guaranty that it will catch all of the places,
> > but better than nothing.
> >
>
> yeah, I missed the call to is_visible_gfn and friends, this fixes it:
>
Thank you. One more kvm_read_guest() in emulate.c.
> diff --git a/arch/arm/kvm/mmu.c b/arch/arm/kvm/mmu.c
> index c806080..f30e131 100644
> --- a/arch/arm/kvm/mmu.c
> +++ b/arch/arm/kvm/mmu.c
> @@ -591,7 +591,7 @@ int kvm_handle_guest_abort(struct kvm_vcpu *vcpu,
> struct kvm_run *run)
> struct kvm_memory_slot *memslot;
> bool is_iabt;
> gfn_t gfn;
> - int ret;
> + int ret, idx;
>
> hsr_ec = vcpu->arch.hsr >> HSR_EC_SHIFT;
> is_iabt = (hsr_ec == HSR_EC_IABT);
> @@ -608,33 +608,43 @@ int kvm_handle_guest_abort(struct kvm_vcpu
> *vcpu, struct kvm_run *run)
> return -EFAULT;
> }
>
> + idx = srcu_read_lock(&vcpu->kvm->srcu);
> +
> gfn = fault_ipa >> PAGE_SHIFT;
> if (!kvm_is_visible_gfn(vcpu->kvm, gfn)) {
> if (is_iabt) {
> /* Prefetch Abort on I/O address */
> kvm_inject_pabt(vcpu, vcpu->arch.hxfar);
> - return 1;
> + ret = 1;
> + goto out_unlock;
> }
>
> if (fault_status != FSC_FAULT) {
> kvm_err("Unsupported fault status on io memory: %#lx\n",
> fault_status);
> - return -EFAULT;
> + ret = -EFAULT;
> + goto out_unlock;
> }
>
> /* Adjust page offset */
> fault_ipa |= vcpu->arch.hxfar & ~PAGE_MASK;
> - return io_mem_abort(vcpu, run, fault_ipa);
> + ret = io_mem_abort(vcpu, run, fault_ipa);
> + goto out_unlock;
> }
>
> memslot = gfn_to_memslot(vcpu->kvm, gfn);
> if (!memslot->user_alloc) {
> kvm_err("non user-alloc memslots not supported\n");
> - return -EINVAL;
> + ret = -EINVAL;
> + goto out_unlock;
> }
>
> ret = user_mem_abort(vcpu, fault_ipa, gfn, memslot, fault_status);
> - return ret ? ret : 1;
> + if (ret == 0)
> + ret = 1;
> +out_unlock:
> + srcu_read_unlock(&vcpu->kvm->srcu, idx);
> + return ret;
> }
>
> static void handle_hva_to_gpa(struct kvm *kvm,
> --
>
> Thanks,
> -Christoffer
--
Gleb.
More information about the linux-arm-kernel
mailing list