]> git.saurik.com Git - apple/xnu.git/blobdiff - bsd/dev/dtrace/systrace.c
xnu-7195.101.1.tar.gz
[apple/xnu.git] / bsd / dev / dtrace / systrace.c
index 35601e943fec93c912b68554450657e5e7abda5f..1342e7785875b4d3a1cf25f8662cfc431bda6ba9 100644 (file)
  * CDDL HEADER END
  */
 /*
- * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
+ * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
  * Use is subject to license terms.
  */
 
-/* #pragma ident       "@(#)systrace.c 1.5     06/03/24 SMI" */
+#include <ptrauth.h>
 
-#if !defined(__APPLE__)
-#include <sys/dtrace.h>
-#include <sys/systrace.h>
-#include <sys/stat.h>
-#include <sys/systm.h>
-#include <sys/conf.h>
-#include <sys/ddi.h>
-#include <sys/sunddi.h>
-#include <sys/atomic.h>
-#define        SYSTRACE_ARTIFICIAL_FRAMES      1
-#else
-
-#ifdef KERNEL
-#ifndef _KERNEL
-#define _KERNEL /* Solaris vs. Darwin */
-#endif
-#endif
-
-#define MACH__POSIX_C_SOURCE_PRIVATE 1 /* pulls in suitable savearea from mach/ppc/thread_status.h */
 #include <kern/thread.h>
 #include <mach/thread_status.h>
+
 /* XXX All of these should really be derived from syscall_sw.h */
-#if defined(__i386__) || defined (__x86_64__)
+#if defined (__x86_64__)
 #define SYSCALL_CLASS_SHIFT 24
 #define SYSCALL_CLASS_MASK  (0xFF << SYSCALL_CLASS_SHIFT)
 #define SYSCALL_NUMBER_MASK (~SYSCALL_CLASS_MASK)
 #define I386_SYSCALL_NUMBER_MASK (0xFFFF)
-
-typedef x86_saved_state_t savearea_t;
 #endif
 
 #include <sys/param.h>
@@ -63,26 +43,32 @@ typedef x86_saved_state_t savearea_t;
 #include <sys/ioctl.h>
 #include <sys/conf.h>
 #include <sys/fcntl.h>
+#include <sys/syscall.h>
 #include <miscfs/devfs/devfs.h>
 
 #include <sys/dtrace.h>
 #include <sys/dtrace_impl.h>
+#include <sys/systrace_args.h>
 #include "systrace.h"
 #include <sys/stat.h>
 #include <sys/systm.h>
 #include <sys/conf.h>
 #include <sys/user.h>
 
-#if defined (__ppc__) || defined (__ppc64__)
-#define        SYSTRACE_ARTIFICIAL_FRAMES      3
-#define MACHTRACE_ARTIFICIAL_FRAMES 4
-#elif defined(__i386__) || defined (__x86_64__)
-#define        SYSTRACE_ARTIFICIAL_FRAMES      2
+#include <machine/pal_routines.h>
+
+#if defined (__x86_64__)
+#define SYSTRACE_ARTIFICIAL_FRAMES      2
+#define MACHTRACE_ARTIFICIAL_FRAMES 3
+#elif defined(__arm__) || defined(__arm64__)
+#define SYSTRACE_ARTIFICIAL_FRAMES  2
 #define MACHTRACE_ARTIFICIAL_FRAMES 3
 #else
 #error Unknown Architecture
 #endif
 
+#define SYSTRACE_NARGS (int)(sizeof(((uthread_t)NULL)->uu_arg) / sizeof(((uthread_t)NULL)->uu_arg[0]))
+
 #include <sys/sysent.h>
 #define sy_callc sy_call /* Map Solaris slot name to Darwin's */
 #define NSYSCALL nsysent /* and is less than 500 or so */
@@ -96,9 +82,14 @@ extern const char *syscallnames[];
 #define LOADABLE_SYSCALL(a) 0 /* Not pertinent to Darwin. */
 #define LOADED_SYSCALL(a) 1 /* Not pertinent to Darwin. */
 
+static LCK_MTX_DECLARE_ATTR(dtrace_systrace_lock,
+    &dtrace_lck_grp, &dtrace_lck_attr);           /* probe state lock */
+
 systrace_sysent_t *systrace_sysent = NULL;
