/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│ ╞══════════════════════════════════════════════════════════════════════════════╡ │ Copyright 2021 Justine Alexandra Roberts Tunney │ │ │ │ Permission to use, copy, modify, and/or distribute this software for │ │ any purpose with or without fee is hereby granted, provided that the │ │ above copyright notice and this permission notice appear in all copies. │ │ │ │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │ │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │ │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │ │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │ │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │ │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │ │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │ │ PERFORMANCE OF THIS SOFTWARE. │ ╚─────────────────────────────────────────────────────────────────────────────*/ #include "libc/calls/strace.internal.h" #include "libc/calls/syscall-sysv.internal.h" #include "libc/dce.h" #include "libc/sysv/consts/ptrace.h" #include "libc/sysv/errfuns.h" static const char *__ptrace_describe_request(int x) { if (x == -1) return "-1"; if (x == PTRACE_TRACEME) return "PTRACE_TRACEME"; if (x == PTRACE_PEEKDATA) return "PTRACE_PEEKDATA"; if (x == PTRACE_GETFPREGS) return "PTRACE_GETFPREGS"; if (x == PTRACE_PEEKTEXT) return "PTRACE_PEEKTEXT"; if (x == PTRACE_POKEDATA) return "PTRACE_POKEDATA"; if (x == PTRACE_PEEKUSER) return "PTRACE_PEEKUSER"; if (x == PTRACE_POKETEXT) return "PTRACE_POKETEXT"; if (x == PTRACE_POKEUSER) return "PTRACE_POKEUSER"; if (x == PTRACE_GETREGS) return "PTRACE_GETREGS"; if (x == PTRACE_GETREGSET) return "PTRACE_GETREGSET"; if (x == PTRACE_SETFPREGS) return "PTRACE_SETFPREGS"; if (x == PTRACE_SETREGS) return "PTRACE_SETREGS"; if (x == PTRACE_SETREGSET) return "PTRACE_SETREGSET"; if (x == PTRACE_GETSIGINFO) return "PTRACE_GETSIGINFO"; if (x == PTRACE_SETSIGINFO) return "PTRACE_SETSIGINFO"; if (x == PTRACE_PEEKSIGINFO) return "PTRACE_PEEKSIGINFO"; if (x == PTRACE_GETSIGMASK) return "PTRACE_GETSIGMASK"; if (x == PTRACE_SETSIGMASK) return "PTRACE_SETSIGMASK"; if (x == PTRACE_SETOPTIONS) return "PTRACE_SETOPTIONS"; if (x == PTRACE_GETEVENTMSG) return "PTRACE_GETEVENTMSG"; if (x == PTRACE_CONT) return "PTRACE_CONT"; if (x == PTRACE_SINGLESTEP) return "PTRACE_SINGLESTEP"; if (x == PTRACE_SYSCALL) return "PTRACE_SYSCALL"; if (x == PTRACE_LISTEN) return "PTRACE_LISTEN"; if (x == PTRACE_KILL) return "PTRACE_KILL"; if (x == PTRACE_INTERRUPT) return "PTRACE_INTERRUPT"; if (x == PTRACE_ATTACH) return "PTRACE_ATTACH"; if (x == PTRACE_SEIZE) return "PTRACE_SEIZE"; if (x == PTRACE_SECCOMP_GET_FILTER) return "PTRACE_SECCOMP_GET_FILTER"; if (x == PTRACE_DETACH) return "PTRACE_DETACH"; return "PTRACE_WUT"; } /** * Traces process. * * @param request can be PTRACE_xxx * @note de facto linux only atm * @vforksafe */ long ptrace(int request, ...) { // TODO(jart): FreeBSD addr and data args are different int pid; va_list va; bool ispeek; long rc, peek; void *addr, *data; va_start(va, request); pid = va_arg(va, int); addr = va_arg(va, void *); data = va_arg(va, void *); va_end(va); if (request == -1) { rc = einval(); /* see consts.sh */ } else { ispeek = IsLinux() && request - 1u < 3; if (ispeek) data = &peek; rc = sys_ptrace(request, pid, addr, data); if (rc != -1 && ispeek) rc = peek; } STRACE("ptrace(%s, %d, %p, %p) → %ld% m", __ptrace_describe_request(request), pid, addr, data); return rc; }