mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-10-24 10:10:59 +00:00
We can put this back the moment someone requests it. Pain-free garbage collection for the C language is pretty cool. All it does is overwrite the return address with a trampoline that calls free(). It's not clear what it should be named if it's made a public API.
990 lines
28 KiB
C
990 lines
28 KiB
C
/*-*- 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/bits/bits.h"
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/sigbits.h"
|
|
#include "libc/calls/struct/sigaction.h"
|
|
#include "libc/calls/struct/sigset.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/fmt/conv.h"
|
|
#include "libc/log/check.h"
|
|
#include "libc/log/log.h"
|
|
#include "libc/mem/mem.h"
|
|
#include "libc/runtime/gc.internal.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/stdio/stdio.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/sysv/consts/ex.h"
|
|
#include "libc/sysv/consts/exit.h"
|
|
#include "libc/sysv/consts/nr.h"
|
|
#include "libc/sysv/consts/o.h"
|
|
#include "libc/sysv/consts/sa.h"
|
|
#include "libc/sysv/consts/sig.h"
|
|
#include "libc/x/x.h"
|
|
#include "third_party/dlmalloc/dlmalloc.internal.h"
|
|
#include "third_party/getopt/getopt.h"
|
|
|
|
/**
|
|
* @fileoverview Pythonic System Call Trace
|
|
*
|
|
* This program invokes `strace` as a subprocess and turns its output
|
|
* into Python data structures. It is useful because strace output is
|
|
* this weird plaintext format that's so famously difficult to parse.
|
|
*
|
|
* For example, you can run this command:
|
|
*
|
|
* pstrace -o trace.pylog echo hello world
|
|
*
|
|
* After which you may parse the output with Python:
|
|
*
|
|
* for line in open('trace.pylog'):
|
|
* pid,time,elap,kind,x = eval(line)
|
|
* if kind == 1:
|
|
* name,ret,args = x
|
|
* print "%s%r -> %d" % (name, args, ret)
|
|
*
|
|
* This program traces the subset of system calls governing processes
|
|
* and files. To do that we must track file descriptor lifetimes too.
|
|
* We also track system calls that are problematic for build configs,
|
|
* such as sockets, since compiling code shouldn't need the Internet.
|
|
*
|
|
* @note this tool is linux only
|
|
* @note freebsd: truss PROC ARGS
|
|
* @note appleos: sudo dtruss PROC ARGS
|
|
* @note openbsd: ktrace PROC ARGS && kdump -f ktrace.out
|
|
* @note windows: https://github.com/rogerorr/NtTrace
|
|
*/
|
|
|
|
#define DEBUG "%ld: %s", lineno, line
|
|
#define READ128BE(S) ((uint128_t)READ64BE(S) << 64 | READ64BE((S) + 8))
|
|
#define APPEND(L) \
|
|
do { \
|
|
if (++L.n > L.c) { \
|
|
L.c = MAX(11, L.c); \
|
|
L.c += L.c >> 1; \
|
|
L.p = realloc(L.p, L.c * sizeof(*L.p)); \
|
|
} \
|
|
memset(L.p + L.n - 1, 0, sizeof(*L.p)); \
|
|
} while (0)
|
|
|
|
struct Trace {
|
|
struct Slices {
|
|
long n, c;
|
|
struct Slice {
|
|
int n, c;
|
|
char *p;
|
|
} * p;
|
|
} slices;
|
|
struct HashTable {
|
|
long i, n;
|
|
struct HashEntry {
|
|
long h;
|
|
long i;
|
|
} * p;
|
|
} sliceindex;
|
|
struct Strlists {
|
|
long n, c;
|
|
struct Strlist {
|
|
long n, c;
|
|
long *p; // slices.p[p[i]]
|
|
} * p;
|
|
} strlists;
|
|
struct Events {
|
|
long n, c;
|
|
struct Event {
|
|
enum EventKind {
|
|
EK_NONE,
|
|
EK_CALL,
|
|
EK_EXIT, // ret is kernel code
|
|
EK_SIGNAL, // ret is signal code
|
|
EK_KILLED, // ret is signal code
|
|
} kind;
|
|
unsigned char arity;
|
|
unsigned char syscall;
|
|
bool is_interrupted;
|
|
int us;
|
|
int elap;
|
|
int pid;
|
|
long sec;
|
|
long ret;
|
|
long lineno;
|
|
struct Arg {
|
|
enum ArgKind {
|
|
AK_LONG, // x
|
|
AK_STR, // slices.p[x]
|
|
AK_STRLIST, // strlists.p[x]
|
|
AK_INTPAIR, // (x&0xffffffff, x>>32)
|
|
} kind;
|
|
long name;
|
|
long x;
|
|
} arg[6];
|
|
} * p;
|
|
} events;
|
|
};
|
|
|
|
static const struct Syscall {
|
|
char name[16];
|
|
} kSyscalls[] = {
|
|
{"accept"}, //
|
|
{"accept4"}, //
|
|
{"access"}, //
|
|
{"bind"}, //
|
|
{"chdir"}, //
|
|
{"chmod"}, //
|
|
{"chown"}, //
|
|
{"chroot"}, //
|
|
{"clone"}, //
|
|
{"close"}, //
|
|
{"connect"}, //
|
|
{"creat"}, //
|
|
{"dup"}, //
|
|
{"dup2"}, //
|
|
{"dup3"}, //
|
|
{"epoll_create"}, //
|
|
{"epoll_create1"}, //
|
|
{"eventfd"}, //
|
|
{"eventfd2"}, //
|
|
{"execve"}, //
|
|
{"execveat"}, //
|
|
{"faccessat"}, //
|
|
{"fchmodat"}, //
|
|
{"fchownat"}, //
|
|
{"fdatasync"}, //
|
|
{"fcntl"}, //
|
|
{"flock"}, //
|
|
{"fork"}, //
|
|
{"fsync"}, //
|
|
{"lchown"}, //
|
|
{"link"}, //
|
|
{"linkat"}, //
|
|
{"listen"}, //
|
|
{"memfd_create"}, //
|
|
{"mkdir"}, //
|
|
{"mkdirat"}, //
|
|
{"mknod"}, //
|
|
{"mknodat"}, //
|
|
{"open"}, //
|
|
{"openat"}, //
|
|
{"pipe"}, //
|
|
{"pipe2"}, //
|
|
{"readlink"}, //
|
|
{"readlinkat"}, //
|
|
{"rename"}, //
|
|
{"renameat"}, //
|
|
{"renameat2"}, //
|
|
{"rmdir"}, //
|
|
{"signalfd"}, //
|
|
{"signalfd4"}, //
|
|
{"socket"}, //
|
|
{"socketpair"}, //
|
|
{"statfs"}, //
|
|
{"symlink"}, //
|
|
{"symlinkat"}, //
|
|
{"sync"}, //
|
|
{"syncfs"}, //
|
|
{"timerfd_create"}, //
|
|
{"truncate"}, //
|
|
{"unlink"}, //
|
|
{"unlinkat"}, //
|
|
{"utimensat"}, //
|
|
{"vfork"}, //
|
|
};
|
|
|
|
static const struct Signal {
|
|
char name[8];
|
|
unsigned char number;
|
|
} kSignals[] = {
|
|
{"SIGABRT", 6}, //
|
|
{"SIGALRM", 14}, //
|
|
{"SIGBUS", 7}, //
|
|
{"SIGCHLD", 17}, //
|
|
{"SIGCONT", 18}, //
|
|
{"SIGFPE", 8}, //
|
|
{"SIGHUP", 1}, //
|
|
{"SIGILL", 4}, //
|
|
{"SIGINT", 2}, //
|
|
{"SIGIO", 29}, //
|
|
{"SIGIOT", 6}, //
|
|
{"SIGKILL", 9}, //
|
|
{"SIGPIPE", 13}, //
|
|
{"SIGPOLL", 29}, //
|
|
{"SIGPROF", 27}, //
|
|
{"SIGPWR", 30}, //
|
|
{"SIGQUIT", 3}, //
|
|
{"SIGSEGV", 11}, //
|
|
{"SIGSTOP", 19}, //
|
|
{"SIGSYS", 31}, //
|
|
{"SIGTERM", 15}, //
|
|
{"SIGTRAP", 5}, //
|
|
{"SIGTSTP", 20}, //
|
|
{"SIGTTIN", 21}, //
|
|
{"SIGTTOU", 22}, //
|
|
{"SIGURG", 23}, //
|
|
{"SIGUSR1", 10}, //
|
|
{"SIGUSR2", 12}, //
|
|
{"SIGWINCH", 28}, //
|
|
{"SIGXCPU", 24}, //
|
|
{"SIGXFSZ", 25}, //
|
|
};
|
|
|
|
static const struct Errno {
|
|
char name[16];
|
|
unsigned char number;
|
|
} kErrnos[] = {
|
|
{"E2BIG", 7}, //
|
|
{"EACCES", 13}, //
|
|
{"EADDRINUSE", 98}, //
|
|
{"EADDRNOTAVAIL", 99}, //
|
|
{"EADV", 68}, //
|
|
{"EAFNOSUPPORT", 97}, //
|
|
{"EAGAIN", 11}, //
|
|
{"EALREADY", 114}, //
|
|
{"EBADE", 52}, //
|
|
{"EBADF", 9}, //
|
|
{"EBADFD", 77}, //
|
|
{"EBADMSG", 74}, //
|
|
{"EBADR", 53}, //
|
|
{"EBADRQC", 56}, //
|
|
{"EBADSLT", 57}, //
|
|
{"EBFONT", 59}, //
|
|
{"EBUSY", 16}, //
|
|
{"ECANCELED", 125}, //
|
|
{"ECHILD", 10}, //
|
|
{"ECHRNG", 44}, //
|
|
{"ECOMM", 70}, //
|
|
{"ECONNABORTED", 103}, //
|
|
{"ECONNREFUSED", 111}, //
|
|
{"ECONNRESET", 104}, //
|
|
{"EDEADLK", 35}, //
|
|
{"EDESTADDRREQ", 89}, //
|
|
{"EDOM", 33}, //
|
|
{"EDOTDOT", 73}, //
|
|
{"EDQUOT", 122}, //
|
|
{"EEXIST", 17}, //
|
|
{"EFAULT", 14}, //
|
|
{"EFBIG", 27}, //
|
|
{"EHOSTDOWN", 112}, //
|
|
{"EHOSTUNREACH", 113}, //
|
|
{"EHWPOISON", 133}, //
|
|
{"EIDRM", 43}, //
|
|
{"EILSEQ", 84}, //
|
|
{"EINPROGRESS", 115}, //
|
|
{"EINTR", 4}, //
|
|
{"EINVAL", 22}, //
|
|
{"EIO", 5}, //
|
|
{"EISCONN", 106}, //
|
|
{"EISDIR", 21}, //
|
|
{"EISNAM", 120}, //
|
|
{"EKEYEXPIRED", 127}, //
|
|
{"EKEYREJECTED", 129}, //
|
|
{"EKEYREVOKED", 128}, //
|
|
{"EL2HLT", 51}, //
|
|
{"EL2NSYNC", 45}, //
|
|
{"EL3HLT", 46}, //
|
|
{"EL3RST", 47}, //
|
|
{"ELIBACC", 79}, //
|
|
{"ELIBBAD", 80}, //
|
|
{"ELIBEXEC", 83}, //
|
|
{"ELIBMAX", 82}, //
|
|
{"ELIBSCN", 81}, //
|
|
{"ELNRNG", 48}, //
|
|
{"ELOOP", 40}, //
|
|
{"EMEDIUMTYPE", 124}, //
|
|
{"EMFILE", 24}, //
|
|
{"EMLINK", 31}, //
|
|
{"EMSGSIZE", 90}, //
|
|
{"EMULTIHOP", 72}, //
|
|
{"ENAMETOOLONG", 36}, //
|
|
{"ENAVAIL", 119}, //
|
|
{"ENETDOWN", 100}, //
|
|
{"ENETRESET", 102}, //
|
|
{"ENETUNREACH", 101}, //
|
|
{"ENFILE", 23}, //
|
|
{"ENOANO", 55}, //
|
|
{"ENOBUFS", 105}, //
|
|
{"ENOCSI", 50}, //
|
|
{"ENODATA", 61}, //
|
|
{"ENODEV", 19}, //
|
|
{"ENOENT", 2}, //
|
|
{"ENOEXEC", 8}, //
|
|
{"ENOKEY", 126}, //
|
|
{"ENOLCK", 37}, //
|
|
{"ENOLINK", 67}, //
|
|
{"ENOMEDIUM", 123}, //
|
|
{"ENOMEM", 12}, //
|
|
{"ENOMSG", 42}, //
|
|
{"ENONET", 64}, //
|
|
{"ENOPKG", 65}, //
|
|
{"ENOPROTOOPT", 92}, //
|
|
{"ENOSPC", 28}, //
|
|
{"ENOSR", 63}, //
|
|
{"ENOSTR", 60}, //
|
|
{"ENOSYS", 38}, //
|
|
{"ENOTBLK", 15}, //
|
|
{"ENOTCONN", 107}, //
|
|
{"ENOTDIR", 20}, //
|
|
{"ENOTEMPTY", 39}, //
|
|
{"ENOTNAM", 118}, //
|
|
{"ENOTRECOVERABLE", 131}, //
|
|
{"ENOTSOCK", 88}, //
|
|
{"ENOTSUP", 95}, //
|
|
{"ENOTTY", 25}, //
|
|
{"ENOTUNIQ", 76}, //
|
|
{"ENXIO", 6}, //
|
|
{"EOPNOTSUPP", 95}, //
|
|
{"EOVERFLOW", 75}, //
|
|
{"EOWNERDEAD", 130}, //
|
|
{"EPERM", 1}, //
|
|
{"EPFNOSUPPORT", 96}, //
|
|
{"EPIPE", 32}, //
|
|
{"EPROTO", 71}, //
|
|
{"EPROTONOSUPPORT", 93}, //
|
|
{"EPROTOTYPE", 91}, //
|
|
{"ERANGE", 34}, //
|
|
{"EREMCHG", 78}, //
|
|
{"EREMOTE", 66}, //
|
|
{"EREMOTEIO", 121}, //
|
|
{"ERESTART", 85}, //
|
|
{"ERFKILL", 132}, //
|
|
{"EROFS", 30}, //
|
|
{"ESHUTDOWN", 108}, //
|
|
{"ESOCKTNOSUPPORT", 94}, //
|
|
{"ESPIPE", 29}, //
|
|
{"ESRCH", 3}, //
|
|
{"ESRMNT", 69}, //
|
|
{"ESTALE", 116}, //
|
|
{"ESTRPIPE", 86}, //
|
|
{"ETIME", 62}, //
|
|
{"ETIMEDOUT", 110}, //
|
|
{"ETOOMANYREFS", 109}, //
|
|
{"ETXTBSY", 26}, //
|
|
{"EUCLEAN", 117}, //
|
|
{"EUNATCH", 49}, //
|
|
{"EUSERS", 87}, //
|
|
{"EWOULDBLOCK", 11}, //
|
|
{"EXDEV", 18}, //
|
|
{"EXFULL", 54}, //
|
|
};
|
|
|
|
static char **strace_args;
|
|
static size_t strace_args_len;
|
|
static volatile bool interrupted;
|
|
|
|
static long Hash(const void *p, size_t n) {
|
|
unsigned h, i;
|
|
for (h = i = 0; i < n; i++) {
|
|
h += ((unsigned char *)p)[i];
|
|
h *= 0x9e3779b1;
|
|
}
|
|
return MAX(1, h);
|
|
}
|
|
|
|
static uint64_t MakeKey64(const char *p, size_t n) {
|
|
char k[8] = {0};
|
|
memcpy(k, p, n);
|
|
return READ64BE(k);
|
|
}
|
|
|
|
static uint128_t MakeKey128(const char *p, size_t n) {
|
|
char k[16] = {0};
|
|
memcpy(k, p, n);
|
|
return READ128BE(k);
|
|
}
|
|
|
|
static int GetSyscall(const char *name, size_t namelen) {
|
|
int m, l, r;
|
|
uint128_t x, y;
|
|
char *endofname;
|
|
if (namelen && namelen <= 16) {
|
|
x = MakeKey128(name, namelen);
|
|
l = 0;
|
|
r = ARRAYLEN(kSyscalls) - 1;
|
|
while (l <= r) {
|
|
m = (l + r) >> 1;
|
|
y = READ128BE(kSyscalls[m].name);
|
|
if (x < y) {
|
|
r = m - 1;
|
|
} else if (x > y) {
|
|
l = m + 1;
|
|
} else {
|
|
return m;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int GetErrno(const char *name, size_t namelen) {
|
|
int m, l, r;
|
|
uint128_t x, y;
|
|
char *endofname;
|
|
if (namelen && namelen <= 16) {
|
|
x = MakeKey128(name, namelen);
|
|
l = 0;
|
|
r = ARRAYLEN(kErrnos) - 1;
|
|
while (l <= r) {
|
|
m = (l + r) >> 1;
|
|
y = READ128BE(kErrnos[m].name);
|
|
if (x < y) {
|
|
r = m - 1;
|
|
} else if (x > y) {
|
|
l = m + 1;
|
|
} else {
|
|
return kErrnos[m].number;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int GetSignal(const char *name, size_t namelen) {
|
|
int m, l, r;
|
|
uint64_t x, y;
|
|
char *endofname;
|
|
if (namelen && namelen <= 8) {
|
|
x = MakeKey64(name, namelen);
|
|
l = 0;
|
|
r = ARRAYLEN(kSignals) - 1;
|
|
while (l <= r) {
|
|
m = (l + r) >> 1;
|
|
y = READ64BE(kSignals[m].name);
|
|
if (x < y) {
|
|
r = m - 1;
|
|
} else if (x > y) {
|
|
l = m + 1;
|
|
} else {
|
|
return kSignals[m].number;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static struct Trace *NewTrace(void) {
|
|
return calloc(1, sizeof(struct Trace));
|
|
}
|
|
|
|
static void FreeTrace(struct Trace *t) {
|
|
long i;
|
|
if (t) {
|
|
for (i = 0; i < t->slices.n; ++i) {
|
|
free(t->slices.p[i].p);
|
|
}
|
|
free(t->slices.p);
|
|
free(t->sliceindex.p);
|
|
for (i = 0; i < t->strlists.n; ++i) {
|
|
free(t->strlists.p[i].p);
|
|
}
|
|
free(t->strlists.p);
|
|
free(t->events.p);
|
|
free(t);
|
|
}
|
|
}
|
|
|
|
static void AppendStrlists(struct Trace *t) {
|
|
APPEND(t->strlists);
|
|
}
|
|
|
|
static void AppendStrlist(struct Strlist *l) {
|
|
APPEND((*l));
|
|
}
|
|
|
|
static void AppendEvent(struct Trace *t) {
|
|
APPEND(t->events);
|
|
}
|
|
|
|
static void AppendSlices(struct Trace *t) {
|
|
APPEND(t->slices);
|
|
}
|
|
|
|
static void AppendSlice(struct Slice *s, int c) {
|
|
APPEND((*s));
|
|
s->p[s->n - 1] = c;
|
|
}
|
|
|
|
static long Intern(struct Trace *t, char *data, long size) {
|
|
struct HashEntry *p;
|
|
long i, j, k, n, m, h, n2;
|
|
h = Hash(data, size);
|
|
n = t->sliceindex.n;
|
|
i = 0;
|
|
if (n) {
|
|
k = 0;
|
|
do {
|
|
i = (h + k + ((k + 1) >> 1)) & (n - 1);
|
|
if (t->sliceindex.p[i].h == h &&
|
|
t->slices.p[t->sliceindex.p[i].i].n == size &&
|
|
!memcmp(t->slices.p[t->sliceindex.p[i].i].p, data, size)) {
|
|
free(data);
|
|
return t->sliceindex.p[i].i;
|
|
}
|
|
++k;
|
|
} while (t->sliceindex.p[i].h);
|
|
}
|
|
if (++t->sliceindex.i >= (n >> 1)) {
|
|
m = n ? n << 1 : 16;
|
|
p = calloc(m, sizeof(struct HashEntry));
|
|
for (j = 0; j < n; ++j) {
|
|
if (t->sliceindex.p[j].h) {
|
|
k = 0;
|
|
do {
|
|
i = (t->sliceindex.p[j].h + k + ((k + 1) >> 1)) & (m - 1);
|
|
++k;
|
|
} while (p[i].h);
|
|
p[i].h = t->sliceindex.p[j].h;
|
|
p[i].i = t->sliceindex.p[j].i;
|
|
}
|
|
}
|
|
k = 0;
|
|
do {
|
|
i = (h + k + ((k + 1) >> 1)) & (m - 1);
|
|
++k;
|
|
} while (p[i].h);
|
|
free(t->sliceindex.p);
|
|
t->sliceindex.p = p;
|
|
t->sliceindex.n = m;
|
|
}
|
|
AppendSlices(t);
|
|
t->slices.p[t->slices.n - 1].p = data;
|
|
t->slices.p[t->slices.n - 1].n = size;
|
|
t->sliceindex.p[i].i = t->slices.n - 1;
|
|
t->sliceindex.p[i].h = h;
|
|
return t->slices.n - 1;
|
|
}
|
|
|
|
static long ReadCharLiteral(struct Slice *buf, long c, char *p, long *i) {
|
|
if (c != '\\') return c;
|
|
switch ((c = p[(*i)++])) {
|
|
case 'a':
|
|
return '\a';
|
|
case 'b':
|
|
return '\b';
|
|
case 't':
|
|
return '\t';
|
|
case 'n':
|
|
return '\n';
|
|
case 'v':
|
|
return '\v';
|
|
case 'f':
|
|
return '\f';
|
|
case 'r':
|
|
return '\r';
|
|
case 'e':
|
|
return 033;
|
|
case 'x':
|
|
if (isxdigit(p[*i])) {
|
|
c = hextoint(p[(*i)++]);
|
|
if (isxdigit(p[*i])) {
|
|
c = c * 16 + hextoint(p[(*i)++]);
|
|
}
|
|
}
|
|
return c;
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
c -= '0';
|
|
if ('0' <= p[*i] && p[*i] <= '7') {
|
|
c = c * 8 + (p[(*i)++] - '0');
|
|
if ('0' <= p[*i] && p[*i] <= '7') {
|
|
c = c * 8 + (p[(*i)++] - '0');
|
|
}
|
|
}
|
|
return c;
|
|
default:
|
|
return c;
|
|
}
|
|
}
|
|
|
|
static long GetDuration(long sec1, long us1, long sec2, long us2) {
|
|
long elap;
|
|
if ((elap = (sec2 - sec1) * 1000000)) {
|
|
return elap + 1000000 - us1 + us2;
|
|
} else {
|
|
return elap + us2 - us1;
|
|
}
|
|
}
|
|
|
|
static void Parse(struct Trace *t, const char *line, long lineno) {
|
|
char *p, *q;
|
|
struct Slice b;
|
|
long c, i, j, k, arg, pid, event, sec, us;
|
|
p = line;
|
|
pid = strtol(p, &p, 10);
|
|
while (*p == ' ') ++p;
|
|
sec = strtol(p, &p, 10);
|
|
CHECK_EQ('.', *p++, DEBUG);
|
|
us = strtol(p, &p, 10);
|
|
CHECK_EQ(' ', *p++, DEBUG);
|
|
if (startswith(p, "<... ")) {
|
|
CHECK_NOTNULL((p = strchr(p, '>')));
|
|
++p;
|
|
for (event = t->events.n; event--;) {
|
|
if (t->events.p[event].pid == pid) {
|
|
CHECK(t->events.p[event].is_interrupted, DEBUG);
|
|
t->events.p[event].is_interrupted = false;
|
|
t->events.p[event].elap =
|
|
GetDuration(t->events.p[event].sec, t->events.p[event].us, sec, us);
|
|
break;
|
|
}
|
|
}
|
|
CHECK_GE(event, 0);
|
|
} else {
|
|
AppendEvent(t);
|
|
event = t->events.n - 1;
|
|
t->events.p[event].pid = pid;
|
|
t->events.p[event].sec = sec;
|
|
t->events.p[event].us = us;
|
|
t->events.p[event].lineno = lineno;
|
|
if (startswith(p, "+++ exited with ")) {
|
|
p += strlen("+++ exited with ");
|
|
t->events.p[event].kind = EK_EXIT;
|
|
t->events.p[event].ret = atoi(p);
|
|
return;
|
|
} else if (startswith(p, "+++ killed by ")) {
|
|
p += strlen("+++ killed by ");
|
|
CHECK((q = strchr(p, ' ')), DEBUG);
|
|
t->events.p[event].kind = EK_KILLED;
|
|
t->events.p[event].ret = GetSignal(p, q - p);
|
|
return;
|
|
} else if (startswith(p, "--- ")) {
|
|
p += 4;
|
|
CHECK(isalpha(*p), DEBUG);
|
|
CHECK((q = strchr(p, ' ')), DEBUG);
|
|
t->events.p[event].kind = EK_SIGNAL;
|
|
t->events.p[event].ret = GetSignal(p, q - p);
|
|
return;
|
|
} else if (isalpha(*p) && (q = strchr(p, '('))) {
|
|
t->events.p[event].kind = EK_CALL;
|
|
CHECK_NE(-1, (t->events.p[event].syscall = GetSyscall(p, q - p)), DEBUG);
|
|
p = q + 1;
|
|
}
|
|
}
|
|
for (;;) {
|
|
if (*p == ',') ++p;
|
|
while (*p == ' ') ++p;
|
|
CHECK(*p, DEBUG);
|
|
if (startswith(p, "<unfinished ...>")) {
|
|
t->events.p[event].is_interrupted = true;
|
|
break;
|
|
} else if (*p == ')') {
|
|
++p;
|
|
while (isspace(*p)) ++p;
|
|
CHECK_EQ('=', *p++, DEBUG);
|
|
while (isspace(*p)) ++p;
|
|
CHECK(isdigit(*p) || *p == '-', DEBUG);
|
|
t->events.p[event].ret = strtol(p, &p, 0);
|
|
if (t->events.p[event].ret == -1) {
|
|
while (isspace(*p)) ++p;
|
|
CHECK((q = strchr(p, ' ')), DEBUG);
|
|
if ((t->events.p[event].ret = GetErrno(p, q - p)) != -1) {
|
|
t->events.p[event].ret = -t->events.p[event].ret;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
CHECK_LT((arg = t->events.p[event].arity++), 6);
|
|
if (isalpha(*p) && !startswith(p, "NULL")) {
|
|
memset(&b, 0, sizeof(b));
|
|
for (; isalpha(*p) || *p == '_'; ++p) {
|
|
AppendSlice(&b, *p);
|
|
}
|
|
t->events.p[event].arg[arg].name = Intern(t, b.p, b.n);
|
|
CHECK_EQ('=', *p++, DEBUG);
|
|
} else {
|
|
t->events.p[event].arg[arg].name = -1;
|
|
}
|
|
if (startswith(p, "NULL")) {
|
|
p += 4;
|
|
t->events.p[event].arg[arg].kind = AK_LONG;
|
|
t->events.p[event].arg[arg].x = 0;
|
|
} else if (*p == '-' || isdigit(*p)) {
|
|
t->events.p[event].arg[arg].kind = AK_LONG;
|
|
for (;;) {
|
|
t->events.p[event].arg[arg].x |= strtol(p, &p, 0);
|
|
if (*p == '|') {
|
|
++p;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
} else if (*p == '{') {
|
|
CHECK_NOTNULL((p = strchr(p, '}')), DEBUG);
|
|
++p;
|
|
} else if (*p == '"') {
|
|
memset(&b, 0, sizeof(b));
|
|
for (j = 0; (c = p[++j]);) {
|
|
if (c == '"') {
|
|
p += j + 1;
|
|
break;
|
|
}
|
|
c = ReadCharLiteral(&b, c, p, &j);
|
|
AppendSlice(&b, c);
|
|
}
|
|
t->events.p[event].arg[arg].kind = AK_STR;
|
|
t->events.p[event].arg[arg].x = Intern(t, b.p, b.n);
|
|
} else if (*p == '[') {
|
|
++p;
|
|
if (isdigit(*p)) {
|
|
t->events.p[event].arg[arg].kind = AK_INTPAIR;
|
|
t->events.p[event].arg[arg].x = strtol(p, &p, 0) & 0xffffffff;
|
|
CHECK_EQ(',', *p++, DEBUG);
|
|
CHECK_EQ(' ', *p++, DEBUG);
|
|
t->events.p[event].arg[arg].x |= strtol(p, &p, 0) << 32;
|
|
CHECK_EQ(']', *p++, DEBUG);
|
|
} else {
|
|
AppendStrlists(t);
|
|
for (j = 0;; ++j) {
|
|
if (*p == ']') {
|
|
++p;
|
|
break;
|
|
}
|
|
if (*p == ',') ++p;
|
|
if (*p == ' ') ++p;
|
|
CHECK_EQ('"', *p, DEBUG);
|
|
memset(&b, 0, sizeof(b));
|
|
for (k = 0; (c = p[++k]);) {
|
|
if (c == '"') {
|
|
p += k + 1;
|
|
break;
|
|
}
|
|
c = ReadCharLiteral(&b, c, p, &k);
|
|
AppendSlice(&b, c);
|
|
}
|
|
AppendStrlist(&t->strlists.p[t->strlists.n - 1]);
|
|
t->strlists.p[t->strlists.n - 1]
|
|
.p[t->strlists.p[t->strlists.n - 1].n - 1] = Intern(t, b.p, b.n);
|
|
}
|
|
t->events.p[event].arg[arg].kind = AK_STRLIST;
|
|
t->events.p[event].arg[arg].x = t->strlists.n - 1;
|
|
}
|
|
} else {
|
|
CHECK(false, DEBUG);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void PrintArg(FILE *f, struct Trace *t, long ev, long arg) {
|
|
long i, x;
|
|
x = t->events.p[ev].arg[arg].name;
|
|
if (x != -1) {
|
|
fprintf(f, "b%`'.*s:", t->slices.p[x].n, t->slices.p[x].p);
|
|
}
|
|
x = t->events.p[ev].arg[arg].x;
|
|
switch (t->events.p[ev].arg[arg].kind) {
|
|
case AK_LONG:
|
|
fprintf(f, "%ld", x);
|
|
break;
|
|
case AK_STR:
|
|
fprintf(f, "b%`'.*s", t->slices.p[x].n, t->slices.p[x].p);
|
|
break;
|
|
case AK_INTPAIR:
|
|
fprintf(f, "(%d,%d)", x >> 32, x);
|
|
break;
|
|
case AK_STRLIST:
|
|
fprintf(f, "(");
|
|
for (i = 0; i < t->strlists.p[x].n; ++i) {
|
|
fprintf(f, "b%`'.*s,", t->slices.p[t->strlists.p[x].p[i]].n,
|
|
t->slices.p[t->strlists.p[x].p[i]].p);
|
|
}
|
|
fprintf(f, ")");
|
|
break;
|
|
default:
|
|
abort();
|
|
}
|
|
}
|
|
|
|
static void PrintEvent(FILE *f, struct Trace *t, long ev) {
|
|
long arg;
|
|
fprintf(f, "(%d,%ld,%d,%d,", t->events.p[ev].pid,
|
|
t->events.p[ev].sec * 1000000 + t->events.p[ev].us,
|
|
t->events.p[ev].elap, t->events.p[ev].kind);
|
|
switch (t->events.p[ev].kind) {
|
|
case EK_EXIT:
|
|
case EK_SIGNAL:
|
|
case EK_KILLED:
|
|
fprintf(f, "%d", t->events.p[ev].ret);
|
|
break;
|
|
case EK_CALL:
|
|
CHECK_LT(t->events.p[ev].syscall, ARRAYLEN(kSyscalls));
|
|
fprintf(f, "(b%`'s,%ld,", kSyscalls[t->events.p[ev].syscall].name,
|
|
t->events.p[ev].ret);
|
|
fprintf(f, "%c",
|
|
t->events.p[ev].arity && t->events.p[ev].arg[0].name != -1 ? '{'
|
|
: '(');
|
|
for (arg = 0; arg < t->events.p[ev].arity; ++arg) {
|
|
PrintArg(f, t, ev, arg);
|
|
fprintf(f, ",");
|
|
}
|
|
fprintf(f, "%c)",
|
|
t->events.p[ev].arity && t->events.p[ev].arg[0].name != -1 ? '}'
|
|
: ')');
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
fprintf(f, ")");
|
|
}
|
|
|
|
static void AppendArg(char *arg) {
|
|
strace_args = realloc(strace_args, ++strace_args_len * sizeof(*strace_args));
|
|
strace_args[strace_args_len - 1] = arg;
|
|
}
|
|
|
|
static wontreturn void PrintUsage(FILE *f, int rc) {
|
|
fprintf(f, "Usage: %s [-o OUT] PROG [ARGS...]\n", program_invocation_name);
|
|
exit(rc);
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
int i;
|
|
int ws;
|
|
int opt;
|
|
int pid;
|
|
long ev;
|
|
FILE *fin;
|
|
FILE *fout;
|
|
char *line;
|
|
long lineno;
|
|
char *strace;
|
|
int pipefds[2];
|
|
struct Trace *t;
|
|
sigset_t block, mask;
|
|
struct sigaction ignore, saveint, savequit;
|
|
|
|
/*
|
|
* parse prefix arguments
|
|
*/
|
|
fout = stderr;
|
|
while ((opt = getopt(argc, argv, "?ho:")) != -1) {
|
|
switch (opt) {
|
|
case 'o':
|
|
fout = fopen(optarg, "w");
|
|
break;
|
|
case 'h':
|
|
case '?':
|
|
PrintUsage(stdout, EXIT_SUCCESS);
|
|
default:
|
|
PrintUsage(stderr, EX_USAGE);
|
|
}
|
|
}
|
|
if (optind == argc) {
|
|
PrintUsage(stderr, EX_USAGE);
|
|
}
|
|
|
|
/*
|
|
* resolve full paths of dependencies
|
|
*/
|
|
if ((strace = commandvenv("STRACE", "strace"))) {
|
|
strace = strdup(strace);
|
|
} else {
|
|
fprintf(stderr, "error: please install strace\n");
|
|
exit(1);
|
|
}
|
|
|
|
/*
|
|
* create strace argument list
|
|
*/
|
|
AppendArg("strace");
|
|
AppendArg("-q"); // don't log attach/detach noise
|
|
AppendArg("-v"); // don't abbreviate arrays
|
|
AppendArg("-f"); // follow subprocesses
|
|
AppendArg("-ttt"); // print unixseconds.micros
|
|
AppendArg("-X"); // print numbers instead of symbols
|
|
AppendArg("raw"); // e.g. 2 vs. O_RDWR
|
|
AppendArg("-s"); // don't abbreviate data
|
|
AppendArg("805306368"); // strace won't let us go higher
|
|
AppendArg("-e"); // system calls that matter
|
|
AppendArg(
|
|
"open,close,access,pipe,dup,dup2,socket,connect,accept,bind,listen,"
|
|
"socketpair,fork,vfork,execve,clone,flock,fsync,fdatasync,truncate,chdir,"
|
|
"rename,mkdir,rmdir,creat,link,unlink,symlink,readlink,chmod,chown,fcntl,"
|
|
"lchown,mknod,mknodat,statfs,chroot,sync,epoll_create,openat,mkdirat,"
|
|
"fchownat,unlinkat,renameat,linkat,symlinkat,readlinkat,fchmodat,fchdir,"
|
|
"faccessat,utimensat,accept4,dup3,pipe2,epoll_create1,signalfd,signalfd4,"
|
|
"eventfd,eventfd2,timerfd_create,syncfs,renameat2,memfd_create,execveat");
|
|
CHECK_NE(-1, pipe(pipefds));
|
|
AppendArg("-o");
|
|
AppendArg(xasprintf("/dev/fd/%d", pipefds[1]));
|
|
for (i = optind; i < argc; ++i) {
|
|
AppendArg(argv[i]);
|
|
}
|
|
AppendArg(NULL);
|
|
|
|
/*
|
|
* spawn strace
|
|
*/
|
|
ignore.sa_flags = 0;
|
|
ignore.sa_handler = SIG_IGN;
|
|
sigemptyset(&ignore.sa_mask);
|
|
sigaction(SIGINT, &ignore, &saveint);
|
|
sigaction(SIGQUIT, &ignore, &savequit);
|
|
sigfillset(&block);
|
|
sigprocmask(SIG_BLOCK, &block, &mask);
|
|
CHECK_NE(-1, (pid = vfork()));
|
|
if (!pid) {
|
|
close(pipefds[0]);
|
|
sigaction(SIGINT, &saveint, NULL);
|
|
sigaction(SIGQUIT, &savequit, NULL);
|
|
sigprocmask(SIG_SETMASK, &mask, NULL);
|
|
execv(strace, strace_args);
|
|
_exit(127);
|
|
}
|
|
close(pipefds[1]);
|
|
sigaddset(&mask, SIGCHLD);
|
|
sigprocmask(SIG_SETMASK, &mask, NULL);
|
|
|
|
/*
|
|
* read output of strace until eof
|
|
*/
|
|
fin = fdopen(pipefds[0], "r");
|
|
t = NewTrace();
|
|
for (ev = 0, lineno = 1; !interrupted && (line = xgetline(fin)); ++lineno) {
|
|
chomp(line);
|
|
Parse(t, line, lineno);
|
|
free(line);
|
|
for (; ev < t->events.n && !t->events.p[ev].is_interrupted; ++ev) {
|
|
PrintEvent(fout, t, ev);
|
|
fprintf(fout, "\n");
|
|
}
|
|
}
|
|
FreeTrace(t);
|
|
CHECK_NE(-1, fclose(fout));
|
|
|
|
/*
|
|
* wait for strace to exit
|
|
*/
|
|
while (waitpid(pid, &ws, 0) == -1) {
|
|
CHECK_EQ(EINTR, errno);
|
|
}
|
|
CHECK_NE(-1, fclose(fin));
|
|
|
|
/*
|
|
* propagate exit
|
|
*/
|
|
if (WIFEXITED(ws)) {
|
|
return WEXITSTATUS(ws);
|
|
} else {
|
|
return 128 + WTERMSIG(ws);
|
|
}
|
|
}
|