Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/aarch64 treat kernel-exec attr and user-exec attr s...



details:   https://anonhg.NetBSD.org/src/rev/8c4b18c01620
branches:  trunk
changeset: 991981:8c4b18c01620
user:      ryo <ryo%NetBSD.org@localhost>
date:      Fri Aug 10 21:06:42 2018 +0000

description:
treat kernel-exec attr and user-exec attr separately.
kernel cannot execute userland exec page, and user cannot execute kernel page.

diffstat:

 sys/arch/aarch64/aarch64/locore.S |   6 ++--
 sys/arch/aarch64/aarch64/pmap.c   |  52 ++++++++++++++++++++++++---------------
 sys/arch/aarch64/include/pmap.h   |   6 ++--
 3 files changed, 38 insertions(+), 26 deletions(-)

diffs (215 lines):

diff -r 7a8acdaa5e4f -r 8c4b18c01620 sys/arch/aarch64/aarch64/locore.S
--- a/sys/arch/aarch64/aarch64/locore.S Fri Aug 10 20:35:52 2018 +0000
+++ b/sys/arch/aarch64/aarch64/locore.S Fri Aug 10 21:06:42 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: locore.S,v 1.17 2018/08/10 17:46:06 maxv Exp $ */
+/*     $NetBSD: locore.S,v 1.18 2018/08/10 21:06:42 ryo Exp $  */
 
 /*
  * Copyright (c) 2017 Ryo Shimizu <ryo%nerv.org@localhost>
@@ -35,7 +35,7 @@
 #include <aarch64/hypervisor.h>
 #include "assym.h"
 
-RCSID("$NetBSD: locore.S,v 1.17 2018/08/10 17:46:06 maxv Exp $")
+RCSID("$NetBSD: locore.S,v 1.18 2018/08/10 21:06:42 ryo Exp $")
 
 /* #define DEBUG_LOCORE */
 /* #define DEBUG_MMU */
@@ -640,7 +640,7 @@
        adr     x2, start               /* physical addr. before MMU */
        and     x2, x2, #L2_BLK_OA      /* L2 block size aligned (2MB) */
        mov     x1, #VM_MIN_KERNEL_ADDRESS
-       mov     x3, #L2_BLKPAG_ATTR_NORMAL_WB
+       mov     x3, #(L2_BLKPAG_ATTR_NORMAL_WB|LX_BLKPAG_UXN)
 
        /* kernelsize = _end - start */
        ldr     x1, =start
diff -r 7a8acdaa5e4f -r 8c4b18c01620 sys/arch/aarch64/aarch64/pmap.c
--- a/sys/arch/aarch64/aarch64/pmap.c   Fri Aug 10 20:35:52 2018 +0000
+++ b/sys/arch/aarch64/aarch64/pmap.c   Fri Aug 10 21:06:42 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap.c,v 1.17 2018/08/06 12:50:56 ryo Exp $    */
+/*     $NetBSD: pmap.c,v 1.18 2018/08/10 21:06:42 ryo Exp $    */
 
 /*
  * Copyright (c) 2017 Ryo Shimizu <ryo%nerv.org@localhost>
@@ -27,7 +27,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.17 2018/08/06 12:50:56 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.18 2018/08/10 21:06:42 ryo Exp $");
 
 #include "opt_arm_debug.h"
 #include "opt_ddb.h"
@@ -163,7 +163,7 @@
 };
 
 static pt_entry_t *_pmap_pte_lookup(struct pmap *, vaddr_t);
-static pt_entry_t _pmap_pte_adjust_prot(pt_entry_t, vm_prot_t, vm_prot_t);
+static pt_entry_t _pmap_pte_adjust_prot(pt_entry_t, vm_prot_t, vm_prot_t, bool);
 static pt_entry_t _pmap_pte_adjust_cacheflags(pt_entry_t, u_int);
 static void _pmap_remove(struct pmap *, vaddr_t, bool);
 static int _pmap_enter(struct pmap *, vaddr_t, paddr_t, vm_prot_t, u_int, bool);
@@ -249,11 +249,15 @@
        oldpte = l2[l2pde_index(va)];
        KDASSERT(!l2pde_valid(oldpte));
 
-       attr = _pmap_pte_adjust_prot(L2_BLOCK, prot, VM_PROT_ALL);
+       attr = _pmap_pte_adjust_prot(L2_BLOCK, prot, VM_PROT_ALL, false);
        attr = _pmap_pte_adjust_cacheflags(attr, flags | PMAP_DEV);
 #ifdef MULTIPROCESSOR
        attr |= LX_BLKPAG_SH_IS;
 #endif
+       /* user cannot execute, and kernel follows the prot */
+       attr |= (LX_BLKPAG_UXN|LX_BLKPAG_PXN);
+       if (prot & VM_PROT_EXECUTE)
+               attr &= ~LX_BLKPAG_PXN;
 
        resid = (size + (L2_SIZE - 1)) & ~(L2_SIZE - 1);
        size = resid;
