aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/kvm/kvm.h
diff options
context:
space:
mode:
authorAvi Kivity <avi@qumranet.com>2007-09-16 12:58:32 -0400
committerAvi Kivity <avi@qumranet.com>2008-01-30 10:52:48 -0500
commitc7addb902054195b995114df154e061c7d604f69 (patch)
tree985910a6c970957126c91e55c55b0e73ae877e0c /drivers/kvm/kvm.h
parent51c6cf662b4b361a09fbd324f4c67875d9bcfbea (diff)
KVM: Allow not-present guest page faults to bypass kvm
There are two classes of page faults trapped by kvm: - host page faults, where the fault is needed to allow kvm to install the shadow pte or update the guest accessed and dirty bits - guest page faults, where the guest has faulted and kvm simply injects the fault back into the guest to handle The second class, guest page faults, is pure overhead. We can eliminate some of it on vmx using the following evil trick: - when we set up a shadow page table entry, if the corresponding guest pte is not present, set up the shadow pte as not present - if the guest pte _is_ present, mark the shadow pte as present but also set one of the reserved bits in the shadow pte - tell the vmx hardware not to trap faults which have the present bit clear With this, normal page-not-present faults go directly to the guest, bypassing kvm entirely. Unfortunately, this trick only works on Intel hardware, as AMD lacks a way to discriminate among page faults based on error code. It is also a little risky since it uses reserved bits which might become unreserved in the future, so a module parameter is provided to disable it. Signed-off-by: Avi Kivity <avi@qumranet.com>
Diffstat (limited to 'drivers/kvm/kvm.h')
-rw-r--r--drivers/kvm/kvm.h3
1 files changed, 3 insertions, 0 deletions
diff --git a/drivers/kvm/kvm.h b/drivers/kvm/kvm.h
index e885b190b798..7de948e9e64e 100644
--- a/drivers/kvm/kvm.h
+++ b/drivers/kvm/kvm.h
@@ -150,6 +150,8 @@ struct kvm_mmu {
150 int (*page_fault)(struct kvm_vcpu *vcpu, gva_t gva, u32 err); 150 int (*page_fault)(struct kvm_vcpu *vcpu, gva_t gva, u32 err);
151 void (*free)(struct kvm_vcpu *vcpu); 151 void (*free)(struct kvm_vcpu *vcpu);
152 gpa_t (*gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t gva); 152 gpa_t (*gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t gva);
153 void (*prefetch_page)(struct kvm_vcpu *vcpu,
154 struct kvm_mmu_page *page);
153 hpa_t root_hpa; 155 hpa_t root_hpa;
154 int root_level; 156 int root_level;
155 int shadow_root_level; 157 int shadow_root_level;
@@ -536,6 +538,7 @@ void kvm_mmu_module_exit(void);
536void kvm_mmu_destroy(struct kvm_vcpu *vcpu); 538void kvm_mmu_destroy(struct kvm_vcpu *vcpu);
537int kvm_mmu_create(struct kvm_vcpu *vcpu); 539int kvm_mmu_create(struct kvm_vcpu *vcpu);
538int kvm_mmu_setup(struct kvm_vcpu *vcpu); 540int kvm_mmu_setup(struct kvm_vcpu *vcpu);
541void kvm_mmu_set_nonpresent_ptes(u64 trap_pte, u64 notrap_pte);
539 542
540int kvm_mmu_reset_context(struct kvm_vcpu *vcpu); 543int kvm_mmu_reset_context(struct kvm_vcpu *vcpu);
541void kvm_mmu_slot_remove_write_access(struct kvm *kvm, int slot); 544void kvm_mmu_slot_remove_write_access(struct kvm *kvm, int slot);