-void (*systrace_probe)(dtrace_id_t, uint64_t, uint64_t,
-    uint64_t, uint64_t, uint64_t);
+void (*systrace_probe)(dtrace_id_t, uint64_t, uint64_t, uint64_t, uint64_t, uint64_t);
+
+static uint64_t systrace_getargval(void *, dtrace_id_t, void *, int, int);
+static void systrace_getargdesc(void *, dtrace_id_t, void *, dtrace_argdesc_t *);
 
 void
 systrace_stub(dtrace_id_t id, uint64_t arg0, uint64_t arg1,
@@ -110,31 +101,17 @@ systrace_stub(dtrace_id_t id, uint64_t arg0, uint64_t arg1,
 int32_t
 dtrace_systrace_syscall(struct proc *pp, void *uap, int *rv)
 {
-       boolean_t           flavor;
-       unsigned short      code;
+       unsigned short      code;       /* The system call number */
 
        systrace_sysent_t *sy;
        dtrace_id_t id;
        int32_t rval;
-#if 0 /* XXX */
-       proc_t *p;
-#endif
        syscall_arg_t *ip = (syscall_arg_t *)uap;
+       uint64_t uargs[SYSTRACE_NARGS] = {0};
 
-#if defined (__ppc__) || defined (__ppc64__)
-       {
-               savearea_t *regs = (savearea_t *)find_user_regs(current_thread());
-
-               flavor = (((unsigned int)regs->save_r0) == 0)? 1: 0;
-
-               if (flavor)
-                       code = regs->save_r3;
-               else
-                       code = regs->save_r0;
-       }
-#elif defined(__i386__) || defined (__x86_64__)
-#pragma unused(flavor)
+#if defined (__x86_64__)
        {
+               pal_register_cache_state(current_thread(), VALID);
                x86_saved_state_t   *tagged_regs = (x86_saved_state_t *)find_user_regs(current_thread());
 
                if (is_saved_state64(tagged_regs)) {
@@ -149,9 +126,55 @@ dtrace_systrace_syscall(struct proc *pp, void *uap, int *rv)
                        }
                } else {
                        code = saved_state32(tagged_regs)->eax & I386_SYSCALL_NUMBER_MASK;
-                       /*
-                        * TODO: handle indirect system calls
-                        */
+
+                       if (code == 0) {
+                               vm_offset_t params = (vm_offset_t) (saved_state32(tagged_regs)->uesp + sizeof(int));
+                               code = fuword(params);
+                       }
+               }
+       }
+#elif defined(__arm__)
+       {
+               /*
+                * On arm, syscall numbers depend on a flavor (indirect or not)
+                * and can be in either r0 or r12  (always u32)
+                */
+
+               /* See bsd/dev/arm/systemcalls.c:arm_get_syscall_number */
+               arm_saved_state_t *arm_regs = (arm_saved_state_t *) find_user_regs(current_thread());
+
+               /* Check for indirect system call */
+               if (arm_regs->r[12] != 0) {
+                       code = arm_regs->r[12];
+               } else {
+                       code = arm_regs->r[0];
+               }
+       }
+#elif defined(__arm64__)
+       {
+               /*
+                * On arm64, syscall numbers depend on a flavor (indirect or not)
+                * ... and for u32 can be in either r0 or r12
+                * ... and for u64 can be in either x0 or x16
+                */
+
+               /* see bsd/dev/arm/systemcalls.c:arm_get_syscall_number */
+               arm_saved_state_t *arm_regs = (arm_saved_state_t *) find_user_regs(current_thread());
+
+               if (is_saved_state32(arm_regs)) {
+                       /* Check for indirect system call */
+                       if (saved_state32(arm_regs)->r[12] != 0) {
+                               code = saved_state32(arm_regs)->r[12];
+                       } else {
+                               code = saved_state32(arm_regs)->r[0];
+                       }
+               } else {
+                       /* Check for indirect system call */
+                       if (saved_state64(arm_regs)->x[ARM64_SYSCALL_CODE_REG_NUM] != 0) {
+                               code = saved_state64(arm_regs)->x[ARM64_SYSCALL_CODE_REG_NUM];
+                       } else {
+                               code = saved_state64(arm_regs)->x[0];
+                       }
                }
        }
 #else
@@ -159,13 +182,29 @@ dtrace_systrace_syscall(struct proc *pp, void *uap, int *rv)
 #endif
 
        // Bounds "check" the value of code a la unix_syscall
-       sy = (code >= NUM_SYSENT) ? &systrace_sysent[63] : &systrace_sysent[code];
+       sy = (code >= nsysent) ? &systrace_sysent[SYS_invalid] : &systrace_sysent[code];
+
+       systrace_args(code, ip, uargs);
+
+       if ((id = sy->stsy_entry) != DTRACE_IDNONE) {
+               uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+               if (uthread) {
+                       uthread->t_dtrace_syscall_args = uargs;
+               }
+
+               static_assert(SYSTRACE_NARGS >= 5, "not enough system call arguments");
+               (*systrace_probe)(id, uargs[0], uargs[1], uargs[2], uargs[3], uargs[4]);
+
+               if (uthread) {
+                       uthread->t_dtrace_syscall_args = NULL;
+               }
+       }
+
 
-       if ((id = sy->stsy_entry) != DTRACE_IDNONE)
-               (*systrace_probe)(id, *ip, *(ip+1), *(ip+2), *(ip+3), *(ip+4));
 
 #if 0 /* XXX */
        /*
+        * APPLE NOTE: Not implemented.
         * We want to explicitly allow DTrace consumers to stop a process
         * before it actually executes the meat of the syscall.
         */
@@ -181,51 +220,78 @@ dtrace_systrace_syscall(struct proc *pp, void *uap, int *rv)
        rval = (*sy->stsy_underlying)(pp, uap, rv);
 
        if ((id = sy->stsy_return) != DTRACE_IDNONE) {
-               uint64_t munged_rv;
-       uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+               uint64_t munged_rv0, munged_rv1;
+               uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
 
-               if (uthread)
+               if (uthread) {
                        uthread->t_dtrace_errno = rval; /* Establish t_dtrace_errno now in case this enabling refers to it. */
-
+               }
                /*
-                * "Decode" rv for use in the call to dtrace_probe()
-                */
+                * "Decode" rv for use in the call to dtrace_probe()
+                */
                if (rval == ERESTART) {
-                       munged_rv = -1LL; /* System call will be reissued in user mode. Make DTrace report a -1 return. */
+                       munged_rv0 = -1LL; /* System call will be reissued in user mode. Make DTrace report a -1 return. */
+                       munged_rv1 = -1LL;
                } else if (rval != EJUSTRETURN) {
                        if (rval) {
-                               munged_rv = -1LL; /* Mimic what libc will do. */
+                               munged_rv0 = -1LL; /* Mimic what libc will do. */
+                               munged_rv1 = -1LL;
                        } else {
                                switch (sy->stsy_return_type) {
                                case _SYSCALL_RET_INT_T:
-                                       munged_rv = rv[0];
+                                       munged_rv0 = rv[0];
+                                       munged_rv1 = rv[1];
                                        break;
                                case _SYSCALL_RET_UINT_T:
-                                       munged_rv = ((u_int)rv[0]);
+                                       munged_rv0 = ((u_int)rv[0]);
+                                       munged_rv1 = ((u_int)rv[1]);
                                        break;
                                case _SYSCALL_RET_OFF_T:
-                                       munged_rv = *(u_int64_t *)rv;
+                               case _SYSCALL_RET_UINT64_T:
+                                       munged_rv0 = *(u_int64_t *)rv;
+                                       munged_rv1 = 0LL;
                                        break;
                                case _SYSCALL_RET_ADDR_T:
                                case _SYSCALL_RET_SIZE_T:
                                case _SYSCALL_RET_SSIZE_T:
-                                       munged_rv = *(user_addr_t *)rv;
+                                       munged_rv0 = *(user_addr_t *)rv;
+                                       munged_rv1 = 0LL;
                                        break;
                                case _SYSCALL_RET_NONE:
-                                       munged_rv = 0LL;
+                                       munged_rv0 = 0LL;
+                                       munged_rv1 = 0LL;
                                        break;
                                default:
-                                       munged_rv = 0LL;
+                                       munged_rv0 = 0LL;
+                                       munged_rv1 = 0LL;
                                        break;
                                }
                        }
-               } else
-                       munged_rv = 0LL;
+               } else {
+                       munged_rv0 = 0LL;
+                       munged_rv1 = 0LL;
+               }
 
-               (*systrace_probe)(id, munged_rv, munged_rv, (uint64_t)rval, 0, 0);
+               /*
+                * <http://mail.opensolaris.org/pipermail/dtrace-discuss/2007-January/003276.html> says:
+                *
+                * "This is a bit of an historical artifact. At first, the syscall provider just
+                * had its return value in arg0, and the fbt and pid providers had their return
+                * values in arg1 (so that we could use arg0 for the offset of the return site).
+                *
+                * We inevitably started writing scripts where we wanted to see the return
+                * values from probes in all three providers, and we made this script easier
+                * to write by replicating the syscall return values in arg1 to match fbt and
+                * pid. We debated briefly about removing the return value from arg0, but
+                * decided that it would be less confusing to have the same data in two places
+                * than to have some non-helpful, non-intuitive value in arg0.
+                *
+                * This change was made 4/23/2003 according to the DTrace project's putback log."
+                */
+               (*systrace_probe)(id, munged_rv0, munged_rv0, munged_rv1, (uint64_t)rval, 0);
        }
 
-       return (rval);
+       return rval;
 }
 
 void
@@ -235,125 +301,135 @@ dtrace_systrace_syscall_return(unsigned short code, int rval, int *rv)
        dtrace_id_t id;
 
        // Bounds "check" the value of code a la unix_syscall_return
