|
@@ -187,6 +187,61 @@ static inline void set_pte_at(struct mm_struct *mm, unsigned long addr,
|
|
|
|
|
|
#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.
|
|
|
*/
|