On Fri, Jul 05, 2013 at 10:55:53PM +0200, Dominik Dingel wrote: > By setting a Kconfig option, the architecture can control when > guest notifications will be presented by the apf backend. > So there is the default batch mechanism, working as before, where the vcpu thread > should pull in this information. On the other hand there is now the direct > mechanism, this will directly push the information to the guest. > > Still the vcpu thread should call check_completion to cleanup leftovers, > that leaves most of the common code untouched. > > Signed-off-by: Dominik Dingel <dingel@xxxxxxxxxxxxxxxxxx> > --- > arch/x86/kvm/mmu.c | 2 +- > include/linux/kvm_host.h | 2 +- > virt/kvm/Kconfig | 4 ++++ > virt/kvm/async_pf.c | 22 +++++++++++++++++++--- > 4 files changed, 25 insertions(+), 5 deletions(-) > > diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c > index 0d094da..b8632e9 100644 > --- a/arch/x86/kvm/mmu.c > +++ b/arch/x86/kvm/mmu.c > @@ -3343,7 +3343,7 @@ static int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn) > arch.direct_map = vcpu->arch.mmu.direct_map; > arch.cr3 = vcpu->arch.mmu.get_cr3(vcpu); > > - return kvm_setup_async_pf(vcpu, gva, gfn, &arch); > + return kvm_setup_async_pf(vcpu, gva, gfn_to_hva(vcpu->kvm, gfn), &arch); > } > > static bool can_do_async_pf(struct kvm_vcpu *vcpu) > diff --git a/include/linux/kvm_host.h b/include/linux/kvm_host.h > index 210f493..969d575 100644 > --- a/include/linux/kvm_host.h > +++ b/include/linux/kvm_host.h > @@ -175,7 +175,7 @@ struct kvm_async_pf { > > void kvm_clear_async_pf_completion_queue(struct kvm_vcpu *vcpu); > void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu); > -int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn, > +int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, unsigned long hva, > struct kvm_arch_async_pf *arch); > int kvm_async_pf_wakeup_all(struct kvm_vcpu *vcpu); > #endif > diff --git a/virt/kvm/Kconfig b/virt/kvm/Kconfig > index 779262f..715e6b5 100644 > --- a/virt/kvm/Kconfig > +++ b/virt/kvm/Kconfig > @@ -22,6 +22,10 @@ config KVM_MMIO > config KVM_ASYNC_PF > bool > > +# Toggle to switch between direct notification and batch job > +config KVM_ASYNC_PF_DIRECT > + bool > + > config HAVE_KVM_MSI > bool > > diff --git a/virt/kvm/async_pf.c b/virt/kvm/async_pf.c > index ea475cd..b8df37a 100644 > --- a/virt/kvm/async_pf.c > +++ b/virt/kvm/async_pf.c > @@ -28,6 +28,21 @@ > #include "async_pf.h" > #include <trace/events/kvm.h> > > +static inline void kvm_async_page_direct_present(struct kvm_vcpu *vcpu, > + struct kvm_async_pf *work) > +{ > +#ifdef CONFIG_KVM_ASYNC_PF_DIRECT > + kvm_arch_async_page_present(vcpu, work); > +#endif > +} > +static inline void kvm_async_page_batch_present(struct kvm_vcpu *vcpu, > + struct kvm_async_pf *work) > +{ > +#ifndef CONFIG_KVM_ASYNC_PF_DIRECT > + kvm_arch_async_page_present(vcpu, work); > +#endif > +} > + I would call them kvm_async_page_present_(async|sync)(). Hmm, to much "sync" in each function name, but I still think it is better. > static struct kmem_cache *async_pf_cache; > > int kvm_async_pf_init(void) > @@ -70,6 +85,7 @@ static void async_pf_execute(struct work_struct *work) > down_read(&mm->mmap_sem); > get_user_pages(current, mm, addr, 1, 1, 0, &page, NULL); > up_read(&mm->mmap_sem); > + kvm_async_page_direct_present(vcpu, apf); > unuse_mm(mm); > > spin_lock(&vcpu->async_pf.lock); > @@ -134,7 +150,7 @@ void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu) > > if (work->page) > kvm_arch_async_page_ready(vcpu, work); > - kvm_arch_async_page_present(vcpu, work); > + kvm_async_page_batch_present(vcpu, work); > > list_del(&work->queue); > vcpu->async_pf.queued--; > @@ -144,7 +160,7 @@ void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu) > } > } > > -int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn, > +int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, unsigned long hva, > struct kvm_arch_async_pf *arch) > { > struct kvm_async_pf *work; > @@ -166,7 +182,7 @@ int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn, > work->done = false; > work->vcpu = vcpu; > work->gva = gva; > - work->addr = gfn_to_hva(vcpu->kvm, gfn); > + work->addr = hva; > work->arch = *arch; > work->mm = current->mm; > atomic_inc(&work->mm->mm_count); > -- > 1.8.2.2 -- Gleb. -- To unsubscribe from this list: send the line "unsubscribe kvm" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html