-       sy = (code >= NUM_SYSENT) ? &systrace_sysent[63] : &systrace_sysent[code];
+       sy = (code >= nsysent) ? &systrace_sysent[SYS_invalid] : &systrace_sysent[code];
 
        if ((id = sy->stsy_return) != DTRACE_IDNONE) {
-               uint64_t munged_rv;
-       uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+               uint64_t munged_rv0, munged_rv1;
+               uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
 
-               if (uthread)
+               if (uthread) {
                        uthread->t_dtrace_errno = rval; /* Establish t_dtrace_errno now in case this enabling refers to it. */
-
+               }
                /*
-                * "Decode" rv for use in the call to dtrace_probe()
-                */
+                * "Decode" rv for use in the call to dtrace_probe()
+                */
                if (rval == ERESTART) {
-                       munged_rv = -1LL; /* System call will be reissued in user mode. Make DTrace report a -1 return. */
+                       munged_rv0 = -1LL; /* System call will be reissued in user mode. Make DTrace report a -1 return. */
+                       munged_rv1 = -1LL;
                } else if (rval != EJUSTRETURN) {
                        if (rval) {
-                               munged_rv = -1LL; /* Mimic what libc will do. */
+                               munged_rv0 = -1LL; /* Mimic what libc will do. */
+                               munged_rv1 = -1LL;
                        } else {
                                switch (sy->stsy_return_type) {
                                case _SYSCALL_RET_INT_T:
-                                       munged_rv = rv[0];
+                                       munged_rv0 = rv[0];
+                                       munged_rv1 = rv[1];
                                        break;
                                case _SYSCALL_RET_UINT_T:
-                                       munged_rv = ((u_int)rv[0]);
+                                       munged_rv0 = ((u_int)rv[0]);
+                                       munged_rv1 = ((u_int)rv[1]);
                                        break;
                                case _SYSCALL_RET_OFF_T:
-                                       munged_rv = *(u_int64_t *)rv;
+                               case _SYSCALL_RET_UINT64_T:
+                                       munged_rv0 = *(u_int64_t *)rv;
+                                       munged_rv1 = 0LL;
                                        break;
                                case _SYSCALL_RET_ADDR_T:
                                case _SYSCALL_RET_SIZE_T:
                                case _SYSCALL_RET_SSIZE_T:
-                                       munged_rv = *(user_addr_t *)rv;
+                                       munged_rv0 = *(user_addr_t *)rv;
+                                       munged_rv1 = 0LL;
                                        break;
                                case _SYSCALL_RET_NONE:
-                                       munged_rv = 0LL;
+                                       munged_rv0 = 0LL;
+                                       munged_rv1 = 0LL;
                                        break;
                                default:
-                                       munged_rv = 0LL;
+                                       munged_rv0 = 0LL;
+                                       munged_rv1 = 0LL;
                                        break;
                                }
                        }
-               } else
-                       munged_rv = 0LL;
+               } else {
+                       munged_rv0 = 0LL;
+                       munged_rv1 = 0LL;
+               }
 
-               (*systrace_probe)(id, munged_rv, munged_rv, (uint64_t)rval, 0, 0);
+               (*systrace_probe)(id, munged_rv0, munged_rv0, munged_rv1, (uint64_t)rval, 0);
        }
 }
-#endif /* __APPLE__ */
 
-#define        SYSTRACE_SHIFT                  16
-#define        SYSTRACE_ISENTRY(x)             ((int)(x) >> SYSTRACE_SHIFT)
-#define        SYSTRACE_SYSNUM(x)              ((int)(x) & ((1 << SYSTRACE_SHIFT) - 1))
-#define        SYSTRACE_ENTRY(id)              ((1 << SYSTRACE_SHIFT) | (id))
-#define        SYSTRACE_RETURN(id)             (id)
+#define SYSTRACE_SHIFT                  16
+#define SYSTRACE_ISENTRY(x)             ((int)(x) >> SYSTRACE_SHIFT)
+#define SYSTRACE_SYSNUM(x)              ((int)(x) & ((1 << SYSTRACE_SHIFT) - 1))
+#define SYSTRACE_ENTRY(id)              ((1 << SYSTRACE_SHIFT) | (id))
+#define SYSTRACE_RETURN(id)             (id)
 
 #if ((1 << SYSTRACE_SHIFT) <= NSYSCALL)
 #error 1 << SYSTRACE_SHIFT must exceed number of system calls
 #endif
 
-static dev_info_t *systrace_devi;
 static dtrace_provider_id_t systrace_id;
 
-#if defined(__APPLE__)
-#define systrace_init _systrace_init /* Avoid name clash with Darwin automagic conf symbol */
-#endif
+/*
+ * APPLE NOTE: Avoid name clash with Darwin automagic conf symbol.
+ * See balanced undef below.
+ */
+#define systrace_init _systrace_init
+
 static void
-systrace_init(struct sysent *actual, systrace_sysent_t **interposed)
+systrace_init(const struct sysent *actual, systrace_sysent_t **interposed)
 {
-       systrace_sysent_t *sysent = *interposed;
-       int i;
+       systrace_sysent_t *ssysent = *interposed;  /* Avoid sysent shadow warning
+                                                   *       from bsd/sys/sysent.h */
+       unsigned int i;
 
-       if (sysent == NULL) {
-               *interposed = sysent = kmem_zalloc(sizeof (systrace_sysent_t) *
+       if (ssysent == NULL) {
+               *interposed = ssysent = kmem_zalloc(sizeof(systrace_sysent_t) *
                    NSYSCALL, KM_SLEEP);
        }
 
        for (i = 0; i < NSYSCALL; i++) {
-               struct sysent *a = &actual[i];
-               systrace_sysent_t *s = &sysent[i];
+               const struct sysent *a = &actual[i];
+               systrace_sysent_t *s = &ssysent[i];
 
-               if (LOADABLE_SYSCALL(a) && !LOADED_SYSCALL(a))
-                       continue;
-
-               if (a->sy_callc == dtrace_systrace_syscall)
+               if (LOADABLE_SYSCALL(a) && !LOADED_SYSCALL(a)) {
                        continue;
+               }
 
-#ifdef _SYSCALL32_IMPL
-               if (a->sy_callc == dtrace_systrace_syscall32)
+               if (a->sy_callc == dtrace_systrace_syscall) {
                        continue;
-#endif
+               }
 
                s->stsy_underlying = a->sy_callc;
-#if defined(__APPLE__)
                s->stsy_return_type = a->sy_return_type;
-#endif
        }
 }
 
+
 /*ARGSUSED*/
 static void
 systrace_provide(void *arg, const dtrace_probedesc_t *desc)
 {
-       int i;
+#pragma unused(arg) /* __APPLE__ */
+       unsigned int i;
 
-       if (desc != NULL)
+       if (desc != NULL) {
                return;
+       }
 
        systrace_init(sysent, &systrace_sysent);
-#ifdef _SYSCALL32_IMPL
-       systrace_init(sysent32, &systrace_sysent32);
-#endif
 
        for (i = 0; i < NSYSCALL; i++) {
-               if (systrace_sysent[i].stsy_underlying == NULL)
+               if (systrace_sysent[i].stsy_underlying == NULL) {
                        continue;
+               }
 
                if (dtrace_probe_lookup(systrace_id, NULL,
-                   syscallnames[i], "entry") != 0)
+                   syscallnames[i], "entry") != 0) {
                        continue;
+               }
 
                (void) dtrace_probe_create(systrace_id, NULL, syscallnames[i],
                    "entry", SYSTRACE_ARTIFICIAL_FRAMES,
@@ -364,158 +440,141 @@ systrace_provide(void *arg, const dtrace_probedesc_t *desc)
 
                systrace_sysent[i].stsy_entry = DTRACE_IDNONE;
                systrace_sysent[i].stsy_return = DTRACE_IDNONE;
-#ifdef _SYSCALL32_IMPL
-               systrace_sysent32[i].stsy_entry = DTRACE_IDNONE;
-               systrace_sysent32[i].stsy_return = DTRACE_IDNONE;
-#endif
        }
 }
-#if defined(__APPLE__)
 #undef systrace_init
-#endif
 
 /*ARGSUSED*/
 static void
 systrace_destroy(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg,id) /* __APPLE__ */
+
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
 
+#pragma unused(sysnum)  /* __APPLE__ */
        /*
         * There's nothing to do here but assert that we have actually been
         * disabled.
         */
        if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
                ASSERT(systrace_sysent[sysnum].stsy_entry == DTRACE_IDNONE);
-#ifdef _SYSCALL32_IMPL
-               ASSERT(systrace_sysent32[sysnum].stsy_entry == DTRACE_IDNONE);
-#endif
        } else {
                ASSERT(systrace_sysent[sysnum].stsy_return == DTRACE_IDNONE);
-#ifdef _SYSCALL32_IMPL
-               ASSERT(systrace_sysent32[sysnum].stsy_return == DTRACE_IDNONE);
-#endif
        }
 }
 
 /*ARGSUSED*/
