aboutsummaryrefslogtreecommitdiffstats
path: root/arch
diff options
context:
space:
mode:
authorGleb Natapov <gleb@redhat.com>2010-10-14 05:22:53 -0400
committerAvi Kivity <avi@redhat.com>2011-01-12 04:23:17 -0500
commit7c90705bf2a373aa238661bdb6446f27299ef489 (patch)
treed3d00b4413b0d33254d53bbb3285be82444494d9 /arch
parent631bc4878220932fe67fc46fc7cf7cccdb1ec597 (diff)
KVM: Inject asynchronous page fault into a PV guest if page is swapped out.
Send async page fault to a PV guest if it accesses swapped out memory. Guest will choose another task to run upon receiving the fault. Allow async page fault injection only when guest is in user mode since otherwise guest may be in non-sleepable context and will not be able to reschedule. Vcpu will be halted if guest will fault on the same page again or if vcpu executes kernel code. Acked-by: Rik van Riel <riel@redhat.com> Signed-off-by: Gleb Natapov <gleb@redhat.com> Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com>
Diffstat (limited to 'arch')
-rw-r--r--arch/x86/include/asm/kvm_host.h3
-rw-r--r--arch/x86/kvm/mmu.c1
-rw-r--r--arch/x86/kvm/x86.c43
3 files changed, 42 insertions, 5 deletions
diff --git a/arch/x86/include/asm/kvm_host.h b/arch/x86/include/asm/kvm_host.h
index 0d7039804b4c..167375cc49ff 100644
--- a/arch/x86/include/asm/kvm_host.h
+++ b/arch/x86/include/asm/kvm_host.h
@@ -421,6 +421,7 @@ struct kvm_vcpu_arch {
421 gfn_t gfns[roundup_pow_of_two(ASYNC_PF_PER_VCPU)]; 421 gfn_t gfns[roundup_pow_of_two(ASYNC_PF_PER_VCPU)];
422 struct gfn_to_hva_cache data; 422 struct gfn_to_hva_cache data;
423 u64 msr_val; 423 u64 msr_val;
424 u32 id;
424 } apf; 425 } apf;
425}; 426};
426 427
@@ -596,6 +597,7 @@ struct kvm_x86_ops {
596}; 597};
597 598
598struct kvm_arch_async_pf { 599struct kvm_arch_async_pf {
600 u32 token;
599 gfn_t gfn; 601 gfn_t gfn;
600}; 602};
601 603
@@ -819,6 +821,7 @@ void kvm_arch_async_page_present(struct kvm_vcpu *vcpu,
819 struct kvm_async_pf *work); 821 struct kvm_async_pf *work);
820void kvm_arch_async_page_ready(struct kvm_vcpu *vcpu, 822void kvm_arch_async_page_ready(struct kvm_vcpu *vcpu,
821 struct kvm_async_pf *work); 823 struct kvm_async_pf *work);
824bool kvm_arch_can_inject_async_page_present(struct kvm_vcpu *vcpu);
822extern bool kvm_find_async_pf_gfn(struct kvm_vcpu *vcpu, gfn_t gfn); 825extern bool kvm_find_async_pf_gfn(struct kvm_vcpu *vcpu, gfn_t gfn);
823 826
824#endif /* _ASM_X86_KVM_HOST_H */ 827#endif /* _ASM_X86_KVM_HOST_H */
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c
index b2c60986a7ce..64f90db369fb 100644
--- a/arch/x86/kvm/mmu.c
+++ b/arch/x86/kvm/mmu.c
@@ -2592,6 +2592,7 @@ static int nonpaging_page_fault(struct kvm_vcpu *vcpu, gva_t gva,
2592int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn) 2592int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn)
2593{ 2593{
2594 struct kvm_arch_async_pf arch; 2594 struct kvm_arch_async_pf arch;
2595 arch.token = (vcpu->arch.apf.id++ << 12) | vcpu->vcpu_id;
2595 arch.gfn = gfn; 2596 arch.gfn = gfn;
2596 2597
2597 return kvm_setup_async_pf(vcpu, gva, gfn, &arch); 2598 return kvm_setup_async_pf(vcpu, gva, gfn, &arch);
diff --git a/arch/x86/kvm/x86.c b/arch/x86/kvm/x86.c
index 063c07296764..ac4c368afd40 100644
--- a/arch/x86/kvm/x86.c
+++ b/arch/x86/kvm/x86.c
@@ -6248,20 +6248,53 @@ static void kvm_del_async_pf_gfn(struct kvm_vcpu *vcpu, gfn_t gfn)
6248 } 6248 }
6249} 6249}
6250 6250
6251static int apf_put_user(struct kvm_vcpu *vcpu, u32 val)
6252{
6253
6254 return kvm_write_guest_cached(vcpu->kvm, &vcpu->arch.apf.data, &val,
6255 sizeof(val));
6256}
6257
6251void kvm_arch_async_page_not_present(struct kvm_vcpu *vcpu, 6258void kvm_arch_async_page_not_present(struct kvm_vcpu *vcpu,
6252 struct kvm_async_pf *work) 6259 struct kvm_async_pf *work)
6253{ 6260{
6254 trace_kvm_async_pf_not_present(work->gva); 6261 trace_kvm_async_pf_not_present(work->arch.token, work->gva);
6255
6256 kvm_make_request(KVM_REQ_APF_HALT, vcpu);
6257 kvm_add_async_pf_gfn(vcpu, work->arch.gfn); 6262 kvm_add_async_pf_gfn(vcpu, work->arch.gfn);
6263
6264 if (!(vcpu->arch.apf.msr_val & KVM_ASYNC_PF_ENABLED) ||
6265 kvm_x86_ops->get_cpl(vcpu) == 0)
6266 kvm_make_request(KVM_REQ_APF_HALT, vcpu);
6267 else if (!apf_put_user(vcpu, KVM_PV_REASON_PAGE_NOT_PRESENT)) {
6268 vcpu->arch.fault.error_code = 0;
6269 vcpu->arch.fault.address = work->arch.token;
6270 kvm_inject_page_fault(vcpu);
6271 }
6258} 6272}
6259 6273
6260void kvm_arch_async_page_present(struct kvm_vcpu *vcpu, 6274void kvm_arch_async_page_present(struct kvm_vcpu *vcpu,
6261 struct kvm_async_pf *work) 6275 struct kvm_async_pf *work)
6262{ 6276{
6263 trace_kvm_async_pf_ready(work->gva); 6277 trace_kvm_async_pf_ready(work->arch.token, work->gva);
6264 kvm_del_async_pf_gfn(vcpu, work->arch.gfn); 6278 if (is_error_page(work->page))
6279 work->arch.token = ~0; /* broadcast wakeup */
6280 else
6281 kvm_del_async_pf_gfn(vcpu, work->arch.gfn);
6282
6283 if ((vcpu->arch.apf.msr_val & KVM_ASYNC_PF_ENABLED) &&
6284 !apf_put_user(vcpu, KVM_PV_REASON_PAGE_READY)) {
6285 vcpu->arch.fault.error_code = 0;
6286 vcpu->arch.fault.address = work->arch.token;
6287 kvm_inject_page_fault(vcpu);
6288 }
6289}
6290
6291bool kvm_arch_can_inject_async_page_present(struct kvm_vcpu *vcpu)
6292{
6293 if (!(vcpu->arch.apf.msr_val & KVM_ASYNC_PF_ENABLED))
6294 return true;
6295 else
6296 return !kvm_event_needs_reinjection(vcpu) &&
6297 kvm_x86_ops->interrupt_allowed(vcpu);
6265} 6298}
6266 6299
6267EXPORT_TRACEPOINT_SYMBOL_GPL(kvm_exit); 6300EXPORT_TRACEPOINT_SYMBOL_GPL(kvm_exit);