strace.c revision 2ee6e45f36566e8735b35ffad40bfcc626a25a98
1/* 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> 5 * Copyright (c) 1996-1999 Wichert Akkerman <wichert@cistron.nl> 6 * 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 33#include <sys/types.h> 34#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> 50#ifdef HAVE_MP_PROCFS 51#include <sys/uio.h> 52#endif 53#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)); 95struct pollfd pollv[MAX_PROCS]; 96 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 113#ifdef HAVE_MP_PROCFS 114#define POLLWANT POLLWRNORM 115#else 116#define POLLWANT POLLPRI 117#endif 118#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()) { 246 fprintf(stderr, "%s: I'm sorry, I can't let you do that, Dave.\n", progname); 247 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 412 if (outf != stderr) close (fileno (outf)); 413#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 */ 422 if (ptrace(PTRACE_TRACEME, 0, (char *) 1, 0) < 0) { 423 perror("strace: ptrace(PTRACE_TRACEME, ...)"); 424 return -1; 425 } 426 if (debug) 427 kill(getpid(), SIGSTOP); 428 429 if (username != NULL || geteuid() == 0) { 430 uid_t run_euid = run_uid; 431 gid_t run_egid = run_gid; 432 433 if (statbuf.st_mode & S_ISUID) 434 run_euid = statbuf.st_uid; 435 if (statbuf.st_mode & S_ISGID) 436 run_egid = statbuf.st_gid; 437 438 /* 439 * It is important to set groups before we 440 * lose privileges on setuid. 441 */ 442 if (username != NULL 443 && initgroups(username, run_gid) < 0) { 444 perror("initgroups"); 445 exit(1); 446 } 447 if (setregid(run_gid, run_egid) < 0) { 448 perror("setregid"); 449 exit(1); 450 } 451 if (setreuid(run_uid, run_euid) < 0) { 452 perror("setreuid"); 453 exit(1); 454 } 455 } 456 else 457 setreuid(run_uid, run_uid); 458#endif /* !SVR4 */ 459 460 execv(pathname, &argv[optind]); 461 perror("strace: exec"); 462 _exit(1); 463 break; 464 } 465 default: 466 if ((tcp = alloctcb(pid)) == NULL) { 467 fprintf(stderr, "tcb table full\n"); 468 cleanup(); 469 exit(1); 470 } 471#ifdef SVR4 472 if (proc_open(tcp, 0) < 0) { 473 fprintf(stderr, "trouble opening proc file\n"); 474 cleanup(); 475 exit(1); 476 } 477#endif /* SVR4 */ 478#ifndef SVR4 479 fake_execve(tcp, pathname, &argv[optind], environ); 480#endif 481 break; 482 } 483 } 484 else if (pflag_seen == 0) 485 usage(stderr, 1); 486 487 sigemptyset(&empty_set); 488 sigemptyset(&blocked_set); 489 sa.sa_handler = SIG_IGN; 490 sigemptyset(&sa.sa_mask); 491 sa.sa_flags = 0; 492 sigaction(SIGTTOU, &sa, NULL); 493 sigaction(SIGTTIN, &sa, NULL); 494 if (interactive) { 495 sigaddset(&blocked_set, SIGHUP); 496 sigaddset(&blocked_set, SIGINT); 497 sigaddset(&blocked_set, SIGQUIT); 498 sigaddset(&blocked_set, SIGPIPE); 499 sigaddset(&blocked_set, SIGTERM); 500 sa.sa_handler = interrupt; 501#ifdef SUNOS4 502 /* POSIX signals on sunos4.1 are a little broken. */ 503 sa.sa_flags = SA_INTERRUPT; 504#endif /* SUNOS4 */ 505 } 506 sigaction(SIGHUP, &sa, NULL); 507 sigaction(SIGINT, &sa, NULL); 508 sigaction(SIGQUIT, &sa, NULL); 509 sigaction(SIGPIPE, &sa, NULL); 510 sigaction(SIGTERM, &sa, NULL); 511#ifdef SVR4 512 sa.sa_handler = reaper; 513 sigaction(SIGCHLD, &sa, NULL); 514#endif /* SVR4 */ 515 516 if (trace() < 0) 517 exit(1); 518 cleanup(); 519 exit(0); 520} 521 522void 523newoutf(tcp) 524struct tcb *tcp; 525{ 526 char name[MAXPATHLEN]; 527 FILE *fp; 528 529 if (outfname && followfork > 1) { 530 sprintf(name, "%s.%u", outfname, tcp->pid); 531#ifndef SVR4 532 setreuid(geteuid(), getuid()); 533#endif 534 fp = fopen(name, "w"); 535#ifndef SVR4 536 setreuid(geteuid(), getuid()); 537#endif 538 if (fp == NULL) { 539 perror("fopen"); 540 return; 541 } 542 tcp->outf = fp; 543 } 544 return; 545} 546 547struct tcb * 548alloctcb(pid) 549int pid; 550{ 551 int i; 552 struct tcb *tcp; 553 554 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) { 555 if ((tcp->flags & TCB_INUSE) == 0) { 556 tcp->pid = pid; 557 tcp->parent = NULL; 558 tcp->nchildren = 0; 559 tcp->flags = TCB_INUSE | TCB_STARTUP; 560 tcp->outf = outf; /* Initialise to current out file */ 561 tcp->stime.tv_sec = 0; 562 tcp->stime.tv_usec = 0; 563 tcp->pfd = -1; 564 nprocs++; 565 return tcp; 566 } 567 } 568 return NULL; 569} 570 571#ifdef SVR4 572int 573proc_open(tcp, attaching) 574struct tcb *tcp; 575int attaching; 576{ 577 char proc[32]; 578 long arg; 579 sysset_t sc_enter, sc_exit; 580 sigset_t signals; 581 fltset_t faults; 582#ifndef HAVE_POLLABLE_PROCFS 583 static int last_pfd; 584#endif 585 586#ifdef HAVE_MP_PROCFS 587 /* Open the process pseudo-files in /proc. */ 588 sprintf(proc, "/proc/%d/ctl", tcp->pid); 589 if ((tcp->pfd = open(proc, O_WRONLY|O_EXCL)) < 0) { 590 perror("strace: open(\"/proc/...\", ...)"); 591 return -1; 592 } 593 if ((arg = fcntl(tcp->pfd, F_GETFD)) < 0) { 594 perror("F_GETFD"); 595 return -1; 596 } 597 if (fcntl(tcp->pfd, F_SETFD, arg|FD_CLOEXEC) < 0) { 598 perror("F_SETFD"); 599 return -1; 600 } 601 sprintf(proc, "/proc/%d/status", tcp->pid); 602 if ((tcp->pfd_stat = open(proc, O_RDONLY|O_EXCL)) < 0) { 603 perror("strace: open(\"/proc/...\", ...)"); 604 return -1; 605 } 606 if ((arg = fcntl(tcp->pfd_stat, F_GETFD)) < 0) { 607 perror("F_GETFD"); 608 return -1; 609 } 610 if (fcntl(tcp->pfd_stat, F_SETFD, arg|FD_CLOEXEC) < 0) { 611 perror("F_SETFD"); 612 return -1; 613 } 614 sprintf(proc, "/proc/%d/as", tcp->pid); 615 if ((tcp->pfd_as = open(proc, O_RDONLY|O_EXCL)) < 0) { 616 perror("strace: open(\"/proc/...\", ...)"); 617 return -1; 618 } 619 if ((arg = fcntl(tcp->pfd_as, F_GETFD)) < 0) { 620 perror("F_GETFD"); 621 return -1; 622 } 623 if (fcntl(tcp->pfd_as, F_SETFD, arg|FD_CLOEXEC) < 0) { 624 perror("F_SETFD"); 625 return -1; 626 } 627#else 628 /* Open the process pseudo-file in /proc. */ 629 sprintf(proc, "/proc/%d", tcp->pid); 630 if ((tcp->pfd = open(proc, O_RDWR|O_EXCL)) < 0) { 631 perror("strace: open(\"/proc/...\", ...)"); 632 return -1; 633 } 634 if ((arg = fcntl(tcp->pfd, F_GETFD)) < 0) { 635 perror("F_GETFD"); 636 return -1; 637 } 638 if (fcntl(tcp->pfd, F_SETFD, arg|FD_CLOEXEC) < 0) { 639 perror("F_SETFD"); 640 return -1; 641 } 642#endif 643 rebuild_pollv(); 644 if (!attaching) { 645 /* 646 * Wait for the child to pause. Because of a race 647 * condition we have to poll for the event. 648 */ 649 for (;;) { 650 if (IOCTL_STATUS (tcp) < 0) { 651 perror("strace: PIOCSTATUS"); 652 return -1; 653 } 654 if (tcp->status.PR_FLAGS & PR_ASLEEP) 655 break; 656 } 657 } 658 /* Stop the process so that we own the stop. */ 659 if (IOCTL(tcp->pfd, PIOCSTOP, NULL) < 0) { 660 perror("strace: PIOCSTOP"); 661 return -1; 662 } 663#ifdef PIOCSET 664 /* Set Run-on-Last-Close. */ 665 arg = PR_RLC; 666 if (IOCTL(tcp->pfd, PIOCSET, &arg) < 0) { 667 perror("PIOCSET PR_RLC"); 668 return -1; 669 } 670 /* Set or Reset Inherit-on-Fork. */ 671 arg = PR_FORK; 672 if (IOCTL(tcp->pfd, followfork ? PIOCSET : PIOCRESET, &arg) < 0) { 673 perror("PIOC{SET,RESET} PR_FORK"); 674 return -1; 675 } 676#else /* !PIOCSET */ 677 if (ioctl(tcp->pfd, PIOCSRLC) < 0) { 678 perror("PIOCSRLC"); 679 return -1; 680 } 681 if (ioctl(tcp->pfd, followfork ? PIOCSFORK : PIOCRFORK) < 0) { 682 perror("PIOC{S,R}FORK"); 683 return -1; 684 } 685#endif /* !PIOCSET */ 686 /* Enable all syscall entries. */ 687 prfillset(&sc_enter); 688 if (IOCTL(tcp->pfd, PIOCSENTRY, &sc_enter) < 0) { 689 perror("PIOCSENTRY"); 690 return -1; 691 } 692 /* Enable all syscall exits. */ 693 prfillset(&sc_exit); 694 if (IOCTL(tcp->pfd, PIOCSEXIT, &sc_exit) < 0) { 695 perror("PIOSEXIT"); 696 return -1; 697 } 698 /* Enable all signals. */ 699 prfillset(&signals); 700 if (IOCTL(tcp->pfd, PIOCSTRACE, &signals) < 0) { 701 perror("PIOCSTRACE"); 702 return -1; 703 } 704 /* Enable all faults. */ 705 prfillset(&faults); 706 if (IOCTL(tcp->pfd, PIOCSFAULT, &faults) < 0) { 707 perror("PIOCSFAULT"); 708 return -1; 709 } 710 if (!attaching) { 711#ifdef MIPS 712 /* 713 * The SGI PRSABORT doesn't work for pause() so 714 * we send it a caught signal to wake it up. 715 */ 716 kill(tcp->pid, SIGINT); 717#else /* !MIPS */ 718 /* The child is in a pause(), abort it. */ 719 arg = PRSABORT; 720 if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) { 721 perror("PIOCRUN"); 722 return -1; 723 } 724#endif /* !MIPS */ 725 for (;;) { 726 /* Wait for the child to do something. */ 727 if (IOCTL_WSTOP (tcp) < 0) { 728 perror("PIOCWSTOP"); 729 return -1; 730 } 731 if (tcp->status.PR_WHY == PR_SYSENTRY) { 732#ifdef HAVE_PR_SYSCALL 733 int scno = tcp->status.pr_syscall; 734#else /* !HAVE_PR_SYSCALL */ 735 int scno = tcp->status.PR_WHAT; 736#endif /* !HAVE_PR_SYSCALL */ 737 if (scno == SYS_execve) 738 break; 739 } 740 /* Set it running: maybe execve will be next. */ 741 arg = 0; 742 if (IOCTL(tcp->pfd, PIOCRUN, &arg) < 0) { 743 perror("PIOCRUN"); 744 return -1; 745 } 746 } 747 } 748#ifndef HAVE_POLLABLE_PROCFS 749 if (proc_poll_pipe[0] != -1) 750 proc_poller(tcp->pfd); 751 else if (nprocs > 1) { 752 proc_poll_open(); 753 proc_poller(last_pfd); 754 proc_poller(tcp->pfd); 755 } 756 last_pfd = tcp->pfd; 757#endif /* !HAVE_POLLABLE_PROCFS */ 758 return 0; 759} 760 761#endif /* SVR4 */ 762 763static struct tcb * 764pid2tcb(pid) 765int pid; 766{ 767 int i; 768 struct tcb *tcp; 769 770 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) { 771 if (pid && tcp->pid != pid) 772 continue; 773 if (tcp->flags & TCB_INUSE) 774 return tcp; 775 } 776 return NULL; 777} 778 779#ifdef SVR4 780 781static struct tcb * 782pfd2tcb(pfd) 783int pfd; 784{ 785 int i; 786 struct tcb *tcp; 787 788 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) { 789 if (tcp->pfd != pfd) 790 continue; 791 if (tcp->flags & TCB_INUSE) 792 return tcp; 793 } 794 return NULL; 795} 796 797#endif /* SVR4 */ 798 799void 800droptcb(tcp) 801struct tcb *tcp; 802{ 803 if (tcp->pid == 0) 804 return; 805 nprocs--; 806 tcp->pid = 0; 807 tcp->flags = 0; 808 if (tcp->pfd != -1) { 809 close(tcp->pfd); 810 tcp->pfd = -1; 811#ifdef SVR4 812 rebuild_pollv(); 813#endif 814 } 815 if (tcp->parent != NULL) { 816 tcp->parent->nchildren--; 817 tcp->parent = NULL; 818 } 819#if 0 820 if (tcp->outf != stderr) 821 fclose(tcp->outf); 822#endif 823 tcp->outf = 0; 824} 825 826#ifndef SVR4 827 828static int 829resume(tcp) 830struct tcb *tcp; 831{ 832 if (tcp == NULL) 833 return -1; 834 835 if (!(tcp->flags & TCB_SUSPENDED)) { 836 fprintf(stderr, "PANIC: pid %u not suspended\n", tcp->pid); 837 return -1; 838 } 839 tcp->flags &= ~TCB_SUSPENDED; 840 841 if (ptrace(PTRACE_SYSCALL, tcp->pid, (char *) 1, 0) < 0) { 842 perror("resume: ptrace(PTRACE_SYSCALL, ...)"); 843 return -1; 844 } 845 846 if (!qflag) 847 fprintf(stderr, "Process %u resumed\n", tcp->pid); 848 return 0; 849} 850 851#endif /* !SVR4 */ 852 853/* detach traced process; continue with sig */ 854 855static int 856detach(tcp, sig) 857struct tcb *tcp; 858int sig; 859{ 860 int error = 0; 861#ifdef LINUX 862 int status; 863#endif 864 865 if (tcp->flags & TCB_BPTSET) 866 sig = SIGKILL; 867 868#ifdef LINUX 869 /* 870 * Linux wrongly insists the child be stopped 871 * before detaching. Arghh. We go through hoops 872 * to make a clean break of things. 873 */ 874#if defined(SPARC) 875#undef PTRACE_DETACH 876#define PTRACE_DETACH PTRACE_SUNDETACH 877#endif 878 if ((error = ptrace(PTRACE_DETACH, tcp->pid, (char *) 1, sig)) == 0) { 879 /* On a clear day, you can see forever. */ 880 } 881 else if (errno != ESRCH) { 882 /* Shouldn't happen. */ 883 perror("detach: ptrace(PTRACE_DETACH, ...)"); 884 } 885 else if (kill(tcp->pid, 0) < 0) { 886 if (errno != ESRCH) 887 perror("detach: checking sanity"); 888 } 889 else if (kill(tcp->pid, SIGSTOP) < 0) { 890 if (errno != ESRCH) 891 perror("detach: stopping child"); 892 } 893 else { 894 for (;;) { 895 if (waitpid(tcp->pid, &status, 0) < 0) { 896 if (errno != ECHILD) 897 perror("detach: waiting"); 898 break; 899 } 900 if (!WIFSTOPPED(status)) { 901 /* Au revoir, mon ami. */ 902 break; 903 } 904 if (WSTOPSIG(status) == SIGSTOP) { 905 if ((error = ptrace(PTRACE_DETACH, 906 tcp->pid, (char *) 1, sig)) < 0) { 907 if (errno != ESRCH) 908 perror("detach: ptrace(PTRACE_DETACH, ...)"); 909 /* I died trying. */ 910 } 911 break; 912 } 913 if ((error = ptrace(PTRACE_CONT, tcp->pid, (char *) 1, 914 WSTOPSIG(status) == SIGTRAP ? 915 0 : WSTOPSIG(status))) < 0) { 916 if (errno != ESRCH) 917 perror("detach: ptrace(PTRACE_CONT, ...)"); 918 break; 919 } 920 } 921 } 922#endif /* LINUX */ 923 924#if defined(SUNOS4) 925 /* PTRACE_DETACH won't respect `sig' argument, so we post it here. */ 926 if (sig && kill(tcp->pid, sig) < 0) 927 perror("detach: kill"); 928 sig = 0; 929 if ((error = ptrace(PTRACE_DETACH, tcp->pid, (char *) 1, sig)) < 0) 930 perror("detach: ptrace(PTRACE_DETACH, ...)"); 931#endif /* SUNOS4 */ 932 933#ifndef SVR4 934 if (waiting_parent(tcp)) 935 error = resume(tcp->parent); 936#endif /* !SVR4 */ 937 938 if (!qflag) 939 fprintf(stderr, "Process %u detached\n", tcp->pid); 940 941 droptcb(tcp); 942 return error; 943} 944 945#ifdef SVR4 946 947static void 948reaper(sig) 949int sig; 950{ 951 int pid; 952 int status; 953 954 while ((pid = waitpid(-1, &status, WNOHANG)) > 0) { 955#if 0 956 struct tcb *tcp; 957 958 tcp = pid2tcb(pid); 959 if (tcp) 960 droptcb(tcp); 961#endif 962 } 963} 964 965#endif /* SVR4 */ 966 967static void 968cleanup() 969{ 970 int i; 971 struct tcb *tcp; 972 973 for (i = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) { 974 if (!(tcp->flags & TCB_INUSE)) 975 continue; 976 if (debug) 977 fprintf(stderr, 978 "cleanup: looking at pid %u\n", tcp->pid); 979 if (tcp_last && 980 (!outfname || followfork < 2 || tcp_last == tcp)) { 981 tprintf(" <unfinished ...>\n"); 982 tcp_last = NULL; 983 } 984 if (tcp->flags & TCB_ATTACHED) 985 detach(tcp, 0); 986 else { 987 kill(tcp->pid, SIGCONT); 988 kill(tcp->pid, SIGTERM); 989 } 990 } 991 if (cflag) 992 call_summary(outf); 993} 994 995static void 996interrupt(sig) 997int sig; 998{ 999 interrupted = 1; 1000} 1001 1002#ifndef HAVE_STRERROR 1003 1004#ifndef SYS_ERRLIST_DECLARED 1005extern int sys_nerr; 1006extern char *sys_errlist[]; 1007#endif /* SYS_ERRLIST_DECLARED */ 1008 1009const char * 1010strerror(errno) 1011int errno; 1012{ 1013 static char buf[64]; 1014 1015 if (errno < 1 || errno >= sys_nerr) { 1016 sprintf(buf, "Unknown error %d", errno); 1017 return buf; 1018 } 1019 return sys_errlist[errno]; 1020} 1021 1022#endif /* HAVE_STERRROR */ 1023 1024#ifndef HAVE_STRSIGNAL 1025 1026#ifndef SYS_SIGLIST_DECLARED 1027#ifdef HAVE__SYS_SIGLIST 1028 extern char *_sys_siglist[]; 1029#else 1030 extern char *sys_siglist[]; 1031#endif 1032#endif /* SYS_SIGLIST_DECLARED */ 1033 1034const char * 1035strsignal(sig) 1036int sig; 1037{ 1038 static char buf[64]; 1039 1040 if (sig < 1 || sig >= NSIG) { 1041 sprintf(buf, "Unknown signal %d", sig); 1042 return buf; 1043 } 1044#ifdef HAVE__SYS_SIGLIST 1045 return _sys_siglist[sig]; 1046#else 1047 return sys_siglist[sig]; 1048#endif 1049} 1050 1051#endif /* HAVE_STRSIGNAL */ 1052 1053#ifdef SVR4 1054 1055static void 1056rebuild_pollv() 1057{ 1058 int i, j; 1059 struct tcb *tcp; 1060 1061 for (i = j = 0, tcp = tcbtab; i < MAX_PROCS; i++, tcp++) { 1062 if (!(tcp->flags & TCB_INUSE)) 1063 continue; 1064 pollv[j].fd = tcp->pfd; 1065 pollv[j].events = POLLWANT; 1066 j++; 1067 } 1068 if (j != nprocs) { 1069 fprintf(stderr, "strace: proc miscount\n"); 1070 exit(1); 1071 } 1072} 1073 1074#ifndef HAVE_POLLABLE_PROCFS 1075 1076static void 1077proc_poll_open() 1078{ 1079 int arg; 1080 int i; 1081 1082 if (pipe(proc_poll_pipe) < 0) { 1083 perror("pipe"); 1084 exit(1); 1085 } 1086 for (i = 0; i < 2; i++) { 1087 if ((arg = fcntl(proc_poll_pipe[i], F_GETFD)) < 0) { 1088 perror("F_GETFD"); 1089 exit(1); 1090 } 1091 if (fcntl(proc_poll_pipe[i], F_SETFD, arg|FD_CLOEXEC) < 0) { 1092 perror("F_SETFD"); 1093 exit(1); 1094 } 1095 } 1096} 1097 1098static int 1099proc_poll(pollv, nfds, timeout) 1100struct pollfd *pollv; 1101int nfds; 1102int timeout; 1103{ 1104 int i; 1105 int n; 1106 struct proc_pollfd pollinfo; 1107 1108 if ((n = read(proc_poll_pipe[0], &pollinfo, sizeof(pollinfo))) < 0) 1109 return n; 1110 if (n != sizeof(struct proc_pollfd)) { 1111 fprintf(stderr, "panic: short read: %d\n", n); 1112 exit(1); 1113 } 1114 for (i = 0; i < nprocs; i++) { 1115 if (pollv[i].fd == pollinfo.fd) 1116 pollv[i].revents = pollinfo.revents; 1117 else 1118 pollv[i].revents = 0; 1119 } 1120 poller_pid = pollinfo.pid; 1121 return 1; 1122} 1123 1124static void 1125wakeup_handler(sig) 1126int sig; 1127{ 1128} 1129 1130static void 1131proc_poller(pfd) 1132int pfd; 1133{ 1134 struct proc_pollfd pollinfo; 1135 struct sigaction sa; 1136 sigset_t blocked_set, empty_set; 1137 int i; 1138 int n; 1139 struct rlimit rl; 1140 1141 switch (fork()) { 1142 case -1: 1143 perror("fork"); 1144 _exit(0); 1145 case 0: 1146 break; 1147 default: 1148 return; 1149 } 1150 1151 sa.sa_handler = interactive ? SIG_DFL : SIG_IGN; 1152 sa.sa_flags = 0; 1153 sigemptyset(&sa.sa_mask); 1154 sigaction(SIGHUP, &sa, NULL); 1155 sigaction(SIGINT, &sa, NULL); 1156 sigaction(SIGQUIT, &sa, NULL); 1157 sigaction(SIGPIPE, &sa, NULL); 1158 sigaction(SIGTERM, &sa, NULL); 1159 sa.sa_handler = wakeup_handler; 1160 sigaction(SIGUSR1, &sa, NULL); 1161 sigemptyset(&blocked_set); 1162 sigaddset(&blocked_set, SIGUSR1); 1163 sigprocmask(SIG_BLOCK, &blocked_set, NULL); 1164 sigemptyset(&empty_set); 1165 1166 if (getrlimit(RLIMIT_NOFILE, &rl) < 0) { 1167 perror("getrlimit(RLIMIT_NOFILE, ...)"); 1168 _exit(0); 1169 } 1170 n = rl.rlim_cur; 1171 for (i = 0; i < n; i++) { 1172 if (i != pfd && i != proc_poll_pipe[1]) 1173 close(i); 1174 } 1175 1176 pollinfo.fd = pfd; 1177 pollinfo.pid = getpid(); 1178 for (;;) { 1179 if (ioctl(pfd, PIOCWSTOP, NULL) < 0) 1180 { 1181 switch (errno) { 1182 case EINTR: 1183 continue; 1184 case EBADF: 1185 pollinfo.revents = POLLERR; 1186 break; 1187 case ENOENT: 1188 pollinfo.revents = POLLHUP; 1189 break; 1190 default: 1191 perror("proc_poller: PIOCWSTOP"); 1192 } 1193 write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo)); 1194 _exit(0); 1195 } 1196 pollinfo.revents = POLLWANT; 1197 write(proc_poll_pipe[1], &pollinfo, sizeof(pollinfo)); 1198 sigsuspend(&empty_set); 1199 } 1200} 1201 1202#endif /* !HAVE_POLLABLE_PROCFS */ 1203 1204static int 1205choose_pfd() 1206{ 1207 int i, j; 1208 struct tcb *tcp; 1209 1210 static int last; 1211 1212 if (followfork < 2 && 1213 last < nprocs && (pollv[last].revents & POLLWANT)) { 1214 /* 1215 * The previous process is ready to run again. We'll 1216 * let it do so if it is currently in a syscall. This 1217 * heuristic improves the readability of the trace. 1218 */ 1219 tcp = pfd2tcb(pollv[last].fd); 1220 if (tcp && (tcp->flags & TCB_INSYSCALL)) 1221 return pollv[last].fd; 1222 } 1223 1224 for (i = 0; i < nprocs; i++) { 1225 /* Let competing children run round robin. */ 1226 j = (i + last + 1) % nprocs; 1227 if (pollv[j].revents & (POLLHUP | POLLERR)) { 1228 tcp = pfd2tcb(pollv[j].fd); 1229 if (!tcp) { 1230 fprintf(stderr, "strace: lost proc\n"); 1231 exit(1); 1232 } 1233 droptcb(tcp); 1234 return -1; 1235 } 1236 if (pollv[j].revents & POLLWANT) { 1237 last = j; 1238 return pollv[j].fd; 1239 } 1240 } 1241 fprintf(stderr, "strace: nothing ready\n"); 1242 exit(1); 1243} 1244 1245static int 1246trace() 1247{ 1248#ifdef POLL_HACK 1249 struct tcb *in_syscall; 1250#endif 1251 struct tcb *tcp; 1252 int pfd; 1253 int what; 1254 int ioctl_result = 0, ioctl_errno = 0; 1255 long arg; 1256 1257 for (;;) { 1258 if (interactive) 1259 sigprocmask(SIG_SETMASK, &empty_set, NULL); 1260 1261 if (nprocs == 0) 1262 break; 1263 1264 switch (nprocs) { 1265 case 1: 1266#ifndef HAVE_POLLABLE_PROCFS 1267 if (proc_poll_pipe[0] == -1) { 1268#endif 1269 tcp = pid2tcb(0); 1270 if (!tcp) 1271 continue; 1272 pfd = tcp->pfd; 1273 if (pfd == -1) 1274 continue; 1275 break; 1276#ifndef HAVE_POLLABLE_PROCFS 1277 } 1278 /* fall through ... */ 1279#endif /* !HAVE_POLLABLE_PROCFS */ 1280 default: 1281#ifdef HAVE_POLLABLE_PROCFS 1282#ifdef POLL_HACK 1283 /* On some systems (e.g. UnixWare) we get too much ugly 1284 "unfinished..." stuff when multiple proceses are in 1285 syscalls. Here's a nasty hack */ 1286 1287 if (in_syscall) { 1288 struct pollfd pv; 1289 tcp = in_syscall; 1290 in_syscall = NULL; 1291 pv.fd = tcp->pfd; 1292 pv.events = POLLWANT; 1293 if ((what = poll (&pv, 1, 1)) < 0) { 1294 if (interrupted) 1295 return 0; 1296 continue; 1297 } 1298 else if (what == 1 && pv.revents & POLLWANT) { 1299 goto FOUND; 1300 } 1301 } 1302#endif 1303 1304 if (poll(pollv, nprocs, INFTIM) < 0) { 1305 if (interrupted) 1306 return 0; 1307 continue; 1308 } 1309#else /* !HAVE_POLLABLE_PROCFS */ 1310 if (proc_poll(pollv, nprocs, INFTIM) < 0) { 1311 if (interrupted) 1312 return 0; 1313 continue; 1314 } 1315#endif /* !HAVE_POLLABLE_PROCFS */ 1316 pfd = choose_pfd(); 1317 if (pfd == -1) 1318 continue; 1319 break; 1320 } 1321 1322 /* Look up `pfd' in our table. */ 1323 if ((tcp = pfd2tcb(pfd)) == NULL) { 1324 fprintf(stderr, "unknown pfd: %u\n", pfd); 1325 exit(1); 1326 } 1327 FOUND: 1328 /* Get the status of the process. */ 1329 if (!interrupted) { 1330 ioctl_result = IOCTL_WSTOP (tcp); 1331 ioctl_errno = errno; 1332#ifndef HAVE_POLLABLE_PROCFS 1333 if (proc_poll_pipe[0] != -1) { 1334 if (ioctl_result < 0) 1335 kill(poller_pid, SIGKILL); 1336 else 1337 kill(poller_pid, SIGUSR1); 1338 } 1339#endif /* !HAVE_POLLABLE_PROCFS */ 1340 } 1341 if (interrupted) 1342 return 0; 1343 1344 if (interactive) 1345 sigprocmask(SIG_BLOCK, &blocked_set, NULL); 1346 1347 if (ioctl_result < 0) { 1348 /* Find out what happened if it failed. */ 1349 switch (ioctl_errno) { 1350 case EINTR: 1351 case EBADF: 1352 continue; 1353 case ENOENT: 1354 droptcb(tcp); 1355 continue; 1356 default: 1357 perror("PIOCWSTOP"); 1358 exit(1); 1359 } 1360 } 1361 1362 /* clear the just started flag */ 1363 tcp->flags &= ~TCB_STARTUP; 1364 1365 /* set current output file */ 1366 outf = tcp->outf; 1367 1368 if (cflag) { 1369 struct timeval stime; 1370 1371 stime.tv_sec = tcp->status.pr_stime.tv_sec; 1372 stime.tv_usec = tcp->status.pr_stime.tv_nsec/1000; 1373 tv_sub(&tcp->dtime, &stime, &tcp->stime); 1374 tcp->stime = stime; 1375 } 1376 1377 what = tcp->status.PR_WHAT; 1378 switch (tcp->status.PR_WHY) { 1379 case PR_REQUESTED: 1380 if (tcp->status.PR_FLAGS & PR_ASLEEP) { 1381 tcp->status.PR_WHY = PR_SYSENTRY; 1382 if (trace_syscall(tcp) < 0) { 1383 fprintf(stderr, "syscall trouble\n"); 1384 exit(1); 1385 } 1386 } 1387 break; 1388 case PR_SYSENTRY: 1389#ifdef POLL_HACK 1390 in_syscall = tcp; 1391#endif 1392 case PR_SYSEXIT: 1393 if (trace_syscall(tcp) < 0) { 1394 fprintf(stderr, "syscall trouble\n"); 1395 exit(1); 1396 } 1397 break; 1398 case PR_SIGNALLED: 1399 if (!cflag && (qual_flags[what] & QUAL_SIGNAL)) { 1400 printleader(tcp); 1401 tprintf("--- %s (%s) ---", 1402 signame(what), strsignal(what)); 1403 printtrailer(tcp); 1404 } 1405 break; 1406 case PR_FAULTED: 1407 if (!cflag && (qual_flags[what] & QUAL_FAULT)) { 1408 printleader(tcp); 1409 tprintf("=== FAULT %d ===", what); 1410 printtrailer(tcp); 1411 } 1412 break; 1413 default: 1414 fprintf(stderr, "odd stop %d\n", tcp->status.PR_WHY); 1415 exit(1); 1416 break; 1417 } 1418 arg = 0; 1419 if (IOCTL (tcp->pfd, PIOCRUN, &arg) < 0) { 1420 perror("PIOCRUN"); 1421 exit(1); 1422 } 1423 } 1424 return 0; 1425} 1426 1427#else /* !SVR4 */ 1428 1429static int 1430trace() 1431{ 1432 int pid; 1433 int wait_errno; 1434 int status; 1435 struct tcb *tcp; 1436#ifdef LINUX 1437 struct rusage ru; 1438#endif /* LINUX */ 1439 1440 while (nprocs != 0) { 1441 if (interactive) 1442 sigprocmask(SIG_SETMASK, &empty_set, NULL); 1443#ifdef LINUX 1444 pid = wait4(-1, &status, 0, cflag ? &ru : NULL); 1445#endif /* LINUX */ 1446#ifdef SUNOS4 1447 pid = wait(&status); 1448#endif /* SUNOS4 */ 1449 wait_errno = errno; 1450 if (interactive) 1451 sigprocmask(SIG_BLOCK, &blocked_set, NULL); 1452 1453 if (interrupted) 1454 return 0; 1455 1456 if (pid == -1) { 1457 switch (wait_errno) { 1458 case EINTR: 1459 continue; 1460 case ECHILD: 1461 /* 1462 * We would like to verify this case 1463 * but sometimes a race in Solbourne's 1464 * version of SunOS sometimes reports 1465 * ECHILD before sending us SIGCHILD. 1466 */ 1467#if 0 1468 if (nprocs == 0) 1469 return 0; 1470 fprintf(stderr, "strace: proc miscount\n"); 1471 exit(1); 1472#endif 1473 return 0; 1474 default: 1475 errno = wait_errno; 1476 perror("strace: wait"); 1477 return -1; 1478 } 1479 } 1480 if (debug) 1481 fprintf(stderr, " [wait(%#x) = %u]\n", status, pid); 1482 1483 /* Look up `pid' in our table. */ 1484 if ((tcp = pid2tcb(pid)) == NULL) { 1485#if 1 /* XXX davidm */ 1486 struct tcb *tcpchild; 1487 1488 if ((tcpchild = alloctcb(pid)) == NULL) { 1489 fprintf(stderr, " [tcb table full]\n"); 1490 kill(pid, SIGKILL); /* XXX */ 1491 return 0; 1492 } 1493 tcpchild->flags |= TCB_ATTACHED; 1494 newoutf(tcpchild); 1495 tcp->nchildren++; 1496 if (!qflag) 1497 fprintf(stderr, "Process %d attached\n", pid); 1498#else 1499 fprintf(stderr, "unknown pid: %u\n", pid); 1500 if (WIFSTOPPED(status)) 1501 ptrace(PTRACE_CONT, pid, (char *) 1, 0); 1502 exit(1); 1503#endif 1504 } 1505 /* set current output file */ 1506 outf = tcp->outf; 1507 if (cflag) { 1508#ifdef LINUX 1509 tv_sub(&tcp->dtime, &ru.ru_stime, &tcp->stime); 1510 tcp->stime = ru.ru_stime; 1511#endif /* !LINUX */ 1512 } 1513 1514 if (tcp->flags & TCB_SUSPENDED) { 1515 /* 1516 * Apparently, doing any ptrace() call on a stopped 1517 * process, provokes the kernel to report the process 1518 * status again on a subsequent wait(), even if the 1519 * process has not been actually restarted. 1520 * Since we have inspected the arguments of suspended 1521 * processes we end up here testing for this case. 1522 */ 1523 continue; 1524 } 1525 if (WIFSIGNALED(status)) { 1526 if (!cflag 1527 && (qual_flags[WTERMSIG(status)] & QUAL_SIGNAL)) { 1528 printleader(tcp); 1529 tprintf("+++ killed by %s +++", 1530 signame(WTERMSIG(status))); 1531 printtrailer(tcp); 1532 } 1533 droptcb(tcp); 1534 continue; 1535 } 1536 if (WIFEXITED(status)) { 1537 if (debug) 1538 fprintf(stderr, "pid %u exited\n", pid); 1539 if (tcp->flags & TCB_ATTACHED) 1540 fprintf(stderr, 1541 "PANIC: attached pid %u exited\n", 1542 pid); 1543 droptcb(tcp); 1544 continue; 1545 } 1546 if (!WIFSTOPPED(status)) { 1547 fprintf(stderr, "PANIC: pid %u not stopped\n", pid); 1548 droptcb(tcp); 1549 continue; 1550 } 1551 if (debug) 1552 fprintf(stderr, "pid %u stopped, [%s]\n", 1553 pid, signame(WSTOPSIG(status))); 1554 1555 if (tcp->flags & TCB_STARTUP) { 1556 /* 1557 * This flag is there to keep us in sync. 1558 * Next time this process stops it should 1559 * really be entering a system call. 1560 */ 1561 tcp->flags &= ~TCB_STARTUP; 1562 if (tcp->flags & TCB_ATTACHED) { 1563 /* 1564 * Interestingly, the process may stop 1565 * with STOPSIG equal to some other signal 1566 * than SIGSTOP if we happend to attach 1567 * just before the process takes a signal. 1568 */ 1569 if (!WIFSTOPPED(status)) { 1570 fprintf(stderr, 1571 "pid %u not stopped\n", pid); 1572 detach(tcp, WSTOPSIG(status)); 1573 continue; 1574 } 1575 } 1576 else { 1577#ifdef SUNOS4 1578 /* A child of us stopped at exec */ 1579 if (WSTOPSIG(status) == SIGTRAP && followvfork) 1580 fixvfork(tcp); 1581#endif /* SUNOS4 */ 1582 } 1583 if (tcp->flags & TCB_BPTSET) { 1584 if (clearbpt(tcp) < 0) /* Pretty fatal */ { 1585 droptcb(tcp); 1586 cleanup(); 1587 return -1; 1588 } 1589 } 1590 goto tracing; 1591 } 1592 1593 if (WSTOPSIG(status) != SIGTRAP) { 1594 if (WSTOPSIG(status) == SIGSTOP && 1595 (tcp->flags & TCB_SIGTRAPPED)) { 1596 /* 1597 * Trapped attempt to block SIGTRAP 1598 * Hope we are back in control now. 1599 */ 1600 tcp->flags &= ~(TCB_INSYSCALL | TCB_SIGTRAPPED); 1601 if (ptrace(PTRACE_SYSCALL, 1602 pid, (char *) 1, 0) < 0) { 1603 perror("trace: ptrace(PTRACE_SYSCALL, ...)"); 1604 cleanup(); 1605 return -1; 1606 } 1607 continue; 1608 } 1609 if (!cflag 1610 && (qual_flags[WSTOPSIG(status)] & QUAL_SIGNAL)) { 1611 printleader(tcp); 1612 tprintf("--- %s (%s) ---", 1613 signame(WSTOPSIG(status)), 1614 strsignal(WSTOPSIG(status))); 1615 printtrailer(tcp); 1616 } 1617 if ((tcp->flags & TCB_ATTACHED) && 1618 !sigishandled(tcp, WSTOPSIG(status))) { 1619 detach(tcp, WSTOPSIG(status)); 1620 continue; 1621 } 1622 if (ptrace(PTRACE_SYSCALL, pid, (char *) 1, 1623 WSTOPSIG(status)) < 0) { 1624 perror("trace: ptrace(PTRACE_SYSCALL, ...)"); 1625 cleanup(); 1626 return -1; 1627 } 1628 tcp->flags &= ~TCB_SUSPENDED; 1629 continue; 1630 } 1631 if (trace_syscall(tcp) < 0) { 1632 if (tcp->flags & TCB_ATTACHED) 1633 detach(tcp, 0); 1634 else { 1635 ptrace(PTRACE_KILL, 1636 tcp->pid, (char *) 1, SIGTERM); 1637 droptcb(tcp); 1638 } 1639 continue; 1640 } 1641 if (tcp->flags & TCB_EXITING) { 1642 if (tcp->flags & TCB_ATTACHED) 1643 detach(tcp, 0); 1644 else if (ptrace(PTRACE_CONT, pid, (char *) 1, 0) < 0) { 1645 perror("strace: ptrace(PTRACE_CONT, ...)"); 1646 cleanup(); 1647 return -1; 1648 } 1649 continue; 1650 } 1651 if (tcp->flags & TCB_SUSPENDED) { 1652 if (!qflag) 1653 fprintf(stderr, "Process %u suspended\n", pid); 1654 continue; 1655 } 1656 tracing: 1657 if (ptrace(PTRACE_SYSCALL, pid, (char *) 1, 0) < 0) { 1658 perror("trace: ptrace(PTRACE_SYSCALL, ...)"); 1659 cleanup(); 1660 return -1; 1661 } 1662 } 1663 return 0; 1664} 1665 1666#endif /* !SVR4 */ 1667 1668static int curcol; 1669 1670#ifdef __STDC__ 1671#include <stdarg.h> 1672#define VA_START(a, b) va_start(a, b) 1673#else 1674#include <varargs.h> 1675#define VA_START(a, b) va_start(a) 1676#endif 1677 1678void 1679#ifdef __STDC__ 1680tprintf(const char *fmt, ...) 1681#else 1682tprintf(fmt, va_alist) 1683char *fmt; 1684va_dcl 1685#endif 1686{ 1687 va_list args; 1688 1689 VA_START(args, fmt); 1690 if (outf) 1691 curcol += vfprintf(outf, fmt, args); 1692 va_end(args); 1693 return; 1694} 1695 1696void 1697printleader(tcp) 1698struct tcb *tcp; 1699{ 1700 if (tcp_last && (!outfname || followfork < 2 || tcp_last == tcp)) { 1701 tcp_last->flags |= TCB_REPRINT; 1702 tprintf(" <unfinished ...>\n"); 1703 } 1704 curcol = 0; 1705 if ((followfork == 1 || pflag_seen > 1) && outfname) 1706 tprintf("%-5d ", tcp->pid); 1707 else if (nprocs > 1 && !outfname) 1708 tprintf("[pid %5u] ", tcp->pid); 1709 if (tflag) { 1710 char str[sizeof("HH:MM:SS")]; 1711 struct timeval tv, dtv; 1712 static struct timeval otv; 1713 1714 gettimeofday(&tv, NULL); 1715 if (rflag) { 1716 if (otv.tv_sec == 0) 1717 otv = tv; 1718 tv_sub(&dtv, &tv, &otv); 1719 tprintf("%6ld.%06ld ", 1720 (long) dtv.tv_sec, (long) dtv.tv_usec); 1721 otv = tv; 1722 } 1723 else if (tflag > 2) { 1724 tprintf("%ld.%06ld ", 1725 (long) tv.tv_sec, (long) tv.tv_usec); 1726 } 1727 else { 1728 time_t local = tv.tv_sec; 1729 strftime(str, sizeof(str), "%T", localtime(&local)); 1730 if (tflag > 1) 1731 tprintf("%s.%06ld ", str, (long) tv.tv_usec); 1732 else 1733 tprintf("%s ", str); 1734 } 1735 } 1736 if (iflag) 1737 printcall(tcp); 1738} 1739 1740void 1741tabto(col) 1742int col; 1743{ 1744 if (curcol < col) 1745 tprintf("%*s", col - curcol, ""); 1746} 1747 1748void 1749printtrailer(tcp) 1750struct tcb *tcp; 1751{ 1752 tprintf("\n"); 1753 tcp_last = NULL; 1754} 1755 1756#ifdef HAVE_MP_PROCFS 1757 1758int mp_ioctl (int fd, int cmd, void *arg, int size) { 1759 1760 struct iovec iov[2]; 1761 int n = 1; 1762 1763 iov[0].iov_base = &cmd; 1764 iov[0].iov_len = sizeof cmd; 1765 if (arg) { 1766 ++n; 1767 iov[1].iov_base = arg; 1768 iov[1].iov_len = size; 1769 } 1770 1771 return writev (fd, iov, n); 1772} 1773 1774#endif 1775