-static void
+static int
 systrace_enable(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg) /* __APPLE__ */
+
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
        int enabled = (systrace_sysent[sysnum].stsy_entry != DTRACE_IDNONE ||
            systrace_sysent[sysnum].stsy_return != DTRACE_IDNONE);
 
        if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
                systrace_sysent[sysnum].stsy_entry = id;
-#ifdef _SYSCALL32_IMPL
-               systrace_sysent32[sysnum].stsy_entry = id;
-#endif
        } else {
                systrace_sysent[sysnum].stsy_return = id;
-#ifdef _SYSCALL32_IMPL
-               systrace_sysent32[sysnum].stsy_return = id;
-#endif
        }
 
        if (enabled) {
                ASSERT(sysent[sysnum].sy_callc == dtrace_systrace_syscall);
-               return;
+               return 0;
        }
 
-       (void) casptr(&sysent[sysnum].sy_callc,
-           (void *)systrace_sysent[sysnum].stsy_underlying,
-           (void *)dtrace_systrace_syscall);
-#ifdef _SYSCALL32_IMPL
-       (void) casptr(&sysent32[sysnum].sy_callc,
-           (void *)systrace_sysent32[sysnum].stsy_underlying,
-           (void *)dtrace_systrace_syscall32);
-#endif
+       lck_mtx_lock(&dtrace_systrace_lock);
+       if (sysent[sysnum].sy_callc == systrace_sysent[sysnum].stsy_underlying) {
+               /* It is not possible to write to sysent[] directly because it is const. */
+               vm_offset_t dss = ptrauth_nop_cast(vm_offset_t, &dtrace_systrace_syscall);
+               ml_nofault_copy((vm_offset_t)&dss, (vm_offset_t)&sysent[sysnum].sy_callc, sizeof(vm_offset_t));
+       }
+       lck_mtx_unlock(&dtrace_systrace_lock);
+
+       return 0;
 }
 
 /*ARGSUSED*/
 static void
 systrace_disable(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg,id) /* __APPLE__ */
+
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
        int disable = (systrace_sysent[sysnum].stsy_entry == DTRACE_IDNONE ||
            systrace_sysent[sysnum].stsy_return == DTRACE_IDNONE);
 
        if (disable) {
-               (void) casptr(&sysent[sysnum].sy_callc,
-                   (void *)dtrace_systrace_syscall,
-                   (void *)systrace_sysent[sysnum].stsy_underlying);
-
-#ifdef _SYSCALL32_IMPL
-               (void) casptr(&sysent32[sysnum].sy_callc,
-                   (void *)dtrace_systrace_syscall32,
-                   (void *)systrace_sysent32[sysnum].stsy_underlying);
-#endif
+               /*
+                * Usage of volatile protects the if statement below from being optimized away.
+                *
+                * Compilers are clever and know that const array values can't change in time
+                * and the if below is always false. That is because it can't see that DTrace
+                * injects dtrace_systrace_syscall dynamically and violates constness of the
+                * array.
+                */
+               volatile const struct sysent *syscallent = &sysent[sysnum];
+
+               lck_mtx_lock(&dtrace_systrace_lock);
+               if (syscallent->sy_callc == dtrace_systrace_syscall) {
+                       ml_nofault_copy((vm_offset_t)&systrace_sysent[sysnum].stsy_underlying,
+                           (vm_offset_t)&syscallent->sy_callc, sizeof(vm_offset_t));
+               }
+               lck_mtx_unlock(&dtrace_systrace_lock);
        }
 
        if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
                systrace_sysent[sysnum].stsy_entry = DTRACE_IDNONE;