@@ -780,9 +784,11 @@
 }
 
 static pt_entry_t
-_pmap_pte_adjust_prot(pt_entry_t pte, vm_prot_t prot, vm_prot_t protmask)
+_pmap_pte_adjust_prot(pt_entry_t pte, vm_prot_t prot, vm_prot_t protmask,
+    bool user)
 {
        vm_prot_t masked;
+       pt_entry_t xn;
 
        masked = prot & protmask;
        pte &= ~(LX_BLKPAG_OS_RWMASK|LX_BLKPAG_AF|LX_BLKPAG_AP);
@@ -820,10 +826,12 @@
                break;
        }
 
-       if ((prot & VM_PROT_EXECUTE) == 0)
-               pte |= (LX_BLKPAG_UXN|LX_BLKPAG_PXN);
-       else
-               pte &= ~(LX_BLKPAG_UXN|LX_BLKPAG_PXN);
+       /* executable for kernel or user? first set never exec both */
+       pte |= (LX_BLKPAG_UXN|LX_BLKPAG_PXN);
+       /* and either to executable */
+       xn = user ? LX_BLKPAG_UXN : LX_BLKPAG_PXN;
+       if (prot & VM_PROT_EXECUTE)
+               pte &= ~xn;                     
 
        return pte;
 }
@@ -1052,6 +1060,7 @@
        pt_entry_t *ptep, pte;
        vm_prot_t pteprot;
        uint32_t mdattr;
+       const bool user = (pv->pv_pmap != pmap_kernel());
 
        UVMHIST_FUNC(__func__);
        UVMHIST_CALLED(pmaphist);
@@ -1073,11 +1082,11 @@
                pteprot |= VM_PROT_READ;
        if ((pte & LX_BLKPAG_AP) == LX_BLKPAG_AP_RW)
                pteprot |= VM_PROT_WRITE;
-       if ((pte & (LX_BLKPAG_UXN|LX_BLKPAG_PXN)) == 0)
+       if (l3pte_executable(pte, user))
                pteprot |= VM_PROT_EXECUTE;
 
        /* new prot = prot & pteprot & mdattr */
-       pte = _pmap_pte_adjust_prot(pte, prot & pteprot, mdattr);
+       pte = _pmap_pte_adjust_prot(pte, prot & pteprot, mdattr, user);
        atomic_swap_64(ptep, pte);
 
 #if 0
