3bb7cd6660
* strace.c: Fix misplaced #endif. * util.c: Indent preprocessor directives, mark code parts which can never be reached by compilation because of the combination of #if directives. These are likely dead code, I want to remove them later.
2925 lines
68 KiB
C
2925 lines
68 KiB
C
/*
|
|
* Copyright (c) 1991, 1992 Paul Kranenburg <pk@cs.few.eur.nl>
|
|
* Copyright (c) 1993 Branko Lankester <branko@hacktic.nl>
|
|
* Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
|
|
* Copyright (c) 1996-1999 Wichert Akkerman <wichert@cistron.nl>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
#include "defs.h"
|
|
|
|
#include <sys/types.h>
|
|
#include <signal.h>
|
|
#include <errno.h>
|
|
#include <sys/param.h>
|
|
#include <fcntl.h>
|
|
#include <sys/resource.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/utsname.h>
|
|
#include <pwd.h>
|
|
#include <grp.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
#include <dirent.h>
|
|
|
|
#ifdef LINUX
|
|
# include <asm/unistd.h>
|
|
# if defined __NR_tgkill
|
|
# define my_tgkill(pid, tid, sig) syscall (__NR_tgkill, (pid), (tid), (sig))
|
|
# elif defined __NR_tkill
|
|
# define my_tgkill(pid, tid, sig) syscall (__NR_tkill, (tid), (sig))
|
|
# else
|
|
/* kill() may choose arbitrarily the target task of the process group
|
|
while we later wait on a that specific TID. PID process waits become
|
|
TID task specific waits for a process under ptrace(2). */
|
|
# warning "Neither tkill(2) nor tgkill(2) available, risk of strace hangs!"
|
|
# define my_tgkill(pid, tid, sig) kill ((tid), (sig))
|
|
# endif
|
|
#endif
|
|
|
|
#if defined(IA64) && defined(LINUX)
|
|
# include <asm/ptrace_offsets.h>
|
|
#endif
|
|
|
|
#ifdef USE_PROCFS
|
|
#include <poll.h>
|
|
#endif
|
|
|
|
#ifdef SVR4
|
|
#include <sys/stropts.h>
|
|
#ifdef HAVE_MP_PROCFS
|
|
#ifdef HAVE_SYS_UIO_H
|
|
#include <sys/uio.h>
|
|
#endif
|
|
#endif
|
|
#endif
|
|
extern char **environ;
|
|
extern int optind;
|
|
extern char *optarg;
|
|
|
|
|
|
int debug = 0, followfork = 0;
|
|
int dtime = 0, cflag = 0, xflag = 0, qflag = 0;
|
|
static int iflag = 0, interactive = 0, pflag_seen = 0, rflag = 0, tflag = 0;
|
|
/*
|
|
* daemonized_tracer supports -D option.
|
|
* With this option, strace forks twice.
|
|
* Unlike normal case, with -D *grandparent* process exec's,
|
|
* becoming a traced process. Child exits (this prevents traced process
|
|
* from having children it doesn't expect to have), and grandchild
|
|
* attaches to grandparent similarly to strace -p PID.
|
|
* This allows for more transparent interaction in cases
|
|
* when process and its parent are communicating via signals,
|
|
* wait() etc. Without -D, strace process gets lodged in between,
|
|
* disrupting parent<->child link.
|
|
*/
|
|
static bool daemonized_tracer = 0;
|
|
|
|
static struct utsname utsname_buf;
|
|
|
|
/* Sometimes we want to print only succeeding syscalls. */
|
|
int not_failing_only = 0;
|
|
|
|
static int exit_code = 0;
|
|
static int strace_child = 0;
|
|
static int ptrace_stop_sig = SIGTRAP;
|
|
static bool ptrace_opts_set;
|
|
|
|
static char *username = NULL;
|
|
uid_t run_uid;
|
|
gid_t run_gid;
|
|
|
|
int acolumn = DEFAULT_ACOLUMN;
|
|
int max_strlen = DEFAULT_STRLEN;
|
|
static char *outfname = NULL;
|
|
FILE *outf;
|
|
struct tcb **tcbtab;
|
|
unsigned int nprocs, tcbtabsize;
|
|
char *progname;
|
|
|
|
static int detach P((struct tcb *tcp, int sig));
|
|
static int trace P((void));
|
|
static void cleanup P((void));
|
|
static void interrupt P((int sig));
|
|
static sigset_t empty_set, blocked_set;
|
|
|
|
#ifdef HAVE_SIG_ATOMIC_T
|
|
static volatile sig_atomic_t interrupted;
|
|
#else /* !HAVE_SIG_ATOMIC_T */
|
|
#ifdef __STDC__
|
|
static volatile int interrupted;
|
|
#else /* !__STDC__ */
|
|
static int interrupted;
|
|
#endif /* !__STDC__ */
|
|
#endif /* !HAVE_SIG_ATOMIC_T */
|
|
|
|
#ifdef USE_PROCFS
|
|
|
|
static struct tcb *pfd2tcb P((int pfd));
|
|
static void reaper P((int sig));
|
|
static void rebuild_pollv P((void));
|
|
static struct pollfd *pollv;
|
|
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
|
|
static void proc_poll_open P((void));
|
|
static void proc_poller P((int pfd));
|
|
|
|
struct proc_pollfd {
|
|
int fd;
|
|
int revents;
|
|
int pid;
|
|
};
|
|
|
|
static int poller_pid;
|
|
static int proc_poll_pipe[2] = { -1, -1 };
|
|
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
|
|
#ifdef HAVE_MP_PROCFS
|
|
#define POLLWANT POLLWRNORM
|
|
#else
|
|
#define POLLWANT POLLPRI
|
|
#endif
|
|
#endif /* USE_PROCFS */
|
|
|
|
static void
|
|
usage(ofp, exitval)
|
|
FILE *ofp;
|
|
int exitval;
|
|
{
|
|
fprintf(ofp, "\
|
|
usage: strace [-dffhiqrtttTvVxx] [-a column] [-e expr] ... [-o file]\n\
|
|
[-p pid] ... [-s strsize] [-u username] [-E var=val] ...\n\
|
|
[command [arg ...]]\n\
|
|
or: strace -c -D [-e expr] ... [-O overhead] [-S sortby] [-E var=val] ...\n\
|
|
[command [arg ...]]\n\
|
|
-c -- count time, calls, and errors for each syscall and report summary\n\
|
|
-f -- follow forks, -ff -- with output into separate files\n\
|
|
-F -- attempt to follow vforks, -h -- print help message\n\
|
|
-i -- print instruction pointer at time of syscall\n\
|
|
-q -- suppress messages about attaching, detaching, etc.\n\
|
|
-r -- print relative timestamp, -t -- absolute timestamp, -tt -- with usecs\n\
|
|
-T -- print time spent in each syscall, -V -- print version\n\
|
|
-v -- verbose mode: print unabbreviated argv, stat, termio[s], etc. args\n\
|
|
-x -- print non-ascii strings in hex, -xx -- print all strings in hex\n\
|
|
-a column -- alignment COLUMN for printing syscall results (default %d)\n\
|
|
-e expr -- a qualifying expression: option=[!]all or option=[!]val1[,val2]...\n\
|
|
options: trace, abbrev, verbose, raw, signal, read, or write\n\
|
|
-o file -- send trace output to FILE instead of stderr\n\
|
|
-O overhead -- set overhead for tracing syscalls to OVERHEAD usecs\n\
|
|
-p pid -- trace process with process id PID, may be repeated\n\
|
|
-D -- run tracer process as a detached grandchild, not as parent\n\
|
|
-s strsize -- limit length of print strings to STRSIZE chars (default %d)\n\
|
|
-S sortby -- sort syscall counts by: time, calls, name, nothing (default %s)\n\
|
|
-u username -- run command as username handling setuid and/or setgid\n\
|
|
-E var=val -- put var=val in the environment for command\n\
|
|
-E var -- remove var from the environment for command\n\
|
|
" /* this is broken, so don't document it
|
|
-z -- print only succeeding syscalls\n\
|
|
*/
|
|
, DEFAULT_ACOLUMN, DEFAULT_STRLEN, DEFAULT_SORTBY);
|
|
exit(exitval);
|
|
}
|
|
|
|
#ifdef SVR4
|
|
#ifdef MIPS
|
|
void
|
|
foobar()
|
|
{
|
|
}
|
|
#endif /* MIPS */
|
|
#endif /* SVR4 */
|
|
|
|
static int
|
|
set_cloexec_flag(int fd)
|
|
{
|
|
int flags, newflags;
|
|
|
|
if ((flags = fcntl(fd, F_GETFD, 0)) < 0)
|
|
{
|
|
fprintf(stderr, "%s: fcntl F_GETFD: %s\n",
|
|
progname, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
newflags = flags | FD_CLOEXEC;
|
|
if (flags == newflags)
|
|
return 0;
|
|
|
|
if (fcntl(fd, F_SETFD, newflags) < 0)
|
|
{
|
|
fprintf(stderr, "%s: fcntl F_SETFD: %s\n",
|
|
progname, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* When strace is setuid executable, we have to swap uids
|
|
* before and after filesystem and process management operations.
|
|
*/
|
|
static void
|
|
swap_uid(void)
|
|
{
|
|
#ifndef SVR4
|
|
int euid = geteuid(), uid = getuid();
|
|
|
|
if (euid != uid && setreuid(euid, uid) < 0)
|
|
{
|
|
fprintf(stderr, "%s: setreuid: %s\n",
|
|
progname, strerror(errno));
|
|
exit(1);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if _LFS64_LARGEFILE
|
|
# define fopen_for_output fopen64
|
|
#else
|
|
# define fopen_for_output fopen
|
|
#endif
|
|
|
|
static FILE *
|
|
strace_fopen(const char *path, const char *mode)
|
|
{
|
|
FILE *fp;
|
|
|
|
swap_uid();
|
|
if ((fp = fopen_for_output(path, mode)) == NULL)
|
|
fprintf(stderr, "%s: can't fopen '%s': %s\n",
|
|
progname, path, strerror(errno));
|
|
swap_uid();
|
|
if (fp && set_cloexec_flag(fileno(fp)) < 0)
|
|
{
|
|
fclose(fp);
|
|
fp = NULL;
|
|
}
|
|
return fp;
|
|
}
|
|
|
|
static int popen_pid = -1;
|
|
|
|
#ifndef _PATH_BSHELL
|
|
# define _PATH_BSHELL "/bin/sh"
|
|
#endif
|
|
|
|
/*
|
|
* We cannot use standard popen(3) here because we have to distinguish
|
|
* popen child process from other processes we trace, and standard popen(3)
|
|
* does not export its child's pid.
|
|
*/
|
|
static FILE *
|
|
strace_popen(const char *command)
|
|
{
|
|
int fds[2];
|
|
|
|
swap_uid();
|
|
if (pipe(fds) < 0)
|
|
{
|
|
fprintf(stderr, "%s: pipe: %s\n",
|
|
progname, strerror(errno));
|
|
swap_uid();
|
|
return NULL;
|
|
}
|
|
|
|
if (set_cloexec_flag(fds[1]) < 0)
|
|
{
|
|
close(fds[0]);
|
|
close(fds[1]);
|
|
swap_uid();
|
|
return NULL;
|
|
}
|
|
|
|
if ((popen_pid = fork()) == -1)
|
|
{
|
|
fprintf(stderr, "%s: fork: %s\n",
|
|
progname, strerror(errno));
|
|
close(fds[0]);
|
|
close(fds[1]);
|
|
swap_uid();
|
|
return NULL;
|
|
}
|
|
|
|
if (popen_pid)
|
|
{
|
|
/* parent */
|
|
close(fds[0]);
|
|
swap_uid();
|
|
return fdopen(fds[1], "w");
|
|
} else
|
|
{
|
|
/* child */
|
|
close(fds[1]);
|
|
if (fds[0] && (dup2(fds[0], 0) || close(fds[0])))
|
|
{
|
|
fprintf(stderr, "%s: dup2: %s\n",
|
|
progname, strerror(errno));
|
|
_exit(1);
|
|
}
|
|
execl(_PATH_BSHELL, "sh", "-c", command, NULL);
|
|
fprintf(stderr, "%s: execl: %s: %s\n",
|
|
progname, _PATH_BSHELL, strerror(errno));
|
|
_exit(1);
|
|
}
|
|
}
|
|
|
|
static int
|
|
newoutf(struct tcb *tcp)
|
|
{
|
|
if (outfname && followfork > 1) {
|
|
char name[520 + sizeof(int) * 3];
|
|
FILE *fp;
|
|
|
|
sprintf(name, "%.512s.%u", outfname, tcp->pid);
|
|
if ((fp = strace_fopen(name, "w")) == NULL)
|
|
return -1;
|
|
tcp->outf = fp;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
startup_attach(void)
|
|
{
|
|
int tcbi;
|
|
struct tcb *tcp;
|
|
|
|
/*
|
|
* Block user interruptions as we would leave the traced
|
|
* process stopped (process state T) if we would terminate in
|
|
* between PTRACE_ATTACH and wait4 () on SIGSTOP.
|
|
* We rely on cleanup () from this point on.
|
|
*/
|
|
if (interactive)
|
|
sigprocmask(SIG_BLOCK, &blocked_set, NULL);
|
|
|
|
if (daemonized_tracer) {
|
|
pid_t pid = fork();
|
|
if (pid < 0) {
|
|
_exit(1);
|
|
}
|
|
if (pid) { /* parent */
|
|
/*
|
|
* Wait for child to attach to straced process
|
|
* (our parent). Child SIGKILLs us after it attached.
|
|
* Parent's wait() is unblocked by our death,
|
|
* it proceeds to exec the straced program.
|
|
*/
|
|
pause();
|
|
_exit(0); /* paranoia */
|
|
}
|
|
}
|
|
|
|
for (tcbi = 0; tcbi < tcbtabsize; tcbi++) {
|
|
tcp = tcbtab[tcbi];
|
|
if (!(tcp->flags & TCB_INUSE) || !(tcp->flags & TCB_ATTACHED))
|
|
continue;
|
|
#ifdef LINUX
|
|
if (tcp->flags & TCB_CLONE_THREAD)
|
|
continue;
|
|
#endif
|
|
/* Reinitialize the output since it may have changed. */
|
|
tcp->outf = outf;
|
|
if (newoutf(tcp) < 0)
|
|
exit(1);
|
|
|
|
#ifdef USE_PROCFS
|
|
if (proc_open(tcp, 1) < 0) {
|
|
fprintf(stderr, "trouble opening proc file\n");
|
|
droptcb(tcp);
|
|
continue;
|
|
}
|
|
#else /* !USE_PROCFS */
|
|
# ifdef LINUX
|
|
if (followfork && !daemonized_tracer) {
|
|
char procdir[sizeof("/proc/%d/task") + sizeof(int) * 3];
|
|
DIR *dir;
|
|
|
|
sprintf(procdir, "/proc/%d/task", tcp->pid);
|
|
dir = opendir(procdir);
|
|
if (dir != NULL) {
|
|
unsigned int ntid = 0, nerr = 0;
|
|
struct dirent *de;
|
|
int tid;
|
|
while ((de = readdir(dir)) != NULL) {
|
|
if (de->d_fileno == 0)
|
|
continue;
|
|
tid = atoi(de->d_name);
|
|
if (tid <= 0)
|
|
continue;
|
|
++ntid;
|
|
if (ptrace(PTRACE_ATTACH, tid, (char *) 1, 0) < 0)
|
|
++nerr;
|
|
else if (tid != tcbtab[tcbi]->pid) {
|
|
tcp = alloctcb(tid);
|
|
tcp->flags |= TCB_ATTACHED|TCB_CLONE_THREAD|TCB_CLONE_DETACHED|TCB_FOLLOWFORK;
|
|
tcbtab[tcbi]->nchildren++;
|
|
tcbtab[tcbi]->nclone_threads++;
|
|
tcbtab[tcbi]->nclone_detached++;
|
|
tcp->parent = tcbtab[tcbi];
|
|
}
|
|
if (interactive) {
|
|
sigprocmask(SIG_SETMASK, &empty_set, NULL);
|
|
if (interrupted)
|
|
return;
|
|
sigprocmask(SIG_BLOCK, &blocked_set, NULL);
|
|
}
|
|
}
|
|
closedir(dir);
|
|
ntid -= nerr;
|
|
if (ntid == 0) {
|
|
perror("attach: ptrace(PTRACE_ATTACH, ...)");
|
|
droptcb(tcp);
|
|
continue;
|
|
}
|
|
if (!qflag) {
|
|
fprintf(stderr, ntid > 1
|
|
? "Process %u attached with %u threads - interrupt to quit\n"
|
|
: "Process %u attached - interrupt to quit\n",
|
|
tcbtab[tcbi]->pid, ntid);
|
|
}
|
|
continue;
|
|
} /* if (opendir worked) */
|
|
} /* if (-f) */
|
|
# endif
|
|
if (ptrace(PTRACE_ATTACH, tcp->pid, (char *) 1, 0) < 0) {
|
|
perror("attach: ptrace(PTRACE_ATTACH, ...)");
|
|
droptcb(tcp);
|
|
continue;
|
|
}
|
|
/* INTERRUPTED is going to be checked at the top of TRACE. */
|
|
|
|
if (daemonized_tracer) {
|
|
/*
|
|
* It is our grandparent we trace, not a -p PID.
|
|
* Don't want to just detach on exit, so...
|
|
*/
|
|
tcp->flags &= ~TCB_ATTACHED;
|
|
/*
|
|
* Make parent go away.
|
|
* Also makes grandparent's wait() unblock.
|
|
*/
|
|
kill(getppid(), SIGKILL);
|
|
}
|
|
|
|
#endif /* !USE_PROCFS */
|
|
if (!qflag)
|
|
fprintf(stderr,
|
|
"Process %u attached - interrupt to quit\n",
|
|
tcp->pid);
|
|
}
|
|
|
|
if (interactive)
|
|
sigprocmask(SIG_SETMASK, &empty_set, NULL);
|
|
}
|
|
|
|
static void
|
|
startup_child (char **argv)
|
|
{
|
|
struct stat statbuf;
|
|
const char *filename;
|
|
char pathname[MAXPATHLEN];
|
|
int pid = 0;
|
|
struct tcb *tcp;
|
|
|
|
filename = argv[0];
|
|
if (strchr(filename, '/')) {
|
|
if (strlen(filename) > sizeof pathname - 1) {
|
|
errno = ENAMETOOLONG;
|
|
perror("strace: exec");
|
|
exit(1);
|
|
}
|
|
strcpy(pathname, filename);
|
|
}
|
|
#ifdef USE_DEBUGGING_EXEC
|
|
/*
|
|
* Debuggers customarily check the current directory
|
|
* first regardless of the path but doing that gives
|
|
* security geeks a panic attack.
|
|
*/
|
|
else if (stat(filename, &statbuf) == 0)
|
|
strcpy(pathname, filename);
|
|
#endif /* USE_DEBUGGING_EXEC */
|
|
else {
|
|
char *path;
|
|
int m, n, len;
|
|
|
|
for (path = getenv("PATH"); path && *path; path += m) {
|
|
if (strchr(path, ':')) {
|
|
n = strchr(path, ':') - path;
|
|
m = n + 1;
|
|
}
|
|
else
|
|
m = n = strlen(path);
|
|
if (n == 0) {
|
|
if (!getcwd(pathname, MAXPATHLEN))
|
|
continue;
|
|
len = strlen(pathname);
|
|
}
|
|
else if (n > sizeof pathname - 1)
|
|
continue;
|
|
else {
|
|
strncpy(pathname, path, n);
|
|
len = n;
|
|
}
|
|
if (len && pathname[len - 1] != '/')
|
|
pathname[len++] = '/';
|
|
strcpy(pathname + len, filename);
|
|
if (stat(pathname, &statbuf) == 0 &&
|
|
/* Accept only regular files
|
|
with some execute bits set.
|
|
XXX not perfect, might still fail */
|
|
S_ISREG(statbuf.st_mode) &&
|
|
(statbuf.st_mode & 0111))
|
|
break;
|
|
}
|
|
}
|
|
if (stat(pathname, &statbuf) < 0) {
|
|
fprintf(stderr, "%s: %s: command not found\n",
|
|
progname, filename);
|
|
exit(1);
|
|
}
|
|
strace_child = pid = fork();
|
|
if (pid < 0) {
|
|
perror("strace: fork");
|
|
cleanup();
|
|
exit(1);
|
|
}
|
|
if ((pid != 0 && daemonized_tracer) /* parent: to become a traced process */
|
|
|| (pid == 0 && !daemonized_tracer) /* child: to become a traced process */
|
|
) {
|
|
pid = getpid();
|
|
#ifdef USE_PROCFS
|
|
if (outf != stderr) close (fileno (outf));
|
|
#ifdef MIPS
|
|
/* Kludge for SGI, see proc_open for details. */
|
|
sa.sa_handler = foobar;
|
|
sa.sa_flags = 0;
|
|
sigemptyset(&sa.sa_mask);
|
|
sigaction(SIGINT, &sa, NULL);
|
|
#endif /* MIPS */
|
|
#ifndef FREEBSD
|
|
pause();
|
|
#else /* FREEBSD */
|
|
kill(pid, SIGSTOP); /* stop HERE */
|
|
#endif /* FREEBSD */
|
|
#else /* !USE_PROCFS */
|
|
if (outf!=stderr)
|
|
close(fileno (outf));
|
|
|
|
if (!daemonized_tracer) {
|
|
if (ptrace(PTRACE_TRACEME, 0, (char *) 1, 0) < 0) {
|
|
perror("strace: ptrace(PTRACE_TRACEME, ...)");
|
|
exit(1);
|
|
}
|
|
if (debug)
|
|
kill(pid, SIGSTOP);
|
|
}
|
|
|
|
if (username != NULL || geteuid() == 0) {
|
|
uid_t run_euid = run_uid;
|
|
gid_t run_egid = run_gid;
|
|
|
|
if (statbuf.st_mode & S_ISUID)
|
|
run_euid = statbuf.st_uid;
|
|
if (statbuf.st_mode & S_ISGID)
|
|
run_egid = statbuf.st_gid;
|
|
|
|
/*
|
|
* It is important to set groups before we
|
|
* lose privileges on setuid.
|
|
*/
|
|
if (username != NULL) {
|
|
if (initgroups(username, run_gid) < 0) {
|
|
perror("initgroups");
|
|
exit(1);
|
|
}
|
|
if (setregid(run_gid, run_egid) < 0) {
|
|
perror("setregid");
|
|
exit(1);
|
|
}
|
|
if (setreuid(run_uid, run_euid) < 0) {
|
|
perror("setreuid");
|
|
exit(1);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
setreuid(run_uid, run_uid);
|
|
|
|
if (!daemonized_tracer) {
|
|
/*
|
|
* Induce an immediate stop so that the parent
|
|
* will resume us with PTRACE_SYSCALL and display
|
|
* this execve call normally.
|
|
*/
|
|
kill(getpid(), SIGSTOP);
|
|
} else {
|
|
struct sigaction sv_sigchld;
|
|
sigaction(SIGCHLD, NULL, &sv_sigchld);
|
|
/*
|
|
* Make sure it is not SIG_IGN, otherwise wait
|
|
* will not block.
|
|
*/
|
|
signal(SIGCHLD, SIG_DFL);
|
|
/*
|
|
* Wait for grandchild to attach to us.
|
|
* It kills child after that, and wait() unblocks.
|
|
*/
|
|
alarm(3);
|
|
wait(NULL);
|
|
alarm(0);
|
|
sigaction(SIGCHLD, &sv_sigchld, NULL);
|
|
}
|
|
#endif /* !USE_PROCFS */
|
|
|
|
execv(pathname, argv);
|
|
perror("strace: exec");
|
|
_exit(1);
|
|
}
|
|
|
|
/* We are the tracer. */
|
|
tcp = alloctcb(daemonized_tracer ? getppid() : pid);
|
|
if (daemonized_tracer) {
|
|
/* We want subsequent startup_attach() to attach to it. */
|
|
tcp->flags |= TCB_ATTACHED;
|
|
}
|
|
#ifdef USE_PROCFS
|
|
if (proc_open(tcp, 0) < 0) {
|
|
fprintf(stderr, "trouble opening proc file\n");
|
|
cleanup();
|
|
exit(1);
|
|
}
|
|
#endif /* USE_PROCFS */
|
|
}
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
struct tcb *tcp;
|
|
int c, pid = 0;
|
|
int optF = 0;
|
|
struct sigaction sa;
|
|
|
|
static char buf[BUFSIZ];
|
|
|
|
progname = argv[0] ? argv[0] : "strace";
|
|
|
|
uname(&utsname_buf);
|
|
|
|
/* Allocate the initial tcbtab. */
|
|
tcbtabsize = argc; /* Surely enough for all -p args. */
|
|
if ((tcbtab = calloc(tcbtabsize, sizeof tcbtab[0])) == NULL) {
|
|
fprintf(stderr, "%s: out of memory\n", progname);
|
|
exit(1);
|
|
}
|
|
if ((tcbtab[0] = calloc(tcbtabsize, sizeof tcbtab[0][0])) == NULL) {
|
|
fprintf(stderr, "%s: out of memory\n", progname);
|
|
exit(1);
|
|
}
|
|
for (tcp = tcbtab[0]; tcp < &tcbtab[0][tcbtabsize]; ++tcp)
|
|
tcbtab[tcp - tcbtab[0]] = &tcbtab[0][tcp - tcbtab[0]];
|
|
|
|
outf = stderr;
|
|
interactive = 1;
|
|
set_sortby(DEFAULT_SORTBY);
|
|
set_personality(DEFAULT_PERSONALITY);
|
|
qualify("trace=all");
|
|
qualify("abbrev=all");
|
|
qualify("verbose=all");
|
|
qualify("signal=all");
|
|
while ((c = getopt(argc, argv,
|
|
"+cdfFhiqrtTvVxz"
|
|
#ifndef USE_PROCFS
|
|
"D"
|
|
#endif
|
|
"a:e:o:O:p:s:S:u:E:")) != EOF) {
|
|
switch (c) {
|
|
case 'c':
|
|
cflag++;
|
|
dtime++;
|
|
break;
|
|
case 'd':
|
|
debug++;
|
|
break;
|
|
#ifndef USE_PROCFS
|
|
/* Experimental, not documented in manpage yet. */
|
|
case 'D':
|
|
daemonized_tracer = 1;
|
|
break;
|
|
#endif
|
|
case 'F':
|
|
optF = 1;
|
|
break;
|
|
case 'f':
|
|
followfork++;
|
|
break;
|
|
case 'h':
|
|
usage(stdout, 0);
|
|
break;
|
|
case 'i':
|
|
iflag++;
|
|
break;
|
|
case 'q':
|
|
qflag++;
|
|
break;
|
|
case 'r':
|
|
rflag++;
|
|
tflag++;
|
|
break;
|
|
case 't':
|
|
tflag++;
|
|
break;
|
|
case 'T':
|
|
dtime++;
|
|
break;
|
|
case 'x':
|
|
xflag++;
|
|
break;
|
|
case 'v':
|
|
qualify("abbrev=none");
|
|
break;
|
|
case 'V':
|
|
printf("%s -- version %s\n", PACKAGE_NAME, VERSION);
|
|
exit(0);
|
|
break;
|
|
case 'z':
|
|
not_failing_only = 1;
|
|
break;
|
|
case 'a':
|
|
acolumn = atoi(optarg);
|
|
break;
|
|
case 'e':
|
|
qualify(optarg);
|
|
break;
|
|
case 'o':
|
|
outfname = strdup(optarg);
|
|
break;
|
|
case 'O':
|
|
set_overhead(atoi(optarg));
|
|
break;
|
|
case 'p':
|
|
if ((pid = atoi(optarg)) <= 0) {
|
|
fprintf(stderr, "%s: Invalid process id: %s\n",
|
|
progname, optarg);
|
|
break;
|
|
}
|
|
if (pid == getpid()) {
|
|
fprintf(stderr, "%s: I'm sorry, I can't let you do that, Dave.\n", progname);
|
|
break;
|
|
}
|
|
tcp = alloc_tcb(pid, 0);
|
|
tcp->flags |= TCB_ATTACHED;
|
|
pflag_seen++;
|
|
break;
|
|
case 's':
|
|
max_strlen = atoi(optarg);
|
|
if (max_strlen < 0) {
|
|
fprintf(stderr,
|
|
"%s: invalid -s argument: %s\n",
|
|
progname, optarg);
|
|
exit(1);
|
|
}
|
|
break;
|
|
case 'S':
|
|
set_sortby(optarg);
|
|
break;
|
|
case 'u':
|
|
username = strdup(optarg);
|
|
break;
|
|
case 'E':
|
|
if (putenv(optarg) < 0) {
|
|
fprintf(stderr, "%s: out of memory\n",
|
|
progname);
|
|
exit(1);
|
|
}
|
|
break;
|
|
default:
|
|
usage(stderr, 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((optind == argc) == !pflag_seen)
|
|
usage(stderr, 1);
|
|
|
|
if (!followfork)
|
|
followfork = optF;
|
|
|
|
if (followfork > 1 && cflag) {
|
|
fprintf(stderr,
|
|
"%s: -c and -ff are mutually exclusive options\n",
|
|
progname);
|
|
exit(1);
|
|
}
|
|
|
|
/* See if they want to run as another user. */
|
|
if (username != NULL) {
|
|
struct passwd *pent;
|
|
|
|
if (getuid() != 0 || geteuid() != 0) {
|
|
fprintf(stderr,
|
|
"%s: you must be root to use the -u option\n",
|
|
progname);
|
|
exit(1);
|
|
}
|
|
if ((pent = getpwnam(username)) == NULL) {
|
|
fprintf(stderr, "%s: cannot find user `%s'\n",
|
|
progname, username);
|
|
exit(1);
|
|
}
|
|
run_uid = pent->pw_uid;
|
|
run_gid = pent->pw_gid;
|
|
}
|
|
else {
|
|
run_uid = getuid();
|
|
run_gid = getgid();
|
|
}
|
|
|
|
/* Check if they want to redirect the output. */
|
|
if (outfname) {
|
|
/* See if they want to pipe the output. */
|
|
if (outfname[0] == '|' || outfname[0] == '!') {
|
|
/*
|
|
* We can't do the <outfname>.PID funny business
|
|
* when using popen, so prohibit it.
|
|
*/
|
|
if (followfork > 1) {
|
|
fprintf(stderr, "\
|
|
%s: piping the output and -ff are mutually exclusive options\n",
|
|
progname);
|
|
exit(1);
|
|
}
|
|
|
|
if ((outf = strace_popen(outfname + 1)) == NULL)
|
|
exit(1);
|
|
}
|
|
else if (followfork <= 1 &&
|
|
(outf = strace_fopen(outfname, "w")) == NULL)
|
|
exit(1);
|
|
}
|
|
|
|
if (!outfname || outfname[0] == '|' || outfname[0] == '!')
|
|
setvbuf(outf, buf, _IOLBF, BUFSIZ);
|
|
if (outfname && optind < argc) {
|
|
interactive = 0;
|
|
qflag = 1;
|
|
}
|
|
/* Valid states here:
|
|
optind < argc pflag_seen outfname interactive
|
|
1 0 0 1
|
|
0 1 0 1
|
|
1 0 1 0
|
|
0 1 1 1
|
|
*/
|
|
|
|
/* STARTUP_CHILD must be called before the signal handlers get
|
|
installed below as they are inherited into the spawned process.
|
|
Also we do not need to be protected by them as during interruption
|
|
in the STARTUP_CHILD mode we kill the spawned process anyway. */
|
|
if (!pflag_seen)
|
|
startup_child(&argv[optind]);
|
|
|
|
sigemptyset(&empty_set);
|
|
sigemptyset(&blocked_set);
|
|
sa.sa_handler = SIG_IGN;
|
|
sigemptyset(&sa.sa_mask);
|
|
sa.sa_flags = 0;
|
|
sigaction(SIGTTOU, &sa, NULL);
|
|
sigaction(SIGTTIN, &sa, NULL);
|
|
if (interactive) {
|
|
sigaddset(&blocked_set, SIGHUP);
|
|
sigaddset(&blocked_set, SIGINT);
|
|
sigaddset(&blocked_set, SIGQUIT);
|
|
sigaddset(&blocked_set, SIGPIPE);
|
|
sigaddset(&blocked_set, SIGTERM);
|
|
sa.sa_handler = interrupt;
|
|
#ifdef SUNOS4
|
|
/* POSIX signals on sunos4.1 are a little broken. */
|
|
sa.sa_flags = SA_INTERRUPT;
|
|
#endif /* SUNOS4 */
|
|
}
|
|
sigaction(SIGHUP, &sa, NULL);
|
|
sigaction(SIGINT, &sa, NULL);
|
|
sigaction(SIGQUIT, &sa, NULL);
|
|
sigaction(SIGPIPE, &sa, NULL);
|
|
sigaction(SIGTERM, &sa, NULL);
|
|
#ifdef USE_PROCFS
|
|
sa.sa_handler = reaper;
|
|
sigaction(SIGCHLD, &sa, NULL);
|
|
#else
|
|
/* Make sure SIGCHLD has the default action so that waitpid
|
|
definitely works without losing track of children. The user
|
|
should not have given us a bogus state to inherit, but he might
|
|
have. Arguably we should detect SIG_IGN here and pass it on
|
|
to children, but probably noone really needs that. */
|
|
sa.sa_handler = SIG_DFL;
|
|
sigaction(SIGCHLD, &sa, NULL);
|
|
#endif /* USE_PROCFS */
|
|
|
|
if (pflag_seen || daemonized_tracer)
|
|
startup_attach();
|
|
|
|
if (trace() < 0)
|
|
exit(1);
|
|
cleanup();
|
|
fflush(NULL);
|
|
if (exit_code > 0xff) {
|
|
/* Child was killed by a signal, mimic that. */
|
|
exit_code &= 0xff;
|
|
signal(exit_code, SIG_DFL);
|
|
raise(exit_code);
|
|
/* Paranoia - what if this signal is not fatal?
|
|
Exit with 128 + signo then. */
|
|
exit_code += 128;
|
|
}
|
|
exit(exit_code);
|
|
}
|
|
|
|
void
|
|
expand_tcbtab(void)
|
|
{
|
|
/* Allocate some more TCBs and expand the table.
|
|
We don't want to relocate the TCBs because our
|
|
callers have pointers and it would be a pain.
|
|
So tcbtab is a table of pointers. Since we never
|
|
free the TCBs, we allocate a single chunk of many. */
|
|
struct tcb **newtab = (struct tcb **)
|
|
realloc(tcbtab, 2 * tcbtabsize * sizeof tcbtab[0]);
|
|
struct tcb *newtcbs = (struct tcb *) calloc(tcbtabsize,
|
|
sizeof *newtcbs);
|
|
int i;
|
|
if (newtab == NULL || newtcbs == NULL) {
|
|
fprintf(stderr, "%s: expand_tcbtab: out of memory\n",
|
|
progname);
|
|
cleanup();
|
|
exit(1);
|
|
}
|
|
for (i = tcbtabsize; i < 2 * tcbtabsize; ++i)
|
|
newtab[i] = &newtcbs[i - tcbtabsize];
|
|
tcbtabsize *= 2;
|
|
tcbtab = newtab;
|
|
}
|
|
|
|
struct tcb *
|
|
alloc_tcb(int pid, int command_options_parsed)
|
|
{
|
|
int i;
|
|
struct tcb *tcp;
|
|
|
|
if (nprocs == tcbtabsize)
|
|
expand_tcbtab();
|
|
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
tcp = tcbtab[i];
|
|
if ((tcp->flags & TCB_INUSE) == 0) {
|
|
memset(tcp, 0, sizeof(*tcp));
|
|
tcp->pid = pid;
|
|
tcp->flags = TCB_INUSE | TCB_STARTUP;
|
|
tcp->outf = outf; /* Initialise to current out file */
|
|
tcp->pfd = -1;
|
|
nprocs++;
|
|
if (command_options_parsed)
|
|
newoutf(tcp);
|
|
return tcp;
|
|
}
|
|
}
|
|
fprintf(stderr, "%s: bug in alloc_tcb\n", progname);
|
|
cleanup();
|
|
exit(1);
|
|
}
|
|
|
|
#ifdef USE_PROCFS
|
|
int
|
|
proc_open(struct tcb *tcp, int attaching)
|
|
{
|
|
char proc[32];
|
|
long arg;
|
|
#ifdef SVR4
|
|
int i;
|
|
sysset_t syscalls;
|
|
sigset_t signals;
|
|
fltset_t faults;
|
|
#endif
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
static int last_pfd;
|
|
#endif
|
|
|
|
#ifdef HAVE_MP_PROCFS
|
|
/* Open the process pseudo-files in /proc. */
|
|
sprintf(proc, "/proc/%d/ctl", tcp->pid);
|
|
if ((tcp->pfd = open(proc, O_WRONLY|O_EXCL)) < 0) {
|
|
perror("strace: open(\"/proc/...\", ...)");
|
|
return -1;
|
|
}
|
|
if (set_cloexec_flag(tcp->pfd) < 0) {
|
|
return -1;
|
|
}
|
|
sprintf(proc, "/proc/%d/status", tcp->pid);
|
|
if ((tcp->pfd_stat = open(proc, O_RDONLY|O_EXCL)) < 0) {
|
|
perror("strace: open(\"/proc/...\", ...)");
|
|
return -1;
|
|
}
|
|
if (set_cloexec_flag(tcp->pfd_stat) < 0) {
|
|
return -1;
|
|
}
|
|
sprintf(proc, "/proc/%d/as", tcp->pid);
|
|
if ((tcp->pfd_as = open(proc, O_RDONLY|O_EXCL)) < 0) {
|
|
perror("strace: open(\"/proc/...\", ...)");
|
|
return -1;
|
|
}
|
|
if (set_cloexec_flag(tcp->pfd_as) < 0) {
|
|
return -1;
|
|
}
|
|
#else
|
|
/* Open the process pseudo-file in /proc. */
|
|
#ifndef FREEBSD
|
|
sprintf(proc, "/proc/%d", tcp->pid);
|
|
if ((tcp->pfd = open(proc, O_RDWR|O_EXCL)) < 0) {
|
|
#else /* FREEBSD */
|
|
sprintf(proc, "/proc/%d/mem", tcp->pid);
|
|
if ((tcp->pfd = open(proc, O_RDWR)) < 0) {
|
|
#endif /* FREEBSD */
|
|
perror("strace: open(\"/proc/...\", ...)");
|
|
return -1;
|
|
}
|
|
if (set_cloexec_flag(tcp->pfd) < 0) {
|
|
return -1;
|
|
}
|
|
#endif
|
|
#ifdef FREEBSD
|
|
sprintf(proc, "/proc/%d/regs", tcp->pid);
|
|
if ((tcp->pfd_reg = open(proc, O_RDONLY)) < 0) {
|
|
perror("strace: open(\"/proc/.../regs\", ...)");
|
|
return -1;
|
|
}
|
|
if (cflag) {
|
|
sprintf(proc, "/proc/%d/status", tcp->pid);
|
|
if ((tcp->pfd_status = open(proc, O_RDONLY)) < 0) {
|
|
perror("strace: open(\"/proc/.../status\", ...)");
|
|
return -1;
|
|
}
|
|
} else
|
|
tcp->pfd_status = -1;
|
|
#endif /* FREEBSD */
|
|
rebuild_pollv();
|
|
if (!attaching) {
|
|
/*
|
|
* Wait for the child to pause. Because of a race
|
|
* condition we have to poll for the event.
|
|
*/
|
|
for (;;) {
|
|
if (IOCTL_STATUS (tcp) < 0) {
|
|
perror("strace: PIOCSTATUS");
|
|
return -1;
|
|
}
|
|
if (tcp->status.PR_FLAGS & PR_ASLEEP)
|
|
break;
|
|
}
|
|
}
|
|
#ifndef FREEBSD
|
|
/* Stop the process so that we own the stop. */
|
|
if (IOCTL(tcp->pfd, PIOCSTOP, (char *)NULL) < 0) {
|
|
perror("strace: PIOCSTOP");
|
|
return -1;
|
|
}
|
|
#endif
|
|
#ifdef PIOCSET
|
|
/* Set Run-on-Last-Close. */
|
|
arg = PR_RLC;
|
|
if (IOCTL(tcp->pfd, PIOCSET, &arg) < 0) {
|
|
perror("PIOCSET PR_RLC");
|
|
return -1;
|
|
}
|
|
/* Set or Reset Inherit-on-Fork. */
|
|
arg = PR_FORK;
|
|
if (IOCTL(tcp->pfd, followfork ? PIOCSET : PIOCRESET, &arg) < 0) {
|
|
perror("PIOC{SET,RESET} PR_FORK");
|
|
return -1;
|
|
}
|
|
#else /* !PIOCSET */
|
|
#ifndef FREEBSD
|
|
if (ioctl(tcp->pfd, PIOCSRLC) < 0) {
|
|
perror("PIOCSRLC");
|
|
return -1;
|
|
}
|
|
if (ioctl(tcp->pfd, followfork ? PIOCSFORK : PIOCRFORK) < 0) {
|
|
perror("PIOC{S,R}FORK");
|
|
return -1;
|
|
}
|
|
#else /* FREEBSD */
|
|
/* just unset the PF_LINGER flag for the Run-on-Last-Close. */
|
|
if (ioctl(tcp->pfd, PIOCGFL, &arg) < 0) {
|
|
perror("PIOCGFL");
|
|
return -1;
|
|
}
|
|
arg &= ~PF_LINGER;
|
|
if (ioctl(tcp->pfd, PIOCSFL, arg) < 0) {
|
|
perror("PIOCSFL");
|
|
return -1;
|
|
}
|
|
#endif /* FREEBSD */
|
|
#endif /* !PIOCSET */
|
|
#ifndef FREEBSD
|
|
/* Enable all syscall entries we care about. */
|
|
premptyset(&syscalls);
|
|
for (i = 1; i < MAX_QUALS; ++i) {
|
|
if (i > (sizeof syscalls) * CHAR_BIT) break;
|
|
if (qual_flags [i] & QUAL_TRACE) praddset (&syscalls, i);
|
|
}
|
|
praddset (&syscalls, SYS_execve);
|
|
if (followfork) {
|
|
praddset (&syscalls, SYS_fork);
|
|
#ifdef SYS_forkall
|
|
praddset (&syscalls, SYS_forkall);
|
|
#endif
|
|
#ifdef SYS_fork1
|
|
praddset (&syscalls, SYS_fork1);
|
|
#endif
|
|
#ifdef SYS_rfork1
|
|
praddset (&syscalls, SYS_rfork1);
|
|
#endif
|
|
#ifdef SYS_rforkall
|
|
praddset (&syscalls, SYS_rforkall);
|
|
#endif
|
|
}
|
|
if (IOCTL(tcp->pfd, PIOCSENTRY, &syscalls) < 0) {
|
|
perror("PIOCSENTRY");
|
|
return -1;
|
|
}
|
|
/* Enable the syscall exits. */
|
|
if (IOCTL(tcp->pfd, PIOCSEXIT, &syscalls) < 0) {
|
|
perror("PIOSEXIT");
|
|
return -1;
|
|
}
|
|
/* Enable signals we care about. */
|
|
premptyset(&signals);
|
|
for (i = 1; i < MAX_QUALS; ++i) {
|
|
if (i > (sizeof signals) * CHAR_BIT) break;
|
|
if (qual_flags [i] & QUAL_SIGNAL) praddset (&signals, i);
|
|
}
|
|
if (IOCTL(tcp->pfd, PIOCSTRACE, &signals) < 0) {
|
|
perror("PIOCSTRACE");
|
|
return -1;
|
|
}
|
|
/* Enable faults we care about */
|
|
premptyset(&faults);
|
|
for (i = 1; i < MAX_QUALS; ++i) {
|
|
if (i > (sizeof faults) * CHAR_BIT) break;
|
|
if (qual_flags [i] & QUAL_FAULT) praddset (&faults, i);
|
|
}
|
|
if (IOCTL(tcp->pfd, PIOCSFAULT, &faults) < 0) {
|
|
perror("PIOCSFAULT");
|
|
return -1;
|
|
}
|
|
#else /* FREEBSD */
|
|
/* set events flags. */
|
|
arg = S_SIG | S_SCE | S_SCX ;
|
|
if(ioctl(tcp->pfd, PIOCBIS, arg) < 0) {
|
|
perror("PIOCBIS");
|
|
return -1;
|
|
}
|
|
#endif /* FREEBSD */
|
|
if (!attaching) {
|
|
#ifdef MIPS
|
|
/*
|
|
* The SGI PRSABORT doesn't work for pause() so
|
|
* we send it a caught signal to wake it up.
|
|
*/
|
|
kill(tcp->pid, SIGINT);
|
|
#else /* !MIPS */
|
|
#ifdef PRSABORT
|
|
/* The child is in a pause(), abort it. */
|
|
arg = PRSABORT;
|
|
if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) {
|
|
perror("PIOCRUN");
|
|
return -1;
|
|
}
|
|
#endif
|
|
#endif /* !MIPS*/
|
|
#ifdef FREEBSD
|
|
/* wake up the child if it received the SIGSTOP */
|
|
kill(tcp->pid, SIGCONT);
|
|
#endif
|
|
for (;;) {
|
|
/* Wait for the child to do something. */
|
|
if (IOCTL_WSTOP (tcp) < 0) {
|
|
perror("PIOCWSTOP");
|
|
return -1;
|
|
}
|
|
if (tcp->status.PR_WHY == PR_SYSENTRY) {
|
|
tcp->flags &= ~TCB_INSYSCALL;
|
|
get_scno(tcp);
|
|
if (known_scno(tcp) == SYS_execve)
|
|
break;
|
|
}
|
|
/* Set it running: maybe execve will be next. */
|
|
#ifndef FREEBSD
|
|
arg = 0;
|
|
if (IOCTL(tcp->pfd, PIOCRUN, &arg) < 0) {
|
|
#else /* FREEBSD */
|
|
if (IOCTL(tcp->pfd, PIOCRUN, 0) < 0) {
|
|
#endif /* FREEBSD */
|
|
perror("PIOCRUN");
|
|
return -1;
|
|
}
|
|
#ifdef FREEBSD
|
|
/* handle the case where we "opened" the child before
|
|
it did the kill -STOP */
|
|
if (tcp->status.PR_WHY == PR_SIGNALLED &&
|
|
tcp->status.PR_WHAT == SIGSTOP)
|
|
kill(tcp->pid, SIGCONT);
|
|
#endif
|
|
}
|
|
#ifndef FREEBSD
|
|
}
|
|
#else /* FREEBSD */
|
|
} else {
|
|
if (attaching < 2) {
|
|
/* We are attaching to an already running process.
|
|
* Try to figure out the state of the process in syscalls,
|
|
* to handle the first event well.
|
|
* This is done by having a look at the "wchan" property of the
|
|
* process, which tells where it is stopped (if it is). */
|
|
FILE * status;
|
|
char wchan[20]; /* should be enough */
|
|
|
|
sprintf(proc, "/proc/%d/status", tcp->pid);
|
|
status = fopen(proc, "r");
|
|
if (status &&
|
|
(fscanf(status, "%*s %*d %*d %*d %*d %*d,%*d %*s %*d,%*d"
|
|
"%*d,%*d %*d,%*d %19s", wchan) == 1) &&
|
|
strcmp(wchan, "nochan") && strcmp(wchan, "spread") &&
|
|
strcmp(wchan, "stopevent")) {
|
|
/* The process is asleep in the middle of a syscall.
|
|
Fake the syscall entry event */
|
|
tcp->flags &= ~(TCB_INSYSCALL|TCB_STARTUP);
|
|
tcp->status.PR_WHY = PR_SYSENTRY;
|
|
trace_syscall(tcp);
|
|
}
|
|
if (status)
|
|
fclose(status);
|
|
} /* otherwise it's a fork being followed */
|
|
}
|
|
#endif /* FREEBSD */
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
if (proc_poll_pipe[0] != -1)
|
|
proc_poller(tcp->pfd);
|
|
else if (nprocs > 1) {
|
|
proc_poll_open();
|
|
proc_poller(last_pfd);
|
|
proc_poller(tcp->pfd);
|
|
}
|
|
last_pfd = tcp->pfd;
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
return 0;
|
|
}
|
|
|
|
#endif /* USE_PROCFS */
|
|
|
|
struct tcb *
|
|
pid2tcb(pid)
|
|
int pid;
|
|
{
|
|
int i;
|
|
struct tcb *tcp;
|
|
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
tcp = tcbtab[i];
|
|
if (pid && tcp->pid != pid)
|
|
continue;
|
|
if (tcp->flags & TCB_INUSE)
|
|
return tcp;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef USE_PROCFS
|
|
|
|
static struct tcb *
|
|
pfd2tcb(pfd)
|
|
int pfd;
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
struct tcb *tcp = tcbtab[i];
|
|
if (tcp->pfd != pfd)
|
|
continue;
|
|
if (tcp->flags & TCB_INUSE)
|
|
return tcp;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
#endif /* USE_PROCFS */
|
|
|
|
void
|
|
droptcb(tcp)
|
|
struct tcb *tcp;
|
|
{
|
|
if (tcp->pid == 0)
|
|
return;
|
|
#ifdef TCB_CLONE_THREAD
|
|
if (tcp->nclone_threads > 0) {
|
|
/* There are other threads left in this process, but this
|
|
is the one whose PID represents the whole process.
|
|
We need to keep this record around as a zombie until
|
|
all the threads die. */
|
|
tcp->flags |= TCB_EXITING;
|
|
return;
|
|
}
|
|
#endif
|
|
nprocs--;
|
|
tcp->pid = 0;
|
|
|
|
if (tcp->parent != NULL) {
|
|
tcp->parent->nchildren--;
|
|
#ifdef TCB_CLONE_THREAD
|
|
if (tcp->flags & TCB_CLONE_DETACHED)
|
|
tcp->parent->nclone_detached--;
|
|
if (tcp->flags & TCB_CLONE_THREAD)
|
|
tcp->parent->nclone_threads--;
|
|
#endif
|
|
#ifdef TCB_CLONE_DETACHED
|
|
if (!(tcp->flags & TCB_CLONE_DETACHED))
|
|
#endif
|
|
tcp->parent->nzombies++;
|
|
#ifdef LINUX
|
|
/* Update `tcp->parent->parent->nchildren' and the other fields
|
|
like NCLONE_DETACHED, only for zombie group leader that has
|
|
already reported and been short-circuited at the top of this
|
|
function. The same condition as at the top of DETACH. */
|
|
if ((tcp->flags & TCB_CLONE_THREAD) &&
|
|
tcp->parent->nclone_threads == 0 &&
|
|
(tcp->parent->flags & TCB_EXITING))
|
|
droptcb(tcp->parent);
|
|
#endif
|
|
tcp->parent = NULL;
|
|
}
|
|
|
|
tcp->flags = 0;
|
|
if (tcp->pfd != -1) {
|
|
close(tcp->pfd);
|
|
tcp->pfd = -1;
|
|
#ifdef FREEBSD
|
|
if (tcp->pfd_reg != -1) {
|
|
close(tcp->pfd_reg);
|
|
tcp->pfd_reg = -1;
|
|
}
|
|
if (tcp->pfd_status != -1) {
|
|
close(tcp->pfd_status);
|
|
tcp->pfd_status = -1;
|
|
}
|
|
#endif /* !FREEBSD */
|
|
#ifdef USE_PROCFS
|
|
rebuild_pollv(); /* Note, flags needs to be cleared by now. */
|
|
#endif
|
|
}
|
|
|
|
if (outfname && followfork > 1 && tcp->outf)
|
|
fclose(tcp->outf);
|
|
|
|
tcp->outf = 0;
|
|
}
|
|
|
|
#ifndef USE_PROCFS
|
|
|
|
static int
|
|
resume(tcp)
|
|
struct tcb *tcp;
|
|
{
|
|
if (tcp == NULL)
|
|
return -1;
|
|
|
|
if (!(tcp->flags & TCB_SUSPENDED)) {
|
|
fprintf(stderr, "PANIC: pid %u not suspended\n", tcp->pid);
|
|
return -1;
|
|
}
|
|
tcp->flags &= ~TCB_SUSPENDED;
|
|
#ifdef TCB_CLONE_THREAD
|
|
if (tcp->flags & TCB_CLONE_THREAD)
|
|
tcp->parent->nclone_waiting--;
|
|
#endif
|
|
|
|
if (ptrace_restart(PTRACE_SYSCALL, tcp, 0) < 0)
|
|
return -1;
|
|
|
|
if (!qflag)
|
|
fprintf(stderr, "Process %u resumed\n", tcp->pid);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
resume_from_tcp (struct tcb *tcp)
|
|
{
|
|
int error = 0;
|
|
int resumed = 0;
|
|
|
|
/* XXX This won't always be quite right (but it never was).
|
|
A waiter with argument 0 or < -1 is waiting for any pid in
|
|
a particular pgrp, which this child might or might not be
|
|
in. The waiter will only wake up if it's argument is -1
|
|
or if it's waiting for tcp->pid's pgrp. It makes a
|
|
difference to wake up a waiter when there might be more
|
|
traced children, because it could get a false ECHILD
|
|
error. OTOH, if this was the last child in the pgrp, then
|
|
it ought to wake up and get ECHILD. We would have to
|
|
search the system for all pid's in the pgrp to be sure.
|
|
|
|
&& (t->waitpid == -1 ||
|
|
(t->waitpid == 0 && getpgid (tcp->pid) == getpgid (t->pid))
|
|
|| (t->waitpid < 0 && t->waitpid == -getpid (t->pid)))
|
|
*/
|
|
|
|
if (tcp->parent &&
|
|
(tcp->parent->flags & TCB_SUSPENDED) &&
|
|
(tcp->parent->waitpid <= 0 || tcp->parent->waitpid == tcp->pid)) {
|
|
error = resume(tcp->parent);
|
|
++resumed;
|
|
}
|
|
#ifdef TCB_CLONE_THREAD
|
|
if (tcp->parent && tcp->parent->nclone_waiting > 0) {
|
|
/* Some other threads of our parent are waiting too. */
|
|
unsigned int i;
|
|
|
|
/* Resume all the threads that were waiting for this PID. */
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
struct tcb *t = tcbtab[i];
|
|
if (t->parent == tcp->parent && t != tcp
|
|
&& ((t->flags & (TCB_CLONE_THREAD|TCB_SUSPENDED))
|
|
== (TCB_CLONE_THREAD|TCB_SUSPENDED))
|
|
&& t->waitpid == tcp->pid) {
|
|
error |= resume (t);
|
|
++resumed;
|
|
}
|
|
}
|
|
if (resumed == 0)
|
|
/* Noone was waiting for this PID in particular,
|
|
so now we might need to resume some wildcarders. */
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
struct tcb *t = tcbtab[i];
|
|
if (t->parent == tcp->parent && t != tcp
|
|
&& ((t->flags
|
|
& (TCB_CLONE_THREAD|TCB_SUSPENDED))
|
|
== (TCB_CLONE_THREAD|TCB_SUSPENDED))
|
|
&& t->waitpid <= 0
|
|
) {
|
|
error |= resume (t);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return error;
|
|
}
|
|
|
|
#endif /* !USE_PROCFS */
|
|
|
|
/* detach traced process; continue with sig
|
|
Never call DETACH twice on the same process as both unattached and
|
|
attached-unstopped processes give the same ESRCH. For unattached process we
|
|
would SIGSTOP it and wait for its SIGSTOP notification forever. */
|
|
|
|
static int
|
|
detach(tcp, sig)
|
|
struct tcb *tcp;
|
|
int sig;
|
|
{
|
|
int error = 0;
|
|
#ifdef LINUX
|
|
int status, catch_sigstop;
|
|
struct tcb *zombie = NULL;
|
|
|
|
/* If the group leader is lingering only because of this other
|
|
thread now dying, then detach the leader as well. */
|
|
if ((tcp->flags & TCB_CLONE_THREAD) &&
|
|
tcp->parent->nclone_threads == 1 &&
|
|
(tcp->parent->flags & TCB_EXITING))
|
|
zombie = tcp->parent;
|
|
#endif
|
|
|
|
if (tcp->flags & TCB_BPTSET)
|
|
sig = SIGKILL;
|
|
|
|
#ifdef LINUX
|
|
/*
|
|
* Linux wrongly insists the child be stopped
|
|
* before detaching. Arghh. We go through hoops
|
|
* to make a clean break of things.
|
|
*/
|
|
#if defined(SPARC)
|
|
#undef PTRACE_DETACH
|
|
#define PTRACE_DETACH PTRACE_SUNDETACH
|
|
#endif
|
|
/*
|
|
* On TCB_STARTUP we did PTRACE_ATTACH but still did not get the
|
|
* expected SIGSTOP. We must catch exactly one as otherwise the
|
|
* detached process would be left stopped (process state T).
|
|
*/
|
|
catch_sigstop = (tcp->flags & TCB_STARTUP);
|
|
if ((error = ptrace(PTRACE_DETACH, tcp->pid, (char *) 1, sig)) == 0) {
|
|
/* On a clear day, you can see forever. */
|
|
}
|
|
else if (errno != ESRCH) {
|
|
/* Shouldn't happen. */
|
|
perror("detach: ptrace(PTRACE_DETACH, ...)");
|
|
}
|
|
else if (my_tgkill((tcp->flags & TCB_CLONE_THREAD ? tcp->parent->pid
|
|
: tcp->pid),
|
|
tcp->pid, 0) < 0) {
|
|
if (errno != ESRCH)
|
|
perror("detach: checking sanity");
|
|
}
|
|
else if (!catch_sigstop && my_tgkill((tcp->flags & TCB_CLONE_THREAD
|
|
? tcp->parent->pid : tcp->pid),
|
|
tcp->pid, SIGSTOP) < 0) {
|
|
if (errno != ESRCH)
|
|
perror("detach: stopping child");
|
|
}
|
|
else
|
|
catch_sigstop = 1;
|
|
if (catch_sigstop) {
|
|
for (;;) {
|
|
#ifdef __WALL
|
|
if (wait4(tcp->pid, &status, __WALL, NULL) < 0) {
|
|
if (errno == ECHILD) /* Already gone. */
|
|
break;
|
|
if (errno != EINVAL) {
|
|
perror("detach: waiting");
|
|
break;
|
|
}
|
|
#endif /* __WALL */
|
|
/* No __WALL here. */
|
|
if (waitpid(tcp->pid, &status, 0) < 0) {
|
|
if (errno != ECHILD) {
|
|
perror("detach: waiting");
|
|
break;
|
|
}
|
|
#ifdef __WCLONE
|
|
/* If no processes, try clones. */
|
|
if (wait4(tcp->pid, &status, __WCLONE,
|
|
NULL) < 0) {
|
|
if (errno != ECHILD)
|
|
perror("detach: waiting");
|
|
break;
|
|
}
|
|
#endif /* __WCLONE */
|
|
}
|
|
#ifdef __WALL
|
|
}
|
|
#endif
|
|
if (!WIFSTOPPED(status)) {
|
|
/* Au revoir, mon ami. */
|
|
break;
|
|
}
|
|
if (WSTOPSIG(status) == SIGSTOP) {
|
|
ptrace_restart(PTRACE_DETACH, tcp, sig);
|
|
break;
|
|
}
|
|
error = ptrace_restart(PTRACE_CONT, tcp,
|
|
WSTOPSIG(status) == ptrace_stop_sig ? 0
|
|
: WSTOPSIG(status));
|
|
if (error < 0)
|
|
break;
|
|
}
|
|
}
|
|
#endif /* LINUX */
|
|
|
|
#if defined(SUNOS4)
|
|
/* PTRACE_DETACH won't respect `sig' argument, so we post it here. */
|
|
if (sig && kill(tcp->pid, sig) < 0)
|
|
perror("detach: kill");
|
|
sig = 0;
|
|
error = ptrace_restart(PTRACE_DETACH, tcp, sig);
|
|
#endif /* SUNOS4 */
|
|
|
|
#ifndef USE_PROCFS
|
|
error |= resume_from_tcp (tcp);
|
|
#endif
|
|
|
|
if (!qflag)
|
|
fprintf(stderr, "Process %u detached\n", tcp->pid);
|
|
|
|
droptcb(tcp);
|
|
|
|
#ifdef LINUX
|
|
if (zombie != NULL) {
|
|
/* TCP no longer exists therefore you must not detach () it. */
|
|
droptcb(zombie);
|
|
}
|
|
#endif
|
|
|
|
return error;
|
|
}
|
|
|
|
#ifdef USE_PROCFS
|
|
|
|
static void
|
|
reaper(sig)
|
|
int sig;
|
|
{
|
|
int pid;
|
|
int status;
|
|
|
|
while ((pid = waitpid(-1, &status, WNOHANG)) > 0) {
|
|
#if 0
|
|
struct tcb *tcp;
|
|
|
|
tcp = pid2tcb(pid);
|
|
if (tcp)
|
|
droptcb(tcp);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
#endif /* USE_PROCFS */
|
|
|
|
static void
|
|
cleanup()
|
|
{
|
|
int i;
|
|
struct tcb *tcp;
|
|
|
|
for (i = 0; i < tcbtabsize; i++) {
|
|
tcp = tcbtab[i];
|
|
if (!(tcp->flags & TCB_INUSE))
|
|
continue;
|
|
if (debug)
|
|
fprintf(stderr,
|
|
"cleanup: looking at pid %u\n", tcp->pid);
|
|
if (tcp_last &&
|
|
(!outfname || followfork < 2 || tcp_last == tcp)) {
|
|
tprintf(" <unfinished ...>");
|
|
printtrailer();
|
|
}
|
|
if (tcp->flags & TCB_ATTACHED)
|
|
detach(tcp, 0);
|
|
else {
|
|
kill(tcp->pid, SIGCONT);
|
|
kill(tcp->pid, SIGTERM);
|
|
}
|
|
}
|
|
if (cflag)
|
|
call_summary(outf);
|
|
}
|
|
|
|
static void
|
|
interrupt(sig)
|
|
int sig;
|
|
{
|
|
interrupted = 1;
|
|
}
|
|
|
|
#ifndef HAVE_STRERROR
|
|
|
|
#if !HAVE_DECL_SYS_ERRLIST
|
|
extern int sys_nerr;
|
|
extern char *sys_errlist[];
|
|
#endif /* HAVE_DECL_SYS_ERRLIST */
|
|
|
|
const char *
|
|
strerror(errno)
|
|
int errno;
|
|
{
|
|
static char buf[64];
|
|
|
|
if (errno < 1 || errno >= sys_nerr) {
|
|
sprintf(buf, "Unknown error %d", errno);
|
|
return buf;
|
|
}
|
|
return sys_errlist[errno];
|
|
}
|
|
|
|
#endif /* HAVE_STERRROR */
|
|
|
|
#ifndef HAVE_STRSIGNAL
|
|
|
|
#if defined HAVE_SYS_SIGLIST && !defined HAVE_DECL_SYS_SIGLIST
|
|
extern char *sys_siglist[];
|
|
#endif
|
|
#if defined HAVE_SYS__SIGLIST && !defined HAVE_DECL__SYS_SIGLIST
|
|
extern char *_sys_siglist[];
|
|
#endif
|
|
|
|
const char *
|
|
strsignal(sig)
|
|
int sig;
|
|
{
|
|
static char buf[64];
|
|
|
|
if (sig < 1 || sig >= NSIG) {
|
|
sprintf(buf, "Unknown signal %d", sig);
|
|
return buf;
|
|
}
|
|
#ifdef HAVE__SYS_SIGLIST
|
|
return _sys_siglist[sig];
|
|
#else
|
|
return sys_siglist[sig];
|
|
#endif
|
|
}
|
|
|
|
#endif /* HAVE_STRSIGNAL */
|
|
|
|
#ifdef USE_PROCFS
|
|
|
|
static void
|
|
rebuild_pollv()
|
|
{
|
|
int i, j;
|
|
|
|
if (pollv != NULL)
|
|
free (pollv);
|
|
pollv = (struct pollfd *) malloc(nprocs * sizeof pollv[0]);
|
|
if (pollv == NULL) {
|
|
fprintf(stderr, "%s: out of memory\n", progname);
|
|
exit(1);
|
|
}
|
|
|
|
for (i = j = 0; i < tcbtabsize; i++) {
|
|
struct tcb *tcp = tcbtab[i];
|
|
if (!(tcp->flags & TCB_INUSE))
|
|
continue;
|
|
pollv[j].fd = tcp->pfd;
|
|
pollv[j].events = POLLWANT;
|
|
j++;
|
|
}
|
|
if (j != nprocs) {
|
|
fprintf(stderr, "strace: proc miscount\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
|
|
static void
|
|
proc_poll_open()
|
|
{
|
|
int i;
|
|
|
|
if (pipe(proc_poll_pipe) < 0) {
|
|
perror("pipe");
|
|
exit(1);
|
|
}
|
|
for (i = 0; i < 2; i++) {
|
|
if (set_cloexec_flag(proc_poll_pipe[i]) < 0) {
|
|
exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int
|
|
proc_poll(pollv, nfds, timeout)
|
|
struct pollfd *pollv;
|
|
int nfds;
|
|
int timeout;
|
|
{
|
|
int i;
|
|
int n;
|
|
struct proc_pollfd pollinfo;
|
|
|
|
if ((n = read(proc_poll_pipe[0], &pollinfo, sizeof(pollinfo))) < 0)
|
|
return n;
|
|
if (n != sizeof(struct proc_pollfd)) {
|
|
fprintf(stderr, "panic: short read: %d\n", n);
|
|
exit(1);
|
|
}
|
|
for (i = 0; i < nprocs; i++) {
|
|
if (pollv[i].fd == pollinfo.fd)
|
|
pollv[i].revents = pollinfo.revents;
|
|
else
|
|
pollv[i].revents = 0;
|
|
}
|
|
poller_pid = pollinfo.pid;
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
wakeup_handler(sig)
|
|
int sig;
|
|
{
|
|
}
|
|
|
|
static void
|
|
proc_poller(pfd)
|
|
int pfd;
|
|
{
|
|
struct proc_pollfd pollinfo;
|
|
struct sigaction sa;
|
|
sigset_t blocked_set, empty_set;
|
|
int i;
|
|
int n;
|
|
struct rlimit rl;
|
|
#ifdef FREEBSD
|
|
struct procfs_status pfs;
|
|
#endif /* FREEBSD */
|
|
|
|
switch (fork()) {
|
|
case -1:
|
|
perror("fork");
|
|
_exit(1);
|
|
case 0:
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
sa.sa_handler = interactive ? SIG_DFL : SIG_IGN;
|
|
sa.sa_flags = 0;
|
|
sigemptyset(&sa.sa_mask);
|
|
sigaction(SIGHUP, &sa, NULL);
|
|
sigaction(SIGINT, &sa, NULL);
|
|
sigaction(SIGQUIT, &sa, NULL);
|
|
sigaction(SIGPIPE, &sa, NULL);
|
|
sigaction(SIGTERM, &sa, NULL);
|
|
sa.sa_handler = wakeup_handler;
|
|
sigaction(SIGUSR1, &sa, NULL);
|
|
sigemptyset(&blocked_set);
|
|
sigaddset(&blocked_set, SIGUSR1);
|
|
sigprocmask(SIG_BLOCK, &blocked_set, NULL);
|
|
sigemptyset(&empty_set);
|
|
|
|
if (getrlimit(RLIMIT_NOFILE, &rl) < 0) {
|
|
perror("getrlimit(RLIMIT_NOFILE, ...)");
|
|
_exit(1);
|
|
}
|
|
n = rl.rlim_cur;
|
|
for (i = 0; i < n; i++) {
|
|
if (i != pfd && i != proc_poll_pipe[1])
|
|
close(i);
|
|
}
|
|
|
|
pollinfo.fd = pfd;
|
|
pollinfo.pid = getpid();
|
|
for (;;) {
|
|
#ifndef FREEBSD
|
|
if (ioctl(pfd, PIOCWSTOP, NULL) < 0)
|
|
#else /* FREEBSD */
|
|
if (ioctl(pfd, PIOCWSTOP, &pfs) < 0)
|
|
#endif /* FREEBSD */
|
|
{
|
|
switch (errno) {
|
|
case EINTR:
|
|
continue;
|
|
case EBADF:
|
|
pollinfo.revents = POLLERR;
|
|
break;
|
|
case ENOENT:
|
|
pollinfo.revents = POLLHUP;
|
|
break;
|
|
default:
|
|
perror("proc_poller: PIOCWSTOP");
|
|
}
|
|
write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo));
|
|
_exit(0);
|
|
}
|
|
pollinfo.revents = POLLWANT;
|
|
write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo));
|
|
sigsuspend(&empty_set);
|
|
}
|
|
}
|
|
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
|
|
static int
|
|
choose_pfd()
|
|
{
|
|
int i, j;
|
|
struct tcb *tcp;
|
|
|
|
static int last;
|
|
|
|
if (followfork < 2 &&
|
|
last < nprocs && (pollv[last].revents & POLLWANT)) {
|
|
/*
|
|
* The previous process is ready to run again. We'll
|
|
* let it do so if it is currently in a syscall. This
|
|
* heuristic improves the readability of the trace.
|
|
*/
|
|
tcp = pfd2tcb(pollv[last].fd);
|
|
if (tcp && (tcp->flags & TCB_INSYSCALL))
|
|
return pollv[last].fd;
|
|
}
|
|
|
|
for (i = 0; i < nprocs; i++) {
|
|
/* Let competing children run round robin. */
|
|
j = (i + last + 1) % nprocs;
|
|
if (pollv[j].revents & (POLLHUP | POLLERR)) {
|
|
tcp = pfd2tcb(pollv[j].fd);
|
|
if (!tcp) {
|
|
fprintf(stderr, "strace: lost proc\n");
|
|
exit(1);
|
|
}
|
|
droptcb(tcp);
|
|
return -1;
|
|
}
|
|
if (pollv[j].revents & POLLWANT) {
|
|
last = j;
|
|
return pollv[j].fd;
|
|
}
|
|
}
|
|
fprintf(stderr, "strace: nothing ready\n");
|
|
exit(1);
|
|
}
|
|
|
|
static int
|
|
trace()
|
|
{
|
|
#ifdef POLL_HACK
|
|
struct tcb *in_syscall = NULL;
|
|
#endif
|
|
struct tcb *tcp;
|
|
int pfd;
|
|
int what;
|
|
int ioctl_result = 0, ioctl_errno = 0;
|
|
long arg;
|
|
|
|
for (;;) {
|
|
if (interactive)
|
|
sigprocmask(SIG_SETMASK, &empty_set, NULL);
|
|
|
|
if (nprocs == 0)
|
|
break;
|
|
|
|
switch (nprocs) {
|
|
case 1:
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
if (proc_poll_pipe[0] == -1) {
|
|
#endif
|
|
tcp = pid2tcb(0);
|
|
if (!tcp)
|
|
continue;
|
|
pfd = tcp->pfd;
|
|
if (pfd == -1)
|
|
continue;
|
|
break;
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
}
|
|
/* fall through ... */
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
default:
|
|
#ifdef HAVE_POLLABLE_PROCFS
|
|
#ifdef POLL_HACK
|
|
/* On some systems (e.g. UnixWare) we get too much ugly
|
|
"unfinished..." stuff when multiple proceses are in
|
|
syscalls. Here's a nasty hack */
|
|
|
|
if (in_syscall) {
|
|
struct pollfd pv;
|
|
tcp = in_syscall;
|
|
in_syscall = NULL;
|
|
pv.fd = tcp->pfd;
|
|
pv.events = POLLWANT;
|
|
if ((what = poll (&pv, 1, 1)) < 0) {
|
|
if (interrupted)
|
|
return 0;
|
|
continue;
|
|
}
|
|
else if (what == 1 && pv.revents & POLLWANT) {
|
|
goto FOUND;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (poll(pollv, nprocs, INFTIM) < 0) {
|
|
if (interrupted)
|
|
return 0;
|
|
continue;
|
|
}
|
|
#else /* !HAVE_POLLABLE_PROCFS */
|
|
if (proc_poll(pollv, nprocs, INFTIM) < 0) {
|
|
if (interrupted)
|
|
return 0;
|
|
continue;
|
|
}
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
pfd = choose_pfd();
|
|
if (pfd == -1)
|
|
continue;
|
|
break;
|
|
}
|
|
|
|
/* Look up `pfd' in our table. */
|
|
if ((tcp = pfd2tcb(pfd)) == NULL) {
|
|
fprintf(stderr, "unknown pfd: %u\n", pfd);
|
|
exit(1);
|
|
}
|
|
#ifdef POLL_HACK
|
|
FOUND:
|
|
#endif
|
|
/* Get the status of the process. */
|
|
if (!interrupted) {
|
|
#ifndef FREEBSD
|
|
ioctl_result = IOCTL_WSTOP (tcp);
|
|
#else /* FREEBSD */
|
|
/* Thanks to some scheduling mystery, the first poller
|
|
sometimes waits for the already processed end of fork
|
|
event. Doing a non blocking poll here solves the problem. */
|
|
if (proc_poll_pipe[0] != -1)
|
|
ioctl_result = IOCTL_STATUS (tcp);
|
|
else
|
|
ioctl_result = IOCTL_WSTOP (tcp);
|
|
#endif /* FREEBSD */
|
|
ioctl_errno = errno;
|
|
#ifndef HAVE_POLLABLE_PROCFS
|
|
if (proc_poll_pipe[0] != -1) {
|
|
if (ioctl_result < 0)
|
|
kill(poller_pid, SIGKILL);
|
|
else
|
|
kill(poller_pid, SIGUSR1);
|
|
}
|
|
#endif /* !HAVE_POLLABLE_PROCFS */
|
|
}
|
|
if (interrupted)
|
|
return 0;
|
|
|
|
if (interactive)
|
|
sigprocmask(SIG_BLOCK, &blocked_set, NULL);
|
|
|
|
if (ioctl_result < 0) {
|
|
/* Find out what happened if it failed. */
|
|
switch (ioctl_errno) {
|
|
case EINTR:
|
|
case EBADF:
|
|
continue;
|
|
#ifdef FREEBSD
|
|
case ENOTTY:
|
|
#endif
|
|
case ENOENT:
|
|
droptcb(tcp);
|
|
continue;
|
|
default:
|
|
perror("PIOCWSTOP");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
#ifdef FREEBSD
|
|
if ((tcp->flags & TCB_STARTUP) && (tcp->status.PR_WHY == PR_SYSEXIT)) {
|
|
/* discard first event for a syscall we never entered */
|
|
IOCTL (tcp->pfd, PIOCRUN, 0);
|
|
continue;
|
|
}
|
|
#endif
|
|
|
|
/* clear the just started flag */
|
|
tcp->flags &= ~TCB_STARTUP;
|
|
|
|
/* set current output file */
|
|
outf = tcp->outf;
|
|
|
|
if (cflag) {
|
|
struct timeval stime;
|
|
#ifdef FREEBSD
|
|
char buf[1024];
|
|
int len;
|
|
|
|
if ((len = pread(tcp->pfd_status, buf, sizeof(buf) - 1, 0)) > 0) {
|
|
buf[len] = '\0';
|
|
sscanf(buf,
|
|
"%*s %*d %*d %*d %*d %*d,%*d %*s %*d,%*d %*d,%*d %ld,%ld",
|
|
&stime.tv_sec, &stime.tv_usec);
|
|
} else
|
|
stime.tv_sec = stime.tv_usec = 0;
|
|
#else /* !FREEBSD */
|
|
stime.tv_sec = tcp->status.pr_stime.tv_sec;
|
|
stime.tv_usec = tcp->status.pr_stime.tv_nsec/1000;
|
|
#endif /* !FREEBSD */
|
|
tv_sub(&tcp->dtime, &stime, &tcp->stime);
|
|
tcp->stime = stime;
|
|
}
|
|
what = tcp->status.PR_WHAT;
|
|
switch (tcp->status.PR_WHY) {
|
|
#ifndef FREEBSD
|
|
case PR_REQUESTED:
|
|
if (tcp->status.PR_FLAGS & PR_ASLEEP) {
|
|
tcp->status.PR_WHY = PR_SYSENTRY;
|
|
if (trace_syscall(tcp) < 0) {
|
|
fprintf(stderr, "syscall trouble\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
break;
|
|
#endif /* !FREEBSD */
|
|
case PR_SYSENTRY:
|
|
#ifdef POLL_HACK
|
|
in_syscall = tcp;
|
|
#endif
|
|
case PR_SYSEXIT:
|
|
if (trace_syscall(tcp) < 0) {
|
|
fprintf(stderr, "syscall trouble\n");
|
|
exit(1);
|
|
}
|
|
break;
|
|
case PR_SIGNALLED:
|
|
if (!cflag && (qual_flags[what] & QUAL_SIGNAL)) {
|
|
printleader(tcp);
|
|
tprintf("--- %s (%s) ---",
|
|
signame(what), strsignal(what));
|
|
printtrailer();
|
|
#ifdef PR_INFO
|
|
if (tcp->status.PR_INFO.si_signo == what) {
|
|
printleader(tcp);
|
|
tprintf(" siginfo=");
|
|
printsiginfo(&tcp->status.PR_INFO, 1);
|
|
printtrailer();
|
|
}
|
|
#endif
|
|
}
|
|
break;
|
|
case PR_FAULTED:
|
|
if (!cflag && (qual_flags[what] & QUAL_FAULT)) {
|
|
printleader(tcp);
|
|
tprintf("=== FAULT %d ===", what);
|
|
printtrailer();
|
|
}
|
|
break;
|
|
#ifdef FREEBSD
|
|
case 0: /* handle case we polled for nothing */
|
|
continue;
|
|
#endif
|
|
default:
|
|
fprintf(stderr, "odd stop %d\n", tcp->status.PR_WHY);
|
|
exit(1);
|
|
break;
|
|
}
|
|
arg = 0;
|
|
#ifndef FREEBSD
|
|
if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) {
|
|
#else
|
|
if (IOCTL (tcp->pfd, PIOCRUN, 0) < 0) {
|
|
#endif
|
|
perror("PIOCRUN");
|
|
exit(1);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#else /* !USE_PROCFS */
|
|
|
|
#ifdef TCB_GROUP_EXITING
|
|
/* Handle an exit detach or death signal that is taking all the
|
|
related clone threads with it. This is called in three circumstances:
|
|
SIG == -1 TCP has already died (TCB_ATTACHED is clear, strace is parent).
|
|
SIG == 0 Continuing TCP will perform an exit_group syscall.
|
|
SIG == other Continuing TCP with SIG will kill the process.
|
|
*/
|
|
static int
|
|
handle_group_exit(struct tcb *tcp, int sig)
|
|
{
|
|
/* We need to locate our records of all the clone threads
|
|
related to TCP, either its children or siblings. */
|
|
struct tcb *leader = NULL;
|
|
|
|
if (tcp->flags & TCB_CLONE_THREAD)
|
|
leader = tcp->parent;
|
|
else if (tcp->nclone_detached > 0)
|
|
leader = tcp;
|
|
|
|
if (sig < 0) {
|
|
if (leader != NULL && leader != tcp
|
|
&& !(leader->flags & TCB_GROUP_EXITING)
|
|
&& !(tcp->flags & TCB_STARTUP)
|
|
) {
|
|
fprintf(stderr,
|
|
"PANIC: handle_group_exit: %d leader %d\n",
|
|
tcp->pid, leader ? leader->pid : -1);
|
|
}
|
|
/* TCP no longer exists therefore you must not detach() it. */
|
|
#ifndef USE_PROCFS
|
|
resume_from_tcp(tcp);
|
|
#endif
|
|
droptcb(tcp); /* Already died. */
|
|
}
|
|
else {
|
|
/* Mark that we are taking the process down. */
|
|
tcp->flags |= TCB_EXITING | TCB_GROUP_EXITING;
|
|
if (tcp->flags & TCB_ATTACHED) {
|
|
detach(tcp, sig);
|
|
if (leader != NULL && leader != tcp)
|
|
leader->flags |= TCB_GROUP_EXITING;
|
|
} else {
|
|
if (ptrace_restart(PTRACE_CONT, tcp, sig) < 0) {
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
if (leader != NULL) {
|
|
leader->flags |= TCB_GROUP_EXITING;
|
|
if (leader != tcp)
|
|
droptcb(tcp);
|
|
}
|
|
/* The leader will report to us as parent now,
|
|
and then we'll get to the SIG==-1 case. */
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static struct tcb *
|
|
collect_stopped_tcbs(void)
|
|
{
|
|
#ifdef LINUX
|
|
static int remembered_pid;
|
|
static int remembered_status;
|
|
#endif
|
|
int pid;
|
|
int wait_errno;
|
|
int status;
|
|
struct tcb *tcp;
|
|
struct tcb *found_tcps;
|
|
#ifdef LINUX
|
|
struct tcb **nextp;
|
|
struct rusage ru;
|
|
struct rusage* ru_ptr = cflag ? &ru : NULL;
|
|
int wnohang = 0;
|
|
#ifdef __WALL
|
|
int wait4_options = __WALL;
|
|
#endif
|
|
|
|
if (remembered_pid > 0) {
|
|
pid = remembered_pid;
|
|
remembered_pid = 0;
|
|
if (debug)
|
|
fprintf(stderr, " [remembered wait(%#x) = %u]\n",
|
|
remembered_status, pid);
|
|
tcp = pid2tcb(pid); /* can't be NULL */
|
|
tcp->wait_status = remembered_status;
|
|
tcp->next_need_service = NULL;
|
|
return tcp;
|
|
}
|
|
nextp = &found_tcps;
|
|
#endif /* LINUX */
|
|
|
|
found_tcps = NULL;
|
|
while (1) {
|
|
#ifdef LINUX
|
|
#ifdef __WALL
|
|
pid = wait4(-1, &status, wait4_options | wnohang, ru_ptr);
|
|
if (pid < 0 && (wait4_options & __WALL) && errno == EINVAL) {
|
|
/* this kernel does not support __WALL */
|
|
wait4_options &= ~__WALL;
|
|
errno = 0;
|
|
pid = wait4(-1, &status, wait4_options | wnohang, ru_ptr);
|
|
}
|
|
if (pid < 0 && !(wait4_options & __WALL) && errno == ECHILD) {
|
|
/* most likely a "cloned" process */
|
|
pid = wait4(-1, &status, __WCLONE | wnohang, ru_ptr);
|
|
if (pid < 0 && errno != ECHILD) {
|
|
fprintf(stderr, "strace: wait4(WCLONE) "
|
|
"failed: %s\n", strerror(errno));
|
|
}
|
|
}
|
|
#else /* !__WALL */
|
|
pid = wait4(-1, &status, wnohang, ru_ptr);
|
|
#endif
|
|
#endif /* LINUX */
|
|
#ifdef SUNOS4
|
|
pid = wait(&status);
|
|
#endif /* SUNOS4 */
|
|
wait_errno = errno;
|
|
if (interactive)
|
|
sigprocmask(SIG_BLOCK, &blocked_set, NULL);
|
|
|
|
if (pid == 0 && wnohang) {
|
|
/* We had at least one successful
|
|
* wait() before. We waited
|
|
* with WNOHANG second time.
|
|
* Stop collecting more tracees,
|
|
* process what we already have.
|
|
*/
|
|
break;
|
|
}
|
|
if (pid == -1) {
|
|
if (wait_errno == EINTR)
|
|
continue;
|
|
if (wait_errno == ECHILD) {
|
|
/*
|
|
* We would like to verify this case
|
|
* but sometimes a race in Solbourne's
|
|
* version of SunOS sometimes reports
|
|
* ECHILD before sending us SIGCHILD.
|
|
*/
|
|
#if 0
|
|
if (nprocs != 0) {
|
|
fprintf(stderr, "strace: proc miscount\n");
|
|
exit(1);
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
errno = wait_errno;
|
|
perror("strace: wait");
|
|
exit(1);
|
|
}
|
|
if (pid == popen_pid) {
|
|
if (WIFEXITED(status) || WIFSIGNALED(status))
|
|
popen_pid = -1;
|
|
continue;
|
|
}
|
|
if (debug)
|
|
fprintf(stderr, " [wait(%#x) = %u]\n", status, pid);
|
|
|
|
/* RHEL5 bug workaround.
|
|
* It can re-report stopped tasks. Happens on SIGSTOPs here.
|
|
* Second (bogus) report has signal# set to 0.
|
|
* Stop collecting and process what we have.
|
|
*/
|
|
if (WIFSTOPPED(status) && WSTOPSIG(status) == 0)
|
|
break;
|
|
|
|
/* Look up `pid' in our table. */
|
|
if ((tcp = pid2tcb(pid)) == NULL) {
|
|
#ifdef LINUX
|
|
if (followfork) {
|
|
/* This is needed to go with the CLONE_PTRACE
|
|
changes in process.c/util.c: we might see
|
|
the child's initial trap before we see the
|
|
parent return from the clone syscall.
|
|
Leave the child suspended until the parent
|
|
returns from its system call. Only then
|
|
will we have the association of parent and
|
|
child so that we know how to do clearbpt
|
|
in the child. */
|
|
tcp = alloctcb(pid);
|
|
tcp->flags |= TCB_ATTACHED | TCB_SUSPENDED;
|
|
if (!qflag)
|
|
fprintf(stderr, "\
|
|
Process %d attached (waiting for parent)\n",
|
|
pid);
|
|
}
|
|
else
|
|
/* This can happen if a clone call used
|
|
CLONE_PTRACE itself. */
|
|
#endif /* LINUX */
|
|
{
|
|
fprintf(stderr, "unknown pid: %u\n", pid);
|
|
if (WIFSTOPPED(status))
|
|
ptrace(PTRACE_CONT, pid, (char *) 1, 0);
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
if (cflag) {
|
|
#ifdef LINUX
|
|
tv_sub(&tcp->dtime, &ru.ru_stime, &tcp->stime);
|
|
tcp->stime = ru.ru_stime;
|
|
#endif
|
|
}
|
|
if (tcp->flags & TCB_SUSPENDED) {
|
|
/*
|
|
* Apparently, doing any ptrace() call on a stopped
|
|
* process, provokes the kernel to report the process
|
|
* status again on a subsequent wait(), even if the
|
|
* process has not been actually restarted.
|
|
* Since we have inspected the arguments of suspended
|
|
* processes we end up here testing for this case.
|
|
*
|
|
* We also end up here when we catch new pid of
|
|
* CLONE_PTRACEd process. Do not process/restart it
|
|
* until we see corresponding clone() syscall exit
|
|
* in its parent.
|
|
*/
|
|
continue;
|
|
}
|
|
|
|
#ifdef LINUX
|
|
/* So far observed only on RHEL5 ia64, but I imagine this
|
|
* can legitimately happen elsewhere.
|
|
* If we waited and got a stopped task notification,
|
|
* subsequent wait may return the same pid again, for example,
|
|
* with SIGKILL notification. SIGKILL kills even stopped tasks.
|
|
* We must not add it to the list
|
|
* (one task can't be inserted twice in the list).
|
|
*/
|
|
{
|
|
struct tcb *f = found_tcps;
|
|
while (f) {
|
|
if (f == tcp) {
|
|
remembered_pid = pid;
|
|
remembered_status = status;
|
|
return found_tcps;
|
|
}
|
|
f = f->next_need_service;
|
|
}
|
|
}
|
|
/* It is important to not invert the order of tasks
|
|
* to process. For one, alloc_tcb() above picks newly forked
|
|
* threads in some order, processing of them and their parent
|
|
* should be in the same order, otherwise bad things happen
|
|
* (misinterpreted SIGSTOPs and such).
|
|
*/
|
|
tcp->wait_status = status;
|
|
*nextp = tcp;
|
|
nextp = &tcp->next_need_service;
|
|
*nextp = NULL;
|
|
wnohang = WNOHANG;
|
|
#endif
|
|
#ifdef SUNOS4
|
|
/* Probably need to replace wait with waitpid
|
|
* and loop on Sun too, but I can't test it. Volunteers?
|
|
*/
|
|
tcp->wait_status = status;
|
|
tcp->next_need_service = NULL;
|
|
found_tcps = tcp;
|
|
break;
|
|
#endif
|
|
} /* while (1) - collecting all stopped/exited tracees */
|
|
|
|
return found_tcps;
|
|
}
|
|
|
|
static int
|
|
handle_stopped_tcbs(struct tcb *tcp)
|
|
{
|
|
for (; tcp; tcp = tcp->next_need_service) {
|
|
int pid;
|
|
int status;
|
|
|
|
outf = tcp->outf;
|
|
status = tcp->wait_status;
|
|
pid = tcp->pid;
|
|
|
|
if (WIFSIGNALED(status)) {
|
|
if (pid == strace_child)
|
|
exit_code = 0x100 | WTERMSIG(status);
|
|
if (!cflag
|
|
&& (qual_flags[WTERMSIG(status)] & QUAL_SIGNAL)) {
|
|
printleader(tcp);
|
|
tprintf("+++ killed by %s %s+++",
|
|
signame(WTERMSIG(status)),
|
|
#ifdef WCOREDUMP
|
|
WCOREDUMP(status) ? "(core dumped) " :
|
|
#endif
|
|
"");
|
|
printtrailer();
|
|
}
|
|
#ifdef TCB_GROUP_EXITING
|
|
handle_group_exit(tcp, -1);
|
|
#else
|
|
droptcb(tcp);
|
|
#endif
|
|
continue;
|
|
}
|
|
if (WIFEXITED(status)) {
|
|
if (pid == strace_child)
|
|
exit_code = WEXITSTATUS(status);
|
|
if (debug)
|
|
fprintf(stderr, "pid %u exited with %d\n", pid, WEXITSTATUS(status));
|
|
if ((tcp->flags & (TCB_ATTACHED|TCB_STARTUP)) == TCB_ATTACHED
|
|
#ifdef TCB_GROUP_EXITING
|
|
&& !(tcp->parent && (tcp->parent->flags & TCB_GROUP_EXITING))
|
|
&& !(tcp->flags & TCB_GROUP_EXITING)
|
|
#endif
|
|
) {
|
|
fprintf(stderr,
|
|
"PANIC: attached pid %u exited with %d\n",
|
|
pid, WEXITSTATUS(status));
|
|
}
|
|
if (tcp == tcp_last) {
|
|
if ((tcp->flags & (TCB_INSYSCALL|TCB_REPRINT)) == TCB_INSYSCALL)
|
|
tprintf(" <unfinished ... exit status %d>\n",
|
|
WEXITSTATUS(status));
|
|
tcp_last = NULL;
|
|
}
|
|
#ifdef TCB_GROUP_EXITING
|
|
handle_group_exit(tcp, -1);
|
|
#else
|
|
droptcb(tcp);
|
|
#endif
|
|
continue;
|
|
}
|
|
if (!WIFSTOPPED(status)) {
|
|
fprintf(stderr, "PANIC: pid %u not stopped\n", pid);
|
|
droptcb(tcp);
|
|
continue;
|
|
}
|
|
if (debug)
|
|
fprintf(stderr, "pid %u stopped, [%s]\n",
|
|
pid, signame(WSTOPSIG(status)));
|
|
|
|
/*
|
|
* Interestingly, the process may stop
|
|
* with STOPSIG equal to some other signal
|
|
* than SIGSTOP if we happend to attach
|
|
* just before the process takes a signal.
|
|
*/
|
|
if ((tcp->flags & TCB_STARTUP) && WSTOPSIG(status) == SIGSTOP) {
|
|
/*
|
|
* This flag is there to keep us in sync.
|
|
* Next time this process stops it should
|
|
* really be entering a system call.
|
|
*/
|
|
tcp->flags &= ~TCB_STARTUP;
|
|
if (tcp->flags & TCB_BPTSET) {
|
|
/*
|
|
* One example is a breakpoint inherited from
|
|
* parent through fork ().
|
|
*/
|
|
if (clearbpt(tcp) < 0) /* Pretty fatal */ {
|
|
droptcb(tcp);
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
}
|
|
/* Add more OSes after you verified it works for them. */
|
|
/* PTRACE_SETOPTIONS may be an enum, not a #define.
|
|
* But sometimes we can test for it by checking PT_SETOPTIONS.
|
|
*/
|
|
#if defined LINUX && (defined PTRACE_SETOPTIONS || defined PT_SETOPTIONS)
|
|
# ifndef PTRACE_O_TRACESYSGOOD
|
|
# define PTRACE_O_TRACESYSGOOD 0x00000001
|
|
# endif
|
|
# ifndef PTRACE_O_TRACEEXEC
|
|
# define PTRACE_O_TRACEEXEC 0x00000010
|
|
# endif
|
|
# ifndef PTRACE_EVENT_EXEC
|
|
# define PTRACE_EVENT_EXEC 4
|
|
# endif
|
|
/*
|
|
* Ask kernel to set signo to SIGTRAP | 0x80
|
|
* on ptrace-generated SIGTRAPs, and mark
|
|
* execve's SIGTRAP with PTRACE_EVENT_EXEC.
|
|
*/
|
|
if (!ptrace_opts_set) {
|
|
char *p;
|
|
ptrace_opts_set = 1;
|
|
|
|
/* RHEL 2.6.18 definitely has crippling bugs */
|
|
/* Vanilla and Fedora 2.6.29 seems to work */
|
|
p = utsname_buf.release;
|
|
if (strtoul(p, &p, 10) < 2 || *p != '.')
|
|
goto tracing;
|
|
if (strtoul(++p, &p, 10) < 6 || *p != '.')
|
|
goto tracing;
|
|
if (strtoul(++p, &p, 10) < 29)
|
|
goto tracing;
|
|
/*
|
|
* NB: even if this "succeeds", we can
|
|
* revert back to SIGTRAP if we later see
|
|
* that it didnt really work.
|
|
* Old kernels are known to lie here.
|
|
*/
|
|
if (ptrace(PTRACE_SETOPTIONS, pid, (char *) 0,
|
|
(void *) (PTRACE_O_TRACESYSGOOD | PTRACE_O_TRACEEXEC)) == 0)
|
|
ptrace_stop_sig = SIGTRAP | 0x80;
|
|
}
|
|
#endif
|
|
goto tracing;
|
|
}
|
|
|
|
#if defined LINUX && (defined PTRACE_SETOPTIONS || defined PT_SETOPTIONS)
|
|
if (ptrace_stop_sig != SIGTRAP && WSTOPSIG(status) == SIGTRAP) {
|
|
/*
|
|
* We told ptrace to report SIGTRAP | 0x80 on this process
|
|
* but got bare SIGTRAP. This can be a genuine SIGTRAP:
|
|
* kill(pid, SIGTRAP), trap insn, etc;
|
|
* but be paranoid about it.
|
|
*/
|
|
if (((unsigned)status >> 16) == PTRACE_EVENT_EXEC) {
|
|
/* It's post-exec ptrace stop. Ignore it,
|
|
* we will get syscall exit ptrace stop later.
|
|
*/
|
|
#ifdef TCB_WAITEXECVE
|
|
tcp->flags &= ~TCB_WAITEXECVE;
|
|
#endif
|
|
goto tracing;
|
|
} else {
|
|
/* Take a better look... */
|
|
siginfo_t si;
|
|
si.si_signo = 0;
|
|
ptrace(PTRACE_GETSIGINFO, pid, (void*) 0, (void*) &si);
|
|
/*
|
|
* Check some fields to make sure we see
|
|
* real SIGTRAP.
|
|
* Otherwise interpret it as ptrace stop.
|
|
* Real SIGTRAPs (int3 insn on x86, kill() etc)
|
|
* have these values:
|
|
* int3: kill -TRAP $pid:
|
|
* si_signo:5 (SIGTRAP) si_signo:5 (SIGTRAP)
|
|
* si_errno:0 si_errno:(?)
|
|
* si_code:128 (SI_KERNEL) si_code:0 (SI_USER)
|
|
* si_pid:0 si_pid:(>0?)
|
|
* si_band:0 si_band:(?)
|
|
* Ptrace stops have garbage there instead.
|
|
*/
|
|
if (si.si_signo != SIGTRAP
|
|
|| (si.si_code != SI_KERNEL && si.si_code != SI_USER)
|
|
) {
|
|
fprintf(stderr, "bogus SIGTRAP (si_code:%x), assuming old kernel\n", si.si_code);
|
|
ptrace_stop_sig = SIGTRAP;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (WSTOPSIG(status) != ptrace_stop_sig) {
|
|
/* This isn't a ptrace stop. */
|
|
|
|
if (WSTOPSIG(status) == SIGSTOP &&
|
|
(tcp->flags & TCB_SIGTRAPPED)) {
|
|
/*
|
|
* Trapped attempt to block SIGTRAP
|
|
* Hope we are back in control now.
|
|
*/
|
|
tcp->flags &= ~(TCB_INSYSCALL | TCB_SIGTRAPPED);
|
|
if (ptrace_restart(PTRACE_SYSCALL, tcp, 0) < 0) {
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
continue;
|
|
}
|
|
if (!cflag
|
|
&& (qual_flags[WSTOPSIG(status)] & QUAL_SIGNAL)) {
|
|
unsigned long addr = 0;
|
|
long pc = 0;
|
|
#if defined(PT_CR_IPSR) && defined(PT_CR_IIP) && defined(PT_GETSIGINFO)
|
|
# define PSR_RI 41
|
|
struct siginfo si;
|
|
long psr;
|
|
|
|
upeek(tcp, PT_CR_IPSR, &psr);
|
|
upeek(tcp, PT_CR_IIP, &pc);
|
|
|
|
pc += (psr >> PSR_RI) & 0x3;
|
|
ptrace(PT_GETSIGINFO, pid, 0, (long) &si);
|
|
addr = (unsigned long) si.si_addr;
|
|
#elif defined PTRACE_GETSIGINFO
|
|
if (WSTOPSIG(status) == SIGSEGV ||
|
|
WSTOPSIG(status) == SIGBUS) {
|
|
siginfo_t si;
|
|
if (ptrace(PTRACE_GETSIGINFO, pid,
|
|
0, &si) == 0)
|
|
addr = (unsigned long)
|
|
si.si_addr;
|
|
}
|
|
#endif
|
|
printleader(tcp);
|
|
tprintf("--- %s (%s) @ %lx (%lx) ---",
|
|
signame(WSTOPSIG(status)),
|
|
strsignal(WSTOPSIG(status)), pc, addr);
|
|
printtrailer();
|
|
}
|
|
if (((tcp->flags & TCB_ATTACHED) ||
|
|
tcp->nclone_threads > 0) &&
|
|
!sigishandled(tcp, WSTOPSIG(status))) {
|
|
#ifdef TCB_GROUP_EXITING
|
|
handle_group_exit(tcp, WSTOPSIG(status));
|
|
#else
|
|
detach(tcp, WSTOPSIG(status));
|
|
#endif
|
|
continue;
|
|
}
|
|
if (ptrace_restart(PTRACE_SYSCALL, tcp, WSTOPSIG(status)) < 0) {
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
tcp->flags &= ~TCB_SUSPENDED;
|
|
continue;
|
|
}
|
|
/* we handled the STATUS, we are permitted to interrupt now. */
|
|
if (interrupted)
|
|
return 0;
|
|
if (trace_syscall(tcp) < 0 && !tcp->ptrace_errno) {
|
|
/* ptrace() failed in trace_syscall() with ESRCH.
|
|
* Likely a result of process disappearing mid-flight.
|
|
* Observed case: exit_group() terminating
|
|
* all processes in thread group. In this case, threads
|
|
* "disappear" in an unpredictable moment without any
|
|
* notification to strace via wait().
|
|
*/
|
|
if (tcp->flags & TCB_ATTACHED) {
|
|
if (tcp_last) {
|
|
/* Do we have dangling line "syscall(param, param"?
|
|
* Finish the line then. We cannot
|
|
*/
|
|
tcp_last->flags |= TCB_REPRINT;
|
|
tprintf(" <unfinished ...>");
|
|
printtrailer();
|
|
}
|
|
detach(tcp, 0);
|
|
} else {
|
|
ptrace(PTRACE_KILL,
|
|
tcp->pid, (char *) 1, SIGTERM);
|
|
droptcb(tcp);
|
|
}
|
|
continue;
|
|
}
|
|
if (tcp->flags & TCB_EXITING) {
|
|
#ifdef TCB_GROUP_EXITING
|
|
if (tcp->flags & TCB_GROUP_EXITING) {
|
|
if (handle_group_exit(tcp, 0) < 0)
|
|
return -1;
|
|
continue;
|
|
}
|
|
#endif
|
|
if (tcp->flags & TCB_ATTACHED)
|
|
detach(tcp, 0);
|
|
else if (ptrace_restart(PTRACE_CONT, tcp, 0) < 0) {
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
continue;
|
|
}
|
|
if (tcp->flags & TCB_SUSPENDED) {
|
|
if (!qflag)
|
|
fprintf(stderr, "Process %u suspended\n", pid);
|
|
continue;
|
|
}
|
|
tracing:
|
|
if (ptrace_restart(PTRACE_SYSCALL, tcp, 0) < 0) {
|
|
cleanup();
|
|
return -1;
|
|
}
|
|
} /* for each tcp */
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
trace()
|
|
{
|
|
int rc;
|
|
struct tcb *tcbs;
|
|
|
|
while (nprocs != 0) {
|
|
if (interrupted)
|
|
return 0;
|
|
if (interactive)
|
|
sigprocmask(SIG_SETMASK, &empty_set, NULL);
|
|
|
|
/* The loop of "wait for one tracee, serve it, repeat"
|
|
* may leave some tracees never served.
|
|
* Kernel provides no guarantees of fairness when you have
|
|
* many waitable tasks.
|
|
* Try strace -f with test/many_looping_threads.c example.
|
|
* To fix it, we collect *all* waitable tasks, then handle
|
|
* them all, then repeat.
|
|
*/
|
|
tcbs = collect_stopped_tcbs();
|
|
if (!tcbs)
|
|
break;
|
|
rc = handle_stopped_tcbs(tcbs);
|
|
if (rc)
|
|
return rc;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#endif /* !USE_PROCFS */
|
|
|
|
static int curcol;
|
|
|
|
#ifdef __STDC__
|
|
#include <stdarg.h>
|
|
#define VA_START(a, b) va_start(a, b)
|
|
#else
|
|
#include <varargs.h>
|
|
#define VA_START(a, b) va_start(a)
|
|
#endif
|
|
|
|
void
|
|
#ifdef __STDC__
|
|
tprintf(const char *fmt, ...)
|
|
#else
|
|
tprintf(fmt, va_alist)
|
|
char *fmt;
|
|
va_dcl
|
|
#endif
|
|
{
|
|
va_list args;
|
|
|
|
VA_START(args, fmt);
|
|
if (outf) {
|
|
int n = vfprintf(outf, fmt, args);
|
|
if (n < 0 && outf != stderr)
|
|
perror(outfname == NULL
|
|
? "<writing to pipe>" : outfname);
|
|
else
|
|
curcol += n;
|
|
}
|
|
va_end(args);
|
|
return;
|
|
}
|
|
|
|
void
|
|
printleader(struct tcb *tcp)
|
|
{
|
|
if (tcp_last) {
|
|
if (tcp_last->ptrace_errno) {
|
|
tcp_last->ptrace_errno = 0;
|
|
if (tcp_last->flags & TCB_INSYSCALL) {
|
|
tprintf(" <unavailable ...>\n");
|
|
tcp_last->flags |= TCB_REPRINT;
|
|
} else {
|
|
tprintf("= ? <unavailable>\n");
|
|
}
|
|
} else if (!outfname || followfork < 2 || tcp_last == tcp) {
|
|
tprintf(" <unfinished ...>\n");
|
|
tcp_last->flags |= TCB_REPRINT;
|
|
}
|
|
}
|
|
curcol = 0;
|
|
if ((followfork == 1 || pflag_seen > 1) && outfname)
|
|
tprintf("%-5d ", tcp->pid);
|
|
else if (nprocs > 1 && !outfname)
|
|
tprintf("[pid %5u] ", tcp->pid);
|
|
if (tflag) {
|
|
char str[sizeof("HH:MM:SS")];
|
|
struct timeval tv, dtv;
|
|
static struct timeval otv;
|
|
|
|
gettimeofday(&tv, NULL);
|
|
if (rflag) {
|
|
if (otv.tv_sec == 0)
|
|
otv = tv;
|
|
tv_sub(&dtv, &tv, &otv);
|
|
tprintf("%6ld.%06ld ",
|
|
(long) dtv.tv_sec, (long) dtv.tv_usec);
|
|
otv = tv;
|
|
}
|
|
else if (tflag > 2) {
|
|
tprintf("%ld.%06ld ",
|
|
(long) tv.tv_sec, (long) tv.tv_usec);
|
|
}
|
|
else {
|
|
time_t local = tv.tv_sec;
|
|
strftime(str, sizeof(str), "%T", localtime(&local));
|
|
if (tflag > 1)
|
|
tprintf("%s.%06ld ", str, (long) tv.tv_usec);
|
|
else
|
|
tprintf("%s ", str);
|
|
}
|
|
}
|
|
if (iflag)
|
|
printcall(tcp);
|
|
}
|
|
|
|
void
|
|
tabto(col)
|
|
int col;
|
|
{
|
|
if (curcol < col)
|
|
tprintf("%*s", col - curcol, "");
|
|
}
|
|
|
|
void
|
|
printtrailer(void)
|
|
{
|
|
tprintf("\n");
|
|
tcp_last = NULL;
|
|
}
|
|
|
|
#ifdef HAVE_MP_PROCFS
|
|
|
|
int
|
|
mp_ioctl(int fd, int cmd, void *arg, int size)
|
|
{
|
|
struct iovec iov[2];
|
|
int n = 1;
|
|
|
|
iov[0].iov_base = &cmd;
|
|
iov[0].iov_len = sizeof cmd;
|
|
if (arg) {
|
|
++n;
|
|
iov[1].iov_base = arg;
|
|
iov[1].iov_len = size;
|
|
}
|
|
|
|
return writev(fd, iov, n);
|
|
}
|
|
|
|
#endif
|