|
@@ -55,3 +55,158 @@ extern void __pgd_error(const char *file, int line, unsigned long val);
|
|
|
* The pgprot_* and protection_map entries will be fixed up at runtime to
|
|
|
* include the cachable and bufferable bits based on memory policy, as well as
|
|
|
* any architecture dependent bits like global/ASID and SMP shared mapping
|
|
|
+ * bits.
|
|
|
+ */
|
|
|
+#define _PAGE_DEFAULT PTE_TYPE_PAGE | PTE_AF
|
|
|
+
|
|
|
+extern pgprot_t pgprot_default;
|
|
|
+
|
|
|
+#define __pgprot_modify(prot,mask,bits) \
|
|
|
+ __pgprot((pgprot_val(prot) & ~(mask)) | (bits))
|
|
|
+
|
|
|
+#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_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_COPY _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
|
+#define PAGE_COPY_EXEC _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_RDONLY)
|
|
|
+#define PAGE_READONLY _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
|
+#define PAGE_READONLY_EXEC _MOD_PROT(pgprot_default, PTE_USER | PTE_NG | PTE_PXN | PTE_RDONLY)
|
|
|
+#define PAGE_KERNEL _MOD_PROT(pgprot_default, PTE_PXN | PTE_UXN | PTE_DIRTY)
|
|
|
+#define PAGE_KERNEL_EXEC _MOD_PROT(pgprot_default, PTE_UXN | PTE_DIRTY)
|
|
|
+
|
|
|
+#define __PAGE_NONE __pgprot(((_PAGE_DEFAULT) & ~PTE_TYPE_MASK) | PTE_PROT_NONE)
|
|
|
+#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_COPY __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
|
+#define __PAGE_COPY_EXEC __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_RDONLY)
|
|
|
+#define __PAGE_READONLY __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_UXN | PTE_RDONLY)
|
|
|
+#define __PAGE_READONLY_EXEC __pgprot(_PAGE_DEFAULT | PTE_USER | PTE_NG | PTE_PXN | PTE_RDONLY)
|
|
|
+
|
|
|
+#endif /* __ASSEMBLY__ */
|
|
|
+
|
|
|
+#define __P000 __PAGE_NONE
|
|
|
+#define __P001 __PAGE_READONLY
|
|
|
+#define __P010 __PAGE_COPY
|
|
|
+#define __P011 __PAGE_COPY
|
|
|
+#define __P100 __PAGE_READONLY_EXEC
|
|
|
+#define __P101 __PAGE_READONLY_EXEC
|
|
|
+#define __P110 __PAGE_COPY_EXEC
|
|
|
+#define __P111 __PAGE_COPY_EXEC
|
|
|
+
|
|
|
+#define __S000 __PAGE_NONE
|
|
|
+#define __S001 __PAGE_READONLY
|
|
|
+#define __S010 __PAGE_SHARED
|
|
|
+#define __S011 __PAGE_SHARED
|
|
|
+#define __S100 __PAGE_READONLY_EXEC
|
|
|
+#define __S101 __PAGE_READONLY_EXEC
|
|
|
+#define __S110 __PAGE_SHARED_EXEC
|
|
|
+#define __S111 __PAGE_SHARED_EXEC
|
|
|
+
|
|
|
+#ifndef __ASSEMBLY__
|
|
|
+/*
|
|
|
+ * ZERO_PAGE is a global shared page that is always zero: used
|
|
|
+ * for zero-mapped memory areas etc..
|
|
|
+ */
|
|
|
+extern struct page *empty_zero_page;
|
|
|
+#define ZERO_PAGE(vaddr) (empty_zero_page)
|
|
|
+
|
|
|
+#define pte_pfn(pte) ((pte_val(pte) & PHYS_MASK) >> PAGE_SHIFT)
|
|
|
+
|
|
|
+#define pfn_pte(pfn,prot) (__pte(((phys_addr_t)(pfn) << PAGE_SHIFT) | pgprot_val(prot)))
|
|
|
+
|
|
|
+#define pte_none(pte) (!pte_val(pte))
|
|
|
+#define pte_clear(mm,addr,ptep) set_pte(ptep, __pte(0))
|
|
|
+#define pte_page(pte) (pfn_to_page(pte_pfn(pte)))
|
|
|
+#define pte_offset_kernel(dir,addr) (pmd_page_vaddr(*(dir)) + __pte_index(addr))
|
|
|
+
|
|
|
+#define pte_offset_map(dir,addr) pte_offset_kernel((dir), (addr))
|
|
|
+#define pte_offset_map_nested(dir,addr) pte_offset_kernel((dir), (addr))
|
|
|
+#define pte_unmap(pte) do { } while (0)
|
|
|
+#define pte_unmap_nested(pte) do { } while (0)
|
|
|
+
|
|
|
+/*
|
|
|
+ * The following only work if pte_present(). Undefined behaviour otherwise.
|
|
|
+ */
|
|
|
+#define pte_present(pte) (pte_val(pte) & (PTE_VALID | PTE_PROT_NONE))
|
|
|
+#define pte_dirty(pte) (pte_val(pte) & PTE_DIRTY)
|
|
|
+#define pte_young(pte) (pte_val(pte) & PTE_AF)
|
|
|
+#define pte_special(pte) (pte_val(pte) & PTE_SPECIAL)
|
|
|
+#define pte_write(pte) (!(pte_val(pte) & PTE_RDONLY))
|
|
|
+#define pte_exec(pte) (!(pte_val(pte) & PTE_UXN))
|
|
|
+
|
|
|
+#define pte_valid_user(pte) \
|
|
|
+ ((pte_val(pte) & (PTE_VALID | PTE_USER)) == (PTE_VALID | PTE_USER))
|
|
|
+
|
|
|
+#define PTE_BIT_FUNC(fn,op) \
|
|
|
+static inline pte_t pte_##fn(pte_t pte) { pte_val(pte) op; return pte; }
|
|
|
+
|
|
|
+PTE_BIT_FUNC(wrprotect, |= PTE_RDONLY);
|
|
|
+PTE_BIT_FUNC(mkwrite, &= ~PTE_RDONLY);
|
|
|
+PTE_BIT_FUNC(mkclean, &= ~PTE_DIRTY);
|
|
|
+PTE_BIT_FUNC(mkdirty, |= PTE_DIRTY);
|
|
|
+PTE_BIT_FUNC(mkold, &= ~PTE_AF);
|
|
|
+PTE_BIT_FUNC(mkyoung, |= PTE_AF);
|
|
|
+PTE_BIT_FUNC(mkspecial, |= PTE_SPECIAL);
|
|
|
+
|
|
|
+static inline void set_pte(pte_t *ptep, pte_t pte)
|
|
|
+{
|
|
|
+ *ptep = pte;
|
|
|
+}
|
|
|
+
|
|
|
+extern void __sync_icache_dcache(pte_t pteval, unsigned long addr);
|
|
|
+
|
|
|
+static inline void set_pte_at(struct mm_struct *mm, unsigned long addr,
|
|
|
+ pte_t *ptep, pte_t pte)
|
|
|
+{
|
|
|
+ if (pte_valid_user(pte)) {
|
|
|
+ if (pte_exec(pte))
|
|
|
+ __sync_icache_dcache(pte, addr);
|
|
|
+ if (!pte_dirty(pte))
|
|
|
+ pte = pte_wrprotect(pte);
|
|
|
+ }
|
|
|
+
|
|
|
+ set_pte(ptep, pte);
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * 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 __HAVE_ARCH_PTE_SPECIAL
|
|
|
+
|
|
|
+/*
|
|
|
+ * Mark the prot value as uncacheable and unbufferable.
|
|
|
+ */
|
|
|
+#define pgprot_noncached(prot) \
|
|
|
+ __pgprot_modify(prot, PTE_ATTRINDX_MASK, PTE_ATTRINDX(MT_DEVICE_nGnRnE))
|
|
|
+#define pgprot_writecombine(prot) \
|
|
|
+ __pgprot_modify(prot, PTE_ATTRINDX_MASK, PTE_ATTRINDX(MT_DEVICE_GRE))
|
|
|
+#define pgprot_dmacoherent(prot) \
|
|
|
+ __pgprot_modify(prot, PTE_ATTRINDX_MASK, PTE_ATTRINDX(MT_NORMAL_NC))
|
|
|
+#define __HAVE_PHYS_MEM_ACCESS_PROT
|
|
|
+struct file;
|
|
|
+extern pgprot_t phys_mem_access_prot(struct file *file, unsigned long pfn,
|
|
|
+ unsigned long size, pgprot_t vma_prot);
|
|
|
+
|
|
|
+#define pmd_none(pmd) (!pmd_val(pmd))
|
|
|
+#define pmd_present(pmd) (pmd_val(pmd))
|
|
|
+
|
|
|
+#define pmd_bad(pmd) (!(pmd_val(pmd) & 2))
|
|
|
+
|
|
|
+static inline void set_pmd(pmd_t *pmdp, pmd_t pmd)
|
|
|
+{
|
|
|
+ *pmdp = pmd;
|
|
|
+ dsb();
|
|
|
+}
|
|
|
+
|
|
|
+static inline void pmd_clear(pmd_t *pmdp)
|
|
|
+{
|
|
|
+ set_pmd(pmdp, __pmd(0));
|
|
|
+}
|
|
|
+
|
|
|
+static inline pte_t *pmd_page_vaddr(pmd_t pmd)
|
|
|
+{
|