aboutsummaryrefslogtreecommitdiffstats
path: root/arch/um/kernel/skas/uaccess.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/um/kernel/skas/uaccess.c')
-rw-r--r--arch/um/kernel/skas/uaccess.c56
1 files changed, 23 insertions, 33 deletions
diff --git a/arch/um/kernel/skas/uaccess.c b/arch/um/kernel/skas/uaccess.c
index d145565b5f4b..05b41dbc1dd9 100644
--- a/arch/um/kernel/skas/uaccess.c
+++ b/arch/um/kernel/skas/uaccess.c
@@ -13,70 +13,60 @@
13#include "kern_util.h" 13#include "kern_util.h"
14#include "os.h" 14#include "os.h"
15 15
16static void *um_virt_to_phys(struct task_struct *task, unsigned long addr, 16pte_t *virt_to_pte(struct mm_struct *mm, unsigned long addr)
17 pte_t *pte_out)
18{ 17{
19 pgd_t *pgd; 18 pgd_t *pgd;
20 pud_t *pud; 19 pud_t *pud;
21 pmd_t *pmd; 20 pmd_t *pmd;
22 pte_t *pte;
23 pte_t ptent;
24 21
25 if (task->mm == NULL) 22 if (mm == NULL)
26 return ERR_PTR(-EINVAL); 23 return NULL;
27 pgd = pgd_offset(task->mm, addr); 24
25 pgd = pgd_offset(mm, addr);
28 if (!pgd_present(*pgd)) 26 if (!pgd_present(*pgd))
29 return ERR_PTR(-EINVAL); 27 return NULL;
30 28
31 pud = pud_offset(pgd, addr); 29 pud = pud_offset(pgd, addr);
32 if (!pud_present(*pud)) 30 if (!pud_present(*pud))
33 return ERR_PTR(-EINVAL); 31 return NULL;
34 32
35 pmd = pmd_offset(pud, addr); 33 pmd = pmd_offset(pud, addr);
36 if (!pmd_present(*pmd)) 34 if (!pmd_present(*pmd))
37 return ERR_PTR(-EINVAL); 35 return NULL;
38
39 pte = pte_offset_kernel(pmd, addr);
40 ptent = *pte;
41 if (!pte_present(ptent))
42 return ERR_PTR(-EINVAL);
43 36
44 if (pte_out != NULL) 37 return pte_offset_kernel(pmd, addr);
45 *pte_out = ptent;
46 return (void *) (pte_val(ptent) & PAGE_MASK) + (addr & ~PAGE_MASK);
47} 38}
48 39
49static unsigned long maybe_map(unsigned long virt, int is_write) 40static pte_t *maybe_map(unsigned long virt, int is_write)
50{ 41{
51 pte_t pte; 42 pte_t *pte = virt_to_pte(current->mm, virt);
52 int err; 43 int err, dummy_code;
53 44
54 void *phys = um_virt_to_phys(current, virt, &pte); 45 if ((pte == NULL) || !pte_present(*pte) ||
55 int dummy_code; 46 (is_write && !pte_write(*pte))) {
56
57 if (IS_ERR(phys) || (is_write && !pte_write(pte))) {
58 err = handle_page_fault(virt, 0, is_write, 1, &dummy_code); 47 err = handle_page_fault(virt, 0, is_write, 1, &dummy_code);
59 if (err) 48 if (err)
60 return -1UL; 49 return NULL;
61 phys = um_virt_to_phys(current, virt, NULL); 50 pte = virt_to_pte(current->mm, virt);
62 } 51 }
63 if (IS_ERR(phys)) 52 if (!pte_present(*pte))
64 phys = (void *) -1; 53 pte = NULL;
65 54
66 return (unsigned long) phys; 55 return pte;
67} 56}
68 57
69static int do_op_one_page(unsigned long addr, int len, int is_write, 58static int do_op_one_page(unsigned long addr, int len, int is_write,
70 int (*op)(unsigned long addr, int len, void *arg), void *arg) 59 int (*op)(unsigned long addr, int len, void *arg), void *arg)
71{ 60{
72 struct page *page; 61 struct page *page;
62 pte_t *pte;
73 int n; 63 int n;
74 64
75 addr = maybe_map(addr, is_write); 65 pte = maybe_map(addr, is_write);
76 if (addr == -1UL) 66 if (pte == NULL)
77 return -1; 67 return -1;
78 68
79 page = phys_to_page(addr); 69 page = pte_page(*pte);
80 addr = (unsigned long) kmap_atomic(page, KM_UML_USERCOPY) + 70 addr = (unsigned long) kmap_atomic(page, KM_UML_USERCOPY) +
81 (addr & ~PAGE_MASK); 71 (addr & ~PAGE_MASK);
82 72