]> git.proxmox.com Git - mirror_ubuntu-zesty-kernel.git/commitdiff
arm64: hugetlb: Fix huge_pte_offset to return poisoned page table entries
authorPunit Agrawal <punit.agrawal@arm.com>
Thu, 8 Jun 2017 17:25:26 +0000 (18:25 +0100)
committerKleber Sacilotto de Souza <kleber.souza@canonical.com>
Wed, 9 Aug 2017 14:23:29 +0000 (16:23 +0200)
When memory failure is enabled, a poisoned hugepage pte is marked as a
swap entry. huge_pte_offset() does not return the poisoned page table
entries when it encounters PUD/PMD hugepages.

This behaviour of huge_pte_offset() leads to error such as below when
munmap is called on poisoned hugepages.

[  344.165544] mm/pgtable-generic.c:33: bad pmd 000000083af00074.

Fix huge_pte_offset() to return the poisoned pte which is then
appropriately handled by the generic layer code.

BugLink: http://launchpad.net/bugs/1696852
Signed-off-by: Punit Agrawal <punit.agrawal@arm.com>
Acked-by: Steve Capper <steve.capper@arm.com>
Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Cc: David Woods <dwoods@mellanox.com>
Tested-by: Manoj Iyer <manoj.iyer@canonical.com>
Signed-off-by: Will Deacon <will.deacon@arm.com>
(cherry picked from commit f02ab08afbe76ee7b0b2a34a9970e7dd200d8b01)
Signed-off-by: Manoj Iyer <manoj.iyer@canonical.com>
Acked-by: Stefan Bader <stefan.bader@canonical.com>
Acked-by: Seth Forshee <seth.forshee@canonical.com>
Signed-off-by: Thadeu Lima de Souza Cascardo <cascardo@canonical.com>
arch/arm64/include/asm/pgtable.h
arch/arm64/mm/hugetlbpage.c

index 61e214015b38ada78f7bf89af962fa556c214560..08aee16f39066a831b810cf71cbff240e56d7ee5 100644 (file)
@@ -431,7 +431,7 @@ static inline phys_addr_t pmd_page_paddr(pmd_t pmd)
 
 #define pud_none(pud)          (!pud_val(pud))
 #define pud_bad(pud)           (!(pud_val(pud) & PUD_TABLE_BIT))
-#define pud_present(pud)       (pud_val(pud))
+#define pud_present(pud)       pte_present(pud_pte(pud))
 
 static inline void set_pud(pud_t *pudp, pud_t pud)
 {
index e25584d723960e73fb8eec8d1a5f48fa57197582..cfe73020d7fe0bffa8ee52beaabf06830f804dba 100644 (file)
@@ -136,36 +136,27 @@ pte_t *huge_pte_offset(struct mm_struct *mm, unsigned long addr)
 {
        pgd_t *pgd;
        pud_t *pud;
-       pmd_t *pmd = NULL;
-       pte_t *pte = NULL;
+       pmd_t *pmd;
 
        pgd = pgd_offset(mm, addr);
        pr_debug("%s: addr:0x%lx pgd:%p\n", __func__, addr, pgd);
        if (!pgd_present(*pgd))
                return NULL;
+
        pud = pud_offset(pgd, addr);
-       if (!pud_present(*pud))
+       if (pud_none(*pud))
                return NULL;
-
-       if (pud_huge(*pud))
+       /* swap or huge page */
+       if (!pud_present(*pud) || pud_huge(*pud))
                return (pte_t *)pud;
+       /* table; check the next level */
+
        pmd = pmd_offset(pud, addr);
-       if (!pmd_present(*pmd))
+       if (pmd_none(*pmd))
                return NULL;
-
-       if (pte_cont(pmd_pte(*pmd))) {
-               pmd = pmd_offset(
-                       pud, (addr & CONT_PMD_MASK));
-               return (pte_t *)pmd;
-       }
-       if (pmd_huge(*pmd))
+       if (!pmd_present(*pmd) || pmd_huge(*pmd))
                return (pte_t *)pmd;
-       pte = pte_offset_kernel(pmd, addr);
-       if (pte_present(*pte) && pte_cont(*pte)) {
-               pte = pte_offset_kernel(
-                       pmd, (addr & CONT_PTE_MASK));
-               return pte;
-       }
+
        return NULL;
 }