diff options
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/kvm/mmu.c | 16 | ||||
-rw-r--r-- | arch/x86/kvm/paging_tmpl.h | 44 |
2 files changed, 27 insertions, 33 deletions
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c index b432a71a1839..d15f908649e7 100644 --- a/arch/x86/kvm/mmu.c +++ b/arch/x86/kvm/mmu.c | |||
@@ -1809,7 +1809,7 @@ static void validate_direct_spte(struct kvm_vcpu *vcpu, u64 *sptep, | |||
1809 | } | 1809 | } |
1810 | } | 1810 | } |
1811 | 1811 | ||
1812 | static void mmu_page_zap_pte(struct kvm *kvm, struct kvm_mmu_page *sp, | 1812 | static bool mmu_page_zap_pte(struct kvm *kvm, struct kvm_mmu_page *sp, |
1813 | u64 *spte) | 1813 | u64 *spte) |
1814 | { | 1814 | { |
1815 | u64 pte; | 1815 | u64 pte; |
@@ -1817,17 +1817,21 @@ static void mmu_page_zap_pte(struct kvm *kvm, struct kvm_mmu_page *sp, | |||
1817 | 1817 | ||
1818 | pte = *spte; | 1818 | pte = *spte; |
1819 | if (is_shadow_present_pte(pte)) { | 1819 | if (is_shadow_present_pte(pte)) { |
1820 | if (is_last_spte(pte, sp->role.level)) | 1820 | if (is_last_spte(pte, sp->role.level)) { |
1821 | drop_spte(kvm, spte); | 1821 | drop_spte(kvm, spte); |
1822 | else { | 1822 | if (is_large_pte(pte)) |
1823 | --kvm->stat.lpages; | ||
1824 | } else { | ||
1823 | child = page_header(pte & PT64_BASE_ADDR_MASK); | 1825 | child = page_header(pte & PT64_BASE_ADDR_MASK); |
1824 | drop_parent_pte(child, spte); | 1826 | drop_parent_pte(child, spte); |
1825 | } | 1827 | } |
1826 | } else if (is_mmio_spte(pte)) | 1828 | return true; |
1829 | } | ||
1830 | |||
1831 | if (is_mmio_spte(pte)) | ||
1827 | mmu_spte_clear_no_track(spte); | 1832 | mmu_spte_clear_no_track(spte); |
1828 | 1833 | ||
1829 | if (is_large_pte(pte)) | 1834 | return false; |
1830 | --kvm->stat.lpages; | ||
1831 | } | 1835 | } |
1832 | 1836 | ||
1833 | static void kvm_mmu_page_unlink_children(struct kvm *kvm, | 1837 | static void kvm_mmu_page_unlink_children(struct kvm *kvm, |
diff --git a/arch/x86/kvm/paging_tmpl.h b/arch/x86/kvm/paging_tmpl.h index 92994100638b..d8d3906649da 100644 --- a/arch/x86/kvm/paging_tmpl.h +++ b/arch/x86/kvm/paging_tmpl.h | |||
@@ -656,6 +656,18 @@ out_unlock: | |||
656 | return 0; | 656 | return 0; |
657 | } | 657 | } |
658 | 658 | ||
659 | static gpa_t FNAME(get_level1_sp_gpa)(struct kvm_mmu_page *sp) | ||
660 | { | ||
661 | int offset = 0; | ||
662 | |||
663 | WARN_ON(sp->role.level != 1); | ||
664 | |||
665 | if (PTTYPE == 32) | ||
666 | offset = sp->role.quadrant << PT64_LEVEL_BITS; | ||
667 | |||
668 | return gfn_to_gpa(sp->gfn) + offset * sizeof(pt_element_t); | ||
669 | } | ||
670 | |||
659 | static void FNAME(invlpg)(struct kvm_vcpu *vcpu, gva_t gva) | 671 | static void FNAME(invlpg)(struct kvm_vcpu *vcpu, gva_t gva) |
660 | { | 672 | { |
661 | struct kvm_shadow_walk_iterator iterator; | 673 | struct kvm_shadow_walk_iterator iterator; |
@@ -663,7 +675,6 @@ static void FNAME(invlpg)(struct kvm_vcpu *vcpu, gva_t gva) | |||
663 | gpa_t pte_gpa = -1; | 675 | gpa_t pte_gpa = -1; |
664 | int level; | 676 | int level; |
665 | u64 *sptep; | 677 | u64 *sptep; |
666 | int need_flush = 0; | ||
667 | 678 | ||
668 | vcpu_clear_mmio_info(vcpu, gva); | 679 | vcpu_clear_mmio_info(vcpu, gva); |
669 | 680 | ||
@@ -675,36 +686,20 @@ static void FNAME(invlpg)(struct kvm_vcpu *vcpu, gva_t gva) | |||
675 | 686 | ||
676 | sp = page_header(__pa(sptep)); | 687 | sp = page_header(__pa(sptep)); |
677 | if (is_last_spte(*sptep, level)) { | 688 | if (is_last_spte(*sptep, level)) { |
678 | int offset, shift; | ||
679 | |||
680 | if (!sp->unsync) | 689 | if (!sp->unsync) |
681 | break; | 690 | break; |
682 | 691 | ||
683 | shift = PAGE_SHIFT - | 692 | pte_gpa = FNAME(get_level1_sp_gpa)(sp); |
684 | (PT_LEVEL_BITS - PT64_LEVEL_BITS) * level; | ||
685 | offset = sp->role.quadrant << shift; | ||
686 | |||
687 | pte_gpa = (sp->gfn << PAGE_SHIFT) + offset; | ||
688 | pte_gpa += (sptep - sp->spt) * sizeof(pt_element_t); | 693 | pte_gpa += (sptep - sp->spt) * sizeof(pt_element_t); |
689 | 694 | ||
690 | if (is_shadow_present_pte(*sptep)) { | 695 | if (mmu_page_zap_pte(vcpu->kvm, sp, sptep)) |
691 | if (is_large_pte(*sptep)) | 696 | kvm_flush_remote_tlbs(vcpu->kvm); |
692 | --vcpu->kvm->stat.lpages; | ||
693 | drop_spte(vcpu->kvm, sptep); | ||
694 | need_flush = 1; | ||
695 | } else if (is_mmio_spte(*sptep)) | ||
696 | mmu_spte_clear_no_track(sptep); | ||
697 | |||
698 | break; | ||
699 | } | 697 | } |
700 | 698 | ||
701 | if (!is_shadow_present_pte(*sptep) || !sp->unsync_children) | 699 | if (!is_shadow_present_pte(*sptep) || !sp->unsync_children) |
702 | break; | 700 | break; |
703 | } | 701 | } |
704 | 702 | ||
705 | if (need_flush) | ||
706 | kvm_flush_remote_tlbs(vcpu->kvm); | ||
707 | |||
708 | atomic_inc(&vcpu->kvm->arch.invlpg_counter); | 703 | atomic_inc(&vcpu->kvm->arch.invlpg_counter); |
709 | 704 | ||
710 | spin_unlock(&vcpu->kvm->mmu_lock); | 705 | spin_unlock(&vcpu->kvm->mmu_lock); |
@@ -769,19 +764,14 @@ static gpa_t FNAME(gva_to_gpa_nested)(struct kvm_vcpu *vcpu, gva_t vaddr, | |||
769 | */ | 764 | */ |
770 | static int FNAME(sync_page)(struct kvm_vcpu *vcpu, struct kvm_mmu_page *sp) | 765 | static int FNAME(sync_page)(struct kvm_vcpu *vcpu, struct kvm_mmu_page *sp) |
771 | { | 766 | { |
772 | int i, offset, nr_present; | 767 | int i, nr_present = 0; |
773 | bool host_writable; | 768 | bool host_writable; |
774 | gpa_t first_pte_gpa; | 769 | gpa_t first_pte_gpa; |
775 | 770 | ||
776 | offset = nr_present = 0; | ||
777 | |||
778 | /* direct kvm_mmu_page can not be unsync. */ | 771 | /* direct kvm_mmu_page can not be unsync. */ |
779 | BUG_ON(sp->role.direct); | 772 | BUG_ON(sp->role.direct); |
780 | 773 | ||
781 | if (PTTYPE == 32) | 774 | first_pte_gpa = FNAME(get_level1_sp_gpa)(sp); |
782 | offset = sp->role.quadrant << PT64_LEVEL_BITS; | ||
783 | |||
784 | first_pte_gpa = gfn_to_gpa(sp->gfn) + offset * sizeof(pt_element_t); | ||
785 | 775 | ||
786 | for (i = 0; i < PT64_ENT_PER_PAGE; i++) { | 776 | for (i = 0; i < PT64_ENT_PER_PAGE; i++) { |
787 | unsigned pte_access; | 777 | unsigned pte_access; |