aboutsummaryrefslogtreecommitdiffstats
path: root/arch/s390/lib
diff options
context:
space:
mode:
Diffstat (limited to 'arch/s390/lib')
-rw-r--r--arch/s390/lib/uaccess_pt.c85
1 files changed, 22 insertions, 63 deletions
diff --git a/arch/s390/lib/uaccess_pt.c b/arch/s390/lib/uaccess_pt.c
index b159a9d65680..dc37ea827f4e 100644
--- a/arch/s390/lib/uaccess_pt.c
+++ b/arch/s390/lib/uaccess_pt.c
@@ -15,6 +15,22 @@
15#include <asm/futex.h> 15#include <asm/futex.h>
16#include "uaccess.h" 16#include "uaccess.h"
17 17
18static inline pte_t *follow_table(struct mm_struct *mm, unsigned long addr)
19{
20 pgd_t *pgd;
21 pmd_t *pmd;
22
23 pgd = pgd_offset(mm, addr);
24 if (pgd_none(*pgd) || unlikely(pgd_bad(*pgd)))
25 return NULL;
26
27 pmd = pmd_offset(pgd, addr);
28 if (pmd_none(*pmd) || unlikely(pmd_bad(*pmd)))
29 return NULL;
30
31 return pte_offset_map(pmd, addr);
32}
33
18static int __handle_fault(struct mm_struct *mm, unsigned long address, 34static int __handle_fault(struct mm_struct *mm, unsigned long address,
19 int write_access) 35 int write_access)
20{ 36{
@@ -85,8 +101,6 @@ static size_t __user_copy_pt(unsigned long uaddr, void *kptr,
85{ 101{
86 struct mm_struct *mm = current->mm; 102 struct mm_struct *mm = current->mm;
87 unsigned long offset, pfn, done, size; 103 unsigned long offset, pfn, done, size;
88 pgd_t *pgd;
89 pmd_t *pmd;
90 pte_t *pte; 104 pte_t *pte;
91 void *from, *to; 105 void *from, *to;
92 106
@@ -94,15 +108,7 @@ static size_t __user_copy_pt(unsigned long uaddr, void *kptr,
94retry: 108retry:
95 spin_lock(&mm->page_table_lock); 109 spin_lock(&mm->page_table_lock);
96 do { 110 do {
97 pgd = pgd_offset(mm, uaddr); 111 pte = follow_table(mm, uaddr);
98 if (pgd_none(*pgd) || unlikely(pgd_bad(*pgd)))
99 goto fault;
100
101 pmd = pmd_offset(pgd, uaddr);
102 if (pmd_none(*pmd) || unlikely(pmd_bad(*pmd)))
103 goto fault;
104
105 pte = pte_offset_map(pmd, uaddr);
106 if (!pte || !pte_present(*pte) || 112 if (!pte || !pte_present(*pte) ||
107 (write_user && !pte_write(*pte))) 113 (write_user && !pte_write(*pte)))
108 goto fault; 114 goto fault;
@@ -142,22 +148,12 @@ static unsigned long __dat_user_addr(unsigned long uaddr)
142{ 148{
143 struct mm_struct *mm = current->mm; 149 struct mm_struct *mm = current->mm;
144 unsigned long pfn, ret; 150 unsigned long pfn, ret;
145 pgd_t *pgd;
146 pmd_t *pmd;
147 pte_t *pte; 151 pte_t *pte;
148 int rc; 152 int rc;
149 153
150 ret = 0; 154 ret = 0;
151retry: 155retry:
152 pgd = pgd_offset(mm, uaddr); 156 pte = follow_table(mm, uaddr);
153 if (pgd_none(*pgd) || unlikely(pgd_bad(*pgd)))
154 goto fault;
155
156 pmd = pmd_offset(pgd, uaddr);
157 if (pmd_none(*pmd) || unlikely(pmd_bad(*pmd)))
158 goto fault;
159
160 pte = pte_offset_map(pmd, uaddr);
161 if (!pte || !pte_present(*pte)) 157 if (!pte || !pte_present(*pte))
162 goto fault; 158 goto fault;
163 159
@@ -229,8 +225,6 @@ static size_t strnlen_user_pt(size_t count, const char __user *src)
229 unsigned long uaddr = (unsigned long) src; 225 unsigned long uaddr = (unsigned long) src;
230 struct mm_struct *mm = current->mm; 226 struct mm_struct *mm = current->mm;
231 unsigned long offset, pfn, done, len; 227 unsigned long offset, pfn, done, len;
232 pgd_t *pgd;
233 pmd_t *pmd;
234 pte_t *pte; 228 pte_t *pte;
235 size_t len_str; 229 size_t len_str;
236 230
@@ -240,15 +234,7 @@ static size_t strnlen_user_pt(size_t count, const char __user *src)
240retry: 234retry:
241 spin_lock(&mm->page_table_lock); 235 spin_lock(&mm->page_table_lock);
242 do { 236 do {
243 pgd = pgd_offset(mm, uaddr); 237 pte = follow_table(mm, uaddr);
244 if (pgd_none(*pgd) || unlikely(pgd_bad(*pgd)))
245 goto fault;
246
247 pmd = pmd_offset(pgd, uaddr);
248 if (pmd_none(*pmd) || unlikely(pmd_bad(*pmd)))
249 goto fault;
250
251 pte = pte_offset_map(pmd, uaddr);
252 if (!pte || !pte_present(*pte)) 238 if (!pte || !pte_present(*pte))
253 goto fault; 239 goto fault;
254 240
@@ -308,8 +294,6 @@ static size_t copy_in_user_pt(size_t n, void __user *to,
308 uaddr, done, size; 294 uaddr, done, size;
309 unsigned long uaddr_from = (unsigned long) from; 295 unsigned long uaddr_from = (unsigned long) from;
310 unsigned long uaddr_to = (unsigned long) to; 296 unsigned long uaddr_to = (unsigned long) to;
311 pgd_t *pgd_from, *pgd_to;
312 pmd_t *pmd_from, *pmd_to;
313 pte_t *pte_from, *pte_to; 297 pte_t *pte_from, *pte_to;
314 int write_user; 298 int write_user;
315 299
@@ -317,39 +301,14 @@ static size_t copy_in_user_pt(size_t n, void __user *to,
317retry: 301retry:
318 spin_lock(&mm->page_table_lock); 302 spin_lock(&mm->page_table_lock);
319 do { 303 do {
320 pgd_from = pgd_offset(mm, uaddr_from); 304 pte_from = follow_table(mm, uaddr_from);
321 if (pgd_none(*pgd_from) || unlikely(pgd_bad(*pgd_from))) {
322 uaddr = uaddr_from;
323 write_user = 0;
324 goto fault;
325 }
326 pgd_to = pgd_offset(mm, uaddr_to);
327 if (pgd_none(*pgd_to) || unlikely(pgd_bad(*pgd_to))) {
328 uaddr = uaddr_to;
329 write_user = 1;
330 goto fault;
331 }
332
333 pmd_from = pmd_offset(pgd_from, uaddr_from);
334 if (pmd_none(*pmd_from) || unlikely(pmd_bad(*pmd_from))) {
335 uaddr = uaddr_from;
336 write_user = 0;
337 goto fault;
338 }
339 pmd_to = pmd_offset(pgd_to, uaddr_to);
340 if (pmd_none(*pmd_to) || unlikely(pmd_bad(*pmd_to))) {
341 uaddr = uaddr_to;
342 write_user = 1;
343 goto fault;
344 }
345
346 pte_from = pte_offset_map(pmd_from, uaddr_from);
347 if (!pte_from || !pte_present(*pte_from)) { 305 if (!pte_from || !pte_present(*pte_from)) {
348 uaddr = uaddr_from; 306 uaddr = uaddr_from;
349 write_user = 0; 307 write_user = 0;
350 goto fault; 308 goto fault;
351 } 309 }
352 pte_to = pte_offset_map(pmd_to, uaddr_to); 310
311 pte_to = follow_table(mm, uaddr_to);
353 if (!pte_to || !pte_present(*pte_to) || !pte_write(*pte_to)) { 312 if (!pte_to || !pte_present(*pte_to) || !pte_write(*pte_to)) {
354 uaddr = uaddr_to; 313 uaddr = uaddr_to;
355 write_user = 1; 314 write_user = 1;