| 
     1 /*-
    2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
    3  *
    4  * Copyright (c) 2017 Edward Tomasz Napierala <trasz@FreeBSD.org>
    5  *
    6  * This software was developed by SRI International and the University of
    7  * Cambridge Computer Laboratory under DARPA/AFRL contract (FA8750-10-C-0237)
    8  * ("CTSRD"), as part of the DARPA CRASH research programme.
    9  *
   10  * Redistribution and use in source and binary forms, with or without
   11  * modification, are permitted provided that the following conditions
   12  * are met:
   13  * 1. Redistributions of source code must retain the above copyright
   14  *    notice, this list of conditions and the following disclaimer.
   15  * 2. Redistributions in binary form must reproduce the above copyright
   16  *    notice, this list of conditions and the following disclaimer in the
   17  *    documentation and/or other materials provided with the distribution.
   18  *
   19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
   20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
   21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
   22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
   23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
   24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
   25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
   26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
   27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
   28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
   29  * SUCH DAMAGE.
   30  */
   31 
   32 #include <sys/cdefs.h>
   33 __FBSDID("$FreeBSD$");
   34 
   35 #include <sys/param.h>
   36 #include <sys/lock.h>
   37 #include <sys/proc.h>
   38 #include <sys/ptrace.h>
   39 #include <sys/sx.h>
   40 #include <sys/syscallsubr.h>
   41 
   42 #include <machine/../linux/linux.h>
   43 #include <machine/../linux/linux_proto.h>
   44 #include <compat/linux/linux_emul.h>
   45 #include <compat/linux/linux_errno.h>
   46 #include <compat/linux/linux_misc.h>
   47 #include <compat/linux/linux_signal.h>
   48 #include <compat/linux/linux_util.h>
   49 
   50 #define LINUX_PTRACE_TRACEME            0
   51 #define LINUX_PTRACE_PEEKTEXT           1
   52 #define LINUX_PTRACE_PEEKDATA           2
   53 #define LINUX_PTRACE_PEEKUSER           3
   54 #define LINUX_PTRACE_POKETEXT           4
   55 #define LINUX_PTRACE_POKEDATA           5
   56 #define LINUX_PTRACE_POKEUSER           6
   57 #define LINUX_PTRACE_CONT               7
   58 #define LINUX_PTRACE_KILL               8
   59 #define LINUX_PTRACE_SINGLESTEP         9
   60 #define LINUX_PTRACE_GETREGS            12
   61 #define LINUX_PTRACE_SETREGS            13
   62 #define LINUX_PTRACE_GETFPREGS          14
   63 #define LINUX_PTRACE_SETFPREGS          15
   64 #define LINUX_PTRACE_ATTACH             16
   65 #define LINUX_PTRACE_DETACH             17
   66 #define LINUX_PTRACE_SYSCALL            24
   67 #define LINUX_PTRACE_SETOPTIONS         0x4200
   68 #define LINUX_PTRACE_GETEVENTMSG        0x4201
   69 #define LINUX_PTRACE_GETSIGINFO         0x4202
   70 #define LINUX_PTRACE_GETREGSET          0x4204
   71 #define LINUX_PTRACE_SEIZE              0x4206
   72 #define LINUX_PTRACE_GET_SYSCALL_INFO   0x420e
   73 
   74 #define LINUX_PTRACE_EVENT_EXEC         4
   75 #define LINUX_PTRACE_EVENT_EXIT         6
   76 
   77 #define LINUX_PTRACE_O_TRACESYSGOOD     1
   78 #define LINUX_PTRACE_O_TRACEFORK        2
   79 #define LINUX_PTRACE_O_TRACEVFORK       4
   80 #define LINUX_PTRACE_O_TRACECLONE       8
   81 #define LINUX_PTRACE_O_TRACEEXEC        16
   82 #define LINUX_PTRACE_O_TRACEVFORKDONE   32
   83 #define LINUX_PTRACE_O_TRACEEXIT        64
   84 #define LINUX_PTRACE_O_TRACESECCOMP     128
   85 #define LINUX_PTRACE_O_EXITKILL         1048576
   86 #define LINUX_PTRACE_O_SUSPEND_SECCOMP  2097152
   87 
   88 #define LINUX_NT_PRSTATUS               0x1
   89 #define LINUX_NT_PRFPREG                0x2
   90 #define LINUX_NT_X86_XSTATE             0x202
   91 
   92 #define LINUX_PTRACE_O_MASK     (LINUX_PTRACE_O_TRACESYSGOOD |  \
   93     LINUX_PTRACE_O_TRACEFORK | LINUX_PTRACE_O_TRACEVFORK |      \
   94     LINUX_PTRACE_O_TRACECLONE | LINUX_PTRACE_O_TRACEEXEC |      \
   95     LINUX_PTRACE_O_TRACEVFORKDONE | LINUX_PTRACE_O_TRACEEXIT |  \
   96     LINUX_PTRACE_O_TRACESECCOMP | LINUX_PTRACE_O_EXITKILL |     \
   97     LINUX_PTRACE_O_SUSPEND_SECCOMP)
   98 
   99 #define LINUX_PTRACE_SYSCALL_INFO_NONE  0
  100 #define LINUX_PTRACE_SYSCALL_INFO_ENTRY 1
  101 #define LINUX_PTRACE_SYSCALL_INFO_EXIT  2
  102 
  103 #define LINUX_PTRACE_PEEKUSER_ORIG_RAX  120
  104 #define LINUX_PTRACE_PEEKUSER_RIP       128
  105 #define LINUX_PTRACE_PEEKUSER_CS        136
  106 #define LINUX_PTRACE_PEEKUSER_DS        184
  107 
  108 static int
  109 map_signum(int lsig, int *bsigp)
  110 {
  111         int bsig;
  112 
  113         if (lsig == 0) {
  114                 *bsigp = 0;
  115                 return (0);
  116         }
  117 
  118         if (lsig < 0 || lsig > LINUX_SIGRTMAX)
  119                 return (EINVAL);
  120 
  121         bsig = linux_to_bsd_signal(lsig);
  122         if (bsig == SIGSTOP)
  123                 bsig = 0;
  124 
  125         *bsigp = bsig;
  126         return (0);
  127 }
  128 
  129 int
  130 linux_ptrace_status(struct thread *td, pid_t pid, int status)
  131 {
  132         struct ptrace_lwpinfo lwpinfo;
  133         struct linux_pemuldata *pem;
  134         register_t saved_retval;
  135         int error;
  136 
  137         saved_retval = td->td_retval[0];
  138         error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
  139         td->td_retval[0] = saved_retval;
  140         if (error != 0) {
  141                 linux_msg(td, "PT_LWPINFO failed with error %d", error);
  142                 return (status);
  143         }
  144 
  145         pem = pem_find(td->td_proc);
  146         KASSERT(pem != NULL, ("%s: proc emuldata not found.\n", __func__));
  147 
  148         LINUX_PEM_SLOCK(pem);
  149         if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACESYSGOOD) &&
  150             lwpinfo.pl_flags & PL_FLAG_SCE)
  151                 status |= (LINUX_SIGTRAP | 0x80) << 8;
  152         if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACESYSGOOD) &&
  153             lwpinfo.pl_flags & PL_FLAG_SCX) {
  154                 if (lwpinfo.pl_flags & PL_FLAG_EXEC)
  155                         status |= (LINUX_SIGTRAP | LINUX_PTRACE_EVENT_EXEC << 8) << 8;
  156                 else
  157                         status |= (LINUX_SIGTRAP | 0x80) << 8;
  158         }
  159         if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACEEXIT) &&
  160             lwpinfo.pl_flags & PL_FLAG_EXITED)
  161                 status |= (LINUX_SIGTRAP | LINUX_PTRACE_EVENT_EXIT << 8) << 8;
  162         LINUX_PEM_SUNLOCK(pem);
  163 
  164         return (status);
  165 }
  166 
  167 static int
  168 linux_ptrace_peek(struct thread *td, pid_t pid, void *addr, void *data)
  169 {
  170         int error;
  171 
  172         error = kern_ptrace(td, PT_READ_I, pid, addr, 0);
  173         if (error == 0)
  174                 error = copyout(td->td_retval, data, sizeof(l_int));
  175         else if (error == ENOMEM)
  176                 error = EIO;
  177         td->td_retval[0] = error;
  178 
  179         return (error);
  180 }
  181 
  182 static int
  183 linux_ptrace_peekuser(struct thread *td, pid_t pid, void *addr, void *data)
  184 {
  185         struct reg b_reg;
  186         uint64_t val;
  187         int error;
  188 
  189         error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
  190         if (error != 0)
  191                 return (error);
  192 
  193         switch ((uintptr_t)addr) {
  194 #ifdef __amd64__
  195         case LINUX_PTRACE_PEEKUSER_ORIG_RAX:
  196                 val = b_reg.r_rax;
  197                 break;
  198         case LINUX_PTRACE_PEEKUSER_RIP:
  199                 val = b_reg.r_rip;
  200                 break;
  201         case LINUX_PTRACE_PEEKUSER_CS:
  202                 val = b_reg.r_cs;
  203                 break;
  204         case LINUX_PTRACE_PEEKUSER_DS:
  205                 val = b_reg.r_ds;
  206                 break;
  207 #endif /* __amd64__ */
  208         default:
  209                 linux_msg(td, "PTRACE_PEEKUSER offset %ld not implemented; "
  210                     "returning EINVAL", (uintptr_t)addr);
  211                 return (EINVAL);
  212         }
  213 
  214         error = copyout(&val, data, sizeof(val));
  215         td->td_retval[0] = error;
  216 
  217         return (error);
  218 }
  219 
  220 static int
  221 linux_ptrace_pokeuser(struct thread *td, pid_t pid, void *addr, void *data)
  222 {
  223 
  224         linux_msg(td, "PTRACE_POKEUSER not implemented; returning EINVAL");
  225         return (EINVAL);
  226 }
  227 
  228 static int
  229 linux_ptrace_setoptions(struct thread *td, pid_t pid, l_ulong data)
  230 {
  231         struct linux_pemuldata *pem;
  232         int mask;
  233 
  234         mask = 0;
  235 
  236         if (data & ~LINUX_PTRACE_O_MASK) {
  237                 linux_msg(td, "unknown ptrace option %lx set; "
  238                     "returning EINVAL",
  239                     data & ~LINUX_PTRACE_O_MASK);
  240                 return (EINVAL);
  241         }
  242 
  243         pem = pem_find(td->td_proc);
  244         KASSERT(pem != NULL, ("%s: proc emuldata not found.\n", __func__));
  245 
  246         /*
  247          * PTRACE_O_EXITKILL is ignored, we do that by default.
  248          */
  249 
  250         LINUX_PEM_XLOCK(pem);
  251         if (data & LINUX_PTRACE_O_TRACESYSGOOD) {
  252                 pem->ptrace_flags |= LINUX_PTRACE_O_TRACESYSGOOD;
  253         } else {
  254                 pem->ptrace_flags &= ~LINUX_PTRACE_O_TRACESYSGOOD;
  255         }
  256         LINUX_PEM_XUNLOCK(pem);
  257 
  258         if (data & LINUX_PTRACE_O_TRACEFORK)
  259                 mask |= PTRACE_FORK;
  260 
  261         if (data & LINUX_PTRACE_O_TRACEVFORK)
  262                 mask |= PTRACE_VFORK;
  263 
  264         if (data & LINUX_PTRACE_O_TRACECLONE)
  265                 mask |= PTRACE_VFORK;
  266 
  267         if (data & LINUX_PTRACE_O_TRACEEXEC)
  268                 mask |= PTRACE_EXEC;
  269 
  270         if (data & LINUX_PTRACE_O_TRACEVFORKDONE)
  271                 mask |= PTRACE_VFORK; /* XXX: Close enough? */
  272 
  273         if (data & LINUX_PTRACE_O_TRACEEXIT) {
  274                 pem->ptrace_flags |= LINUX_PTRACE_O_TRACEEXIT;
  275         } else {
  276                 pem->ptrace_flags &= ~LINUX_PTRACE_O_TRACEEXIT;
  277         }
  278 
  279         return (kern_ptrace(td, PT_SET_EVENT_MASK, pid, &mask, sizeof(mask)));
  280 }
  281 
  282 static int
  283 linux_ptrace_geteventmsg(struct thread *td, pid_t pid, l_ulong data)
  284 {
  285 
  286         linux_msg(td, "PTRACE_GETEVENTMSG not implemented; returning EINVAL");
  287         return (EINVAL);
  288 }
  289 
  290 static int
  291 linux_ptrace_getsiginfo(struct thread *td, pid_t pid, l_ulong data)
  292 {
  293         struct ptrace_lwpinfo lwpinfo;
  294         l_siginfo_t l_siginfo;
  295         int error, sig;
  296 
  297         error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
  298         if (error != 0) {
  299                 linux_msg(td, "PT_LWPINFO failed with error %d", error);
  300                 return (error);
  301         }
  302 
  303         if ((lwpinfo.pl_flags & PL_FLAG_SI) == 0) {
  304                 error = EINVAL;
  305                 linux_msg(td, "no PL_FLAG_SI, returning %d", error);
  306                 return (error);
  307         }
  308 
  309         sig = bsd_to_linux_signal(lwpinfo.pl_siginfo.si_signo);
  310         memset(&l_siginfo, 0, sizeof(l_siginfo));
  311         siginfo_to_lsiginfo(&lwpinfo.pl_siginfo, &l_siginfo, sig);
  312         error = copyout(&l_siginfo, (void *)data, sizeof(l_siginfo));
  313         return (error);
  314 }
  315 
  316 static int
  317 linux_ptrace_getregs(struct thread *td, pid_t pid, void *data)
  318 {
  319         struct reg b_reg;
  320         struct linux_pt_regset l_regset;
  321         int error;
  322 
  323         error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
  324         if (error != 0)
  325                 return (error);
  326 
  327         bsd_to_linux_regset(&b_reg, &l_regset);
  328         error = linux_ptrace_getregs_machdep(td, pid, &l_regset);
  329         if (error != 0)
  330                 return (error);
  331 
  332         error = copyout(&l_regset, (void *)data, sizeof(l_regset));
  333         return (error);
  334 }
  335 
  336 static int
  337 linux_ptrace_setregs(struct thread *td, pid_t pid, void *data)
  338 {
  339         struct reg b_reg;
  340         struct linux_pt_regset l_regset;
  341         int error;
  342 
  343         error = copyin(data, &l_regset, sizeof(l_regset));
  344         if (error != 0)
  345                 return (error);
  346         linux_to_bsd_regset(&b_reg, &l_regset);
  347         error = kern_ptrace(td, PT_SETREGS, pid, &b_reg, 0);
  348         return (error);
  349 }
  350 
  351 static int
  352 linux_ptrace_getregset_prstatus(struct thread *td, pid_t pid, l_ulong data)
  353 {
  354         struct reg b_reg;
  355         struct linux_pt_regset l_regset;
  356         struct iovec iov;
  357         size_t len;
  358         int error;
  359 
  360         error = copyin((const void *)data, &iov, sizeof(iov));
  361         if (error != 0) {
  362                 linux_msg(td, "copyin error %d", error);
  363                 return (error);
  364         }
  365 
  366         error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
  367         if (error != 0)
  368                 return (error);
  369 
  370         bsd_to_linux_regset(&b_reg, &l_regset);
  371         error = linux_ptrace_getregs_machdep(td, pid, &l_regset);
  372         if (error != 0)
  373                 return (error);
  374 
  375         len = MIN(iov.iov_len, sizeof(l_regset));
  376         error = copyout(&l_regset, (void *)iov.iov_base, len);
  377         if (error != 0) {
  378                 linux_msg(td, "copyout error %d", error);
  379                 return (error);
  380         }
  381 
  382         iov.iov_len = len;
  383         error = copyout(&iov, (void *)data, sizeof(iov));
  384         if (error != 0) {
  385                 linux_msg(td, "iov copyout error %d", error);
  386                 return (error);
  387         }
  388 
  389         return (error);
  390 }
  391 
  392 static int
  393 linux_ptrace_getregset(struct thread *td, pid_t pid, l_ulong addr, l_ulong data)
  394 {
  395 
  396         switch (addr) {
  397         case LINUX_NT_PRSTATUS:
  398                 return (linux_ptrace_getregset_prstatus(td, pid, data));
  399         case LINUX_NT_PRFPREG:
  400                 linux_msg(td, "PTRAGE_GETREGSET NT_PRFPREG not implemented; "
  401                     "returning EINVAL");
  402                 return (EINVAL);
  403         case LINUX_NT_X86_XSTATE:
  404                 linux_msg(td, "PTRAGE_GETREGSET NT_X86_XSTATE not implemented; "
  405                     "returning EINVAL");
  406                 return (EINVAL);
  407         default:
  408                 linux_msg(td, "PTRACE_GETREGSET request %#lx not implemented; "
  409                     "returning EINVAL", addr);
  410                 return (EINVAL);
  411         }
  412 }
  413 
  414 static int
  415 linux_ptrace_seize(struct thread *td, pid_t pid, l_ulong addr, l_ulong data)
  416 {
  417 
  418         linux_msg(td, "PTRACE_SEIZE not implemented; returning EINVAL");
  419         return (EINVAL);
  420 }
  421 
  422 static int
  423 linux_ptrace_get_syscall_info(struct thread *td, pid_t pid,
  424     l_ulong len, l_ulong data)
  425 {
  426         struct ptrace_lwpinfo lwpinfo;
  427         struct ptrace_sc_ret sr;
  428         struct reg b_reg;
  429         struct syscall_info si;
  430         int error;
  431 
  432         error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
  433         if (error != 0) {
  434                 linux_msg(td, "PT_LWPINFO failed with error %d", error);
  435                 return (error);
  436         }
  437 
  438         memset(&si, 0, sizeof(si));
  439 
  440         if (lwpinfo.pl_flags & PL_FLAG_SCE) {
  441                 si.op = LINUX_PTRACE_SYSCALL_INFO_ENTRY;
  442                 si.entry.nr = lwpinfo.pl_syscall_code;
  443                 /*
  444                  * The use of PT_GET_SC_ARGS there is special,
  445                  * implementation of PT_GET_SC_ARGS for Linux-ABI
  446                  * callers emulates Linux bug which strace(1) depends
  447                  * on: at initialization it tests whether ptrace works
  448                  * by calling close(2), or some other single-argument
  449                  * syscall, _with six arguments_, and then verifies
  450                  * whether it can fetch them all using this API;
  451                  * otherwise it bails out.
  452                  */
  453                 error = kern_ptrace(td, PT_GET_SC_ARGS, pid,
  454                     &si.entry.args, sizeof(si.entry.args));
  455                 if (error != 0) {
  456                         linux_msg(td, "PT_GET_SC_ARGS failed with error %d",
  457                             error);
  458                         return (error);
  459                 }
  460         } else if (lwpinfo.pl_flags & PL_FLAG_SCX) {
  461                 si.op = LINUX_PTRACE_SYSCALL_INFO_EXIT;
  462                 error = kern_ptrace(td, PT_GET_SC_RET, pid, &sr, sizeof(sr));
  463 
  464                 if (error != 0) {
  465                         linux_msg(td, "PT_GET_SC_RET failed with error %d",
  466                             error);
  467                         return (error);
  468                 }
  469 
  470                 if (sr.sr_error == 0) {
  471                         si.exit.rval = sr.sr_retval[0];
  472                         si.exit.is_error = 0;
  473                 } else if (sr.sr_error == EJUSTRETURN) {
  474                         /*
  475                          * EJUSTRETURN means the actual value to return
  476                          * has already been put into td_frame; instead
  477                          * of extracting it and trying to determine whether
  478                          * it's an error or not just bail out and let
  479                          * the ptracing process fall back to another method.
  480                          */
  481                         si.op = LINUX_PTRACE_SYSCALL_INFO_NONE;
  482                 } else if (sr.sr_error == ERESTART) {
  483                         si.exit.rval = -LINUX_ERESTARTSYS;
  484                         si.exit.is_error = 1;
  485                 } else {
  486                         si.exit.rval = bsd_to_linux_errno(sr.sr_error);
  487                         si.exit.is_error = 1;
  488                 }
  489         } else {
  490                 si.op = LINUX_PTRACE_SYSCALL_INFO_NONE;
  491         }
  492 
  493         error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
  494         if (error != 0)
  495                 return (error);
  496 
  497         linux_ptrace_get_syscall_info_machdep(&b_reg, &si);
  498 
  499         len = MIN(len, sizeof(si));
  500         error = copyout(&si, (void *)data, len);
  501         if (error == 0)
  502                 td->td_retval[0] = sizeof(si);
  503 
  504         return (error);
  505 }
  506 
  507 int
  508 linux_ptrace(struct thread *td, struct linux_ptrace_args *uap)
  509 {
  510         void *addr;
  511         pid_t pid;
  512         int error, sig;
  513 
  514         if (!allow_ptrace)
  515                 return (ENOSYS);
  516 
  517         pid  = (pid_t)uap->pid;
  518         addr = (void *)uap->addr;
  519 
  520         switch (uap->req) {
  521         case LINUX_PTRACE_TRACEME:
  522                 error = kern_ptrace(td, PT_TRACE_ME, 0, 0, 0);
  523                 break;
  524         case LINUX_PTRACE_PEEKTEXT:
  525         case LINUX_PTRACE_PEEKDATA:
  526                 error = linux_ptrace_peek(td, pid, addr, (void *)uap->data);
  527                 if (error != 0)
  528                         goto out;
  529                 /*
  530                  * Linux expects this syscall to read 64 bits, not 32.
  531                  */
  532                 error = linux_ptrace_peek(td, pid,
  533                     (void *)(uap->addr + 4), (void *)(uap->data + 4));
  534                 break;
  535         case LINUX_PTRACE_PEEKUSER:
  536                 error = linux_ptrace_peekuser(td, pid, addr, (void *)uap->data);
  537                 break;
  538         case LINUX_PTRACE_POKETEXT:
  539         case LINUX_PTRACE_POKEDATA:
  540                 error = kern_ptrace(td, PT_WRITE_D, pid, addr, uap->data);
  541                 if (error != 0)
  542                         goto out;
  543                 /*
  544                  * Linux expects this syscall to write 64 bits, not 32.
  545                  */
  546                 error = kern_ptrace(td, PT_WRITE_D, pid,
  547                     (void *)(uap->addr + 4), uap->data >> 32);
  548                 break;
  549         case LINUX_PTRACE_POKEUSER:
  550                 error = linux_ptrace_pokeuser(td, pid, addr, (void *)uap->data);
  551                 break;
  552         case LINUX_PTRACE_CONT:
  553                 error = map_signum(uap->data, &sig);
  554                 if (error != 0)
  555                         break;
  556                 error = kern_ptrace(td, PT_CONTINUE, pid, (void *)1, sig);
  557                 break;
  558         case LINUX_PTRACE_KILL:
  559                 error = kern_ptrace(td, PT_KILL, pid, addr, uap->data);
  560                 break;
  561         case LINUX_PTRACE_SINGLESTEP:
  562                 error = map_signum(uap->data, &sig);
  563                 if (error != 0)
  564                         break;
  565                 error = kern_ptrace(td, PT_STEP, pid, (void *)1, sig);
  566                 break;
  567         case LINUX_PTRACE_GETREGS:
  568                 error = linux_ptrace_getregs(td, pid, (void *)uap->data);
  569                 break;
  570         case LINUX_PTRACE_SETREGS:
  571                 error = linux_ptrace_setregs(td, pid, (void *)uap->data);
  572                 break;
  573         case LINUX_PTRACE_ATTACH:
  574                 error = kern_ptrace(td, PT_ATTACH, pid, addr, uap->data);
  575                 break;
  576         case LINUX_PTRACE_DETACH:
  577                 error = map_signum(uap->data, &sig);
  578                 if (error != 0)
  579                         break;
  580                 error = kern_ptrace(td, PT_DETACH, pid, (void *)1, sig);
  581                 break;
  582         case LINUX_PTRACE_SYSCALL:
  583                 error = map_signum(uap->data, &sig);
  584                 if (error != 0)
  585                         break;
  586                 error = kern_ptrace(td, PT_SYSCALL, pid, (void *)1, sig);
  587                 break;
  588         case LINUX_PTRACE_SETOPTIONS:
  589                 error = linux_ptrace_setoptions(td, pid, uap->data);
  590                 break;
  591         case LINUX_PTRACE_GETEVENTMSG:
  592                 error = linux_ptrace_geteventmsg(td, pid, uap->data);
  593                 break;
  594         case LINUX_PTRACE_GETSIGINFO:
  595                 error = linux_ptrace_getsiginfo(td, pid, uap->data);
  596                 break;
  597         case LINUX_PTRACE_GETREGSET:
  598                 error = linux_ptrace_getregset(td, pid, uap->addr, uap->data);
  599                 break;
  600         case LINUX_PTRACE_SEIZE:
  601                 error = linux_ptrace_seize(td, pid, uap->addr, uap->data);
  602                 break;
  603         case LINUX_PTRACE_GET_SYSCALL_INFO:
  604                 error = linux_ptrace_get_syscall_info(td, pid, uap->addr, uap->data);
  605                 break;
  606         default:
  607                 linux_msg(td, "ptrace(%ld, ...) not implemented; "
  608                     "returning EINVAL", uap->req);
  609                 error = EINVAL;
  610                 break;
  611         }
  612 
  613 out:
  614         if (error == EBUSY)
  615                 error = ESRCH;
  616 
  617         return (error);
  618 }
Cache object: 2170c282c02a6b1d4aa6bad0635f284a 
 
 |