aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/kvm/x86.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/kvm/x86.c')
-rw-r--r--drivers/kvm/x86.c25
1 files changed, 25 insertions, 0 deletions
diff --git a/drivers/kvm/x86.c b/drivers/kvm/x86.c
index ac09f381f47f..15e1203faef0 100644
--- a/drivers/kvm/x86.c
+++ b/drivers/kvm/x86.c
@@ -166,6 +166,26 @@ out:
166 return ret; 166 return ret;
167} 167}
168 168
169static bool pdptrs_changed(struct kvm_vcpu *vcpu)
170{
171 u64 pdpte[ARRAY_SIZE(vcpu->pdptrs)];
172 bool changed = true;
173 int r;
174
175 if (is_long_mode(vcpu) || !is_pae(vcpu))
176 return false;
177
178 mutex_lock(&vcpu->kvm->lock);
179 r = kvm_read_guest(vcpu->kvm, vcpu->cr3 & ~31u, pdpte, sizeof(pdpte));
180 if (r < 0)
181 goto out;
182 changed = memcmp(pdpte, vcpu->pdptrs, sizeof(pdpte)) != 0;
183out:
184 mutex_unlock(&vcpu->kvm->lock);
185
186 return changed;
187}
188
169void set_cr0(struct kvm_vcpu *vcpu, unsigned long cr0) 189void set_cr0(struct kvm_vcpu *vcpu, unsigned long cr0)
170{ 190{
171 if (cr0 & CR0_RESERVED_BITS) { 191 if (cr0 & CR0_RESERVED_BITS) {
@@ -271,6 +291,11 @@ EXPORT_SYMBOL_GPL(set_cr4);
271 291
272void set_cr3(struct kvm_vcpu *vcpu, unsigned long cr3) 292void set_cr3(struct kvm_vcpu *vcpu, unsigned long cr3)
273{ 293{
294 if (cr3 == vcpu->cr3 && !pdptrs_changed(vcpu)) {
295 kvm_mmu_flush_tlb(vcpu);
296 return;
297 }
298
274 if (is_long_mode(vcpu)) { 299 if (is_long_mode(vcpu)) {
275 if (cr3 & CR3_L_MODE_RESERVED_BITS) { 300 if (cr3 & CR3_L_MODE_RESERVED_BITS) {
276 printk(KERN_DEBUG "set_cr3: #GP, reserved bits\n"); 301 printk(KERN_DEBUG "set_cr3: #GP, reserved bits\n");