-#ifdef _SYSCALL32_IMPL
-               systrace_sysent32[sysnum].stsy_entry = DTRACE_IDNONE;
-#endif
        } else {
                systrace_sysent[sysnum].stsy_return = DTRACE_IDNONE;
-#ifdef _SYSCALL32_IMPL
-               systrace_sysent32[sysnum].stsy_return = DTRACE_IDNONE;
-#endif
        }
 }
 
 static dtrace_pattr_t systrace_attr = {
-{ DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
-{ DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
+       { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
+       { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
 };
 
 static dtrace_pops_t systrace_pops = {
-       systrace_provide,
-       NULL,
-       systrace_enable,
-       systrace_disable,
-       NULL,
-       NULL,
-       NULL,
-       NULL,
-       NULL,
-       systrace_destroy
+       .dtps_provide =         systrace_provide,
+       .dtps_provide_module =  NULL,
+       .dtps_enable =          systrace_enable,
+       .dtps_disable =         systrace_disable,
+       .dtps_suspend =         NULL,
+       .dtps_resume =          NULL,
+       .dtps_getargdesc =      systrace_getargdesc,
+       .dtps_getargval =       systrace_getargval,
+       .dtps_usermode =        NULL,
+       .dtps_destroy =         systrace_destroy
 };
 
 static int
-systrace_attach(dev_info_t *devi, ddi_attach_cmd_t cmd)
+systrace_attach(dev_info_t *devi)
 {
-       switch (cmd) {
-       case DDI_ATTACH:
-               break;
-       case DDI_RESUME:
-               return (DDI_SUCCESS);
-       default:
-               return (DDI_FAILURE);
-       }
-
-       systrace_probe = dtrace_probe;
+       systrace_probe = (void*)&dtrace_probe;
        membar_enter();
 
        if (ddi_create_minor_node(devi, "systrace", S_IFCHR, 0,
-           DDI_PSEUDO, NULL) == DDI_FAILURE ||
+           DDI_PSEUDO, 0) == DDI_FAILURE ||
            dtrace_register("syscall", &systrace_attr, DTRACE_PRIV_USER, NULL,
            &systrace_pops, NULL, &systrace_id) != 0) {
                systrace_probe = systrace_stub;
                ddi_remove_minor_node(devi, NULL);
-               return (DDI_FAILURE);
+               return DDI_FAILURE;
        }
 
-       ddi_report_dev(devi);
-       systrace_devi = devi;
-
-       return (DDI_SUCCESS);
+       return DDI_SUCCESS;
 }
 
+
+/*
+ * APPLE NOTE:  systrace_detach not implemented
+ */
 #if !defined(__APPLE__)
 static int
 systrace_detach(dev_info_t *devi, ddi_detach_cmd_t cmd)
@@ -524,287 +583,55 @@ systrace_detach(dev_info_t *devi, ddi_detach_cmd_t cmd)
        case DDI_DETACH:
                break;
        case DDI_SUSPEND:
-               return (DDI_SUCCESS);
+               return DDI_SUCCESS;
        default:
-               return (DDI_FAILURE);
+               return DDI_FAILURE;
        }
 
-       if (dtrace_unregister(systrace_id) != 0)
-               return (DDI_FAILURE);
+       if (dtrace_unregister(systrace_id) != 0) {
+               return DDI_FAILURE;
+       }
 
        ddi_remove_minor_node(devi, NULL);
        systrace_probe = systrace_stub;
-       return (DDI_SUCCESS);
-}
-
-/*ARGSUSED*/
-static int
-systrace_info(dev_info_t *dip, ddi_info_cmd_t infocmd, void *arg, void **result)
-{
-       int error;
-
-       switch (infocmd) {
-       case DDI_INFO_DEVT2DEVINFO:
-               *result = (void *)systrace_devi;
-               error = DDI_SUCCESS;
-               break;
-       case DDI_INFO_DEVT2INSTANCE:
-               *result = (void *)0;
-               error = DDI_SUCCESS;
-               break;
-       default:
-               error = DDI_FAILURE;
-       }
-       return (error);
-}
-
-/*ARGSUSED*/
-static int
-systrace_open(dev_t *devp, int flag, int otyp, cred_t *cred_p)
-{
-       return (0);
-}
-
-static struct cb_ops systrace_cb_ops = {
-       systrace_open,          /* open */
-       nodev,                  /* close */
-       nulldev,                /* strategy */
-       nulldev,                /* print */
-       nodev,                  /* dump */
-       nodev,                  /* read */
-       nodev,                  /* write */
-       nodev,                  /* ioctl */
-       nodev,                  /* devmap */
-       nodev,                  /* mmap */
-       nodev,                  /* segmap */
-       nochpoll,               /* poll */
-       ddi_prop_op,            /* cb_prop_op */
-       0,                      /* streamtab  */
-       D_NEW | D_MP            /* Driver compatibility flag */
-};
-
-static struct dev_ops systrace_ops = {
-       DEVO_REV,               /* devo_rev, */
-       0,                      /* refcnt  */
-       systrace_info,          /* get_dev_info */
-       nulldev,                /* identify */
-       nulldev,                /* probe */
-       systrace_attach,        /* attach */
-       systrace_detach,        /* detach */
-       nodev,                  /* reset */
-       &systrace_cb_ops,       /* driver operations */
-       NULL,                   /* bus operations */
-       nodev                   /* dev power */
-};
-
-/*
- * Module linkage information for the kernel.
- */
-static struct modldrv modldrv = {
-       &mod_driverops,         /* module type (this is a pseudo driver) */
-       "System Call Tracing",  /* name of module */
-       &systrace_ops,          /* driver ops */
-};
-
-static struct modlinkage modlinkage = {
-       MODREV_1,
-       (void *)&modldrv,
-       NULL
-};
-
-int
-_init(void)
-{
-       return (mod_install(&modlinkage));
+       return DDI_SUCCESS;
 }
+#endif /* __APPLE__ */
 
-int
-_info(struct modinfo *modinfop)
-{
-       return (mod_info(&modlinkage, modinfop));
-}
 
-int
-_fini(void)
-{
-       return (mod_remove(&modlinkage));
-}
-#else
 typedef kern_return_t (*mach_call_t)(void *);
 
-/* XXX From #include <kern/syscall_sw.h> which may be changed for 64 bit! */
-typedef void    mach_munge_t(const void *, void *);
+/* APPLE NOTE: From #include <kern/syscall_sw.h> which may be changed for 64 bit! */
+typedef void    mach_munge_t(void *);
 
 typedef struct {
-        int                     mach_trap_arg_count;
-        int                     (*mach_trap_function)(void);
-#if defined(__i386__)
-        boolean_t               mach_trap_stack;
-#else
-        mach_munge_t            *mach_trap_arg_munge32; /* system call arguments for 32-bit */
-        mach_munge_t            *mach_trap_arg_munge64; /* system call arguments for 64-bit */
+       int                     mach_trap_arg_count;
+       kern_return_t           (*mach_trap_function)(void *);
+#if defined(__arm64__) || defined(__x86_64__)
+       mach_munge_t            *mach_trap_arg_munge32; /* system call arguments for 32-bit */
 #endif
-#if     !MACH_ASSERT
-        int                     mach_trap_unused;
-#else
-        const char*             mach_trap_name;
-#endif /* !MACH_ASSERT */
+       int                     mach_trap_u32_words;
+#if     MACH_ASSERT
+       const char*             mach_trap_name;
+#endif /* MACH_ASSERT */
 } mach_trap_t;
 
-#define MACH_TRAP_TABLE_COUNT   128
-
-extern mach_trap_t              mach_trap_table[];
-extern int                      mach_trap_count;
-
-#define MACH_TRAP(name, foo, bar, baz) #name
-
-/* XXX From osfmk/kern/syscall_sw.c */
-static const char *    mach_name_table[MACH_TRAP_TABLE_COUNT] = {
-/* 0 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 1 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 2 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 3 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 4 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 5 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 6 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 7 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 8 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 9 */                MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 10 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 11 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 12 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 13 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 14 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 15 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 16 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 17 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 18 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 19 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 20 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 21 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 22 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 23 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 24 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 25 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 26 */       MACH_TRAP(mach_reply_port, 0, NULL, NULL),
-/* 27 */       MACH_TRAP(thread_self_trap, 0, NULL, NULL),
-/* 28 */       MACH_TRAP(task_self_trap, 0, NULL, NULL),
-/* 29 */       MACH_TRAP(host_self_trap, 0, NULL, NULL),
-/* 30 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 31 */       MACH_TRAP(mach_msg_trap, 7, munge_wwwwwww, munge_ddddddd),
-/* 32 */       MACH_TRAP(mach_msg_overwrite_trap, 8, munge_wwwwwwww, munge_dddddddd),
-/* 33 */       MACH_TRAP(semaphore_signal_trap, 1, munge_w, munge_d),
-/* 34 */       MACH_TRAP(semaphore_signal_all_trap, 1, munge_w, munge_d),
-/* 35 */       MACH_TRAP(semaphore_signal_thread_trap, 2, munge_ww, munge_dd),
-/* 36 */       MACH_TRAP(semaphore_wait_trap, 1, munge_w, munge_d),
-/* 37 */       MACH_TRAP(semaphore_wait_signal_trap, 2, munge_ww, munge_dd),
-/* 38 */       MACH_TRAP(semaphore_timedwait_trap, 3, munge_www, munge_ddd),
-/* 39 */       MACH_TRAP(semaphore_timedwait_signal_trap, 4, munge_wwww, munge_dddd),
-/* 40 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 41 */       MACH_TRAP(init_process, 0, NULL, NULL),
-/* 42 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 43 */       MACH_TRAP(map_fd, 5, munge_wwwww, munge_ddddd),
-/* 44 */       MACH_TRAP(task_name_for_pid, 3, munge_www, munge_ddd),
-/* 45 */       MACH_TRAP(task_for_pid, 3, munge_www, munge_ddd),
-/* 46 */       MACH_TRAP(pid_for_task, 2, munge_ww,munge_dd),
-/* 47 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 48 */       MACH_TRAP(macx_swapon, 4, munge_wwww, munge_dddd),
-/* 49 */       MACH_TRAP(macx_swapoff, 2, munge_ww, munge_dd),
-/* 50 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 51 */       MACH_TRAP(macx_triggers, 4, munge_wwww, munge_dddd),
-/* 52 */       MACH_TRAP(macx_backing_store_suspend, 1, munge_w, munge_d),
-/* 53 */       MACH_TRAP(macx_backing_store_recovery, 1, munge_w, munge_d),
-/* 54 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 55 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 56 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 57 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 58 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 59 */       MACH_TRAP(swtch_pri, 0, NULL, NULL),
-/* 60 */       MACH_TRAP(swtch, 0, NULL, NULL),
-/* 61 */       MACH_TRAP(thread_switch, 3, munge_www, munge_ddd),
-/* 62 */       MACH_TRAP(clock_sleep_trap, 5, munge_wwwww, munge_ddddd),
-/* 63 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* traps 64 - 95 reserved (debo) */
-/* 64 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 65 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 66 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 67 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 68 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 69 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 70 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 71 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 72 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 73 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 74 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 75 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 76 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 77 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 78 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 79 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 80 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 81 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 82 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 83 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 84 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 85 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 86 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 87 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 88 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 89 */       MACH_TRAP(mach_timebase_info_trap, 1, munge_w, munge_d),
-/* 90 */       MACH_TRAP(mach_wait_until_trap, 2, munge_l, munge_d),
-/* 91 */       MACH_TRAP(mk_timer_create_trap, 0, NULL, NULL),
-/* 92 */       MACH_TRAP(mk_timer_destroy_trap, 1, munge_w, munge_d),
-/* 93 */       MACH_TRAP(mk_timer_arm_trap, 3, munge_wl, munge_dd),
-/* 94 */       MACH_TRAP(mk_timer_cancel_trap, 2, munge_ww, munge_dd),         
-/* 95 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),         
-/* traps 64 - 95 reserved (debo) */
-/* 96 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 97 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 98 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 99 */       MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* traps 100-107 reserved for iokit (esb) */ 
-/* 100 */      MACH_TRAP(iokit_user_client_trap, 8, munge_wwwwwwww, munge_dddddddd),
-/* 101 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 102 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 103 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 104 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 105 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 106 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 107 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* traps 108-127 unused */                     
-/* 108 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 109 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 110 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 111 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 112 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 113 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 114 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 115 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 116 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 117 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 118 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 119 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 120 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 121 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 122 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 123 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 124 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 125 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 126 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),
-/* 127 */      MACH_TRAP(kern_invalid, 0, NULL, NULL),                 
-};
+extern const mach_trap_t mach_trap_table[]; /* syscall_sw.h now declares this as const */
+extern const int         mach_trap_count;
+
+extern const char *const mach_syscall_name_table[];
 
 /* XXX From osfmk/i386/bsd_i386.c */
 struct mach_call_args {
-        syscall_arg_t arg1;
-        syscall_arg_t arg2;
-        syscall_arg_t arg3;
-        syscall_arg_t arg4;
-        syscall_arg_t arg5;
-        syscall_arg_t arg6;
-        syscall_arg_t arg7;
-        syscall_arg_t arg8;
-        syscall_arg_t arg9;
+       syscall_arg_t arg1;
+       syscall_arg_t arg2;
+       syscall_arg_t arg3;
+       syscall_arg_t arg4;
+       syscall_arg_t arg5;
+       syscall_arg_t arg6;
+       syscall_arg_t arg7;
+       syscall_arg_t arg8;
+       syscall_arg_t arg9;
 };
 
 #undef NSYSCALL
@@ -814,21 +641,26 @@ struct mach_call_args {
 #error 1 << SYSTRACE_SHIFT must exceed number of Mach traps
 #endif
 
-typedef systrace_sysent_t machtrace_sysent_t;
+typedef struct machtrace_sysent {
+       dtrace_id_t     stsy_entry;
+       dtrace_id_t     stsy_return;
+       kern_return_t   (*stsy_underlying)(void *);
+       int32_t         stsy_return_type;
+} machtrace_sysent_t;
 
 static machtrace_sysent_t *machtrace_sysent = NULL;
 
 void (*machtrace_probe)(dtrace_id_t, uint64_t, uint64_t,
     uint64_t, uint64_t, uint64_t);
 
-static dev_info_t *machtrace_devi;
+static uint64_t machtrace_getarg(void *, dtrace_id_t, void *, int, int);
+
 static dtrace_provider_id_t machtrace_id;
 
 static kern_return_t
 dtrace_machtrace_syscall(struct mach_call_args *args)
 {
-       boolean_t           flavor;
-       unsigned short      code;
+       int code;       /* The mach call number */
 
        machtrace_sysent_t *sy;
        dtrace_id_t id;
@@ -839,20 +671,9 @@ dtrace_machtrace_syscall(struct mach_call_args *args)
        syscall_arg_t *ip = (syscall_arg_t *)args;
        mach_call_t mach_call;
 
-#if defined (__ppc__) || defined (__ppc64__)
-       {
-               savearea_t *regs = (savearea_t *)find_user_regs(current_thread());
-
-               flavor = (((unsigned int)regs->save_r0) == 0)? 1: 0;
-
-               if (flavor)
-                       code = -regs->save_r3;
-               else
-                       code = -regs->save_r0;
-       }
-#elif defined(__i386__) || defined (__x86_64__)
-#pragma unused(flavor)
+#if defined (__x86_64__)
        {
+               pal_register_cache_state(current_thread(), VALID);
                x86_saved_state_t   *tagged_regs = (x86_saved_state_t *)find_user_regs(current_thread());
 
                if (is_saved_state64(tagged_regs)) {
@@ -861,17 +682,51 @@ dtrace_machtrace_syscall(struct mach_call_args *args)
                        code = -saved_state32(tagged_regs)->eax;
                }
        }
+#elif defined(__arm__)
+       {
+               /* r12 has the machcall number, but it is -ve */
+               arm_saved_state_t *arm_regs = (arm_saved_state_t *) find_user_regs(current_thread());
+               code = (int)arm_regs->r[12];
+               ASSERT(code < 0);    /* Otherwise it would be a Unix syscall */
+               code = -code;
+       }
+#elif defined(__arm64__)
+       {
+               /* From arm/thread_status.h:get_saved_state_svc_number */
+               arm_saved_state_t *arm_regs = (arm_saved_state_t *) find_user_regs(current_thread());
+               if (is_saved_state32(arm_regs)) {
+                       code = (int)saved_state32(arm_regs)->r[12];
+               } else {
+                       code = (int)saved_state64(arm_regs)->x[ARM64_SYSCALL_CODE_REG_NUM];
+               }
+
+               /* From bsd/arm64.c:mach_syscall */
+               ASSERT(code < 0);    /* Otherwise it would be a Unix syscall */
+               code = -code;
+       }
 #else
 #error Unknown Architecture
 #endif
 
        sy = &machtrace_sysent[code];
 
-       if ((id = sy->stsy_entry) != DTRACE_IDNONE)
-               (*machtrace_probe)(id, *ip, *(ip+1), *(ip+2), *(ip+3), *(ip+4));
+       if ((id = sy->stsy_entry) != DTRACE_IDNONE) {
+               uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+
+               if (uthread) {
+                       uthread->t_dtrace_syscall_args = (void *)ip;
+               }
+
+               (*machtrace_probe)(id, *ip, *(ip + 1), *(ip + 2), *(ip + 3), *(ip + 4));
+
+               if (uthread) {
+                       uthread->t_dtrace_syscall_args = (void *)0;
+               }
+       }
 
 #if 0 /* XXX */
        /*
+        * APPLE NOTE:  Not implemented.
         * We want to explicitly allow DTrace consumers to stop a process
         * before it actually executes the meat of the syscall.
         */
@@ -887,32 +742,35 @@ dtrace_machtrace_syscall(struct mach_call_args *args)
        mach_call = (mach_call_t)(*sy->stsy_underlying);
        rval = mach_call(args);
 
-       if ((id = sy->stsy_return) != DTRACE_IDNONE)
+       if ((id = sy->stsy_return) != DTRACE_IDNONE) {
                (*machtrace_probe)(id, (uint64_t)rval, 0, 0, 0, 0);
+       }
 
-       return (rval);
+       return rval;
 }
 
 static void
-machtrace_init(mach_trap_t *actual, machtrace_sysent_t **interposed)
+machtrace_init(const mach_trap_t *actual, machtrace_sysent_t **interposed)
 {
        machtrace_sysent_t *msysent = *interposed;
        int i;
 
        if (msysent == NULL) {
-               *interposed = msysent = kmem_zalloc(sizeof (machtrace_sysent_t) *
-                               NSYSCALL, KM_SLEEP);
+               *interposed = msysent = kmem_zalloc(sizeof(machtrace_sysent_t) *
+                   NSYSCALL, KM_SLEEP);
        }
 
        for (i = 0; i < NSYSCALL; i++) {
-               mach_trap_t *a = &actual[i];
+               const mach_trap_t *a = &actual[i];
                machtrace_sysent_t *s = &msysent[i];
 
-               if (LOADABLE_SYSCALL(a) && !LOADED_SYSCALL(a))
+               if (LOADABLE_SYSCALL(a) && !LOADED_SYSCALL(a)) {
                        continue;
+               }
 
-               if ((mach_call_t)(a->mach_trap_function) == (mach_call_t)(dtrace_machtrace_syscall))
+               if (a->mach_trap_function == (mach_call_t)(dtrace_machtrace_syscall)) {
                        continue;
+               }
 
                s->stsy_underlying = a->mach_trap_function;
        }
@@ -922,28 +780,32 @@ machtrace_init(mach_trap_t *actual, machtrace_sysent_t **interposed)
 static void
 machtrace_provide(void *arg, const dtrace_probedesc_t *desc)
 {
+#pragma unused(arg) /* __APPLE__ */
+
        int i;
 
-       if (desc != NULL)
+       if (desc != NULL) {
                return;
+       }
 
        machtrace_init(mach_trap_table, &machtrace_sysent);
 
        for (i = 0; i < NSYSCALL; i++) {
-               
-               if (machtrace_sysent[i].stsy_underlying == NULL)
+               if (machtrace_sysent[i].stsy_underlying == NULL) {
                        continue;
+               }
 
                if (dtrace_probe_lookup(machtrace_id, NULL,
-                                       mach_name_table[i], "entry") != 0)
+                   mach_syscall_name_table[i], "entry") != 0) {
                        continue;
+               }
 
-               (void) dtrace_probe_create(machtrace_id, NULL, mach_name_table[i],
-                                          "entry", MACHTRACE_ARTIFICIAL_FRAMES,
-                                          (void *)((uintptr_t)SYSTRACE_ENTRY(i)));
-               (void) dtrace_probe_create(machtrace_id, NULL, mach_name_table[i],
-                                          "return", MACHTRACE_ARTIFICIAL_FRAMES,
-                                          (void *)((uintptr_t)SYSTRACE_RETURN(i)));
+               (void) dtrace_probe_create(machtrace_id, NULL, mach_syscall_name_table[i],
+                   "entry", MACHTRACE_ARTIFICIAL_FRAMES,
+                   (void *)((uintptr_t)SYSTRACE_ENTRY(i)));
+               (void) dtrace_probe_create(machtrace_id, NULL, mach_syscall_name_table[i],
+                   "return", MACHTRACE_ARTIFICIAL_FRAMES,
+                   (void *)((uintptr_t)SYSTRACE_RETURN(i)));
 
                machtrace_sysent[i].stsy_entry = DTRACE_IDNONE;
                machtrace_sysent[i].stsy_return = DTRACE_IDNONE;
@@ -954,8 +816,11 @@ machtrace_provide(void *arg, const dtrace_probedesc_t *desc)
 static void
 machtrace_destroy(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg,id) /* __APPLE__ */
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
 
+#pragma unused(sysnum) /* __APPLE__ */
+
        /*
         * There's nothing to do here but assert that we have actually been
         * disabled.
@@ -968,12 +833,14 @@ machtrace_destroy(void *arg, dtrace_id_t id, void *parg)
 }
 
 /*ARGSUSED*/
-static void
+static int
 machtrace_enable(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg) /* __APPLE__ */
+
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
        int enabled = (machtrace_sysent[sysnum].stsy_entry != DTRACE_IDNONE ||
-                       machtrace_sysent[sysnum].stsy_return != DTRACE_IDNONE);
+           machtrace_sysent[sysnum].stsy_return != DTRACE_IDNONE);
 
        if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
                machtrace_sysent[sysnum].stsy_entry = id;
@@ -982,28 +849,50 @@ machtrace_enable(void *arg, dtrace_id_t id, void *parg)
        }
 
        if (enabled) {
-               ASSERT(sysent[sysnum].sy_callc == dtrace_machtrace_syscall);
-               return;
+               ASSERT(mach_trap_table[sysnum].mach_trap_function == (void *)dtrace_machtrace_syscall);
+               return 0;
        }
 
-       (void) casptr(&mach_trap_table[sysnum].mach_trap_function,
-                     (void *)machtrace_sysent[sysnum].stsy_underlying,
-                     (void *)dtrace_machtrace_syscall);
+       lck_mtx_lock(&dtrace_systrace_lock);
+
+       if (mach_trap_table[sysnum].mach_trap_function == machtrace_sysent[sysnum].stsy_underlying) {
+               /* It is not possible to write to mach_trap_table[] directly because it is const. */
+               vm_offset_t dss = ptrauth_nop_cast(vm_offset_t, &dtrace_machtrace_syscall);
+               ml_nofault_copy((vm_offset_t)&dss, (vm_offset_t)&mach_trap_table[sysnum].mach_trap_function, sizeof(vm_offset_t));
+       }
+
+       lck_mtx_unlock(&dtrace_systrace_lock);
+
+       return 0;
 }
 
 /*ARGSUSED*/
 static void
 machtrace_disable(void *arg, dtrace_id_t id, void *parg)
 {
+#pragma unused(arg,id) /* __APPLE__ */
+
        int sysnum = SYSTRACE_SYSNUM((uintptr_t)parg);
        int disable = (machtrace_sysent[sysnum].stsy_entry == DTRACE_IDNONE ||
-                       machtrace_sysent[sysnum].stsy_return == DTRACE_IDNONE);
+           machtrace_sysent[sysnum].stsy_return == DTRACE_IDNONE);
 
        if (disable) {
-               (void) casptr(&mach_trap_table[sysnum].mach_trap_function,
-                             (void *)dtrace_machtrace_syscall,
-                             (void *)machtrace_sysent[sysnum].stsy_underlying);
-
+               /*
+                * Usage of volatile protects the if statement below from being optimized away.
+                *
+                * Compilers are clever and know that const array values can't change in time
+                * and the if below is always false. That is because it can't see that DTrace
+                * injects dtrace_machtrace_syscall dynamically and violates constness of the
+                * array.
+                */
+               volatile const mach_trap_t *machtrap = &mach_trap_table[sysnum];
+
+               lck_mtx_lock(&dtrace_systrace_lock);
+               if (machtrap->mach_trap_function == (mach_call_t)dtrace_machtrace_syscall) {
+                       ml_nofault_copy((vm_offset_t)&machtrace_sysent[sysnum].stsy_underlying,
+                           (vm_offset_t)&machtrap->mach_trap_function, sizeof(vm_offset_t));
+               }
+               lck_mtx_unlock(&dtrace_systrace_lock);
        }
 
        if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
