blob: f96eaa04d88c45fd2132078972343445f33feab4 [file] [log] [blame]
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001/*
2 * Copyright (c) 1991, 1992 Paul Kranenburg <pk@cs.few.eur.nl>
3 * Copyright (c) 1993 Branko Lankester <branko@hacktic.nl>
4 * Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
Wichert Akkerman4dc8a2a1999-12-23 14:20:14 +00005 * Copyright (c) 1996-1999 Wichert Akkerman <wichert@cistron.nl>
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00006 * All rights reserved.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. The name of the author may not be used to endorse or promote products
17 * derived from this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 *
30 * $Id$
31 */
32
Wichert Akkerman2ee6e452000-02-18 15:36:12 +000033#include <sys/types.h>
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000034#include "defs.h"
35
36#include <signal.h>
37#include <errno.h>
38#include <sys/param.h>
39#include <fcntl.h>
40#include <sys/resource.h>
41#include <sys/wait.h>
42#include <sys/stat.h>
43#include <pwd.h>
44#include <grp.h>
45#include <string.h>
46
47#ifdef SVR4
48#include <sys/stropts.h>
49#include <poll.h>
Wichert Akkermanea78f0f1999-11-29 15:34:02 +000050#ifdef HAVE_MP_PROCFS
Wichert Akkerman9ce1a631999-08-29 23:15:07 +000051#include <sys/uio.h>
52#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000053#endif
54
55int debug = 0, followfork = 0, followvfork = 0, interactive = 0;
56int rflag = 0, tflag = 0, dtime = 0, cflag = 0;
57int iflag = 0, xflag = 0, qflag = 0;
58int pflag_seen = 0;
59
60char *username = NULL;
61uid_t run_uid;
62gid_t run_gid;
63
64int acolumn = DEFAULT_ACOLUMN;
65int max_strlen = DEFAULT_STRLEN;
66char *outfname = NULL;
67FILE *outf;
68struct tcb tcbtab[MAX_PROCS];
69int nprocs;
70char *progname;
71extern char version[];
72extern char **environ;
73
74static struct tcb *pid2tcb P((int pid));
75static int trace P((void));
76static void cleanup P((void));
77static void interrupt P((int sig));
78static sigset_t empty_set, blocked_set;
79
80#ifdef HAVE_SIG_ATOMIC_T
81static volatile sig_atomic_t interrupted;
82#else /* !HAVE_SIG_ATOMIC_T */
83#ifdef __STDC__
84static volatile int interrupted;
85#else /* !__STDC__ */
86static int interrupted;
87#endif /* !__STDC__ */
88#endif /* !HAVE_SIG_ATOMIC_T */
89
90#ifdef SVR4
91
92static struct tcb *pfd2tcb P((int pfd));
93static void reaper P((int sig));
94static void rebuild_pollv P((void));
Wichert Akkermane68d61c1999-06-28 13:17:16 +000095struct pollfd pollv[MAX_PROCS];
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000096
97#ifndef HAVE_POLLABLE_PROCFS
98
99static void proc_poll_open P((void));
100static void proc_poller P((int pfd));
101
102struct proc_pollfd {
103 int fd;
104 int revents;
105 int pid;
106};
107
108static int poller_pid;
109static int proc_poll_pipe[2] = { -1, -1 };
110
111#endif /* !HAVE_POLLABLE_PROCFS */
112
Wichert Akkermanea78f0f1999-11-29 15:34:02 +0000113#ifdef HAVE_MP_PROCFS
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000114#define POLLWANT POLLWRNORM
115#else
116#define POLLWANT POLLPRI
117#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000118#endif /* SVR4 */
119
120static void
121usage(ofp, exitval)
122FILE *ofp;
123int exitval;
124{
125 fprintf(ofp, "\
126usage: strace [-dffhiqrtttTvVxx] [-a column] [-e expr] ... [-o file]\n\
127 [-p pid] ... [-s strsize] [-u username] [command [arg ...]]\n\
128 or: strace -c [-e expr] ... [-O overhead] [-S sortby] [command [arg ...]]\n\
129-c -- count time, calls, and errors for each syscall and report summary\n\
130-f -- follow forks, -ff -- with output into separate files\n\
131-F -- attempt to follow vforks, -h -- print help message\n\
132-i -- print instruction pointer at time of syscall\n\
133-q -- suppress messages about attaching, detaching, etc.\n\
134-r -- print relative timestamp, -t -- absolute timestamp, -tt -- with usecs\n\
135-T -- print time spent in each syscall, -V -- print version\n\
136-v -- verbose mode: print unabbreviated argv, stat, termio[s], etc. args\n\
137-x -- print non-ascii strings in hex, -xx -- print all strings in hex\n\
138-a column -- alignment COLUMN for printing syscall results (default %d)\n\
139-e expr -- a qualifying expression: option=[!]all or option=[!]val1[,val2]...\n\
140 options: trace, abbrev, verbose, raw, signal, read, or write\n\
141-o file -- send trace output to FILE instead of stderr\n\
142-O overhead -- set overhead for tracing syscalls to OVERHEAD usecs\n\
143-p pid -- trace process with process id PID, may be repeated\n\
144-s strsize -- limit length of print strings to STRSIZE chars (default %d)\n\
145-S sortby -- sort syscall counts by: time, calls, name, nothing (default %s)\n\
146-u username -- run command as username handling setuid and/or setgid\n\
147", DEFAULT_ACOLUMN, DEFAULT_STRLEN, DEFAULT_SORTBY);
148 exit(exitval);
149}
150
151#ifdef SVR4
152#ifdef MIPS
153void
154foobar()
155{
156}
157#endif /* MIPS */
158#endif /* SVR4 */
159
160int
161main(argc, argv)
162int argc;
163char *argv[];
164{
165 extern int optind;
166 extern char *optarg;
167 struct tcb *tcp;
168 int c, pid = 0;
169 struct sigaction sa;
170
171 static char buf[BUFSIZ];
172
173 progname = argv[0];
174 outf = stderr;
175 interactive = 1;
176 qualify("trace=all");
177 qualify("abbrev=all");
178 qualify("verbose=all");
179 qualify("signal=all");
180 set_sortby(DEFAULT_SORTBY);
181 set_personality(DEFAULT_PERSONALITY);
182 while ((c = getopt(argc, argv,
183 "+cdfFhiqrtTvVxa:e:o:O:p:s:S:u:")) != EOF) {
184 switch (c) {
185 case 'c':
186 cflag++;
187 dtime++;
188 break;
189 case 'd':
190 debug++;
191 break;
192 case 'f':
193 followfork++;
194 break;
195 case 'F':
196 followvfork++;
197 break;
198 case 'h':
199 usage(stdout, 0);
200 break;
201 case 'i':
202 iflag++;
203 break;
204 case 'q':
205 qflag++;
206 break;
207 case 'r':
208 rflag++;
209 tflag++;
210 break;
211 case 't':
212 tflag++;
213 break;
214 case 'T':
215 dtime++;
216 break;
217 case 'x':
218 xflag++;
219 break;
220 case 'v':
221 qualify("abbrev=none");
222 break;
223 case 'V':
224 printf("%s\n", version);
225 exit(0);
226 break;
227 case 'a':
228 acolumn = atoi(optarg);
229 break;
230 case 'e':
231 qualify(optarg);
232 break;
233 case 'o':
234 outfname = strdup(optarg);
235 break;
236 case 'O':
237 set_overhead(atoi(optarg));
238 break;
239 case 'p':
240 if ((pid = atoi(optarg)) == 0) {
241 fprintf(stderr, "%s: Invalid process id: %s\n",
242 progname, optarg);
243 break;
244 }
245 if (pid == getpid()) {
Wichert Akkerman54a47671999-10-17 00:57:34 +0000246 fprintf(stderr, "%s: I'm sorry, I can't let you do that, Dave.\n", progname);
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000247 break;
248 }
249 if ((tcp = alloctcb(pid)) == NULL) {
250 fprintf(stderr, "%s: tcb table full, please recompile strace\n",
251 progname);
252 exit(1);
253 }
254 tcp->flags |= TCB_ATTACHED;
255 pflag_seen++;
256 break;
257 case 's':
258 max_strlen = atoi(optarg);
259 break;
260 case 'S':
261 set_sortby(optarg);
262 break;
263 case 'u':
264 username = strdup(optarg);
265 break;
266 default:
267 usage(stderr, 1);
268 break;
269 }
270 }
271
272 /* See if they want to run as another user. */
273 if (username != NULL) {
274 struct passwd *pent;
275
276 if (getuid() != 0 || geteuid() != 0) {
277 fprintf(stderr,
278 "%s: you must be root to use the -u option\n",
279 progname);
280 exit(1);
281 }
282 if ((pent = getpwnam(username)) == NULL) {
283 fprintf(stderr, "%s: cannot find user `%s'\n",
284 progname, optarg);
285 exit(1);
286 }
287 run_uid = pent->pw_uid;
288 run_gid = pent->pw_gid;
289 }
290 else {
291 run_uid = getuid();
292 run_gid = getgid();
293 }
294
295#ifndef SVR4
296 setreuid(geteuid(), getuid());
297#endif
298
299 /* See if they want to pipe the output. */
300 if (outfname && (outfname[0] == '|' || outfname[0] == '!')) {
301 if ((outf = popen(outfname + 1, "w")) == NULL) {
302 fprintf(stderr, "%s: can't popen '%s': %s\n",
303 progname, outfname + 1, strerror(errno));
304 exit(1);
305 }
306 free(outfname);
307 outfname = NULL;
308 }
309
310 /* Check if they want to redirect the output. */
311 if (outfname) {
312 if ((outf = fopen(outfname, "w")) == NULL) {
313 fprintf(stderr, "%s: can't fopen '%s': %s\n",
314 progname, outfname, strerror(errno));
315 exit(1);
316 }
317 }
318
319#ifndef SVR4
320 setreuid(geteuid(), getuid());
321#endif
322
323 if (!outfname) {
324 qflag = 1;
325 setvbuf(outf, buf, _IOLBF, BUFSIZ);
326 }
327 else if (optind < argc)
328 interactive = 0;
329 else
330 qflag = 1;
331
332 for (c = 0, tcp = tcbtab; c < MAX_PROCS; c++, tcp++) {
333 /* Reinitialize the output since it may have changed. */
334 tcp->outf = outf;
335 if (!(tcp->flags & TCB_INUSE) || !(tcp->flags & TCB_ATTACHED))
336 continue;
337#ifdef SVR4
338 if (proc_open(tcp, 1) < 0) {
339 fprintf(stderr, "trouble opening proc file\n");
340 droptcb(tcp);
341 continue;
342 }
343#else /* !SVR4 */
344 if (ptrace(PTRACE_ATTACH, tcp->pid, (char *) 1, 0) < 0) {
345 perror("attach: ptrace(PTRACE_ATTACH, ...)");
346 droptcb(tcp);
347 continue;
348 }
349#endif /* !SVR4 */
350 if (!qflag)
351 fprintf(stderr,
352 "Process %u attached - interrupt to quit\n",
353 pid);
354 }
355
356 if (optind < argc) {
357 struct stat statbuf;
358 char *filename;
359 char pathname[MAXPATHLEN];
360
361 filename = argv[optind];
362 if (strchr(filename, '/'))
363 strcpy(pathname, filename);
364#ifdef USE_DEBUGGING_EXEC
365 /*
366 * Debuggers customarily check the current directory
367 * first regardless of the path but doing that gives
368 * security geeks a panic attack.
369 */
370 else if (stat(filename, &statbuf) == 0)
371 strcpy(pathname, filename);
372#endif /* USE_DEBUGGING_EXEC */
373 else {
374 char *path;
375 int m, n, len;
376
377 for (path = getenv("PATH"); path && *path; path += m) {
378 if (strchr(path, ':')) {
379 n = strchr(path, ':') - path;
380 m = n + 1;
381 }
382 else
383 m = n = strlen(path);
384 if (n == 0) {
385 getcwd(pathname, MAXPATHLEN);
386 len = strlen(pathname);
387 }
388 else {
389 strncpy(pathname, path, n);
390 len = n;
391 }
392 if (len && pathname[len - 1] != '/')
393 pathname[len++] = '/';
394 strcpy(pathname + len, filename);
395 if (stat(pathname, &statbuf) == 0)
396 break;
397 }
398 }
399 if (stat(pathname, &statbuf) < 0) {
400 fprintf(stderr, "%s: %s: command not found\n",
401 progname, filename);
402 exit(1);
403 }
404 switch (pid = fork()) {
405 case -1:
406 perror("strace: fork");
407 cleanup();
408 exit(1);
409 break;
410 case 0: {
411#ifdef SVR4
Wichert Akkerman789ed351999-06-14 10:45:01 +0000412 if (outf != stderr) close (fileno (outf));
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000413#ifdef MIPS
414 /* Kludge for SGI, see proc_open for details. */
415 sa.sa_handler = foobar;
416 sa.sa_flags = 0;
417 sigemptyset(&sa.sa_mask);
418 sigaction(SIGINT, &sa, NULL);
419#endif /* MIPS */
420 pause();
421#else /* !SVR4 */
Wichert Akkermanbd4125c2000-06-27 17:28:06 +0000422 if (outf!=stderr)
Wichert Akkerman7987cdf2000-07-05 16:05:39 +0000423 close(fileno (outf));
Wichert Akkermanbd4125c2000-06-27 17:28:06 +0000424
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000425 if (ptrace(PTRACE_TRACEME, 0, (char *) 1, 0) < 0) {
426 perror("strace: ptrace(PTRACE_TRACEME, ...)");
427 return -1;
428 }
429 if (debug)
430 kill(getpid(), SIGSTOP);
431
432 if (username != NULL || geteuid() == 0) {
433 uid_t run_euid = run_uid;
434 gid_t run_egid = run_gid;
435
436 if (statbuf.st_mode & S_ISUID)
437 run_euid = statbuf.st_uid;
438 if (statbuf.st_mode & S_ISGID)
439 run_egid = statbuf.st_gid;
440
441 /*
442 * It is important to set groups before we
443 * lose privileges on setuid.
444 */
Wichert Akkerman5ae21ea2000-05-01 01:53:59 +0000445 if (username != NULL) {
446 if (initgroups(username, run_gid) < 0) {
447 perror("initgroups");
448 exit(1);
449 }
450 if (setregid(run_gid, run_egid) < 0) {
451 perror("setregid");
452 exit(1);
453 }
454 if (setreuid(run_uid, run_euid) < 0) {
455 perror("setreuid");
456 exit(1);
457 }
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000458 }
459 }
460 else
461 setreuid(run_uid, run_uid);
462#endif /* !SVR4 */
463
464 execv(pathname, &argv[optind]);
465 perror("strace: exec");
466 _exit(1);
467 break;
468 }
469 default:
470 if ((tcp = alloctcb(pid)) == NULL) {
471 fprintf(stderr, "tcb table full\n");
472 cleanup();
473 exit(1);
474 }
475#ifdef SVR4
476 if (proc_open(tcp, 0) < 0) {
477 fprintf(stderr, "trouble opening proc file\n");
478 cleanup();
479 exit(1);
480 }
481#endif /* SVR4 */
482#ifndef SVR4
483 fake_execve(tcp, pathname, &argv[optind], environ);
484#endif
485 break;
486 }
487 }
488 else if (pflag_seen == 0)
489 usage(stderr, 1);
490
491 sigemptyset(&empty_set);
492 sigemptyset(&blocked_set);
493 sa.sa_handler = SIG_IGN;
494 sigemptyset(&sa.sa_mask);
495 sa.sa_flags = 0;
496 sigaction(SIGTTOU, &sa, NULL);
497 sigaction(SIGTTIN, &sa, NULL);
498 if (interactive) {
499 sigaddset(&blocked_set, SIGHUP);
500 sigaddset(&blocked_set, SIGINT);
501 sigaddset(&blocked_set, SIGQUIT);
502 sigaddset(&blocked_set, SIGPIPE);
503 sigaddset(&blocked_set, SIGTERM);
504 sa.sa_handler = interrupt;
505#ifdef SUNOS4
506 /* POSIX signals on sunos4.1 are a little broken. */
507 sa.sa_flags = SA_INTERRUPT;
508#endif /* SUNOS4 */
509 }
510 sigaction(SIGHUP, &sa, NULL);
511 sigaction(SIGINT, &sa, NULL);
512 sigaction(SIGQUIT, &sa, NULL);
513 sigaction(SIGPIPE, &sa, NULL);
514 sigaction(SIGTERM, &sa, NULL);
515#ifdef SVR4
516 sa.sa_handler = reaper;
517 sigaction(SIGCHLD, &sa, NULL);
518#endif /* SVR4 */
519
520 if (trace() < 0)
521 exit(1);
522 cleanup();
523 exit(0);
524}
525
526void
527newoutf(tcp)
528struct tcb *tcp;
529{
530 char name[MAXPATHLEN];
531 FILE *fp;
532
533 if (outfname && followfork > 1) {
534 sprintf(name, "%s.%u", outfname, tcp->pid);
535#ifndef SVR4
536 setreuid(geteuid(), getuid());
537#endif
538 fp = fopen(name, "w");
539#ifndef SVR4
540 setreuid(geteuid(), getuid());
541#endif
542 if (fp == NULL) {
543 perror("fopen");
544 return;
545 }
546 tcp->outf = fp;
547 }
548 return;
549}
550
551struct tcb *
552alloctcb(pid)
553int pid;
554{
555 int i;
556 struct tcb *tcp;
557
558 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) {
559 if ((tcp->flags & TCB_INUSE) == 0) {
560 tcp->pid = pid;
561 tcp->parent = NULL;
562 tcp->nchildren = 0;
563 tcp->flags = TCB_INUSE | TCB_STARTUP;
564 tcp->outf = outf; /* Initialise to current out file */
565 tcp->stime.tv_sec = 0;
566 tcp->stime.tv_usec = 0;
567 tcp->pfd = -1;
568 nprocs++;
569 return tcp;
570 }
571 }
572 return NULL;
573}
574
575#ifdef SVR4
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000576int
577proc_open(tcp, attaching)
578struct tcb *tcp;
579int attaching;
580{
581 char proc[32];
582 long arg;
583 sysset_t sc_enter, sc_exit;
584 sigset_t signals;
585 fltset_t faults;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000586#ifndef HAVE_POLLABLE_PROCFS
587 static int last_pfd;
588#endif
589
Wichert Akkermanea78f0f1999-11-29 15:34:02 +0000590#ifdef HAVE_MP_PROCFS
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000591 /* Open the process pseudo-files in /proc. */
592 sprintf(proc, "/proc/%d/ctl", tcp->pid);
593 if ((tcp->pfd = open(proc, O_WRONLY|O_EXCL)) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000594 perror("strace: open(\"/proc/...\", ...)");
595 return -1;
596 }
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000597 if ((arg = fcntl(tcp->pfd, F_GETFD)) < 0) {
598 perror("F_GETFD");
599 return -1;
600 }
601 if (fcntl(tcp->pfd, F_SETFD, arg|FD_CLOEXEC) < 0) {
602 perror("F_SETFD");
603 return -1;
604 }
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000605 sprintf(proc, "/proc/%d/status", tcp->pid);
606 if ((tcp->pfd_stat = open(proc, O_RDONLY|O_EXCL)) < 0) {
607 perror("strace: open(\"/proc/...\", ...)");
608 return -1;
609 }
610 if ((arg = fcntl(tcp->pfd_stat, F_GETFD)) < 0) {
611 perror("F_GETFD");
612 return -1;
613 }
614 if (fcntl(tcp->pfd_stat, F_SETFD, arg|FD_CLOEXEC) < 0) {
615 perror("F_SETFD");
616 return -1;
617 }
618 sprintf(proc, "/proc/%d/as", tcp->pid);
619 if ((tcp->pfd_as = open(proc, O_RDONLY|O_EXCL)) < 0) {
620 perror("strace: open(\"/proc/...\", ...)");
621 return -1;
622 }
623 if ((arg = fcntl(tcp->pfd_as, F_GETFD)) < 0) {
624 perror("F_GETFD");
625 return -1;
626 }
627 if (fcntl(tcp->pfd_as, F_SETFD, arg|FD_CLOEXEC) < 0) {
628 perror("F_SETFD");
629 return -1;
630 }
631#else
632 /* Open the process pseudo-file in /proc. */
633 sprintf(proc, "/proc/%d", tcp->pid);
634 if ((tcp->pfd = open(proc, O_RDWR|O_EXCL)) < 0) {
635 perror("strace: open(\"/proc/...\", ...)");
636 return -1;
637 }
638 if ((arg = fcntl(tcp->pfd, F_GETFD)) < 0) {
639 perror("F_GETFD");
640 return -1;
641 }
642 if (fcntl(tcp->pfd, F_SETFD, arg|FD_CLOEXEC) < 0) {
643 perror("F_SETFD");
644 return -1;
645 }
646#endif
647 rebuild_pollv();
648 if (!attaching) {
649 /*
650 * Wait for the child to pause. Because of a race
651 * condition we have to poll for the event.
652 */
653 for (;;) {
654 if (IOCTL_STATUS (tcp) < 0) {
655 perror("strace: PIOCSTATUS");
656 return -1;
657 }
658 if (tcp->status.PR_FLAGS & PR_ASLEEP)
659 break;
660 }
661 }
662 /* Stop the process so that we own the stop. */
Wichert Akkerman16a03d22000-08-10 02:14:04 +0000663 if (IOCTL(tcp->pfd, PIOCSTOP, (char *)NULL) < 0) {
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000664 perror("strace: PIOCSTOP");
665 return -1;
666 }
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000667#ifdef PIOCSET
668 /* Set Run-on-Last-Close. */
669 arg = PR_RLC;
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000670 if (IOCTL(tcp->pfd, PIOCSET, &arg) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000671 perror("PIOCSET PR_RLC");
672 return -1;
673 }
674 /* Set or Reset Inherit-on-Fork. */
675 arg = PR_FORK;
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000676 if (IOCTL(tcp->pfd, followfork ? PIOCSET : PIOCRESET, &arg) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000677 perror("PIOC{SET,RESET} PR_FORK");
678 return -1;
679 }
680#else /* !PIOCSET */
681 if (ioctl(tcp->pfd, PIOCSRLC) < 0) {
682 perror("PIOCSRLC");
683 return -1;
684 }
685 if (ioctl(tcp->pfd, followfork ? PIOCSFORK : PIOCRFORK) < 0) {
686 perror("PIOC{S,R}FORK");
687 return -1;
688 }
689#endif /* !PIOCSET */
690 /* Enable all syscall entries. */
691 prfillset(&sc_enter);
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000692 if (IOCTL(tcp->pfd, PIOCSENTRY, &sc_enter) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000693 perror("PIOCSENTRY");
694 return -1;
695 }
696 /* Enable all syscall exits. */
697 prfillset(&sc_exit);
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000698 if (IOCTL(tcp->pfd, PIOCSEXIT, &sc_exit) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000699 perror("PIOSEXIT");
700 return -1;
701 }
702 /* Enable all signals. */
703 prfillset(&signals);
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000704 if (IOCTL(tcp->pfd, PIOCSTRACE, &signals) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000705 perror("PIOCSTRACE");
706 return -1;
707 }
708 /* Enable all faults. */
709 prfillset(&faults);
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000710 if (IOCTL(tcp->pfd, PIOCSFAULT, &faults) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000711 perror("PIOCSFAULT");
712 return -1;
713 }
714 if (!attaching) {
715#ifdef MIPS
716 /*
717 * The SGI PRSABORT doesn't work for pause() so
718 * we send it a caught signal to wake it up.
719 */
720 kill(tcp->pid, SIGINT);
721#else /* !MIPS */
722 /* The child is in a pause(), abort it. */
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000723 arg = PRSABORT;
724 if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000725 perror("PIOCRUN");
726 return -1;
727 }
728#endif /* !MIPS */
729 for (;;) {
730 /* Wait for the child to do something. */
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000731 if (IOCTL_WSTOP (tcp) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000732 perror("PIOCWSTOP");
733 return -1;
734 }
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000735 if (tcp->status.PR_WHY == PR_SYSENTRY) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000736#ifdef HAVE_PR_SYSCALL
737 int scno = tcp->status.pr_syscall;
738#else /* !HAVE_PR_SYSCALL */
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000739 int scno = tcp->status.PR_WHAT;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000740#endif /* !HAVE_PR_SYSCALL */
741 if (scno == SYS_execve)
742 break;
743 }
744 /* Set it running: maybe execve will be next. */
Wichert Akkerman9ce1a631999-08-29 23:15:07 +0000745 arg = 0;
746 if (IOCTL(tcp->pfd, PIOCRUN, &arg) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000747 perror("PIOCRUN");
748 return -1;
749 }
750 }
751 }
752#ifndef HAVE_POLLABLE_PROCFS
753 if (proc_poll_pipe[0] != -1)
754 proc_poller(tcp->pfd);
755 else if (nprocs > 1) {
756 proc_poll_open();
757 proc_poller(last_pfd);
758 proc_poller(tcp->pfd);
759 }
760 last_pfd = tcp->pfd;
761#endif /* !HAVE_POLLABLE_PROCFS */
762 return 0;
763}
764
765#endif /* SVR4 */
766
767static struct tcb *
768pid2tcb(pid)
769int pid;
770{
771 int i;
772 struct tcb *tcp;
773
774 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) {
775 if (pid && tcp->pid != pid)
776 continue;
777 if (tcp->flags & TCB_INUSE)
778 return tcp;
779 }
780 return NULL;
781}
782
783#ifdef SVR4
784
785static struct tcb *
786pfd2tcb(pfd)
787int pfd;
788{
789 int i;
790 struct tcb *tcp;
791
792 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) {
793 if (tcp->pfd != pfd)
794 continue;
795 if (tcp->flags & TCB_INUSE)
796 return tcp;
797 }
798 return NULL;
799}
800
801#endif /* SVR4 */
802
803void
804droptcb(tcp)
805struct tcb *tcp;
806{
807 if (tcp->pid == 0)
808 return;
809 nprocs--;
810 tcp->pid = 0;
811 tcp->flags = 0;
812 if (tcp->pfd != -1) {
813 close(tcp->pfd);
814 tcp->pfd = -1;
815#ifdef SVR4
816 rebuild_pollv();
817#endif
818 }
819 if (tcp->parent != NULL) {
820 tcp->parent->nchildren--;
821 tcp->parent = NULL;
822 }
823#if 0
824 if (tcp->outf != stderr)
825 fclose(tcp->outf);
826#endif
827 tcp->outf = 0;
828}
829
830#ifndef SVR4
831
832static int
833resume(tcp)
834struct tcb *tcp;
835{
836 if (tcp == NULL)
837 return -1;
838
839 if (!(tcp->flags & TCB_SUSPENDED)) {
840 fprintf(stderr, "PANIC: pid %u not suspended\n", tcp->pid);
841 return -1;
842 }
843 tcp->flags &= ~TCB_SUSPENDED;
844
845 if (ptrace(PTRACE_SYSCALL, tcp->pid, (char *) 1, 0) < 0) {
846 perror("resume: ptrace(PTRACE_SYSCALL, ...)");
847 return -1;
848 }
849
850 if (!qflag)
851 fprintf(stderr, "Process %u resumed\n", tcp->pid);
852 return 0;
853}
854
855#endif /* !SVR4 */
856
857/* detach traced process; continue with sig */
858
859static int
860detach(tcp, sig)
861struct tcb *tcp;
862int sig;
863{
864 int error = 0;
865#ifdef LINUX
866 int status;
867#endif
868
869 if (tcp->flags & TCB_BPTSET)
870 sig = SIGKILL;
871
872#ifdef LINUX
873 /*
874 * Linux wrongly insists the child be stopped
875 * before detaching. Arghh. We go through hoops
876 * to make a clean break of things.
877 */
Wichert Akkermandacfb6e1999-06-03 14:21:07 +0000878#if defined(SPARC)
879#undef PTRACE_DETACH
880#define PTRACE_DETACH PTRACE_SUNDETACH
881#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000882 if ((error = ptrace(PTRACE_DETACH, tcp->pid, (char *) 1, sig)) == 0) {
883 /* On a clear day, you can see forever. */
884 }
885 else if (errno != ESRCH) {
886 /* Shouldn't happen. */
887 perror("detach: ptrace(PTRACE_DETACH, ...)");
888 }
889 else if (kill(tcp->pid, 0) < 0) {
890 if (errno != ESRCH)
891 perror("detach: checking sanity");
892 }
893 else if (kill(tcp->pid, SIGSTOP) < 0) {
894 if (errno != ESRCH)
895 perror("detach: stopping child");
896 }
897 else {
898 for (;;) {
899 if (waitpid(tcp->pid, &status, 0) < 0) {
900 if (errno != ECHILD)
901 perror("detach: waiting");
902 break;
903 }
904 if (!WIFSTOPPED(status)) {
905 /* Au revoir, mon ami. */
906 break;
907 }
908 if (WSTOPSIG(status) == SIGSTOP) {
909 if ((error = ptrace(PTRACE_DETACH,
910 tcp->pid, (char *) 1, sig)) < 0) {
911 if (errno != ESRCH)
912 perror("detach: ptrace(PTRACE_DETACH, ...)");
913 /* I died trying. */
914 }
915 break;
916 }
917 if ((error = ptrace(PTRACE_CONT, tcp->pid, (char *) 1,
918 WSTOPSIG(status) == SIGTRAP ?
919 0 : WSTOPSIG(status))) < 0) {
920 if (errno != ESRCH)
921 perror("detach: ptrace(PTRACE_CONT, ...)");
922 break;
923 }
924 }
925 }
926#endif /* LINUX */
927
928#if defined(SUNOS4)
929 /* PTRACE_DETACH won't respect `sig' argument, so we post it here. */
930 if (sig && kill(tcp->pid, sig) < 0)
931 perror("detach: kill");
932 sig = 0;
933 if ((error = ptrace(PTRACE_DETACH, tcp->pid, (char *) 1, sig)) < 0)
934 perror("detach: ptrace(PTRACE_DETACH, ...)");
935#endif /* SUNOS4 */
936
937#ifndef SVR4
938 if (waiting_parent(tcp))
939 error = resume(tcp->parent);
940#endif /* !SVR4 */
941
942 if (!qflag)
943 fprintf(stderr, "Process %u detached\n", tcp->pid);
944
945 droptcb(tcp);
946 return error;
947}
948
949#ifdef SVR4
950
951static void
952reaper(sig)
953int sig;
954{
955 int pid;
956 int status;
957
958 while ((pid = waitpid(-1, &status, WNOHANG)) > 0) {
959#if 0
960 struct tcb *tcp;
961
962 tcp = pid2tcb(pid);
963 if (tcp)
964 droptcb(tcp);
965#endif
966 }
967}
968
969#endif /* SVR4 */
970
971static void
972cleanup()
973{
974 int i;
975 struct tcb *tcp;
976
977 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) {
978 if (!(tcp->flags & TCB_INUSE))
979 continue;
980 if (debug)
981 fprintf(stderr,
982 "cleanup: looking at pid %u\n", tcp->pid);
983 if (tcp_last &&
984 (!outfname || followfork < 2 || tcp_last == tcp)) {
985 tprintf(" <unfinished ...>\n");
986 tcp_last = NULL;
987 }
988 if (tcp->flags & TCB_ATTACHED)
989 detach(tcp, 0);
990 else {
991 kill(tcp->pid, SIGCONT);
992 kill(tcp->pid, SIGTERM);
993 }
994 }
995 if (cflag)
996 call_summary(outf);
997}
998
999static void
1000interrupt(sig)
1001int sig;
1002{
1003 interrupted = 1;
1004}
1005
1006#ifndef HAVE_STRERROR
1007
1008#ifndef SYS_ERRLIST_DECLARED
1009extern int sys_nerr;
1010extern char *sys_errlist[];
1011#endif /* SYS_ERRLIST_DECLARED */
1012
1013const char *
1014strerror(errno)
1015int errno;
1016{
1017 static char buf[64];
1018
1019 if (errno < 1 || errno >= sys_nerr) {
1020 sprintf(buf, "Unknown error %d", errno);
1021 return buf;
1022 }
1023 return sys_errlist[errno];
1024}
1025
1026#endif /* HAVE_STERRROR */
1027
1028#ifndef HAVE_STRSIGNAL
1029
1030#ifndef SYS_SIGLIST_DECLARED
1031#ifdef HAVE__SYS_SIGLIST
1032 extern char *_sys_siglist[];
1033#else
1034 extern char *sys_siglist[];
1035#endif
1036#endif /* SYS_SIGLIST_DECLARED */
1037
1038const char *
1039strsignal(sig)
1040int sig;
1041{
1042 static char buf[64];
1043
1044 if (sig < 1 || sig >= NSIG) {
1045 sprintf(buf, "Unknown signal %d", sig);
1046 return buf;
1047 }
1048#ifdef HAVE__SYS_SIGLIST
1049 return _sys_siglist[sig];
1050#else
1051 return sys_siglist[sig];
1052#endif
1053}
1054
1055#endif /* HAVE_STRSIGNAL */
1056
1057#ifdef SVR4
1058
1059static void
1060rebuild_pollv()
1061{
1062 int i, j;
1063 struct tcb *tcp;
1064
1065 for (i = j = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) {
1066 if (!(tcp->flags & TCB_INUSE))
1067 continue;
1068 pollv[j].fd = tcp->pfd;
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001069 pollv[j].events = POLLWANT;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001070 j++;
1071 }
1072 if (j != nprocs) {
1073 fprintf(stderr, "strace: proc miscount\n");
1074 exit(1);
1075 }
1076}
1077
1078#ifndef HAVE_POLLABLE_PROCFS
1079
1080static void
1081proc_poll_open()
1082{
1083 int arg;
1084 int i;
1085
1086 if (pipe(proc_poll_pipe) < 0) {
1087 perror("pipe");
1088 exit(1);
1089 }
1090 for (i = 0; i < 2; i++) {
1091 if ((arg = fcntl(proc_poll_pipe[i], F_GETFD)) < 0) {
1092 perror("F_GETFD");
1093 exit(1);
1094 }
1095 if (fcntl(proc_poll_pipe[i], F_SETFD, arg|FD_CLOEXEC) < 0) {
1096 perror("F_SETFD");
1097 exit(1);
1098 }
1099 }
1100}
1101
1102static int
1103proc_poll(pollv, nfds, timeout)
1104struct pollfd *pollv;
1105int nfds;
1106int timeout;
1107{
1108 int i;
1109 int n;
1110 struct proc_pollfd pollinfo;
1111
1112 if ((n = read(proc_poll_pipe[0], &pollinfo, sizeof(pollinfo))) < 0)
1113 return n;
1114 if (n != sizeof(struct proc_pollfd)) {
1115 fprintf(stderr, "panic: short read: %d\n", n);
1116 exit(1);
1117 }
1118 for (i = 0; i < nprocs; i++) {
1119 if (pollv[i].fd == pollinfo.fd)
1120 pollv[i].revents = pollinfo.revents;
1121 else
1122 pollv[i].revents = 0;
1123 }
1124 poller_pid = pollinfo.pid;
1125 return 1;
1126}
1127
1128static void
1129wakeup_handler(sig)
1130int sig;
1131{
1132}
1133
1134static void
1135proc_poller(pfd)
1136int pfd;
1137{
1138 struct proc_pollfd pollinfo;
1139 struct sigaction sa;
1140 sigset_t blocked_set, empty_set;
1141 int i;
1142 int n;
1143 struct rlimit rl;
1144
1145 switch (fork()) {
1146 case -1:
1147 perror("fork");
1148 _exit(0);
1149 case 0:
1150 break;
1151 default:
1152 return;
1153 }
1154
1155 sa.sa_handler = interactive ? SIG_DFL : SIG_IGN;
1156 sa.sa_flags = 0;
1157 sigemptyset(&sa.sa_mask);
1158 sigaction(SIGHUP, &sa, NULL);
1159 sigaction(SIGINT, &sa, NULL);
1160 sigaction(SIGQUIT, &sa, NULL);
1161 sigaction(SIGPIPE, &sa, NULL);
1162 sigaction(SIGTERM, &sa, NULL);
1163 sa.sa_handler = wakeup_handler;
1164 sigaction(SIGUSR1, &sa, NULL);
1165 sigemptyset(&blocked_set);
1166 sigaddset(&blocked_set, SIGUSR1);
1167 sigprocmask(SIG_BLOCK, &blocked_set, NULL);
1168 sigemptyset(&empty_set);
1169
1170 if (getrlimit(RLIMIT_NOFILE, &rl) < 0) {
1171 perror("getrlimit(RLIMIT_NOFILE, ...)");
1172 _exit(0);
1173 }
1174 n = rl.rlim_cur;
1175 for (i = 0; i < n; i++) {
1176 if (i != pfd && i != proc_poll_pipe[1])
1177 close(i);
1178 }
1179
1180 pollinfo.fd = pfd;
1181 pollinfo.pid = getpid();
1182 for (;;) {
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001183 if (ioctl(pfd, PIOCWSTOP, NULL) < 0)
1184 {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001185 switch (errno) {
1186 case EINTR:
1187 continue;
1188 case EBADF:
1189 pollinfo.revents = POLLERR;
1190 break;
1191 case ENOENT:
1192 pollinfo.revents = POLLHUP;
1193 break;
1194 default:
1195 perror("proc_poller: PIOCWSTOP");
1196 }
1197 write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo));
1198 _exit(0);
1199 }
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001200 pollinfo.revents = POLLWANT;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001201 write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo));
1202 sigsuspend(&empty_set);
1203 }
1204}
1205
1206#endif /* !HAVE_POLLABLE_PROCFS */
1207
1208static int
1209choose_pfd()
1210{
1211 int i, j;
1212 struct tcb *tcp;
1213
1214 static int last;
1215
1216 if (followfork < 2 &&
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001217 last < nprocs && (pollv[last].revents & POLLWANT)) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001218 /*
1219 * The previous process is ready to run again. We'll
1220 * let it do so if it is currently in a syscall. This
1221 * heuristic improves the readability of the trace.
1222 */
1223 tcp = pfd2tcb(pollv[last].fd);
1224 if (tcp && (tcp->flags & TCB_INSYSCALL))
1225 return pollv[last].fd;
1226 }
1227
1228 for (i = 0; i < nprocs; i++) {
1229 /* Let competing children run round robin. */
1230 j = (i + last + 1) % nprocs;
1231 if (pollv[j].revents & (POLLHUP | POLLERR)) {
1232 tcp = pfd2tcb(pollv[j].fd);
1233 if (!tcp) {
1234 fprintf(stderr, "strace: lost proc\n");
1235 exit(1);
1236 }
1237 droptcb(tcp);
1238 return -1;
1239 }
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001240 if (pollv[j].revents & POLLWANT) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001241 last = j;
1242 return pollv[j].fd;
1243 }
1244 }
1245 fprintf(stderr, "strace: nothing ready\n");
1246 exit(1);
1247}
1248
1249static int
1250trace()
1251{
Wichert Akkerman9dbf1541999-11-26 13:11:29 +00001252#ifdef POLL_HACK
1253 struct tcb *in_syscall;
1254#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001255 struct tcb *tcp;
1256 int pfd;
1257 int what;
1258 int ioctl_result = 0, ioctl_errno = 0;
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001259 long arg;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001260
1261 for (;;) {
1262 if (interactive)
1263 sigprocmask(SIG_SETMASK, &empty_set, NULL);
1264
1265 if (nprocs == 0)
1266 break;
1267
1268 switch (nprocs) {
1269 case 1:
1270#ifndef HAVE_POLLABLE_PROCFS
1271 if (proc_poll_pipe[0] == -1) {
1272#endif
1273 tcp = pid2tcb(0);
1274 if (!tcp)
1275 continue;
1276 pfd = tcp->pfd;
1277 if (pfd == -1)
1278 continue;
1279 break;
1280#ifndef HAVE_POLLABLE_PROCFS
1281 }
1282 /* fall through ... */
1283#endif /* !HAVE_POLLABLE_PROCFS */
1284 default:
1285#ifdef HAVE_POLLABLE_PROCFS
Wichert Akkerman9dbf1541999-11-26 13:11:29 +00001286#ifdef POLL_HACK
1287 /* On some systems (e.g. UnixWare) we get too much ugly
1288 "unfinished..." stuff when multiple proceses are in
1289 syscalls. Here's a nasty hack */
1290
1291 if (in_syscall) {
1292 struct pollfd pv;
1293 tcp = in_syscall;
1294 in_syscall = NULL;
1295 pv.fd = tcp->pfd;
1296 pv.events = POLLWANT;
1297 if ((what = poll (&pv, 1, 1)) < 0) {
1298 if (interrupted)
1299 return 0;
1300 continue;
1301 }
1302 else if (what == 1 && pv.revents & POLLWANT) {
1303 goto FOUND;
1304 }
1305 }
1306#endif
1307
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001308 if (poll(pollv, nprocs, INFTIM) < 0) {
1309 if (interrupted)
1310 return 0;
1311 continue;
1312 }
1313#else /* !HAVE_POLLABLE_PROCFS */
1314 if (proc_poll(pollv, nprocs, INFTIM) < 0) {
1315 if (interrupted)
1316 return 0;
1317 continue;
1318 }
1319#endif /* !HAVE_POLLABLE_PROCFS */
1320 pfd = choose_pfd();
1321 if (pfd == -1)
1322 continue;
1323 break;
1324 }
1325
1326 /* Look up `pfd' in our table. */
1327 if ((tcp = pfd2tcb(pfd)) == NULL) {
1328 fprintf(stderr, "unknown pfd: %u\n", pfd);
1329 exit(1);
1330 }
Wichert Akkerman9dbf1541999-11-26 13:11:29 +00001331 FOUND:
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001332 /* Get the status of the process. */
1333 if (!interrupted) {
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001334 ioctl_result = IOCTL_WSTOP (tcp);
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001335 ioctl_errno = errno;
1336#ifndef HAVE_POLLABLE_PROCFS
1337 if (proc_poll_pipe[0] != -1) {
1338 if (ioctl_result < 0)
1339 kill(poller_pid, SIGKILL);
1340 else
1341 kill(poller_pid, SIGUSR1);
1342 }
1343#endif /* !HAVE_POLLABLE_PROCFS */
1344 }
1345 if (interrupted)
1346 return 0;
1347
1348 if (interactive)
1349 sigprocmask(SIG_BLOCK, &blocked_set, NULL);
1350
1351 if (ioctl_result < 0) {
1352 /* Find out what happened if it failed. */
1353 switch (ioctl_errno) {
1354 case EINTR:
1355 case EBADF:
1356 continue;
1357 case ENOENT:
1358 droptcb(tcp);
1359 continue;
1360 default:
1361 perror("PIOCWSTOP");
1362 exit(1);
1363 }
1364 }
1365
1366 /* clear the just started flag */
1367 tcp->flags &= ~TCB_STARTUP;
1368
1369 /* set current output file */
1370 outf = tcp->outf;
1371
1372 if (cflag) {
1373 struct timeval stime;
1374
1375 stime.tv_sec = tcp->status.pr_stime.tv_sec;
1376 stime.tv_usec = tcp->status.pr_stime.tv_nsec/1000;
1377 tv_sub(&tcp->dtime, &stime, &tcp->stime);
1378 tcp->stime = stime;
1379 }
1380
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001381 what = tcp->status.PR_WHAT;
1382 switch (tcp->status.PR_WHY) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001383 case PR_REQUESTED:
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001384 if (tcp->status.PR_FLAGS & PR_ASLEEP) {
1385 tcp->status.PR_WHY = PR_SYSENTRY;
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001386 if (trace_syscall(tcp) < 0) {
1387 fprintf(stderr, "syscall trouble\n");
1388 exit(1);
1389 }
1390 }
1391 break;
1392 case PR_SYSENTRY:
Wichert Akkerman9dbf1541999-11-26 13:11:29 +00001393#ifdef POLL_HACK
1394 in_syscall = tcp;
1395#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001396 case PR_SYSEXIT:
1397 if (trace_syscall(tcp) < 0) {
1398 fprintf(stderr, "syscall trouble\n");
1399 exit(1);
1400 }
1401 break;
1402 case PR_SIGNALLED:
1403 if (!cflag && (qual_flags[what] & QUAL_SIGNAL)) {
1404 printleader(tcp);
1405 tprintf("--- %s (%s) ---",
Nate Sammonsce780fc1999-03-29 23:23:13 +00001406 signame(what), strsignal(what));
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001407 printtrailer(tcp);
1408 }
1409 break;
1410 case PR_FAULTED:
1411 if (!cflag && (qual_flags[what] & QUAL_FAULT)) {
1412 printleader(tcp);
1413 tprintf("=== FAULT %d ===", what);
1414 printtrailer(tcp);
1415 }
1416 break;
1417 default:
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001418 fprintf(stderr, "odd stop %d\n", tcp->status.PR_WHY);
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001419 exit(1);
1420 break;
1421 }
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001422 arg = 0;
1423 if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) {
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001424 perror("PIOCRUN");
1425 exit(1);
1426 }
1427 }
1428 return 0;
1429}
1430
1431#else /* !SVR4 */
1432
1433static int
1434trace()
1435{
1436 int pid;
1437 int wait_errno;
1438 int status;
1439 struct tcb *tcp;
1440#ifdef LINUX
1441 struct rusage ru;
1442#endif /* LINUX */
1443
1444 while (nprocs != 0) {
1445 if (interactive)
1446 sigprocmask(SIG_SETMASK, &empty_set, NULL);
1447#ifdef LINUX
1448 pid = wait4(-1, &status, 0, cflag ? &ru : NULL);
1449#endif /* LINUX */
1450#ifdef SUNOS4
1451 pid = wait(&status);
1452#endif /* SUNOS4 */
1453 wait_errno = errno;
1454 if (interactive)
1455 sigprocmask(SIG_BLOCK, &blocked_set, NULL);
1456
1457 if (interrupted)
1458 return 0;
1459
1460 if (pid == -1) {
1461 switch (wait_errno) {
1462 case EINTR:
1463 continue;
1464 case ECHILD:
1465 /*
1466 * We would like to verify this case
1467 * but sometimes a race in Solbourne's
1468 * version of SunOS sometimes reports
1469 * ECHILD before sending us SIGCHILD.
1470 */
1471#if 0
1472 if (nprocs == 0)
1473 return 0;
1474 fprintf(stderr, "strace: proc miscount\n");
1475 exit(1);
1476#endif
1477 return 0;
1478 default:
1479 errno = wait_errno;
1480 perror("strace: wait");
1481 return -1;
1482 }
1483 }
1484 if (debug)
1485 fprintf(stderr, " [wait(%#x) = %u]\n", status, pid);
1486
1487 /* Look up `pid' in our table. */
1488 if ((tcp = pid2tcb(pid)) == NULL) {
Wichert Akkermanfaf72222000-02-19 23:59:03 +00001489#if 0 /* XXX davidm */ /* WTA: disabled again */
Wichert Akkerman8b1b40c2000-02-03 21:58:30 +00001490 struct tcb *tcpchild;
1491
1492 if ((tcpchild = alloctcb(pid)) == NULL) {
1493 fprintf(stderr, " [tcb table full]\n");
1494 kill(pid, SIGKILL); /* XXX */
1495 return 0;
1496 }
1497 tcpchild->flags |= TCB_ATTACHED;
1498 newoutf(tcpchild);
1499 tcp->nchildren++;
1500 if (!qflag)
1501 fprintf(stderr, "Process %d attached\n", pid);
1502#else
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001503 fprintf(stderr, "unknown pid: %u\n", pid);
1504 if (WIFSTOPPED(status))
1505 ptrace(PTRACE_CONT, pid, (char *) 1, 0);
1506 exit(1);
Wichert Akkerman8b1b40c2000-02-03 21:58:30 +00001507#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001508 }
1509 /* set current output file */
1510 outf = tcp->outf;
1511 if (cflag) {
1512#ifdef LINUX
1513 tv_sub(&tcp->dtime, &ru.ru_stime, &tcp->stime);
1514 tcp->stime = ru.ru_stime;
1515#endif /* !LINUX */
1516 }
1517
1518 if (tcp->flags & TCB_SUSPENDED) {
1519 /*
1520 * Apparently, doing any ptrace() call on a stopped
1521 * process, provokes the kernel to report the process
1522 * status again on a subsequent wait(), even if the
1523 * process has not been actually restarted.
1524 * Since we have inspected the arguments of suspended
1525 * processes we end up here testing for this case.
1526 */
1527 continue;
1528 }
1529 if (WIFSIGNALED(status)) {
1530 if (!cflag
1531 && (qual_flags[WTERMSIG(status)] & QUAL_SIGNAL)) {
1532 printleader(tcp);
1533 tprintf("+++ killed by %s +++",
Nate Sammonsce780fc1999-03-29 23:23:13 +00001534 signame(WTERMSIG(status)));
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001535 printtrailer(tcp);
1536 }
1537 droptcb(tcp);
1538 continue;
1539 }
1540 if (WIFEXITED(status)) {
1541 if (debug)
1542 fprintf(stderr, "pid %u exited\n", pid);
1543 if (tcp->flags & TCB_ATTACHED)
1544 fprintf(stderr,
1545 "PANIC: attached pid %u exited\n",
1546 pid);
1547 droptcb(tcp);
1548 continue;
1549 }
1550 if (!WIFSTOPPED(status)) {
1551 fprintf(stderr, "PANIC: pid %u not stopped\n", pid);
1552 droptcb(tcp);
1553 continue;
1554 }
1555 if (debug)
1556 fprintf(stderr, "pid %u stopped, [%s]\n",
Nate Sammonsce780fc1999-03-29 23:23:13 +00001557 pid, signame(WSTOPSIG(status)));
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001558
1559 if (tcp->flags & TCB_STARTUP) {
1560 /*
1561 * This flag is there to keep us in sync.
1562 * Next time this process stops it should
1563 * really be entering a system call.
1564 */
1565 tcp->flags &= ~TCB_STARTUP;
1566 if (tcp->flags & TCB_ATTACHED) {
1567 /*
1568 * Interestingly, the process may stop
1569 * with STOPSIG equal to some other signal
1570 * than SIGSTOP if we happend to attach
1571 * just before the process takes a signal.
1572 */
1573 if (!WIFSTOPPED(status)) {
1574 fprintf(stderr,
1575 "pid %u not stopped\n", pid);
1576 detach(tcp, WSTOPSIG(status));
1577 continue;
1578 }
1579 }
1580 else {
1581#ifdef SUNOS4
1582 /* A child of us stopped at exec */
1583 if (WSTOPSIG(status) == SIGTRAP && followvfork)
1584 fixvfork(tcp);
1585#endif /* SUNOS4 */
1586 }
1587 if (tcp->flags & TCB_BPTSET) {
1588 if (clearbpt(tcp) < 0) /* Pretty fatal */ {
1589 droptcb(tcp);
1590 cleanup();
1591 return -1;
1592 }
1593 }
1594 goto tracing;
1595 }
1596
1597 if (WSTOPSIG(status) != SIGTRAP) {
1598 if (WSTOPSIG(status) == SIGSTOP &&
1599 (tcp->flags & TCB_SIGTRAPPED)) {
1600 /*
1601 * Trapped attempt to block SIGTRAP
1602 * Hope we are back in control now.
1603 */
1604 tcp->flags &= ~(TCB_INSYSCALL | TCB_SIGTRAPPED);
1605 if (ptrace(PTRACE_SYSCALL,
1606 pid, (char *) 1, 0) < 0) {
1607 perror("trace: ptrace(PTRACE_SYSCALL, ...)");
1608 cleanup();
1609 return -1;
1610 }
1611 continue;
1612 }
1613 if (!cflag
1614 && (qual_flags[WSTOPSIG(status)] & QUAL_SIGNAL)) {
1615 printleader(tcp);
1616 tprintf("--- %s (%s) ---",
Nate Sammonsce780fc1999-03-29 23:23:13 +00001617 signame(WSTOPSIG(status)),
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001618 strsignal(WSTOPSIG(status)));
1619 printtrailer(tcp);
1620 }
1621 if ((tcp->flags & TCB_ATTACHED) &&
1622 !sigishandled(tcp, WSTOPSIG(status))) {
1623 detach(tcp, WSTOPSIG(status));
1624 continue;
1625 }
1626 if (ptrace(PTRACE_SYSCALL, pid, (char *) 1,
1627 WSTOPSIG(status)) < 0) {
1628 perror("trace: ptrace(PTRACE_SYSCALL, ...)");
1629 cleanup();
1630 return -1;
1631 }
1632 tcp->flags &= ~TCB_SUSPENDED;
1633 continue;
1634 }
1635 if (trace_syscall(tcp) < 0) {
1636 if (tcp->flags & TCB_ATTACHED)
1637 detach(tcp, 0);
1638 else {
1639 ptrace(PTRACE_KILL,
1640 tcp->pid, (char *) 1, SIGTERM);
1641 droptcb(tcp);
1642 }
1643 continue;
1644 }
1645 if (tcp->flags & TCB_EXITING) {
1646 if (tcp->flags & TCB_ATTACHED)
1647 detach(tcp, 0);
1648 else if (ptrace(PTRACE_CONT, pid, (char *) 1, 0) < 0) {
1649 perror("strace: ptrace(PTRACE_CONT, ...)");
1650 cleanup();
1651 return -1;
1652 }
1653 continue;
1654 }
1655 if (tcp->flags & TCB_SUSPENDED) {
1656 if (!qflag)
1657 fprintf(stderr, "Process %u suspended\n", pid);
1658 continue;
1659 }
1660 tracing:
1661 if (ptrace(PTRACE_SYSCALL, pid, (char *) 1, 0) < 0) {
1662 perror("trace: ptrace(PTRACE_SYSCALL, ...)");
1663 cleanup();
1664 return -1;
1665 }
1666 }
1667 return 0;
1668}
1669
1670#endif /* !SVR4 */
1671
1672static int curcol;
1673
1674#ifdef __STDC__
1675#include <stdarg.h>
1676#define VA_START(a, b) va_start(a, b)
1677#else
1678#include <varargs.h>
1679#define VA_START(a, b) va_start(a)
1680#endif
1681
1682void
1683#ifdef __STDC__
1684tprintf(const char *fmt, ...)
1685#else
1686tprintf(fmt, va_alist)
1687char *fmt;
1688va_dcl
1689#endif
1690{
1691 va_list args;
1692
1693 VA_START(args, fmt);
1694 if (outf)
1695 curcol += vfprintf(outf, fmt, args);
1696 va_end(args);
1697 return;
1698}
1699
1700void
1701printleader(tcp)
1702struct tcb *tcp;
1703{
1704 if (tcp_last && (!outfname || followfork < 2 || tcp_last == tcp)) {
1705 tcp_last->flags |= TCB_REPRINT;
1706 tprintf(" <unfinished ...>\n");
1707 }
1708 curcol = 0;
1709 if ((followfork == 1 || pflag_seen > 1) && outfname)
1710 tprintf("%-5d ", tcp->pid);
1711 else if (nprocs > 1 && !outfname)
1712 tprintf("[pid %5u] ", tcp->pid);
1713 if (tflag) {
1714 char str[sizeof("HH:MM:SS")];
1715 struct timeval tv, dtv;
1716 static struct timeval otv;
1717
1718 gettimeofday(&tv, NULL);
1719 if (rflag) {
1720 if (otv.tv_sec == 0)
1721 otv = tv;
1722 tv_sub(&dtv, &tv, &otv);
1723 tprintf("%6ld.%06ld ",
1724 (long) dtv.tv_sec, (long) dtv.tv_usec);
1725 otv = tv;
1726 }
1727 else if (tflag > 2) {
1728 tprintf("%ld.%06ld ",
1729 (long) tv.tv_sec, (long) tv.tv_usec);
1730 }
1731 else {
1732 time_t local = tv.tv_sec;
1733 strftime(str, sizeof(str), "%T", localtime(&local));
1734 if (tflag > 1)
1735 tprintf("%s.%06ld ", str, (long) tv.tv_usec);
1736 else
1737 tprintf("%s ", str);
1738 }
1739 }
1740 if (iflag)
1741 printcall(tcp);
1742}
1743
1744void
1745tabto(col)
1746int col;
1747{
1748 if (curcol < col)
1749 tprintf("%*s", col - curcol, "");
1750}
1751
1752void
1753printtrailer(tcp)
1754struct tcb *tcp;
1755{
1756 tprintf("\n");
1757 tcp_last = NULL;
1758}
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001759
Wichert Akkermanea78f0f1999-11-29 15:34:02 +00001760#ifdef HAVE_MP_PROCFS
Wichert Akkerman9ce1a631999-08-29 23:15:07 +00001761
1762int mp_ioctl (int fd, int cmd, void *arg, int size) {
1763
1764 struct iovec iov[2];
1765 int n = 1;
1766
1767 iov[0].iov_base = &cmd;
1768 iov[0].iov_len = sizeof cmd;
1769 if (arg) {
1770 ++n;
1771 iov[1].iov_base = arg;
1772 iov[1].iov_len = size;
1773 }
1774
1775 return writev (fd, iov, n);
1776}
1777
1778#endif