]> git.hungrycats.org Git - linux/commitdiff
powerpc/thp: Handle combo pages in invalidate
authorAneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Wed, 13 Aug 2014 07:02:00 +0000 (12:32 +0530)
committerJiri Slaby <jslaby@suse.cz>
Wed, 17 Sep 2014 14:55:04 +0000 (16:55 +0200)
commit fc0479557572375100ef16c71170b29a98e0d69a upstream.

If we changed base page size of the segment, either via sub_page_protect
or via remap_4k_pfn, we do a demote_segment which doesn't flush the hash
table entries. We do a lazy hash page table flush for all mapped pages
in the demoted segment. This happens when we handle hash page fault for
these pages.

We use _PAGE_COMBO bit along with _PAGE_HASHPTE to indicate whether a
pte is backed by 4K hash pte. If we find _PAGE_COMBO not set on the pte,
that implies that we could possibly have older 64K hash pte entries in
the hash page table and we need to invalidate those entries.

Use _PAGE_COMBO to determine the page size with which we should
invalidate the hash table entries on unmap.

Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Jiri Slaby <jslaby@suse.cz>
arch/powerpc/include/asm/pgtable-ppc64.h
arch/powerpc/mm/pgtable_64.c
arch/powerpc/mm/tlb_hash64.c

index 46db09414a1063415fccc9ef74b9f4b664de02f3..832a39d042d438e8092ad51ad642f1d1449f3b1a 100644 (file)
@@ -409,7 +409,7 @@ static inline char *get_hpte_slot_array(pmd_t *pmdp)
 }
 
 extern void hpte_do_hugepage_flush(struct mm_struct *mm, unsigned long addr,
-                                  pmd_t *pmdp);
+                                  pmd_t *pmdp, unsigned long old_pmd);
 #ifdef CONFIG_TRANSPARENT_HUGEPAGE
 extern pmd_t pfn_pmd(unsigned long pfn, pgprot_t pgprot);
 extern pmd_t mk_pmd(struct page *page, pgprot_t pgprot);
index 48bda6700404e8d54a3fa2326e4f7f10479e13c9..c9379a2d600611a66fd09799ab54ea9eabc3f00f 100644 (file)
@@ -524,7 +524,7 @@ unsigned long pmd_hugepage_update(struct mm_struct *mm, unsigned long addr,
        *pmdp = __pmd(old & ~clr);
 #endif
        if (old & _PAGE_HASHPTE)
-               hpte_do_hugepage_flush(mm, addr, pmdp);
+               hpte_do_hugepage_flush(mm, addr, pmdp, old);
        return old;
 }
 
@@ -631,7 +631,7 @@ void pmdp_splitting_flush(struct vm_area_struct *vma,
        if (!(old & _PAGE_SPLITTING)) {
                /* We need to flush the hpte */
                if (old & _PAGE_HASHPTE)
-                       hpte_do_hugepage_flush(vma->vm_mm, address, pmdp);
+                       hpte_do_hugepage_flush(vma->vm_mm, address, pmdp, old);
        }
 }
 
@@ -704,7 +704,7 @@ void pmdp_invalidate(struct vm_area_struct *vma, unsigned long address,
  * neesd to be flushed.
  */
 void hpte_do_hugepage_flush(struct mm_struct *mm, unsigned long addr,
-                           pmd_t *pmdp)
+                           pmd_t *pmdp, unsigned long old_pmd)
 {
        int ssize, i;
        unsigned long s_addr;
@@ -727,7 +727,15 @@ void hpte_do_hugepage_flush(struct mm_struct *mm, unsigned long addr,
                return;
 
        /* get the base page size,vsid and segment size */
+#ifdef CONFIG_DEBUG_VM
        psize = get_slice_psize(mm, s_addr);
+       BUG_ON(psize == MMU_PAGE_16M);
+#endif
+       if (old_pmd & _PAGE_COMBO)
+               psize = MMU_PAGE_4K;
+       else
+               psize = MMU_PAGE_64K;
+
        if (!is_kernel_addr(s_addr)) {
                ssize = user_segment_size(s_addr);
                vsid = get_vsid(mm->context.id, s_addr, ssize);
index 36e44b4260ebd25ea5fa34660d7cbd81f71bf760..c66e445d9890a068c6eeaaeabf96e139ea71259b 100644 (file)
@@ -217,7 +217,7 @@ void __flush_hash_table_range(struct mm_struct *mm, unsigned long start,
                if (!(pte & _PAGE_HASHPTE))
                        continue;
                if (unlikely(hugepage_shift && pmd_trans_huge(*(pmd_t *)pte)))
-                       hpte_do_hugepage_flush(mm, start, (pmd_t *)pte);
+                       hpte_do_hugepage_flush(mm, start, (pmd_t *)ptep, pte);
                else
                        hpte_need_flush(mm, start, ptep, pte, 0);
        }