mm: thp: KVM: Explicitly check for THP when populating secondary MMU
authorSean Christopherson <sean.j.christopherson@intel.com>
Wed, 8 Jan 2020 20:24:36 +0000 (12:24 -0800)
committerPaolo Bonzini <pbonzini@redhat.com>
Mon, 27 Jan 2020 19:00:01 +0000 (20:00 +0100)
Add a helper, is_transparent_hugepage(), to explicitly check whether a
compound page is a THP and use it when populating KVM's secondary MMU.
The explicit check fixes a bug where a remapped compound page, e.g. for
an XDP Rx socket, is mapped into a KVM guest and is mistaken for a THP,
which results in KVM incorrectly creating a huge page in its secondary
MMU.

Fixes: 936a5fe6e6148 ("thp: kvm mmu transparent hugepage support")
Reported-by: syzbot+c9d1fb51ac9d0d10c39d@syzkaller.appspotmail.com
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: stable@vger.kernel.org
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
arch/x86/kvm/mmu/mmu.c
include/linux/huge_mm.h
include/linux/kvm_host.h
mm/huge_memory.c
virt/kvm/arm/mmu.c
virt/kvm/kvm_main.c

index db597f57cdc26f274d68e082f42ecb5dfc34692c..7eb21a22cc1351d27ad217d9d29b08e20664a482 100644 (file)
@@ -3344,7 +3344,7 @@ static void transparent_hugepage_adjust(struct kvm_vcpu *vcpu,
         */
        if (!is_error_noslot_pfn(pfn) && !kvm_is_reserved_pfn(pfn) &&
            !kvm_is_zone_device_pfn(pfn) && level == PT_PAGE_TABLE_LEVEL &&
-           PageTransCompoundMap(pfn_to_page(pfn))) {
+           kvm_is_transparent_hugepage(pfn)) {
                unsigned long mask;
 
                /*
@@ -5961,7 +5961,7 @@ restart:
                 */
                if (sp->role.direct && !kvm_is_reserved_pfn(pfn) &&
                    !kvm_is_zone_device_pfn(pfn) &&
-                   PageTransCompoundMap(pfn_to_page(pfn))) {
+                   kvm_is_transparent_hugepage(pfn)) {
                        pte_list_remove(rmap_head, sptep);
 
                        if (kvm_available_flush_tlb_with_range())
index 0b84e13e88e2d4ab7d1d5bcb1cbfa5c562ae131a..5aca3d1bdb32c959b5258e7c792a8e8cc1254bc1 100644 (file)
@@ -160,6 +160,7 @@ extern unsigned long thp_get_unmapped_area(struct file *filp,
 
 extern void prep_transhuge_page(struct page *page);
 extern void free_transhuge_page(struct page *page);
+bool is_transparent_hugepage(struct page *page);
 
 bool can_split_huge_page(struct page *page, int *pextra_pins);
 int split_huge_page_to_list(struct page *page, struct list_head *list);
@@ -308,6 +309,11 @@ static inline bool transhuge_vma_suitable(struct vm_area_struct *vma,
 
 static inline void prep_transhuge_page(struct page *page) {}
 
+static inline bool is_transparent_hugepage(struct page *page)
+{
+       return false;
+}
+
 #define transparent_hugepage_flags 0UL
 
 #define thp_get_unmapped_area  NULL
index 48e139c293c23192f99c8f7db4c3ff1314d105f1..46fdb753367875753a36762ef446c988a24c1862 100644 (file)
@@ -976,6 +976,7 @@ int kvm_cpu_has_pending_timer(struct kvm_vcpu *vcpu);
 
 bool kvm_is_reserved_pfn(kvm_pfn_t pfn);
 bool kvm_is_zone_device_pfn(kvm_pfn_t pfn);
+bool kvm_is_transparent_hugepage(kvm_pfn_t pfn);
 
 struct kvm_irq_ack_notifier {
        struct hlist_node link;
index 41a0fbddc96bb27a0717e4033bbbaa511aac4899..9b3ee79d0edf271673d79f0a5ee40056a20312fd 100644 (file)
@@ -527,6 +527,17 @@ void prep_transhuge_page(struct page *page)
        set_compound_page_dtor(page, TRANSHUGE_PAGE_DTOR);
 }
 
+bool is_transparent_hugepage(struct page *page)
+{
+       if (!PageCompound(page))
+               return 0;
+
+       page = compound_head(page);
+       return is_huge_zero_page(page) ||
+              page[1].compound_dtor == TRANSHUGE_PAGE_DTOR;
+}
+EXPORT_SYMBOL_GPL(is_transparent_hugepage);
+
 static unsigned long __thp_get_unmapped_area(struct file *filp, unsigned long len,
                loff_t off, unsigned long flags, unsigned long size)
 {
index 0b32a904a1bb3a5636363803eceb34a45766cd25..dc8254bf30ea0a24f9b716f2990bf3a6fe7289ee 100644 (file)
@@ -1377,14 +1377,8 @@ static bool transparent_hugepage_adjust(kvm_pfn_t *pfnp, phys_addr_t *ipap)
 {
        kvm_pfn_t pfn = *pfnp;
        gfn_t gfn = *ipap >> PAGE_SHIFT;
-       struct page *page = pfn_to_page(pfn);
 
-       /*
-        * PageTransCompoundMap() returns true for THP and
-        * hugetlbfs. Make sure the adjustment is done only for THP
-        * pages.
-        */
-       if (!PageHuge(page) && PageTransCompoundMap(page)) {
+       if (kvm_is_transparent_hugepage(pfn)) {
                unsigned long mask;
                /*
                 * The address we faulted on is backed by a transparent huge
index ffec9f427b55801be5d16d2bf21359b114f8424f..64e9e9d65ed4e95fe6ef1dd81d19e17832fe4f64 100644 (file)
@@ -191,6 +191,16 @@ bool kvm_is_reserved_pfn(kvm_pfn_t pfn)
        return true;
 }
 
+bool kvm_is_transparent_hugepage(kvm_pfn_t pfn)
+{
+       struct page *page = pfn_to_page(pfn);
+
+       if (!PageTransCompoundMap(page))
+               return false;
+
+       return is_transparent_hugepage(compound_head(page));
+}
+
 /*
  * Switches to specified vcpu, until a matching vcpu_put()
  */