aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMarcelo Tosatti <mtosatti@redhat.com>2008-06-11 19:32:40 -0400
committerAvi Kivity <avi@qumranet.com>2008-06-24 05:18:18 -0400
commit3094538739415a9225afd2a6c78cb0fe1c1f641b (patch)
treeb6f4b62812fd55ffa60d4383af1e7079e8b29a30
parent6597ca09e6c0e5aec7ffd2b8ab48c671d3c28414 (diff)
KVM: MMU: large page update_pte issue with non-PAE 32-bit guests (resend)
kvm_mmu_pte_write() does not handle 32-bit non-PAE large page backed guests properly. It will instantiate two 2MB sptes pointing to the same physical 2MB page when a guest large pte update is trapped. Instead of duplicating code to handle this, disallow directory level updates to happen through kvm_mmu_pte_write(), so the two 2MB sptes emulating one guest 4MB pte can be correctly created by the page fault handling path. Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com> Signed-off-by: Avi Kivity <avi@qumranet.com>
-rw-r--r--arch/x86/kvm/mmu.c12
1 files changed, 7 insertions, 5 deletions
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c
index 9628091c574d..baa6503894d3 100644
--- a/arch/x86/kvm/mmu.c
+++ b/arch/x86/kvm/mmu.c
@@ -1581,11 +1581,13 @@ static void mmu_pte_write_new_pte(struct kvm_vcpu *vcpu,
1581 u64 *spte, 1581 u64 *spte,
1582 const void *new) 1582 const void *new)
1583{ 1583{
1584 if ((sp->role.level != PT_PAGE_TABLE_LEVEL) 1584 if (sp->role.level != PT_PAGE_TABLE_LEVEL) {
1585 && !vcpu->arch.update_pte.largepage) { 1585 if (!vcpu->arch.update_pte.largepage ||
1586 ++vcpu->kvm->stat.mmu_pde_zapped; 1586 sp->role.glevels == PT32_ROOT_LEVEL) {
1587 return; 1587 ++vcpu->kvm->stat.mmu_pde_zapped;
1588 } 1588 return;
1589 }
1590 }
1589 1591
1590 ++vcpu->kvm->stat.mmu_pte_updated; 1592 ++vcpu->kvm->stat.mmu_pte_updated;
1591 if (sp->role.glevels == PT32_ROOT_LEVEL) 1593 if (sp->role.glevels == PT32_ROOT_LEVEL)