diff options
Diffstat (limited to 'arch/arm64/include')
-rw-r--r-- | arch/arm64/include/asm/futex.h | 2 | ||||
-rw-r--r-- | arch/arm64/include/asm/page.h | 1 | ||||
-rw-r--r-- | arch/arm64/include/asm/pgtable.h | 21 |
3 files changed, 13 insertions, 11 deletions
diff --git a/arch/arm64/include/asm/futex.h b/arch/arm64/include/asm/futex.h index 007a69fc4f40..5f3ab8c1db55 100644 --- a/arch/arm64/include/asm/futex.h +++ b/arch/arm64/include/asm/futex.h | |||
@@ -121,6 +121,7 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, | |||
121 | return -EFAULT; | 121 | return -EFAULT; |
122 | 122 | ||
123 | asm volatile("// futex_atomic_cmpxchg_inatomic\n" | 123 | asm volatile("// futex_atomic_cmpxchg_inatomic\n" |
124 | ALTERNATIVE("nop", SET_PSTATE_PAN(0), ARM64_HAS_PAN, CONFIG_ARM64_PAN) | ||
124 | " prfm pstl1strm, %2\n" | 125 | " prfm pstl1strm, %2\n" |
125 | "1: ldxr %w1, %2\n" | 126 | "1: ldxr %w1, %2\n" |
126 | " sub %w3, %w1, %w4\n" | 127 | " sub %w3, %w1, %w4\n" |
@@ -137,6 +138,7 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, | |||
137 | " .align 3\n" | 138 | " .align 3\n" |
138 | " .quad 1b, 4b, 2b, 4b\n" | 139 | " .quad 1b, 4b, 2b, 4b\n" |
139 | " .popsection\n" | 140 | " .popsection\n" |
141 | ALTERNATIVE("nop", SET_PSTATE_PAN(1), ARM64_HAS_PAN, CONFIG_ARM64_PAN) | ||
140 | : "+r" (ret), "=&r" (val), "+Q" (*uaddr), "=&r" (tmp) | 142 | : "+r" (ret), "=&r" (val), "+Q" (*uaddr), "=&r" (tmp) |
141 | : "r" (oldval), "r" (newval), "Ir" (-EFAULT) | 143 | : "r" (oldval), "r" (newval), "Ir" (-EFAULT) |
142 | : "memory"); | 144 | : "memory"); |
diff --git a/arch/arm64/include/asm/page.h b/arch/arm64/include/asm/page.h index 9b2f5a9d019d..ae615b9d9a55 100644 --- a/arch/arm64/include/asm/page.h +++ b/arch/arm64/include/asm/page.h | |||
@@ -39,6 +39,7 @@ | |||
39 | 39 | ||
40 | #ifndef __ASSEMBLY__ | 40 | #ifndef __ASSEMBLY__ |
41 | 41 | ||
42 | #include <linux/personality.h> /* for READ_IMPLIES_EXEC */ | ||
42 | #include <asm/pgtable-types.h> | 43 | #include <asm/pgtable-types.h> |
43 | 44 | ||
44 | extern void __cpu_clear_user_page(void *p, unsigned long user); | 45 | extern void __cpu_clear_user_page(void *p, unsigned long user); |
diff --git a/arch/arm64/include/asm/pgtable.h b/arch/arm64/include/asm/pgtable.h index 2d545d7aa80b..bf464de33f52 100644 --- a/arch/arm64/include/asm/pgtable.h +++ b/arch/arm64/include/asm/pgtable.h | |||
@@ -67,11 +67,11 @@ extern void __pgd_error(const char *file, int line, unsigned long val); | |||
67 | #define PROT_DEFAULT (PTE_TYPE_PAGE | PTE_AF | PTE_SHARED) | 67 | #define PROT_DEFAULT (PTE_TYPE_PAGE | PTE_AF | PTE_SHARED) |
68 | #define PROT_SECT_DEFAULT (PMD_TYPE_SECT | PMD_SECT_AF | PMD_SECT_S) | 68 | #define PROT_SECT_DEFAULT (PMD_TYPE_SECT | PMD_SECT_AF | PMD_SECT_S) |
69 | 69 | ||
70 | #define PROT_DEVICE_nGnRnE (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_ATTRINDX(MT_DEVICE_nGnRnE)) | 70 | #define PROT_DEVICE_nGnRnE (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_ATTRINDX(MT_DEVICE_nGnRnE)) |
71 | #define PROT_DEVICE_nGnRE (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_ATTRINDX(MT_DEVICE_nGnRE)) | 71 | #define PROT_DEVICE_nGnRE (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_ATTRINDX(MT_DEVICE_nGnRE)) |
72 | #define PROT_NORMAL_NC (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_ATTRINDX(MT_NORMAL_NC)) | 72 | #define PROT_NORMAL_NC (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_ATTRINDX(MT_NORMAL_NC)) |
73 | #define PROT_NORMAL_WT (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_ATTRINDX(MT_NORMAL_WT)) | 73 | #define PROT_NORMAL_WT (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_ATTRINDX(MT_NORMAL_WT)) |
74 | #define PROT_NORMAL (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_ATTRINDX(MT_NORMAL)) | 74 | #define PROT_NORMAL (PROT_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_ATTRINDX(MT_NORMAL)) |
75 | 75 | ||
76 | #define PROT_SECT_DEVICE_nGnRE (PROT_SECT_DEFAULT | PMD_SECT_PXN | PMD_SECT_UXN | PMD_ATTRINDX(MT_DEVICE_nGnRE)) | 76 | #define PROT_SECT_DEVICE_nGnRE (PROT_SECT_DEFAULT | PMD_SECT_PXN | PMD_SECT_UXN | PMD_ATTRINDX(MT_DEVICE_nGnRE)) |
77 | #define PROT_SECT_NORMAL (PROT_SECT_DEFAULT | PMD_SECT_PXN | PMD_SECT_UXN | PMD_ATTRINDX(MT_NORMAL)) | 77 | #define PROT_SECT_NORMAL (PROT_SECT_DEFAULT | PMD_SECT_PXN | PMD_SECT_UXN | PMD_ATTRINDX(MT_NORMAL)) |
@@ -81,7 +81,7 @@ extern void __pgd_error(const char *file, int line, unsigned long val); | |||
81 | 81 | ||
82 | #define PAGE_KERNEL __pgprot(_PAGE_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE) | 82 | #define PAGE_KERNEL __pgprot(_PAGE_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_WRITE) |
83 | #define PAGE_KERNEL_RO __pgprot(_PAGE_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_RDONLY) | 83 | #define PAGE_KERNEL_RO __pgprot(_PAGE_DEFAULT | PTE_PXN | PTE_UXN | PTE_DIRTY | PTE_RDONLY) |
84 | #define PAGE_KERNEL_ROX __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_RDONLY) | 84 | #define PAGE_KERNEL_ROX __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_RDONLY) |
85 | #define PAGE_KERNEL_EXEC __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_WRITE) | 85 | #define PAGE_KERNEL_EXEC __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_WRITE) |
86 | #define PAGE_KERNEL_EXEC_CONT __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_CONT) | 86 | #define PAGE_KERNEL_EXEC_CONT __pgprot(_PAGE_DEFAULT | PTE_UXN | PTE_DIRTY | PTE_WRITE | PTE_CONT) |
87 | 87 | ||
@@ -153,6 +153,7 @@ extern struct page *empty_zero_page; | |||
153 | #define pte_write(pte) (!!(pte_val(pte) & PTE_WRITE)) | 153 | #define pte_write(pte) (!!(pte_val(pte) & PTE_WRITE)) |
154 | #define pte_exec(pte) (!(pte_val(pte) & PTE_UXN)) | 154 | #define pte_exec(pte) (!(pte_val(pte) & PTE_UXN)) |
155 | #define pte_cont(pte) (!!(pte_val(pte) & PTE_CONT)) | 155 | #define pte_cont(pte) (!!(pte_val(pte) & PTE_CONT)) |
156 | #define pte_user(pte) (!!(pte_val(pte) & PTE_USER)) | ||
156 | 157 | ||
157 | #ifdef CONFIG_ARM64_HW_AFDBM | 158 | #ifdef CONFIG_ARM64_HW_AFDBM |
158 | #define pte_hw_dirty(pte) (pte_write(pte) && !(pte_val(pte) & PTE_RDONLY)) | 159 | #define pte_hw_dirty(pte) (pte_write(pte) && !(pte_val(pte) & PTE_RDONLY)) |
@@ -163,8 +164,6 @@ extern struct page *empty_zero_page; | |||
163 | #define pte_dirty(pte) (pte_sw_dirty(pte) || pte_hw_dirty(pte)) | 164 | #define pte_dirty(pte) (pte_sw_dirty(pte) || pte_hw_dirty(pte)) |
164 | 165 | ||
165 | #define pte_valid(pte) (!!(pte_val(pte) & PTE_VALID)) | 166 | #define pte_valid(pte) (!!(pte_val(pte) & PTE_VALID)) |
166 | #define pte_valid_user(pte) \ | ||
167 | ((pte_val(pte) & (PTE_VALID | PTE_USER)) == (PTE_VALID | PTE_USER)) | ||
168 | #define pte_valid_not_user(pte) \ | 167 | #define pte_valid_not_user(pte) \ |
169 | ((pte_val(pte) & (PTE_VALID | PTE_USER)) == PTE_VALID) | 168 | ((pte_val(pte) & (PTE_VALID | PTE_USER)) == PTE_VALID) |
170 | #define pte_valid_young(pte) \ | 169 | #define pte_valid_young(pte) \ |
@@ -278,13 +277,13 @@ extern void __sync_icache_dcache(pte_t pteval, unsigned long addr); | |||
278 | static inline void set_pte_at(struct mm_struct *mm, unsigned long addr, | 277 | static inline void set_pte_at(struct mm_struct *mm, unsigned long addr, |
279 | pte_t *ptep, pte_t pte) | 278 | pte_t *ptep, pte_t pte) |
280 | { | 279 | { |
281 | if (pte_valid_user(pte)) { | 280 | if (pte_valid(pte)) { |
282 | if (!pte_special(pte) && pte_exec(pte)) | ||
283 | __sync_icache_dcache(pte, addr); | ||
284 | if (pte_sw_dirty(pte) && pte_write(pte)) | 281 | if (pte_sw_dirty(pte) && pte_write(pte)) |
285 | pte_val(pte) &= ~PTE_RDONLY; | 282 | pte_val(pte) &= ~PTE_RDONLY; |
286 | else | 283 | else |
287 | pte_val(pte) |= PTE_RDONLY; | 284 | pte_val(pte) |= PTE_RDONLY; |
285 | if (pte_user(pte) && pte_exec(pte) && !pte_special(pte)) | ||
286 | __sync_icache_dcache(pte, addr); | ||
288 | } | 287 | } |
289 | 288 | ||
290 | /* | 289 | /* |