|
@@ -25,8 +25,8 @@
|
|
* Software defined PTE bits definition.
|
|
* Software defined PTE bits definition.
|
|
*/
|
|
*/
|
|
#define PTE_VALID (_AT(pteval_t, 1) << 0)
|
|
#define PTE_VALID (_AT(pteval_t, 1) << 0)
|
|
-#define PTE_PROT_NONE (_AT(pteval_t, 1) << 1) /* only when !PTE_VALID */
|
|
|
|
-#define PTE_FILE (_AT(pteval_t, 1) << 2) /* only when !pte_present() */
|
|
|
|
|
|
+#define PTE_PROT_NONE (_AT(pteval_t, 1) << 2) /* only when !PTE_VALID */
|
|
|
|
+#define PTE_FILE (_AT(pteval_t, 1) << 3) /* only when !pte_present() */
|
|
#define PTE_DIRTY (_AT(pteval_t, 1) << 55)
|
|
#define PTE_DIRTY (_AT(pteval_t, 1) << 55)
|
|
#define PTE_SPECIAL (_AT(pteval_t, 1) << 56)
|
|
#define PTE_SPECIAL (_AT(pteval_t, 1) << 56)
|
|
|
|
|
|
@@ -66,7 +66,7 @@ extern pgprot_t pgprot_default;
|
|
|
|
|
|
#define _MOD_PROT(p, b) __pgprot_modify(p, 0, b)
|
|
#define _MOD_PROT(p, b) __pgprot_modify(p, 0, b)
|
|
|
|
|
|
-#define PAGE_NONE __pgprot_modify(pgprot_default, PTE_TYPE_MASK, PTE_PROT_NONE)
|
|
|
|
|
|
+#define PAGE_NONE __pgprot_modify(pgprot_default, PTE_TYPE_MASK, PTE_PROT_NONE | PTE_RDONLY | PTE_PXN | PTE_UXN)
|
|
#define PAGE_SHARED _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN)
|
|
#define PAGE_SHARED _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN)
|
|
#define PAGE_SHARED_EXEC _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN)
|
|
#define PAGE_SHARED_EXEC _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN)
|
|
#define PAGE_COPY _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
#define PAGE_COPY _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
@@ -82,7 +82,7 @@ extern pgprot_t pgprot_default;
|
|
#define PAGE_S2 __pgprot_modify(pgprot_default, PTE_S2_MEMATTR_MASK, PTE_S2_MEMATTR(MT_S2_NORMAL) | PTE_S2_RDONLY)
|
|
#define PAGE_S2 __pgprot_modify(pgprot_default, PTE_S2_MEMATTR_MASK, PTE_S2_MEMATTR(MT_S2_NORMAL) | PTE_S2_RDONLY)
|
|
#define PAGE_S2_DEVICE __pgprot(PROT_DEFAULT | PTE_S2_MEMATTR(MT_S2_DEVICE_nGnRE) | PTE_S2_RDWR | PTE_UXN)
|
|
#define PAGE_S2_DEVICE __pgprot(PROT_DEFAULT | PTE_S2_MEMATTR(MT_S2_DEVICE_nGnRE) | PTE_S2_RDWR | PTE_UXN)
|
|
|
|
|
|
-#define __PAGE_NONE __pgprot(((_PAGE_DEFAULT) & ~PTE_TYPE_MASK) | PTE_PROT_NONE)
|
|
|
|
|
|
+#define __PAGE_NONE __pgprot(((_PAGE_DEFAULT) & ~PTE_TYPE_MASK) | PTE_PROT_NONE | PTE_RDONLY | PTE_PXN | PTE_UXN)
|
|
#define __PAGE_SHARED __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN)
|
|
#define __PAGE_SHARED __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN)
|
|
#define __PAGE_SHARED_EXEC __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN)
|
|
#define __PAGE_SHARED_EXEC __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN)
|
|
#define __PAGE_COPY __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
#define __PAGE_COPY __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
@@ -179,11 +179,75 @@ static inline void set_pte_at(struct mm_struct *mm, unsigned long addr,
|
|
/*
|
|
/*
|
|
* Huge pte definitions.
|
|
* Huge pte definitions.
|
|
*/
|
|
*/
|
|
-#define pte_huge(pte) ((pte_val(pte) & PTE_TYPE_MASK) == PTE_TYPE_HUGEPAGE)
|
|
|
|
-#define pte_mkhuge(pte) (__pte((pte_val(pte) & ~PTE_TYPE_MASK) | PTE_TYPE_HUGEPAGE))
|
|
|
|
|
|
+#define pte_huge(pte) (!(pte_val(pte) & PTE_TABLE_BIT))
|
|
|
|
+#define pte_mkhuge(pte) (__pte(pte_val(pte) & ~PTE_TABLE_BIT))
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
+ * Hugetlb definitions.
|
|
|
|
+ */
|
|
|
|
+#define HUGE_MAX_HSTATE 2
|
|
|
|
+#define HPAGE_SHIFT PMD_SHIFT
|
|
|
|
+#define HPAGE_SIZE (_AC(1, UL) << HPAGE_SHIFT)
|
|
|
|
+#define HPAGE_MASK (~(HPAGE_SIZE - 1))
|
|
|
|
+#define HUGETLB_PAGE_ORDER (HPAGE_SHIFT - PAGE_SHIFT)
|
|
|
|
|
|
#define __HAVE_ARCH_PTE_SPECIAL
|
|
#define __HAVE_ARCH_PTE_SPECIAL
|
|
|
|
|
|
|
|
+/*
|
|
|
|
+ * Software PMD bits for THP
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+#define PMD_SECT_DIRTY (_AT(pmdval_t, 1) << 55)
|
|
|
|
+#define PMD_SECT_SPLITTING (_AT(pmdval_t, 1) << 57)
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
+ * THP definitions.
|
|
|
|
+ */
|
|
|
|
+#define pmd_young(pmd) (pmd_val(pmd) & PMD_SECT_AF)
|
|
|
|
+
|
|
|
|
+#define __HAVE_ARCH_PMD_WRITE
|
|
|
|
+#define pmd_write(pmd) (!(pmd_val(pmd) & PMD_SECT_RDONLY))
|
|
|
|
+
|
|
|
|
+#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
|
|
|
+#define pmd_trans_huge(pmd) (pmd_val(pmd) && !(pmd_val(pmd) & PMD_TABLE_BIT))
|
|
|
|
+#define pmd_trans_splitting(pmd) (pmd_val(pmd) & PMD_SECT_SPLITTING)
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
+#define PMD_BIT_FUNC(fn,op) \
|
|
|
|
+static inline pmd_t pmd_##fn(pmd_t pmd) { pmd_val(pmd) op; return pmd; }
|
|
|
|
+
|
|
|
|
+PMD_BIT_FUNC(wrprotect, |= PMD_SECT_RDONLY);
|
|
|
|
+PMD_BIT_FUNC(mkold, &= ~PMD_SECT_AF);
|
|
|
|
+PMD_BIT_FUNC(mksplitting, |= PMD_SECT_SPLITTING);
|
|
|
|
+PMD_BIT_FUNC(mkwrite, &= ~PMD_SECT_RDONLY);
|
|
|
|
+PMD_BIT_FUNC(mkdirty, |= PMD_SECT_DIRTY);
|
|
|
|
+PMD_BIT_FUNC(mkyoung, |= PMD_SECT_AF);
|
|
|
|
+PMD_BIT_FUNC(mknotpresent, &= ~PMD_TYPE_MASK);
|
|
|
|
+
|
|
|
|
+#define pmd_mkhuge(pmd) (__pmd(pmd_val(pmd) & ~PMD_TABLE_BIT))
|
|
|
|
+
|
|
|
|
+#define pmd_pfn(pmd) (((pmd_val(pmd) & PMD_MASK) & PHYS_MASK) >> PAGE_SHIFT)
|
|
|
|
+#define pfn_pmd(pfn,prot) (__pmd(((phys_addr_t)(pfn) << PAGE_SHIFT) | pgprot_val(prot)))
|
|
|
|
+#define mk_pmd(page,prot) pfn_pmd(page_to_pfn(page),prot)
|
|
|
|
+
|
|
|
|
+#define pmd_page(pmd) pfn_to_page(__phys_to_pfn(pmd_val(pmd) & PHYS_MASK))
|
|
|
|
+
|
|
|
|
+static inline pmd_t pmd_modify(pmd_t pmd, pgprot_t newprot)
|
|
|
|
+{
|
|
|
|
+ const pmdval_t mask = PMD_SECT_USER | PMD_SECT_PXN | PMD_SECT_UXN |
|
|
|
|
+ PMD_SECT_RDONLY | PMD_SECT_PROT_NONE |
|
|
|
|
+ PMD_SECT_VALID;
|
|
|
|
+ pmd_val(pmd) = (pmd_val(pmd) & ~mask) | (pgprot_val(newprot) & mask);
|
|
|
|
+ return pmd;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+#define set_pmd_at(mm, addr, pmdp, pmd) set_pmd(pmdp, pmd)
|
|
|
|
+
|
|
|
|
+static inline int has_transparent_hugepage(void)
|
|
|
|
+{
|
|
|
|
+ return 1;
|
|
|
|
+}
|
|
|
|
+
|
|
/*
|
|
/*
|
|
* Mark the prot value as uncacheable and unbufferable.
|
|
* Mark the prot value as uncacheable and unbufferable.
|
|
*/
|
|
*/
|
|
@@ -293,12 +357,12 @@ extern pgd_t idmap_pg_dir[PTRS_PER_PGD];
|
|
|
|
|
|
/*
|
|
/*
|
|
* Encode and decode a swap entry:
|
|
* Encode and decode a swap entry:
|
|
- * bits 0-1: present (must be zero)
|
|
|
|
- * bit 2: PTE_FILE
|
|
|
|
- * bits 3-8: swap type
|
|
|
|
|
|
+ * bits 0, 2: present (must both be zero)
|
|
|
|
+ * bit 3: PTE_FILE
|
|
|
|
+ * bits 4-8: swap type
|
|
* bits 9-63: swap offset
|
|
* bits 9-63: swap offset
|
|
*/
|
|
*/
|
|
-#define __SWP_TYPE_SHIFT 3
|
|
|
|
|
|
+#define __SWP_TYPE_SHIFT 4
|
|
#define __SWP_TYPE_BITS 6
|
|
#define __SWP_TYPE_BITS 6
|
|
#define __SWP_TYPE_MASK ((1 << __SWP_TYPE_BITS) - 1)
|
|
#define __SWP_TYPE_MASK ((1 << __SWP_TYPE_BITS) - 1)
|
|
#define __SWP_OFFSET_SHIFT (__SWP_TYPE_BITS + __SWP_TYPE_SHIFT)
|
|
#define __SWP_OFFSET_SHIFT (__SWP_TYPE_BITS + __SWP_TYPE_SHIFT)
|
|
@@ -318,15 +382,15 @@ extern pgd_t idmap_pg_dir[PTRS_PER_PGD];
|
|
|
|
|
|
/*
|
|
/*
|
|
* Encode and decode a file entry:
|
|
* Encode and decode a file entry:
|
|
- * bits 0-1: present (must be zero)
|
|
|
|
- * bit 2: PTE_FILE
|
|
|
|
- * bits 3-63: file offset / PAGE_SIZE
|
|
|
|
|
|
+ * bits 0, 2: present (must both be zero)
|
|
|
|
+ * bit 3: PTE_FILE
|
|
|
|
+ * bits 4-63: file offset / PAGE_SIZE
|
|
*/
|
|
*/
|
|
#define pte_file(pte) (pte_val(pte) & PTE_FILE)
|
|
#define pte_file(pte) (pte_val(pte) & PTE_FILE)
|
|
-#define pte_to_pgoff(x) (pte_val(x) >> 3)
|
|
|
|
-#define pgoff_to_pte(x) __pte(((x) << 3) | PTE_FILE)
|
|
|
|
|
|
+#define pte_to_pgoff(x) (pte_val(x) >> 4)
|
|
|
|
+#define pgoff_to_pte(x) __pte(((x) << 4) | PTE_FILE)
|
|
|
|
|
|
-#define PTE_FILE_MAX_BITS 61
|
|
|
|
|
|
+#define PTE_FILE_MAX_BITS 60
|
|
|
|
|
|
extern int kern_addr_valid(unsigned long addr);
|
|
extern int kern_addr_valid(unsigned long addr);
|
|
|
|
|