@@ -1014,59 +903,48 @@ machtrace_disable(void *arg, dtrace_id_t id, void *parg)
 }
 
 static dtrace_pattr_t machtrace_attr = {
-{ DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
-{ DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
-{ DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
+       { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
+       { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_COMMON },
+       { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
 };
 
 static dtrace_pops_t machtrace_pops = {
-       machtrace_provide,
-       NULL,
-       machtrace_enable,
-       machtrace_disable,
-       NULL,
-       NULL,
-       NULL,
-       NULL,
-       NULL,
-       machtrace_destroy
+       .dtps_provide =         machtrace_provide,
+       .dtps_provide_module =  NULL,
+       .dtps_enable =          machtrace_enable,
+       .dtps_disable =         machtrace_disable,
+       .dtps_suspend =         NULL,
+       .dtps_resume =          NULL,
+       .dtps_getargdesc =      NULL,
+       .dtps_getargval =       machtrace_getarg,
+       .dtps_usermode =        NULL,
+       .dtps_destroy =         machtrace_destroy
 };
 
 static int
-machtrace_attach(dev_info_t *devi, ddi_attach_cmd_t cmd)
+machtrace_attach(dev_info_t *devi)
 {
-       switch (cmd) {
-               case DDI_ATTACH:
-                       break;
-               case DDI_RESUME:
-                       return (DDI_SUCCESS);
-               default:
-                       return (DDI_FAILURE);
-       }
-
        machtrace_probe = dtrace_probe;
        membar_enter();
 
        if (ddi_create_minor_node(devi, "machtrace", S_IFCHR, 0,
-                               DDI_PSEUDO, NULL) == DDI_FAILURE ||
-                       dtrace_register("mach_trap", &machtrace_attr, DTRACE_PRIV_USER, NULL,
-                               &machtrace_pops, NULL, &machtrace_id) != 0) {
-               machtrace_probe = systrace_stub;
+           DDI_PSEUDO, 0) == DDI_FAILURE ||
+           dtrace_register("mach_trap", &machtrace_attr, DTRACE_PRIV_USER, NULL,
+           &machtrace_pops, NULL, &machtrace_id) != 0) {
+               machtrace_probe = (void*)&systrace_stub;
                ddi_remove_minor_node(devi, NULL);
-               return (DDI_FAILURE);
+               return DDI_FAILURE;
        }
 
-       ddi_report_dev(devi);
-       machtrace_devi = devi;
-
-       return (DDI_SUCCESS);
+       return DDI_SUCCESS;
 }
 
 d_open_t _systrace_open;
 
-int _systrace_open(dev_t dev, int flags, int devtype, struct proc *p)
+int
+_systrace_open(dev_t dev, int flags, int devtype, struct proc *p)
 {
 #pragma unused(dev,flags,devtype,p)
        return 0;
@@ -1074,49 +952,117 @@ int _systrace_open(dev_t dev, int flags, int devtype, struct proc *p)
 
 #define SYSTRACE_MAJOR  -24 /* let the kernel pick the device number */
 
-/*
- * A struct describing which functions will get invoked for certain
- * actions.
- */
 static struct cdevsw systrace_cdevsw =
 {
-       _systrace_open,         /* open */
-       eno_opcl,               /* close */
-       eno_rdwrt,                      /* read */
-       eno_rdwrt,                      /* write */
-       eno_ioctl,              /* ioctl */
-       (stop_fcn_t *)nulldev, /* stop */
-       (reset_fcn_t *)nulldev, /* reset */
-       NULL,                           /* tty's */
-       eno_select,                     /* select */
-       eno_mmap,                       /* mmap */
-       eno_strat,                      /* strategy */
-       eno_getc,                       /* getc */
-       eno_putc,                       /* putc */
-       0                                       /* type */
+       .d_open = _systrace_open,
+       .d_close = eno_opcl,
+       .d_read = eno_rdwrt,
+       .d_write = eno_rdwrt,
+       .d_ioctl = eno_ioctl,
+       .d_stop = (stop_fcn_t *)nulldev,
+       .d_reset = (reset_fcn_t *)nulldev,
+       .d_select = eno_select,
+       .d_mmap = eno_mmap,
+       .d_strategy = eno_strat,
+       .d_reserved_1 = eno_getc,
+       .d_reserved_2 = eno_putc,
 };
 
-static int gSysTraceInited = 0;
-
 void systrace_init( void );
 
-void systrace_init( void )
+void
+systrace_init( void )
 {
-       if (0 == gSysTraceInited) {
-               int majdevno = cdevsw_add(SYSTRACE_MAJOR, &systrace_cdevsw);
+       if (dtrace_sdt_probes_restricted()) {
+               return;
+       }
 
-               if (majdevno < 0) {
-                       printf("systrace_init: failed to allocate a major number!\n");
-                       gSysTraceInited = 0;
-                       return;
-               }
+       int majdevno = cdevsw_add(SYSTRACE_MAJOR, &systrace_cdevsw);
 
-               systrace_attach( (dev_info_t    *)majdevno, DDI_ATTACH );
-               machtrace_attach( (dev_info_t   *)majdevno, DDI_ATTACH );
+       if (majdevno < 0) {
+               printf("systrace_init: failed to allocate a major number!\n");
+               return;
+       }
 
-               gSysTraceInited = 1;
-       } else
-               panic("systrace_init: called twice!\n");
+       systrace_attach((dev_info_t*)(uintptr_t)majdevno);
+       machtrace_attach((dev_info_t*)(uintptr_t)majdevno);
 }
 #undef SYSTRACE_MAJOR
-#endif /* __APPLE__ */
+
+static uint64_t
+systrace_getargval(void *arg, dtrace_id_t id, void *parg, int argno, int aframes)
+{
+#pragma unused(arg,id,parg,aframes)     /* __APPLE__ */
+       uint64_t val = 0;
+       uint64_t *uargs = NULL;
+
+       uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+
+       if (uthread) {
+               uargs = uthread->t_dtrace_syscall_args;
+       }
+       if (!uargs) {
+               return 0;
+       }
+       if (argno < 0 || argno >= SYSTRACE_NARGS) {
+               return 0;
+       }
+
+       DTRACE_CPUFLAG_SET(CPU_DTRACE_NOFAULT);
+       val = uargs[argno];
+       DTRACE_CPUFLAG_CLEAR(CPU_DTRACE_NOFAULT);
+       return val;
+}
+
+static void
+systrace_getargdesc(void *arg, dtrace_id_t id, void *parg,
+    dtrace_argdesc_t *desc)
+{
+#pragma unused(arg, id)
+       int sysnum = SYSTRACE_SYSNUM(parg);
+       uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+       uint64_t *uargs = NULL;
+
+       if (!uthread) {
+               desc->dtargd_ndx = DTRACE_ARGNONE;
+               return;
+       }
+
+       uargs = uthread->t_dtrace_syscall_args;
+
+       if (SYSTRACE_ISENTRY((uintptr_t)parg)) {
+               systrace_entry_setargdesc(sysnum, desc->dtargd_ndx,
+                   desc->dtargd_native, sizeof(desc->dtargd_native));
+       } else {
+               systrace_return_setargdesc(sysnum, desc->dtargd_ndx,
+                   desc->dtargd_native, sizeof(desc->dtargd_native));
+       }
+
+       if (desc->dtargd_native[0] == '\0') {
+               desc->dtargd_ndx = DTRACE_ARGNONE;
+       }
+}
+
+static uint64_t
+machtrace_getarg(void *arg, dtrace_id_t id, void *parg, int argno, int aframes)
+{
+#pragma unused(arg,id,parg,aframes)     /* __APPLE__ */
+       uint64_t val = 0;
+       syscall_arg_t *stack = (syscall_arg_t *)NULL;
+
+       uthread_t uthread = (uthread_t)get_bsdthread_info(current_thread());
+
+       if (uthread) {
+               stack = (syscall_arg_t *)uthread->t_dtrace_syscall_args;
+       }
+
+       if (!stack) {
+               return 0;
+       }
+
+       DTRACE_CPUFLAG_SET(CPU_DTRACE_NOFAULT);
+       /* dtrace_probe arguments arg0 .. arg4 are 64bits wide */
+       val = (uint64_t)*(stack + argno);
+       DTRACE_CPUFLAG_CLEAR(CPU_DTRACE_NOFAULT);
+       return val;
+}