2020-06-15 14:18:57 +00:00
|
|
|
/*-*- 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│
|
|
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
2021-01-25 21:08:05 +00:00
|
|
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
2020-06-15 14:18:57 +00:00
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
│ 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. │
|
2020-06-15 14:18:57 +00:00
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
│ 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. │
|
2020-06-15 14:18:57 +00:00
|
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
2021-01-25 21:08:05 +00:00
|
|
|
#include "libc/calls/internal.h"
|
2022-03-24 15:00:21 +00:00
|
|
|
#include "libc/calls/strace.internal.h"
|
|
|
|
#include "libc/dce.h"
|
|
|
|
#include "libc/sysv/consts/ptrace.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
#include "libc/sysv/errfuns.h"
|
|
|
|
|
2022-03-24 15:00:21 +00:00
|
|
|
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";
|
|
|
|
}
|
|
|
|
|
2021-01-25 21:08:05 +00:00
|
|
|
/**
|
|
|
|
* Traces process.
|
|
|
|
*
|
|
|
|
* @param request can be PTRACE_xxx
|
|
|
|
* @note de facto linux only atm
|
2022-04-06 16:12:05 +00:00
|
|
|
* @vforksafe
|
2021-01-25 21:08:05 +00:00
|
|
|
*/
|
2022-03-24 15:00:21 +00:00
|
|
|
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;
|
2021-01-25 21:08:05 +00:00
|
|
|
}
|