@@ -1093,6 +1102,7 @@
 pmap_protect(struct pmap *pm, vaddr_t sva, vaddr_t eva, vm_prot_t prot)
 {
        vaddr_t va;
+       const bool user = (pm != pmap_kernel());
 
        KASSERT((prot & VM_PROT_READ) || !(prot & VM_PROT_WRITE));
 
@@ -1159,8 +1169,8 @@
 #ifdef UVMHIST
                opte = pte;
 #endif
-               executable = l3pte_executable(pte);
-               pte = _pmap_pte_adjust_prot(pte, prot, mdattr);
+               executable = l3pte_executable(pte, user);
+               pte = _pmap_pte_adjust_prot(pte, prot, mdattr, user);
 
                if (!executable && (prot & VM_PROT_EXECUTE)) {
                        /* non-exec -> exec */
@@ -1419,7 +1429,7 @@
 #ifdef UVMHIST
        opte = pte;
 #endif
-       executable = l3pte_executable(pte);
+       executable = l3pte_executable(pte, user);
 
        if (l3pte_valid(pte)) {
                KASSERT(!kenter);       /* pmap_kenter_pa() cannot override */
@@ -1486,7 +1496,7 @@
        }
 #endif
 
-       attr = _pmap_pte_adjust_prot(L3_PAGE, prot, mdattr);
+       attr = _pmap_pte_adjust_prot(L3_PAGE, prot, mdattr, user);
        attr = _pmap_pte_adjust_cacheflags(attr, flags);
        if (VM_MAXUSER_ADDRESS > va)
                attr |= LX_BLKPAG_APUSER;
@@ -1780,7 +1790,7 @@
                pmap_prot = (VM_PROT_READ|VM_PROT_WRITE);
                break;
        }
-       if ((pte & (LX_BLKPAG_UXN|LX_BLKPAG_PXN)) == 0)
+       if (l3pte_executable(pte, pm != pmap_kernel()))
                pmap_prot |= VM_PROT_EXECUTE;
 
        UVMHIST_LOG(pmaphist, "va=%016lx, pmapprot=%08x, accessprot=%08x",
@@ -2063,10 +2073,10 @@
        }
 
        if ((prot & VM_PROT_EXECUTE) == 0) {
-               pte |= (LX_BLKPAG_UXN|LX_BLKPAG_PXN);
+               pte |= LX_BLKPAG_PXN;
        } else {
                pte |= LX_BLKPAG_AF;
-               pte &= ~(LX_BLKPAG_UXN|LX_BLKPAG_PXN);
+               pte &= ~LX_BLKPAG_PXN;
        }
 
        *ptep = pte;
@@ -2152,8 +2162,10 @@
                        pr(", pmap_read");
                if (pte & LX_BLKPAG_OS_WRITE)
                        pr(", pmap_write");
-               if ((pte & (LX_BLKPAG_UXN|LX_BLKPAG_PXN)) == 0)
-                       pr(", executable");
+               if ((pte & LX_BLKPAG_UXN) == 0)
+                       pr(", user-executable");
+               if ((pte & LX_BLKPAG_PXN) == 0)
+                       pr(", kernel-executable");
 
        } else {
                /* L1 and L2 pde */
diff -r 7a8acdaa5e4f -r 8c4b18c01620 sys/arch/aarch64/include/pmap.h
--- a/sys/arch/aarch64/include/pmap.h   Fri Aug 10 20:35:52 2018 +0000
+++ b/sys/arch/aarch64/include/pmap.h   Fri Aug 10 21:06:42 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pmap.h,v 1.7 2018/08/06 12:50:56 ryo Exp $ */
+/* $NetBSD: pmap.h,v 1.8 2018/08/10 21:06:42 ryo Exp $ */
 
 /*-
  * Copyright (c) 2014 The NetBSD Foundation, Inc.
@@ -95,8 +95,8 @@
 #define l2pde_is_table(pde)    (((pde) & LX_TYPE) == LX_TYPE_TBL)
 
 #define l3pte_pa(pde)          ((paddr_t)((pde) & LX_TBL_PA))
-#define l3pte_executable(pde)  \
-    (((pde) & (LX_BLKPAG_UXN|LX_BLKPAG_PXN)) != (LX_BLKPAG_UXN|LX_BLKPAG_PXN))
+#define l3pte_executable(pde,user)     \
+    (((pde) & ((user) ? LX_BLKPAG_UXN : LX_BLKPAG_PXN)) == 0)
 #define l3pte_readable(pde)    ((pde) & LX_BLKPAG_AF)
 #define l3pte_writable(pde)    \
     (((pde) & (LX_BLKPAG_AF|LX_BLKPAG_AP)) == (LX_BLKPAG_AF|LX_BLKPAG_AP_RW))



Home | Main Index | Thread Index | Old Index