1999-02-19 00:21:36 +00:00
/*
* 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 >
1999-12-23 14:20:14 +00:00
* Copyright ( c ) 1996 - 1999 Wichert Akkerman < wichert @ cistron . nl >
1999-02-19 00:21:36 +00:00
* 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"
2005-02-02 04:44:57 +00:00
# include <sys/types.h>
2011-05-23 21:29:03 +02:00
# include <stdarg.h>
1999-02-19 00:21:36 +00:00
# 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 <pwd.h>
# include <grp.h>
# include <string.h>
2001-10-19 08:59:12 +00:00
# include <limits.h>
2004-04-09 00:25:21 +00:00
# include <dirent.h>
1999-02-19 00:21:36 +00:00
2007-06-02 00:07:33 +00:00
# ifdef LINUX
# include <asm / unistd.h>
# if defined __NR_tgkill
2011-06-07 12:13:24 +02:00
# define my_tgkill(pid, tid, sig) syscall(__NR_tgkill, (pid), (tid), (sig))
2007-06-02 00:07:33 +00:00
# elif defined __NR_tkill
2011-06-07 12:13:24 +02:00
# define my_tgkill(pid, tid, sig) syscall(__NR_tkill, (tid), (sig))
2007-06-02 00:07:33 +00:00
# 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!"
2011-06-07 12:13:24 +02:00
# define my_tgkill(pid, tid, sig) kill((tid), (sig))
2007-06-02 00:07:33 +00:00
# endif
# endif
2001-10-09 23:47:38 +00:00
# if defined(IA64) && defined(LINUX)
# include <asm / ptrace_offsets.h>
# endif
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
# include <poll.h>
# endif
1999-02-19 00:21:36 +00:00
# ifdef SVR4
# include <sys/stropts.h>
1999-11-29 15:34:02 +00:00
# ifdef HAVE_MP_PROCFS
2001-07-10 13:48:44 +00:00
# ifdef HAVE_SYS_UIO_H
1999-08-29 23:15:07 +00:00
# include <sys/uio.h>
# endif
1999-02-19 00:21:36 +00:00
# endif
2001-07-10 13:48:44 +00:00
# endif
2008-12-29 19:13:27 +00:00
extern char * * environ ;
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
extern int optind ;
extern char * optarg ;
2008-12-29 19:13:27 +00:00
1999-02-19 00:21:36 +00:00
2008-07-18 00:25:10 +00:00
int debug = 0 , followfork = 0 ;
2011-06-21 14:34:10 +02:00
unsigned int ptrace_setoptions = 0 ;
2011-05-23 21:29:03 +02:00
/* Which WSTOPSIG(status) value marks syscall traps? */
2011-05-27 14:36:01 +02:00
static unsigned int syscall_trap_sig = SIGTRAP ;
2010-03-28 19:24:54 +00:00
int dtime = 0 , xflag = 0 , qflag = 0 ;
cflag_t cflag = CFLAG_NONE ;
2006-12-13 16:59:44 +00:00
static int iflag = 0 , interactive = 0 , pflag_seen = 0 , rflag = 0 , tflag = 0 ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
/*
* 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 ;
1999-02-19 00:21:36 +00:00
2002-11-06 13:17:21 +00:00
/* Sometimes we want to print only succeeding syscalls. */
int not_failing_only = 0 ;
2011-04-07 20:25:40 +00:00
/* Show path associated with fd arguments */
int show_fd_path = 0 ;
/* are we filtering traces based on paths? */
int tracing_paths = 0 ;
2008-11-10 17:14:58 +00:00
static int exit_code = 0 ;
static int strace_child = 0 ;
2011-05-27 14:36:01 +02:00
static int strace_tracer_pid = 0 ;
2009-06-02 16:49:22 -07:00
2006-12-13 16:59:44 +00:00
static char * username = NULL ;
1999-02-19 00:21:36 +00:00
uid_t run_uid ;
gid_t run_gid ;
int acolumn = DEFAULT_ACOLUMN ;
int max_strlen = DEFAULT_STRLEN ;
2006-12-13 16:59:44 +00:00
static char * outfname = NULL ;
1999-02-19 00:21:36 +00:00
FILE * outf ;
2009-10-27 16:27:13 +01:00
static int curcol ;
2002-12-18 04:16:10 +00:00
struct tcb * * tcbtab ;
2011-06-22 12:45:25 +02:00
static unsigned int nprocs , tcbtabsize ;
2010-09-06 22:08:24 +00:00
const char * progname ;
1999-02-19 00:21:36 +00:00
2009-10-27 16:56:43 +01:00
static int detach ( struct tcb * tcp , int sig ) ;
static int trace ( void ) ;
static void cleanup ( void ) ;
static void interrupt ( int sig ) ;
1999-02-19 00:21:36 +00:00
static sigset_t empty_set , blocked_set ;
# ifdef HAVE_SIG_ATOMIC_T
static volatile sig_atomic_t interrupted ;
# else /* !HAVE_SIG_ATOMIC_T */
static volatile int interrupted ;
# endif /* !HAVE_SIG_ATOMIC_T */
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
2009-10-27 16:56:43 +01:00
static struct tcb * pfd2tcb ( int pfd ) ;
static void reaper ( int sig ) ;
static void rebuild_pollv ( void ) ;
2002-12-18 04:16:10 +00:00
static struct pollfd * pollv ;
1999-02-19 00:21:36 +00:00
# ifndef HAVE_POLLABLE_PROCFS
2009-10-27 16:56:43 +01:00
static void proc_poll_open ( void ) ;
static void proc_poller ( int pfd ) ;
1999-02-19 00:21:36 +00:00
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 */
1999-11-29 15:34:02 +00:00
# ifdef HAVE_MP_PROCFS
1999-08-29 23:15:07 +00:00
# define POLLWANT POLLWRNORM
# else
# define POLLWANT POLLPRI
# endif
2000-09-01 21:03:06 +00:00
# endif /* USE_PROCFS */
1999-02-19 00:21:36 +00:00
static void
usage ( ofp , exitval )
FILE * ofp ;
int exitval ;
{
fprintf ( ofp , " \
2011-04-07 20:25:40 +00:00
usage : strace [ - CdDffhiqrtttTvVxxy ] [ - a column ] [ - e expr ] . . . [ - o file ] \ n \
2003-01-24 04:31:23 +00:00
[ - p pid ] . . . [ - s strsize ] [ - u username ] [ - E var = val ] . . . \ n \
2011-04-07 20:25:40 +00:00
[ - P path ] [ command [ arg . . . ] ] \ n \
2010-06-11 15:49:36 +02:00
or : strace - c [ - D ] [ - e expr ] . . . [ - O overhead ] [ - S sortby ] [ - E var = val ] . . . \ n \
2003-01-24 04:31:23 +00:00
[ command [ arg . . . ] ] \ n \
1999-02-19 00:21:36 +00:00
- c - - count time , calls , and errors for each syscall and report summary \ n \
2010-06-11 15:49:36 +02:00
- C - - like - c but also print regular output while processes are running \ n \
1999-02-19 00:21:36 +00:00
- 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 \
2011-04-07 20:25:40 +00:00
- y - - print paths associated with file descriptor arguments \ n \
1999-02-19 00:21:36 +00:00
- 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 \
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
- D - - run tracer process as a detached grandchild , not as parent \ n \
1999-02-19 00:21:36 +00:00
- 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 \
2003-01-24 04:31:23 +00:00
- E var = val - - put var = val in the environment for command \ n \
- E var - - remove var from the environment for command \ n \
2011-04-07 20:25:40 +00:00
- P path - - trace accesses to path \ n \
2003-01-24 04:31:23 +00:00
" /* this is broken, so don't document it
2002-11-06 13:17:21 +00:00
- z - - print only succeeding syscalls \ n \
2003-01-24 04:31:23 +00:00
*/
, DEFAULT_ACOLUMN , DEFAULT_STRLEN , DEFAULT_SORTBY ) ;
1999-02-19 00:21:36 +00:00
exit ( exitval ) ;
}
2011-05-27 14:36:01 +02:00
static void die ( void ) __attribute__ ( ( noreturn ) ) ;
static void die ( void )
{
if ( strace_tracer_pid = = getpid ( ) ) {
cflag = 0 ;
cleanup ( ) ;
}
exit ( 1 ) ;
}
static void verror_msg ( int err_no , const char * fmt , va_list p )
2011-05-23 21:29:03 +02:00
{
2011-06-09 15:50:41 +00:00
fflush ( NULL ) ;
fprintf ( stderr , " %s: " , progname ) ;
vfprintf ( stderr , fmt , p ) ;
if ( err_no )
fprintf ( stderr , " : %s \n " , strerror ( err_no ) ) ;
else
putc ( ' \n ' , stderr ) ;
fflush ( stderr ) ;
2011-05-27 14:36:01 +02:00
}
void error_msg ( const char * fmt , . . . )
{
va_list p ;
va_start ( p , fmt ) ;
verror_msg ( 0 , fmt , p ) ;
2011-05-23 21:29:03 +02:00
va_end ( p ) ;
2011-05-27 14:36:01 +02:00
}
2011-05-23 21:29:03 +02:00
2011-05-27 14:36:01 +02:00
void error_msg_and_die ( const char * fmt , . . . )
{
va_list p ;
va_start ( p , fmt ) ;
verror_msg ( 0 , fmt , p ) ;
die ( ) ;
}
void perror_msg ( const char * fmt , . . . )
{
va_list p ;
va_start ( p , fmt ) ;
verror_msg ( errno , fmt , p ) ;
va_end ( p ) ;
}
void perror_msg_and_die ( const char * fmt , . . . )
{
va_list p ;
va_start ( p , fmt ) ;
verror_msg ( errno , fmt , p ) ;
die ( ) ;
2011-05-23 21:29:03 +02:00
}
1999-02-19 00:21:36 +00:00
# ifdef SVR4
# ifdef MIPS
void
foobar ( )
{
}
# endif /* MIPS */
# endif /* SVR4 */
2009-10-07 20:41:29 -04:00
/* Glue for systems without a MMU that cannot provide fork() */
# ifdef HAVE_FORK
# define strace_vforked 0
# else
# define strace_vforked 1
# define fork() vfork()
# endif
2011-06-22 13:11:23 +02:00
static void
2006-12-13 21:45:31 +00:00
set_cloexec_flag ( int fd )
{
2011-06-22 13:11:23 +02:00
int flags , newflags ;
flags = fcntl ( fd , F_GETFD ) ;
if ( flags < 0 ) {
/* Can happen only if fd is bad.
* Should never happen : if it does , we have a bug
* in the caller . Therefore we just abort
* instead of propagating the error .
*/
perror_msg_and_die ( " fcntl(%d, F_GETFD) " , fd ) ;
2006-12-13 21:45:31 +00:00
}
newflags = flags | FD_CLOEXEC ;
if ( flags = = newflags )
2011-06-22 13:11:23 +02:00
return ;
2006-12-13 21:45:31 +00:00
2011-06-22 13:11:23 +02:00
fcntl ( fd , F_SETFD , newflags ) ; /* never fails */
2006-12-13 21:45:31 +00:00
}
/*
* 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
}
2007-07-05 20:03:16 +00:00
# if _LFS64_LARGEFILE
# define fopen_for_output fopen64
# else
# define fopen_for_output fopen
# endif
2006-12-13 21:45:31 +00:00
static FILE *
strace_fopen ( const char * path , const char * mode )
{
FILE * fp ;
swap_uid ( ) ;
2007-07-05 20:03:16 +00:00
if ( ( fp = fopen_for_output ( path , mode ) ) = = NULL )
2006-12-13 21:45:31 +00:00
fprintf ( stderr , " %s: can't fopen '%s': %s \n " ,
progname , path , strerror ( errno ) ) ;
swap_uid ( ) ;
2011-06-22 13:11:23 +02:00
if ( fp )
set_cloexec_flag ( fileno ( fp ) ) ;
2006-12-13 21:45:31 +00:00
return fp ;
}
2011-06-22 13:03:56 +02:00
static int popen_pid = 0 ;
2006-12-13 21:45:31 +00:00
# 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 )
{
2011-06-22 13:03:56 +02:00
FILE * fp ;
int fds [ 2 ] ;
2006-12-13 21:45:31 +00:00
swap_uid ( ) ;
if ( pipe ( fds ) < 0 )
2011-06-22 13:03:56 +02:00
perror_msg_and_die ( " pipe " ) ;
2006-12-13 21:45:31 +00:00
2011-06-22 13:03:56 +02:00
set_cloexec_flag ( fds [ 1 ] ) ; /* never fails */
2006-12-13 21:45:31 +00:00
2011-06-22 13:03:56 +02:00
popen_pid = vfork ( ) ;
if ( popen_pid = = - 1 )
perror_msg_and_die ( " vfork " ) ;
2006-12-13 21:45:31 +00:00
2011-06-22 13:03:56 +02:00
if ( popen_pid = = 0 ) {
2006-12-13 21:45:31 +00:00
/* child */
close ( fds [ 1 ] ) ;
2011-06-22 13:03:56 +02:00
if ( fds [ 0 ] ! = 0 ) {
if ( dup2 ( fds [ 0 ] , 0 ) )
perror_msg_and_die ( " dup2 " ) ;
close ( fds [ 0 ] ) ;
2006-12-13 21:45:31 +00:00
}
execl ( _PATH_BSHELL , " sh " , " -c " , command , NULL ) ;
2011-06-22 13:03:56 +02:00
perror_msg_and_die ( " Can't execute '%s' " , _PATH_BSHELL ) ;
2006-12-13 21:45:31 +00:00
}
2011-06-22 13:03:56 +02:00
/* parent */
close ( fds [ 0 ] ) ;
swap_uid ( ) ;
fp = fdopen ( fds [ 1 ] , " w " ) ;
if ( ! fp )
error_msg_and_die ( " Out of memory " ) ;
return fp ;
2006-12-13 21:45:31 +00:00
}
static int
newoutf ( struct tcb * tcp )
{
if ( outfname & & followfork > 1 ) {
2009-01-29 20:38:20 +00:00
char name [ 520 + sizeof ( int ) * 3 ] ;
2006-12-13 21:45:31 +00:00
FILE * fp ;
2009-01-29 20:38:20 +00:00
sprintf ( name , " %.512s.%u " , outfname , tcp - > pid ) ;
2006-12-13 21:45:31 +00:00
if ( ( fp = strace_fopen ( name , " w " ) ) = = NULL )
return - 1 ;
tcp - > outf = fp ;
}
return 0 ;
}
2007-06-11 22:06:31 +00:00
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 .
2011-06-07 12:13:24 +02:00
* We rely on cleanup ( ) from this point on .
2007-06-11 22:06:31 +00:00
*/
if ( interactive )
sigprocmask ( SIG_BLOCK , & blocked_set , NULL ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( daemonized_tracer ) {
pid_t pid = fork ( ) ;
if ( pid < 0 ) {
_exit ( 1 ) ;
}
if ( pid ) { /* parent */
/*
2011-05-27 14:36:01 +02:00
* Wait for grandchild to attach to straced process
* ( grandparent ) . Grandchild SIGKILLs us after it attached .
* Grandparent ' s wait ( ) is unblocked by our death ,
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
* it proceeds to exec the straced program .
*/
pause ( ) ;
_exit ( 0 ) ; /* paranoia */
}
2011-05-27 14:36:01 +02:00
/* grandchild */
/* We will be the tracer process. Remember our new pid: */
strace_tracer_pid = getpid ( ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
}
2007-06-11 22:06:31 +00:00
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
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( followfork & & ! daemonized_tracer ) {
2009-01-29 20:38:20 +00:00
char procdir [ sizeof ( " /proc/%d/task " ) + sizeof ( int ) * 3 ] ;
2007-06-11 22:06:31 +00:00
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 ) {
2009-01-29 20:38:20 +00:00
if ( de - > d_fileno = = 0 )
2007-06-11 22:06:31 +00:00
continue ;
tid = atoi ( de - > d_name ) ;
if ( tid < = 0 )
continue ;
+ + ntid ;
2009-03-17 14:46:54 +00:00
if ( ptrace ( PTRACE_ATTACH , tid , ( char * ) 1 , 0 ) < 0 )
2007-06-11 22:06:31 +00:00
+ + nerr ;
2009-03-17 14:46:54 +00:00
else if ( tid ! = tcbtab [ tcbi ] - > pid ) {
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
tcp = alloctcb ( tid ) ;
2010-08-27 17:43:16 +08:00
tcp - > flags | = TCB_ATTACHED | TCB_CLONE_THREAD | TCB_FOLLOWFORK ;
2007-06-11 22:06:31 +00:00
tcbtab [ tcbi ] - > nclone_threads + + ;
tcp - > parent = tcbtab [ tcbi ] ;
}
2009-03-17 14:46:54 +00:00
if ( interactive ) {
sigprocmask ( SIG_SETMASK , & empty_set , NULL ) ;
if ( interrupted )
return ;
sigprocmask ( SIG_BLOCK , & blocked_set , NULL ) ;
}
2007-06-11 22:06:31 +00:00
}
closedir ( dir ) ;
2009-01-29 20:38:20 +00:00
ntid - = nerr ;
if ( ntid = = 0 ) {
2007-06-11 22:06:31 +00:00
perror ( " attach: ptrace(PTRACE_ATTACH, ...) " ) ;
droptcb ( tcp ) ;
continue ;
}
if ( ! qflag ) {
2009-01-29 20:38:20 +00:00
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 ) ;
2007-06-11 22:06:31 +00:00
}
continue ;
2009-01-29 20:38:20 +00:00
} /* if (opendir worked) */
} /* if (-f) */
2007-06-11 22:06:31 +00:00
# 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. */
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
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 ) ;
}
2007-06-11 22:06:31 +00:00
# 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
2011-05-30 14:00:14 +02:00
startup_child ( char * * argv )
2007-06-11 22:06:31 +00:00
{
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 {
2010-09-06 22:08:24 +00:00
const char * path ;
2007-06-11 22:06:31 +00:00
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 ) ;
}
2008-11-10 17:14:58 +00:00
strace_child = pid = fork ( ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( pid < 0 ) {
2007-06-11 22:06:31 +00:00
perror ( " strace: fork " ) ;
cleanup ( ) ;
exit ( 1 ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
}
2011-05-27 14:36:01 +02:00
if ( ( pid ! = 0 & & daemonized_tracer ) /* -D: parent to become a traced process */
| | ( pid = = 0 & & ! daemonized_tracer ) /* not -D: child to become a traced process */
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
) {
pid = getpid ( ) ;
2007-06-11 22:06:31 +00:00
# ifdef USE_PROCFS
2011-06-07 12:13:24 +02:00
if ( outf ! = stderr ) close ( fileno ( outf ) ) ;
2007-06-11 22:06:31 +00:00
# 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 */
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
kill ( pid , SIGSTOP ) ; /* stop HERE */
2007-06-11 22:06:31 +00:00
# endif /* FREEBSD */
# else /* !USE_PROCFS */
2011-06-07 12:13:24 +02:00
if ( outf ! = stderr )
close ( fileno ( outf ) ) ;
2007-06-11 22:06:31 +00:00
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( ! daemonized_tracer ) {
if ( ptrace ( PTRACE_TRACEME , 0 , ( char * ) 1 , 0 ) < 0 ) {
perror ( " strace: ptrace(PTRACE_TRACEME, ...) " ) ;
exit ( 1 ) ;
}
if ( debug )
kill ( pid , SIGSTOP ) ;
2007-06-11 22:06:31 +00:00
}
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 ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( ! daemonized_tracer ) {
/*
* Induce an immediate stop so that the parent
* will resume us with PTRACE_SYSCALL and display
* this execve call normally .
2009-10-07 20:41:29 -04:00
* Unless of course we ' re on a no - MMU system where
* we vfork ( ) - ed , so we cannot stop the child .
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
*/
2009-10-07 20:41:29 -04:00
if ( ! strace_vforked )
kill ( getpid ( ) , SIGSTOP ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
} 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 ) ;
}
2007-06-11 22:06:31 +00:00
# endif /* !USE_PROCFS */
execv ( pathname , argv ) ;
perror ( " strace: exec " ) ;
_exit ( 1 ) ;
}
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
/* We are the tracer. */
2011-05-27 14:36:01 +02:00
/* With -D, we are *child* here, IOW: different pid. Fetch it. */
strace_tracer_pid = getpid ( ) ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
tcp = alloctcb ( daemonized_tracer ? getppid ( ) : pid ) ;
if ( daemonized_tracer ) {
/* We want subsequent startup_attach() to attach to it. */
tcp - > flags | = TCB_ATTACHED ;
}
2007-06-11 22:06:31 +00:00
# ifdef USE_PROCFS
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( proc_open ( tcp , 0 ) < 0 ) {
fprintf ( stderr , " trouble opening proc file \n " ) ;
cleanup ( ) ;
exit ( 1 ) ;
2007-06-11 22:06:31 +00:00
}
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
# endif /* USE_PROCFS */
2007-06-11 22:06:31 +00:00
}
2010-11-12 17:25:19 +08:00
# ifdef LINUX
/*
2010-12-02 20:56:43 +00:00
* Test whether the kernel support PTRACE_O_TRACECLONE et al options .
2010-11-12 17:25:19 +08:00
* First fork a new child , call ptrace with PTRACE_SETOPTIONS on it ,
2010-12-02 20:56:43 +00:00
* and then see which options are supported by the kernel .
2010-11-12 17:25:19 +08:00
*/
static int
2011-05-23 21:29:03 +02:00
test_ptrace_setoptions_followfork ( void )
2010-11-12 17:25:19 +08:00
{
2011-02-19 21:33:50 +00:00
int pid , expected_grandchild = 0 , found_grandchild = 0 ;
const unsigned int test_options = PTRACE_O_TRACECLONE |
PTRACE_O_TRACEFORK |
PTRACE_O_TRACEVFORK ;
2010-11-12 17:25:19 +08:00
if ( ( pid = fork ( ) ) < 0 )
return - 1 ;
else if ( pid = = 0 ) {
2010-12-02 20:56:43 +00:00
if ( ptrace ( PTRACE_TRACEME , 0 , ( char * ) 1 , 0 ) < 0 )
2010-11-12 17:25:19 +08:00
_exit ( 1 ) ;
kill ( getpid ( ) , SIGSTOP ) ;
2010-12-02 20:56:43 +00:00
_exit ( fork ( ) < 0 ) ;
2010-11-12 17:25:19 +08:00
}
2010-12-02 20:56:43 +00:00
while ( 1 ) {
int status , tracee_pid ;
tracee_pid = wait ( & status ) ;
if ( tracee_pid = = - 1 ) {
if ( errno = = EINTR )
continue ;
else if ( errno = = ECHILD )
2010-11-12 17:25:19 +08:00
break ;
2011-05-23 21:29:03 +02:00
perror ( " test_ptrace_setoptions_followfork " ) ;
2010-12-02 20:56:43 +00:00
return - 1 ;
}
if ( tracee_pid ! = pid ) {
2011-02-19 21:33:50 +00:00
found_grandchild = tracee_pid ;
2010-12-02 20:56:43 +00:00
if ( ptrace ( PTRACE_CONT , tracee_pid , 0 , 0 ) < 0 & &
errno ! = ESRCH )
kill ( tracee_pid , SIGKILL ) ;
}
else if ( WIFSTOPPED ( status ) ) {
2011-02-19 21:33:50 +00:00
switch ( WSTOPSIG ( status ) ) {
case SIGSTOP :
if ( ptrace ( PTRACE_SETOPTIONS , pid ,
NULL , test_options ) < 0 ) {
2010-11-12 17:25:19 +08:00
kill ( pid , SIGKILL ) ;
2010-12-02 20:56:43 +00:00
return - 1 ;
}
2011-02-19 21:33:50 +00:00
break ;
case SIGTRAP :
if ( status > > 16 = = PTRACE_EVENT_FORK ) {
long msg = 0 ;
if ( ptrace ( PTRACE_GETEVENTMSG , pid ,
NULL , ( long ) & msg ) = = 0 )
expected_grandchild = msg ;
}
break ;
2010-11-12 17:25:19 +08:00
}
2010-12-02 20:56:43 +00:00
if ( ptrace ( PTRACE_SYSCALL , pid , 0 , 0 ) < 0 & &
errno ! = ESRCH )
kill ( pid , SIGKILL ) ;
2010-11-12 17:25:19 +08:00
}
}
2011-02-19 21:33:50 +00:00
if ( expected_grandchild & & expected_grandchild = = found_grandchild )
2011-06-21 14:34:10 +02:00
ptrace_setoptions | = test_options ;
2010-11-12 17:25:19 +08:00
return 0 ;
}
2011-05-23 21:29:03 +02:00
/*
* Test whether the kernel support PTRACE_O_TRACESYSGOOD .
* First fork a new child , call ptrace ( PTRACE_SETOPTIONS ) on it ,
* and then see whether it will stop with ( SIGTRAP | 0x80 ) .
*
* Use of this option enables correct handling of user - generated SIGTRAPs ,
* and SIGTRAPs generated by special instructions such as int3 on x86 :
* _start : . globl _start
* int3
* movl $ 42 , % ebx
* movl $ 1 , % eax
* int $ 0x80
* ( compile with : " gcc -nostartfiles -nostdlib -o int3 int3.S " )
*/
static void
test_ptrace_setoptions_for_all ( void )
{
const unsigned int test_options = PTRACE_O_TRACESYSGOOD | PTRACE_O_TRACEEXEC ;
int pid ;
int it_worked = 0 ;
pid = fork ( ) ;
if ( pid < 0 )
2011-05-27 14:36:01 +02:00
perror_msg_and_die ( " fork " ) ;
2011-05-23 21:29:03 +02:00
if ( pid = = 0 ) {
pid = getpid ( ) ;
if ( ptrace ( PTRACE_TRACEME , 0L , 0L , 0L ) < 0 )
2011-05-27 14:36:01 +02:00
/* Note: exits with exitcode 1 */
perror_msg_and_die ( " %s: PTRACE_TRACEME doesn't work " , __func__ ) ;
2011-05-23 21:29:03 +02:00
kill ( pid , SIGSTOP ) ;
_exit ( 0 ) ; /* parent should see entry into this syscall */
}
while ( 1 ) {
int status , tracee_pid ;
errno = 0 ;
tracee_pid = wait ( & status ) ;
if ( tracee_pid < = 0 ) {
if ( errno = = EINTR )
continue ;
kill ( pid , SIGKILL ) ;
2011-05-27 14:36:01 +02:00
perror_msg_and_die ( " %s: unexpected wait result %d " , __func__ , tracee_pid ) ;
}
if ( WIFEXITED ( status ) ) {
if ( WEXITSTATUS ( status ) = = 0 )
break ;
/* PTRACE_TRACEME failed in child. This is fatal. */
exit ( 1 ) ;
2011-05-23 21:29:03 +02:00
}
if ( ! WIFSTOPPED ( status ) ) {
kill ( pid , SIGKILL ) ;
error_msg_and_die ( " %s: unexpected wait status %x " , __func__ , status ) ;
}
if ( WSTOPSIG ( status ) = = SIGSTOP ) {
/*
* We don ' t check " options aren't accepted " error .
* If it happens , we ' ll never get ( SIGTRAP | 0x80 ) ,
* and thus will decide to not use the option .
* IOW : the outcome of the test will be correct .
*/
2011-05-27 14:36:01 +02:00
if ( ptrace ( PTRACE_SETOPTIONS , pid , 0L , test_options ) < 0 )
if ( errno ! = EINVAL )
perror_msg ( " PTRACE_SETOPTIONS " ) ;
2011-05-23 21:29:03 +02:00
}
if ( WSTOPSIG ( status ) = = ( SIGTRAP | 0x80 ) ) {
it_worked = 1 ;
}
if ( ptrace ( PTRACE_SYSCALL , pid , 0L , 0L ) < 0 ) {
kill ( pid , SIGKILL ) ;
2011-05-27 14:36:01 +02:00
perror_msg_and_die ( " PTRACE_SYSCALL doesn't work " ) ;
2011-05-23 21:29:03 +02:00
}
}
if ( it_worked ) {
2011-05-27 14:36:01 +02:00
syscall_trap_sig = ( SIGTRAP | 0x80 ) ;
2011-06-21 14:34:10 +02:00
ptrace_setoptions | = test_options ;
2011-05-23 21:29:03 +02:00
if ( debug )
2011-06-21 14:34:10 +02:00
fprintf ( stderr , " ptrace_setoptions = %#x \n " ,
ptrace_setoptions ) ;
2011-05-23 21:29:03 +02:00
return ;
}
fprintf ( stderr ,
" Test for PTRACE_O_TRACESYSGOOD failed, giving up using this feature. \n " ) ;
}
2010-11-12 17:25:19 +08:00
# endif
1999-02-19 00:21:36 +00:00
int
2007-10-08 21:04:41 +00:00
main ( int argc , char * argv [ ] )
1999-02-19 00:21:36 +00:00
{
struct tcb * tcp ;
int c , pid = 0 ;
2008-07-25 15:42:34 +00:00
int optF = 0 ;
1999-02-19 00:21:36 +00:00
struct sigaction sa ;
static char buf [ BUFSIZ ] ;
2007-10-08 21:04:41 +00:00
progname = argv [ 0 ] ? argv [ 0 ] : " strace " ;
2011-05-27 14:36:01 +02:00
strace_tracer_pid = getpid ( ) ;
2002-12-18 04:16:10 +00:00
/* Allocate the initial tcbtab. */
tcbtabsize = argc ; /* Surely enough for all -p args. */
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
if ( ( tcbtab = calloc ( tcbtabsize , sizeof tcbtab [ 0 ] ) ) = = NULL ) {
2007-10-08 21:04:41 +00:00
fprintf ( stderr , " %s: out of memory \n " , progname ) ;
exit ( 1 ) ;
}
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
if ( ( tcbtab [ 0 ] = calloc ( tcbtabsize , sizeof tcbtab [ 0 ] [ 0 ] ) ) = = NULL ) {
2007-10-08 21:04:41 +00:00
fprintf ( stderr , " %s: out of memory \n " , progname ) ;
exit ( 1 ) ;
}
2002-12-18 04:16:10 +00:00
for ( tcp = tcbtab [ 0 ] ; tcp < & tcbtab [ 0 ] [ tcbtabsize ] ; + + tcp )
tcbtab [ tcp - tcbtab [ 0 ] ] = & tcbtab [ 0 ] [ tcp - tcbtab [ 0 ] ] ;
1999-02-19 00:21:36 +00:00
outf = stderr ;
interactive = 1 ;
2006-01-12 09:50:49 +00:00
set_sortby ( DEFAULT_SORTBY ) ;
set_personality ( DEFAULT_PERSONALITY ) ;
1999-02-19 00:21:36 +00:00
qualify ( " trace=all " ) ;
qualify ( " abbrev=all " ) ;
qualify ( " verbose=all " ) ;
qualify ( " signal=all " ) ;
while ( ( c = getopt ( argc , argv ,
2011-04-07 20:25:40 +00:00
" +cCdfFhiqrtTvVxyz "
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
# ifndef USE_PROCFS
" D "
# endif
2011-04-07 20:25:40 +00:00
" a:e:o:O:p:s:S:u:E:P: " ) ) ! = EOF ) {
1999-02-19 00:21:36 +00:00
switch ( c ) {
case ' c ' :
2010-03-28 19:24:54 +00:00
if ( cflag = = CFLAG_BOTH ) {
fprintf ( stderr , " %s: -c and -C are mutually exclusive options \n " ,
progname ) ;
exit ( 1 ) ;
}
cflag = CFLAG_ONLY_STATS ;
break ;
case ' C ' :
if ( cflag = = CFLAG_ONLY_STATS ) {
fprintf ( stderr , " %s: -c and -C are mutually exclusive options \n " ,
progname ) ;
exit ( 1 ) ;
}
cflag = CFLAG_BOTH ;
1999-02-19 00:21:36 +00:00
break ;
case ' d ' :
debug + + ;
break ;
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
# ifndef USE_PROCFS
case ' D ' :
daemonized_tracer = 1 ;
break ;
# endif
2008-07-18 00:25:10 +00:00
case ' F ' :
2008-07-25 15:42:34 +00:00
optF = 1 ;
break ;
1999-02-19 00:21:36 +00:00
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 ;
2011-04-07 20:25:40 +00:00
case ' y ' :
show_fd_path = 1 ;
break ;
1999-02-19 00:21:36 +00:00
case ' v ' :
qualify ( " abbrev=none " ) ;
break ;
case ' V ' :
2003-02-20 02:56:29 +00:00
printf ( " %s -- version %s \n " , PACKAGE_NAME , VERSION ) ;
1999-02-19 00:21:36 +00:00
exit ( 0 ) ;
break ;
2002-11-06 13:17:21 +00:00
case ' z ' :
not_failing_only = 1 ;
break ;
1999-02-19 00:21:36 +00:00
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 ' :
2003-01-24 04:31:23 +00:00
if ( ( pid = atoi ( optarg ) ) < = 0 ) {
1999-02-19 00:21:36 +00:00
fprintf ( stderr , " %s: Invalid process id: %s \n " ,
progname , optarg ) ;
break ;
}
2011-05-27 14:36:01 +02:00
if ( pid = = strace_tracer_pid ) {
1999-10-17 00:57:34 +00:00
fprintf ( stderr , " %s: I'm sorry, I can't let you do that, Dave. \n " , progname ) ;
1999-02-19 00:21:36 +00:00
break ;
}
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
tcp = alloc_tcb ( pid , 0 ) ;
1999-02-19 00:21:36 +00:00
tcp - > flags | = TCB_ATTACHED ;
pflag_seen + + ;
break ;
2011-04-07 20:25:40 +00:00
case ' P ' :
tracing_paths = 1 ;
if ( pathtrace_select ( optarg ) ) {
2011-06-07 12:13:24 +02:00
fprintf ( stderr , " %s : failed to select path '%s' \n " , progname , optarg ) ;
2011-04-07 20:25:40 +00:00
exit ( 1 ) ;
}
break ;
1999-02-19 00:21:36 +00:00
case ' s ' :
max_strlen = atoi ( optarg ) ;
2005-05-09 07:45:47 +00:00
if ( max_strlen < 0 ) {
fprintf ( stderr ,
" %s: invalid -s argument: %s \n " ,
progname , optarg ) ;
exit ( 1 ) ;
}
1999-02-19 00:21:36 +00:00
break ;
case ' S ' :
set_sortby ( optarg ) ;
break ;
case ' u ' :
username = strdup ( optarg ) ;
break ;
2003-01-24 04:31:23 +00:00
case ' E ' :
if ( putenv ( optarg ) < 0 ) {
fprintf ( stderr , " %s: out of memory \n " ,
progname ) ;
exit ( 1 ) ;
}
break ;
1999-02-19 00:21:36 +00:00
default :
usage ( stderr , 1 ) ;
break ;
}
}
2006-04-25 07:39:40 +00:00
if ( ( optind = = argc ) = = ! pflag_seen )
2003-11-11 21:24:23 +00:00
usage ( stderr , 1 ) ;
2010-08-05 14:30:11 +08:00
if ( pflag_seen & & daemonized_tracer ) {
fprintf ( stderr ,
" %s: -D and -p are mutually exclusive options \n " ,
progname ) ;
exit ( 1 ) ;
}
2008-07-25 15:42:34 +00:00
if ( ! followfork )
followfork = optF ;
2006-04-25 07:48:03 +00:00
if ( followfork > 1 & & cflag ) {
fprintf ( stderr ,
2010-03-28 19:24:54 +00:00
" %s: (-c or -C) and -ff are mutually exclusive options \n " ,
2006-04-25 07:48:03 +00:00
progname ) ;
exit ( 1 ) ;
}
1999-02-19 00:21:36 +00:00
/* 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 " ,
2007-07-05 19:31:49 +00:00
progname , username ) ;
1999-02-19 00:21:36 +00:00
exit ( 1 ) ;
}
run_uid = pent - > pw_uid ;
run_gid = pent - > pw_gid ;
}
else {
run_uid = getuid ( ) ;
run_gid = getgid ( ) ;
}
2010-12-07 12:50:49 +00:00
# ifdef LINUX
if ( followfork ) {
2011-05-23 21:29:03 +02:00
if ( test_ptrace_setoptions_followfork ( ) < 0 ) {
2010-12-07 12:50:49 +00:00
fprintf ( stderr ,
" Test for options supported by PTRACE_SETOPTIONS "
" failed, giving up using this feature. \n " ) ;
2011-06-21 14:34:10 +02:00
ptrace_setoptions = 0 ;
2010-12-07 12:50:49 +00:00
}
if ( debug )
2011-06-21 14:34:10 +02:00
fprintf ( stderr , " ptrace_setoptions = %#x \n " ,
ptrace_setoptions ) ;
2010-12-07 12:50:49 +00:00
}
2011-05-23 21:29:03 +02:00
test_ptrace_setoptions_for_all ( ) ;
2010-12-07 12:50:49 +00:00
# endif
1999-02-19 00:21:36 +00:00
/* Check if they want to redirect the output. */
if ( outfname ) {
2003-11-07 02:26:54 +00:00
/* 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 .
*/
2011-06-22 13:03:56 +02:00
if ( followfork > 1 )
error_msg_and_die ( " Piping the output and -ff are mutually exclusive " ) ;
outf = strace_popen ( outfname + 1 ) ;
2001-08-03 11:43:35 +00:00
}
2006-12-13 21:45:31 +00:00
else if ( followfork < = 1 & &
( outf = strace_fopen ( outfname , " w " ) ) = = NULL )
2001-08-03 11:43:35 +00:00
exit ( 1 ) ;
1999-02-19 00:21:36 +00:00
}
2003-11-07 02:26:54 +00:00
if ( ! outfname | | outfname [ 0 ] = = ' | ' | | outfname [ 0 ] = = ' ! ' )
1999-02-19 00:21:36 +00:00
setvbuf ( outf , buf , _IOLBF , BUFSIZ ) ;
2003-11-07 02:26:54 +00:00
if ( outfname & & optind < argc ) {
1999-02-19 00:21:36 +00:00
interactive = 0 ;
qflag = 1 ;
2003-06-03 01:35:20 +00:00
}
2010-11-12 17:25:19 +08:00
2007-11-03 23:34:11 +00:00
/* 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 ] ) ;
1999-02-19 00:21:36 +00:00
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 ) ;
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
sa . sa_handler = reaper ;
sigaction ( SIGCHLD , & sa , NULL ) ;
2002-12-16 20:40:39 +00:00
# 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 ) ;
2000-09-01 21:03:06 +00:00
# endif /* USE_PROCFS */
1999-02-19 00:21:36 +00:00
Experimental support for -D option.
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.
* strace.c: Add global flag variable daemonized_tracer for -D option.
(startup_attach): If -D, fork and block parent in pause().
In this case we are already a child, we in fact created a grandchild.
After attaching to grandparent, grandchild SIGKILLs parent.
(startup_child): If -D, parent blocks in wait(), then
execs the program to strace. Normally (w/o -D), it is child
who execs the program.
(main): Detect -D option, call startup_attach() if it is given.
2008-12-30 20:51:30 +00:00
if ( pflag_seen | | daemonized_tracer )
2007-06-11 22:06:31 +00:00
startup_attach ( ) ;
1999-02-19 00:21:36 +00:00
if ( trace ( ) < 0 )
exit ( 1 ) ;
cleanup ( ) ;
2008-11-10 17:14:58 +00:00
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 ) ;
1999-02-19 00:21:36 +00:00
}
2011-06-22 12:45:25 +02:00
static void
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
expand_tcbtab ( void )
2004-06-04 01:50:45 +00:00
{
/* 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 . */
2011-06-22 12:41:57 +02:00
int i = tcbtabsize ;
struct tcb * newtcbs = calloc ( tcbtabsize , sizeof ( newtcbs [ 0 ] ) ) ;
struct tcb * * newtab = realloc ( tcbtab , tcbtabsize * 2 * sizeof ( tcbtab [ 0 ] ) ) ;
if ( newtab = = NULL | | newtcbs = = NULL )
error_msg_and_die ( " expand_tcbtab: out of memory " ) ;
2004-06-04 01:50:45 +00:00
tcbtabsize * = 2 ;
tcbtab = newtab ;
2011-06-22 12:41:57 +02:00
while ( i < tcbtabsize )
tcbtab [ i + + ] = newtcbs + + ;
2004-06-04 01:50:45 +00:00
}
1999-02-19 00:21:36 +00:00
struct tcb *
2006-12-13 21:45:31 +00:00
alloc_tcb ( int pid , int command_options_parsed )
1999-02-19 00:21:36 +00:00
{
int i ;
struct tcb * tcp ;
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
if ( nprocs = = tcbtabsize )
expand_tcbtab ( ) ;
2002-12-18 04:16:10 +00:00
for ( i = 0 ; i < tcbtabsize ; i + + ) {
tcp = tcbtab [ i ] ;
1999-02-19 00:21:36 +00:00
if ( ( tcp - > flags & TCB_INUSE ) = = 0 ) {
2011-06-22 12:41:57 +02:00
memset ( tcp , 0 , sizeof ( * tcp ) ) ;
1999-02-19 00:21:36 +00:00
tcp - > pid = pid ;
tcp - > flags = TCB_INUSE | TCB_STARTUP ;
tcp - > outf = outf ; /* Initialise to current out file */
tcp - > pfd = - 1 ;
nprocs + + ;
2006-12-13 21:45:31 +00:00
if ( command_options_parsed )
newoutf ( tcp ) ;
1999-02-19 00:21:36 +00:00
return tcp ;
}
}
2011-06-22 12:41:57 +02:00
error_msg_and_die ( " bug in alloc_tcb " ) ;
1999-02-19 00:21:36 +00:00
}
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
int
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
proc_open ( struct tcb * tcp , int attaching )
1999-02-19 00:21:36 +00:00
{
char proc [ 32 ] ;
long arg ;
2000-09-01 21:03:06 +00:00
# ifdef SVR4
2001-10-19 08:59:12 +00:00
int i ;
sysset_t syscalls ;
1999-02-19 00:21:36 +00:00
sigset_t signals ;
fltset_t faults ;
2000-09-01 21:03:06 +00:00
# endif
1999-02-19 00:21:36 +00:00
# ifndef HAVE_POLLABLE_PROCFS
static int last_pfd ;
# endif
1999-11-29 15:34:02 +00:00
# ifdef HAVE_MP_PROCFS
1999-08-29 23:15:07 +00:00
/* 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 ;
}
2011-06-22 13:11:23 +02:00
set_cloexec_flag ( tcp - > pfd ) ;
1999-08-29 23:15:07 +00:00
sprintf ( proc , " /proc/%d/status " , tcp - > pid ) ;
if ( ( tcp - > pfd_stat = open ( proc , O_RDONLY | O_EXCL ) ) < 0 ) {
perror ( " strace: open( \" /proc/... \" , ...) " ) ;
return - 1 ;
}
2011-06-22 13:11:23 +02:00
set_cloexec_flag ( tcp - > pfd_stat ) ;
1999-08-29 23:15:07 +00:00
sprintf ( proc , " /proc/%d/as " , tcp - > pid ) ;
if ( ( tcp - > pfd_as = open ( proc , O_RDONLY | O_EXCL ) ) < 0 ) {
perror ( " strace: open( \" /proc/... \" , ...) " ) ;
return - 1 ;
}
2011-06-22 13:11:23 +02:00
set_cloexec_flag ( tcp - > pfd_as ) ;
1999-08-29 23:15:07 +00:00
# else
1999-02-19 00:21:36 +00:00
/* Open the process pseudo-file in /proc. */
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
1999-02-19 00:21:36 +00:00
sprintf ( proc , " /proc/%d " , tcp - > pid ) ;
2010-07-09 11:49:27 +02:00
tcp - > pfd = open ( proc , O_RDWR | O_EXCL ) ;
2000-09-01 21:03:06 +00:00
# else /* FREEBSD */
sprintf ( proc , " /proc/%d/mem " , tcp - > pid ) ;
2010-07-09 11:49:27 +02:00
tcp - > pfd = open ( proc , O_RDWR ) ;
2000-09-01 21:03:06 +00:00
# endif /* FREEBSD */
2010-07-09 11:49:27 +02:00
if ( tcp - > pfd < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " strace: open( \" /proc/... \" , ...) " ) ;
return - 1 ;
}
2011-06-22 13:11:23 +02:00
set_cloexec_flag ( tcp - > pfd ) ;
1999-08-29 23:15:07 +00:00
# endif
2000-09-01 21:03:06 +00:00
# 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 */
1999-02-19 00:21:36 +00:00
rebuild_pollv ( ) ;
if ( ! attaching ) {
/*
* Wait for the child to pause . Because of a race
* condition we have to poll for the event .
*/
for ( ; ; ) {
2011-06-07 12:13:24 +02:00
if ( IOCTL_STATUS ( tcp ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " strace: PIOCSTATUS " ) ;
return - 1 ;
}
1999-08-29 23:15:07 +00:00
if ( tcp - > status . PR_FLAGS & PR_ASLEEP )
2000-09-01 21:03:06 +00:00
break ;
1999-02-19 00:21:36 +00:00
}
}
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
1999-02-19 00:21:36 +00:00
/* Stop the process so that we own the stop. */
2000-08-10 02:14:04 +00:00
if ( IOCTL ( tcp - > pfd , PIOCSTOP , ( char * ) NULL ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " strace: PIOCSTOP " ) ;
return - 1 ;
}
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
# ifdef PIOCSET
/* Set Run-on-Last-Close. */
arg = PR_RLC ;
1999-08-29 23:15:07 +00:00
if ( IOCTL ( tcp - > pfd , PIOCSET , & arg ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCSET PR_RLC " ) ;
return - 1 ;
}
/* Set or Reset Inherit-on-Fork. */
arg = PR_FORK ;
1999-08-29 23:15:07 +00:00
if ( IOCTL ( tcp - > pfd , followfork ? PIOCSET : PIOCRESET , & arg ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOC{SET,RESET} PR_FORK " ) ;
return - 1 ;
}
# else /* !PIOCSET */
2002-12-16 20:40:39 +00:00
# ifndef FREEBSD
1999-02-19 00:21:36 +00:00
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 ;
}
2000-09-01 21:03:06 +00:00
# else /* FREEBSD */
/* just unset the PF_LINGER flag for the Run-on-Last-Close. */
if ( ioctl ( tcp - > pfd , PIOCGFL , & arg ) < 0 ) {
perror ( " PIOCGFL " ) ;
2009-02-27 20:32:52 +00:00
return - 1 ;
2000-09-01 21:03:06 +00:00
}
arg & = ~ PF_LINGER ;
if ( ioctl ( tcp - > pfd , PIOCSFL , arg ) < 0 ) {
2009-02-27 20:32:52 +00:00
perror ( " PIOCSFL " ) ;
return - 1 ;
2000-09-01 21:03:06 +00:00
}
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
# endif /* !PIOCSET */
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
2001-10-19 08:59:12 +00:00
/* Enable all syscall entries we care about. */
premptyset ( & syscalls ) ;
for ( i = 1 ; i < MAX_QUALS ; + + i ) {
if ( i > ( sizeof syscalls ) * CHAR_BIT ) break ;
2011-06-07 12:13:24 +02:00
if ( qual_flags [ i ] & QUAL_TRACE ) praddset ( & syscalls , i ) ;
2001-10-19 08:59:12 +00:00
}
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_execve ) ;
2001-10-19 08:59:12 +00:00
if ( followfork ) {
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_fork ) ;
2001-10-19 08:59:12 +00:00
# ifdef SYS_forkall
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_forkall ) ;
2001-10-19 08:59:12 +00:00
# endif
2002-12-16 20:40:39 +00:00
# ifdef SYS_fork1
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_fork1 ) ;
2001-10-19 08:59:12 +00:00
# endif
# ifdef SYS_rfork1
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_rfork1 ) ;
2001-10-19 08:59:12 +00:00
# endif
# ifdef SYS_rforkall
2011-06-07 12:13:24 +02:00
praddset ( & syscalls , SYS_rforkall ) ;
2001-10-19 08:59:12 +00:00
# endif
}
if ( IOCTL ( tcp - > pfd , PIOCSENTRY , & syscalls ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCSENTRY " ) ;
return - 1 ;
}
2001-10-19 08:59:12 +00:00
/* Enable the syscall exits. */
if ( IOCTL ( tcp - > pfd , PIOCSEXIT , & syscalls ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOSEXIT " ) ;
return - 1 ;
}
2001-10-19 08:59:12 +00:00
/* Enable signals we care about. */
premptyset ( & signals ) ;
for ( i = 1 ; i < MAX_QUALS ; + + i ) {
if ( i > ( sizeof signals ) * CHAR_BIT ) break ;
2011-06-07 12:13:24 +02:00
if ( qual_flags [ i ] & QUAL_SIGNAL ) praddset ( & signals , i ) ;
2001-10-19 08:59:12 +00:00
}
1999-08-29 23:15:07 +00:00
if ( IOCTL ( tcp - > pfd , PIOCSTRACE , & signals ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCSTRACE " ) ;
return - 1 ;
}
2001-10-19 08:59:12 +00:00
/* Enable faults we care about */
premptyset ( & faults ) ;
for ( i = 1 ; i < MAX_QUALS ; + + i ) {
if ( i > ( sizeof faults ) * CHAR_BIT ) break ;
2011-06-07 12:13:24 +02:00
if ( qual_flags [ i ] & QUAL_FAULT ) praddset ( & faults , i ) ;
2001-10-19 08:59:12 +00:00
}
1999-08-29 23:15:07 +00:00
if ( IOCTL ( tcp - > pfd , PIOCSFAULT , & faults ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCSFAULT " ) ;
return - 1 ;
}
2000-09-01 21:03:06 +00:00
# else /* FREEBSD */
/* set events flags. */
2011-06-07 12:13:24 +02:00
arg = S_SIG | S_SCE | S_SCX ;
if ( ioctl ( tcp - > pfd , PIOCBIS , arg ) < 0 ) {
2000-09-01 21:03:06 +00:00
perror ( " PIOCBIS " ) ;
return - 1 ;
}
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
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 */
2002-12-16 20:40:39 +00:00
# ifdef PRSABORT
1999-02-19 00:21:36 +00:00
/* The child is in a pause(), abort it. */
1999-08-29 23:15:07 +00:00
arg = PRSABORT ;
2011-06-07 12:13:24 +02:00
if ( IOCTL ( tcp - > pfd , PIOCRUN , & arg ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCRUN " ) ;
return - 1 ;
}
2002-12-16 20:40:39 +00:00
# endif
2000-09-01 21:03:06 +00:00
# endif /* !MIPS*/
# ifdef FREEBSD
/* wake up the child if it received the SIGSTOP */
kill ( tcp - > pid , SIGCONT ) ;
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
for ( ; ; ) {
/* Wait for the child to do something. */
2011-06-07 12:13:24 +02:00
if ( IOCTL_WSTOP ( tcp ) < 0 ) {
1999-02-19 00:21:36 +00:00
perror ( " PIOCWSTOP " ) ;
return - 1 ;
}
1999-08-29 23:15:07 +00:00
if ( tcp - > status . PR_WHY = = PR_SYSENTRY ) {
2000-09-01 21:03:06 +00:00
tcp - > flags & = ~ TCB_INSYSCALL ;
get_scno ( tcp ) ;
2005-06-07 23:21:31 +00:00
if ( known_scno ( tcp ) = = SYS_execve )
1999-02-19 00:21:36 +00:00
break ;
}
/* Set it running: maybe execve will be next. */
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
1999-08-29 23:15:07 +00:00
arg = 0 ;
if ( IOCTL ( tcp - > pfd , PIOCRUN , & arg ) < 0 ) {
2000-09-01 21:03:06 +00:00
# else /* FREEBSD */
if ( IOCTL ( tcp - > pfd , PIOCRUN , 0 ) < 0 ) {
2002-12-16 20:40:39 +00:00
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
perror ( " PIOCRUN " ) ;
return - 1 ;
}
2000-09-01 21:03:06 +00:00
# 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 ) ;
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
}
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
1999-02-19 00:21:36 +00:00
}
2000-09-01 21:03:06 +00:00
# else /* FREEBSD */
} else {
2002-12-16 20:40:39 +00:00
if ( attaching < 2 ) {
2000-09-03 23:57:48 +00:00
/* 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 */
2002-12-16 20:40:39 +00:00
2000-09-03 23:57:48 +00:00
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 */
2000-09-01 21:03:06 +00:00
}
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
# 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 ;
}
2000-09-01 21:03:06 +00:00
# endif /* USE_PROCFS */
1999-02-19 00:21:36 +00:00
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
struct tcb *
2010-09-14 18:59:20 -07:00
pid2tcb ( int pid )
1999-02-19 00:21:36 +00:00
{
int i ;
2010-09-14 18:59:20 -07:00
if ( pid < = 0 )
return NULL ;
1999-02-19 00:21:36 +00:00
2002-12-18 04:16:10 +00:00
for ( i = 0 ; i < tcbtabsize ; i + + ) {
2010-09-14 18:59:20 -07:00
struct tcb * tcp = tcbtab [ i ] ;
if ( tcp - > pid = = pid & & ( tcp - > flags & TCB_INUSE ) )
1999-02-19 00:21:36 +00:00
return tcp ;
}
2010-09-14 18:59:20 -07:00
1999-02-19 00:21:36 +00:00
return NULL ;
}
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
2010-09-14 18:59:20 -07:00
static struct tcb *
first_used_tcb ( void )
{
int i ;
struct tcb * tcp ;
for ( i = 0 ; i < tcbtabsize ; i + + ) {
tcp = tcbtab [ i ] ;
if ( tcp - > flags & TCB_INUSE )
return tcp ;
}
return NULL ;
}
1999-02-19 00:21:36 +00:00
static struct tcb *
2011-05-30 14:00:14 +02:00
pfd2tcb ( int pfd )
1999-02-19 00:21:36 +00:00
{
int i ;
2003-01-10 19:55:28 +00:00
for ( i = 0 ; i < tcbtabsize ; i + + ) {
struct tcb * tcp = tcbtab [ i ] ;
1999-02-19 00:21:36 +00:00
if ( tcp - > pfd ! = pfd )
continue ;
if ( tcp - > flags & TCB_INUSE )
return tcp ;
}
return NULL ;
}
2000-09-01 21:03:06 +00:00
# endif /* USE_PROCFS */
1999-02-19 00:21:36 +00:00
void
2011-05-30 14:00:14 +02:00
droptcb ( struct tcb * tcp )
1999-02-19 00:21:36 +00:00
{
if ( tcp - > pid = = 0 )
return ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# 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
1999-02-19 00:21:36 +00:00
nprocs - - ;
tcp - > pid = 0 ;
2002-04-01 17:48:02 +00:00
2003-01-10 20:14:20 +00:00
if ( tcp - > parent ! = NULL ) {
# ifdef TCB_CLONE_THREAD
if ( tcp - > flags & TCB_CLONE_THREAD )
tcp - > parent - > nclone_threads - - ;
# endif
2007-11-13 08:12:12 +00:00
# ifdef LINUX
2011-06-21 16:06:28 +02:00
/* Update fields like NCLONE_DETACHED, only
for zombie group leader that has already reported
and been short - circuited at the top of this
2007-11-13 08:12:12 +00:00
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
2003-01-10 20:14:20 +00:00
tcp - > parent = NULL ;
}
tcp - > flags = 0 ;
1999-02-19 00:21:36 +00:00
if ( tcp - > pfd ! = - 1 ) {
close ( tcp - > pfd ) ;
tcp - > pfd = - 1 ;
2000-09-01 21:03:06 +00:00
# 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 ;
}
2002-12-16 20:40:39 +00:00
# endif /* !FREEBSD */
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
2003-01-10 20:14:20 +00:00
rebuild_pollv ( ) ; /* Note, flags needs to be cleared by now. */
1999-02-19 00:21:36 +00:00
# endif
}
2002-04-01 17:48:02 +00:00
2002-04-03 10:55:14 +00:00
if ( outfname & & followfork > 1 & & tcp - > outf )
1999-02-19 00:21:36 +00:00
fclose ( tcp - > outf ) ;
2002-04-01 17:48:02 +00:00
1999-02-19 00:21:36 +00:00
tcp - > outf = 0 ;
}
2007-07-05 18:43:16 +00:00
/* 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 . */
1999-02-19 00:21:36 +00:00
static int
2011-05-30 14:00:14 +02:00
detach ( struct tcb * tcp , int sig )
1999-02-19 00:21:36 +00:00
{
int error = 0 ;
2003-01-10 19:55:28 +00:00
# ifdef LINUX
2007-08-03 10:02:00 +00:00
int status , catch_sigstop ;
2005-08-03 11:23:46 +00:00
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 ;
2003-01-10 19:55:28 +00:00
# endif
1999-02-19 00:21:36 +00:00
if ( tcp - > flags & TCB_BPTSET )
2010-01-12 11:16:32 +01:00
clearbpt ( tcp ) ;
1999-02-19 00:21:36 +00:00
# ifdef LINUX
/*
* Linux wrongly insists the child be stopped
2002-12-16 20:42:50 +00:00
* before detaching . Arghh . We go through hoops
* to make a clean break of things .
1999-02-19 00:21:36 +00:00
*/
2002-12-16 20:42:50 +00:00
# if defined(SPARC)
# undef PTRACE_DETACH
# define PTRACE_DETACH PTRACE_SUNDETACH
# endif
2007-06-11 22:06:31 +00:00
/*
* 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 ) ;
1999-02-19 00:21:36 +00:00
if ( ( error = ptrace ( PTRACE_DETACH , tcp - > pid , ( char * ) 1 , sig ) ) = = 0 ) {
/* On a clear day, you can see forever. */
2002-12-16 20:42:50 +00:00
}
else if ( errno ! = ESRCH ) {
/* Shouldn't happen. */
perror ( " detach: ptrace(PTRACE_DETACH, ...) " ) ;
}
2007-06-02 00:07:33 +00:00
else if ( my_tgkill ( ( tcp - > flags & TCB_CLONE_THREAD ? tcp - > parent - > pid
: tcp - > pid ) ,
tcp - > pid , 0 ) < 0 ) {
2002-12-16 20:42:50 +00:00
if ( errno ! = ESRCH )
perror ( " detach: checking sanity " ) ;
}
2007-06-11 22:06:31 +00:00
else if ( ! catch_sigstop & & my_tgkill ( ( tcp - > flags & TCB_CLONE_THREAD
? tcp - > parent - > pid : tcp - > pid ) ,
tcp - > pid , SIGSTOP ) < 0 ) {
2002-12-16 20:42:50 +00:00
if ( errno ! = ESRCH )
perror ( " detach: stopping child " ) ;
}
2007-06-11 22:06:31 +00:00
else
catch_sigstop = 1 ;
2009-01-06 21:45:06 +00:00
if ( catch_sigstop ) {
1999-02-19 00:21:36 +00:00
for ( ; ; ) {
2002-12-17 10:48:05 +00:00
# ifdef __WALL
if ( wait4 ( tcp - > pid , & status , __WALL , NULL ) < 0 ) {
if ( errno = = ECHILD ) /* Already gone. */
break ;
if ( errno ! = EINVAL ) {
1999-02-19 00:21:36 +00:00
perror ( " detach: waiting " ) ;
2002-12-17 10:48:05 +00:00
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
1999-02-19 00:21:36 +00:00
}
2002-12-17 10:48:05 +00:00
# endif
1999-02-19 00:21:36 +00:00
if ( ! WIFSTOPPED ( status ) ) {
/* Au revoir, mon ami. */
break ;
}
if ( WSTOPSIG ( status ) = = SIGSTOP ) {
2008-12-17 19:21:59 +00:00
ptrace_restart ( PTRACE_DETACH , tcp , sig ) ;
1999-02-19 00:21:36 +00:00
break ;
}
2008-12-17 19:21:59 +00:00
error = ptrace_restart ( PTRACE_CONT , tcp ,
2011-05-27 14:36:01 +02:00
WSTOPSIG ( status ) = = syscall_trap_sig ? 0
2008-12-17 19:21:59 +00:00
: WSTOPSIG ( status ) ) ;
if ( error < 0 )
1999-02-19 00:21:36 +00:00
break ;
}
2009-01-06 21:45:06 +00:00
}
2002-12-16 20:42:50 +00:00
# endif /* LINUX */
1999-02-19 00:21:36 +00:00
# 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 ;
2008-12-17 19:21:59 +00:00
error = ptrace_restart ( PTRACE_DETACH , tcp , sig ) ;
1999-02-19 00:21:36 +00:00
# endif /* SUNOS4 */
if ( ! qflag )
fprintf ( stderr , " Process %u detached \n " , tcp - > pid ) ;
droptcb ( tcp ) ;
2005-08-03 11:23:46 +00:00
# ifdef LINUX
2007-07-05 18:43:16 +00:00
if ( zombie ! = NULL ) {
2011-06-07 12:13:24 +02:00
/* TCP no longer exists therefore you must not detach() it. */
2007-07-05 18:43:16 +00:00
droptcb ( zombie ) ;
}
2005-08-03 11:23:46 +00:00
# endif
1999-02-19 00:21:36 +00:00
return error ;
}
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
Remove dead code
* defs.h (tv_tv): Remove.
* net.c (sys_xsetsockaddr): Remove commented out dead code.
* process.c (setarg, sys_execv, sys_execve, struct_user_offsets):
Likewise.
* signal.c (sys_sigsuspend): Likewise.
* strace.c (reaper, trace): Likewise.
* stream.c (internal_stream_ioctl): Likewise.
* syscall.c (trace_syscall): Likewise.
* term.c (term_ioctl): Likewise.
* util.c (tv_tv, umoven, uload, getpc, fixvfork, setbpt, clearbpt):
Likewise.
2009-12-31 22:50:49 +00:00
static void reaper ( int sig )
1999-02-19 00:21:36 +00:00
{
int pid ;
int status ;
while ( ( pid = waitpid ( - 1 , & status , WNOHANG ) ) > 0 ) {
}
}
2000-09-01 21:03:06 +00:00
# endif /* USE_PROCFS */
1999-02-19 00:21:36 +00:00
static void
2011-05-30 14:00:14 +02:00
cleanup ( void )
1999-02-19 00:21:36 +00:00
{
int i ;
struct tcb * tcp ;
2002-12-18 04:16:10 +00:00
for ( i = 0 ; i < tcbtabsize ; i + + ) {
tcp = tcbtab [ i ] ;
1999-02-19 00:21:36 +00:00
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 ) ) {
2009-01-06 21:45:06 +00:00
tprintf ( " <unfinished ...> " ) ;
printtrailer ( ) ;
1999-02-19 00:21:36 +00:00
}
if ( tcp - > flags & TCB_ATTACHED )
detach ( tcp , 0 ) ;
else {
kill ( tcp - > pid , SIGCONT ) ;
kill ( tcp - > pid , SIGTERM ) ;
}
}
if ( cflag )
call_summary ( outf ) ;
}
static void
2011-05-30 14:00:14 +02:00
interrupt ( int sig )
1999-02-19 00:21:36 +00:00
{
interrupted = 1 ;
}
# ifndef HAVE_STRERROR
2002-12-30 00:51:30 +00:00
# if !HAVE_DECL_SYS_ERRLIST
1999-02-19 00:21:36 +00:00
extern int sys_nerr ;
extern char * sys_errlist [ ] ;
2002-12-30 00:51:30 +00:00
# endif /* HAVE_DECL_SYS_ERRLIST */
1999-02-19 00:21:36 +00:00
const char *
2011-05-30 14:00:14 +02:00
strerror ( int err_no )
1999-02-19 00:21:36 +00:00
{
static char buf [ 64 ] ;
2011-05-25 15:33:26 +02:00
if ( err_no < 1 | | err_no > = sys_nerr ) {
sprintf ( buf , " Unknown error %d " , err_no ) ;
1999-02-19 00:21:36 +00:00
return buf ;
}
2011-05-25 15:33:26 +02:00
return sys_errlist [ err_no ] ;
1999-02-19 00:21:36 +00:00
}
# endif /* HAVE_STERRROR */
# ifndef HAVE_STRSIGNAL
2003-01-14 07:53:33 +00:00
# if defined HAVE_SYS_SIGLIST && !defined HAVE_DECL_SYS_SIGLIST
2002-12-30 00:51:30 +00:00
extern char * sys_siglist [ ] ;
1999-02-19 00:21:36 +00:00
# endif
2003-01-14 07:53:33 +00:00
# if defined HAVE_SYS__SIGLIST && !defined HAVE_DECL__SYS_SIGLIST
extern char * _sys_siglist [ ] ;
# endif
1999-02-19 00:21:36 +00:00
const char *
2011-05-30 14:00:14 +02:00
strsignal ( int sig )
1999-02-19 00:21:36 +00:00
{
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 */
2000-09-01 21:03:06 +00:00
# ifdef USE_PROCFS
1999-02-19 00:21:36 +00:00
static void
2011-05-30 14:00:14 +02:00
rebuild_pollv ( void )
1999-02-19 00:21:36 +00:00
{
int i , j ;
2002-12-18 04:16:10 +00:00
if ( pollv ! = NULL )
2011-06-07 12:13:24 +02:00
free ( pollv ) ;
2003-01-10 20:05:56 +00:00
pollv = ( struct pollfd * ) malloc ( nprocs * sizeof pollv [ 0 ] ) ;
2002-12-18 04:16:10 +00:00
if ( pollv = = NULL ) {
2005-06-01 18:55:42 +00:00
fprintf ( stderr , " %s: out of memory \n " , progname ) ;
2002-12-18 04:16:10 +00:00
exit ( 1 ) ;
}
2003-01-10 19:55:28 +00:00
for ( i = j = 0 ; i < tcbtabsize ; i + + ) {
struct tcb * tcp = tcbtab [ i ] ;
1999-02-19 00:21:36 +00:00
if ( ! ( tcp - > flags & TCB_INUSE ) )
continue ;
pollv [ j ] . fd = tcp - > pfd ;
1999-08-29 23:15:07 +00:00
pollv [ j ] . events = POLLWANT ;
1999-02-19 00:21:36 +00:00
j + + ;
}
if ( j ! = nprocs ) {
fprintf ( stderr , " strace: proc miscount \n " ) ;
exit ( 1 ) ;
}
}
# ifndef HAVE_POLLABLE_PROCFS
static void
2011-05-30 14:00:14 +02:00
proc_poll_open ( void )
1999-02-19 00:21:36 +00:00
{
int i ;
if ( pipe ( proc_poll_pipe ) < 0 ) {
perror ( " pipe " ) ;
exit ( 1 ) ;
}
for ( i = 0 ; i < 2 ; i + + ) {
2011-06-22 13:11:23 +02:00
set_cloexec_flag ( proc_poll_pipe [ i ] ) ;
1999-02-19 00:21:36 +00:00
}
}
static int
2011-05-30 14:00:14 +02:00
proc_poll ( struct pollfd * pollv , int nfds , int timeout )
1999-02-19 00:21:36 +00:00
{
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
2011-05-30 14:00:14 +02:00
wakeup_handler ( int sig )
1999-02-19 00:21:36 +00:00
{
}
static void
2011-05-30 14:00:14 +02:00
proc_poller ( int pfd )
1999-02-19 00:21:36 +00:00
{
struct proc_pollfd pollinfo ;
struct sigaction sa ;
sigset_t blocked_set , empty_set ;
int i ;
int n ;
struct rlimit rl ;
2000-09-01 21:03:06 +00:00
# ifdef FREEBSD
struct procfs_status pfs ;
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
switch ( fork ( ) ) {
case - 1 :
perror ( " fork " ) ;
2008-11-10 17:14:58 +00:00
_exit ( 1 ) ;
1999-02-19 00:21:36 +00:00
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, ...) " ) ;
2008-11-10 17:14:58 +00:00
_exit ( 1 ) ;
1999-02-19 00:21:36 +00:00
}
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 ( ; ; ) {
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
2009-02-27 20:32:52 +00:00
if ( ioctl ( pfd , PIOCWSTOP , NULL ) < 0 )
# else
if ( ioctl ( pfd , PIOCWSTOP , & pfs ) < 0 )
# endif
1999-08-29 23:15:07 +00:00
{
1999-02-19 00:21:36 +00:00
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 ) ;
}
1999-08-29 23:15:07 +00:00
pollinfo . revents = POLLWANT ;
1999-02-19 00:21:36 +00:00
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 & &
1999-08-29 23:15:07 +00:00
last < nprocs & & ( pollv [ last ] . revents & POLLWANT ) ) {
1999-02-19 00:21:36 +00:00
/*
* 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 ;
}
1999-08-29 23:15:07 +00:00
if ( pollv [ j ] . revents & POLLWANT ) {
1999-02-19 00:21:36 +00:00
last = j ;
return pollv [ j ] . fd ;
}
}
fprintf ( stderr , " strace: nothing ready \n " ) ;
exit ( 1 ) ;
}
static int
2011-05-30 14:00:14 +02:00
trace ( void )
1999-02-19 00:21:36 +00:00
{
1999-11-26 13:11:29 +00:00
# ifdef POLL_HACK
2002-05-21 11:24:18 +00:00
struct tcb * in_syscall = NULL ;
1999-11-26 13:11:29 +00:00
# endif
1999-02-19 00:21:36 +00:00
struct tcb * tcp ;
int pfd ;
int what ;
int ioctl_result = 0 , ioctl_errno = 0 ;
1999-08-29 23:15:07 +00:00
long arg ;
1999-02-19 00:21:36 +00:00
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
2010-09-14 18:59:20 -07:00
tcp = first_used_tcb ( ) ;
1999-02-19 00:21:36 +00:00
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
1999-11-26 13:11:29 +00:00
# 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 */
2002-12-16 20:40:39 +00:00
1999-11-26 13:11:29 +00:00
if ( in_syscall ) {
struct pollfd pv ;
tcp = in_syscall ;
in_syscall = NULL ;
pv . fd = tcp - > pfd ;
pv . events = POLLWANT ;
2011-06-07 12:13:24 +02:00
if ( ( what = poll ( & pv , 1 , 1 ) ) < 0 ) {
1999-11-26 13:11:29 +00:00
if ( interrupted )
return 0 ;
continue ;
}
else if ( what = = 1 & & pv . revents & POLLWANT ) {
goto FOUND ;
}
}
# endif
1999-02-19 00:21:36 +00:00
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 ) ;
}
2002-05-23 11:02:22 +00:00
# ifdef POLL_HACK
1999-11-26 13:11:29 +00:00
FOUND :
2002-05-23 11:02:22 +00:00
# endif
1999-02-19 00:21:36 +00:00
/* Get the status of the process. */
if ( ! interrupted ) {
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
2011-06-07 12:13:24 +02:00
ioctl_result = IOCTL_WSTOP ( tcp ) ;
2000-09-01 21:03:06 +00:00
# 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 )
2011-06-07 12:13:24 +02:00
ioctl_result = IOCTL_STATUS ( tcp ) ;
2000-09-01 21:03:06 +00:00
else
2011-06-07 12:13:24 +02:00
ioctl_result = IOCTL_WSTOP ( tcp ) ;
2002-12-16 20:40:39 +00:00
# endif /* FREEBSD */
1999-02-19 00:21:36 +00:00
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 ;
2000-09-01 21:03:06 +00:00
# ifdef FREEBSD
case ENOTTY :
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
case ENOENT :
droptcb ( tcp ) ;
continue ;
default :
perror ( " PIOCWSTOP " ) ;
exit ( 1 ) ;
}
}
2000-09-03 23:57:48 +00:00
# ifdef FREEBSD
if ( ( tcp - > flags & TCB_STARTUP ) & & ( tcp - > status . PR_WHY = = PR_SYSEXIT ) ) {
/* discard first event for a syscall we never entered */
2011-06-07 12:13:24 +02:00
IOCTL ( tcp - > pfd , PIOCRUN , 0 ) ;
2000-09-03 23:57:48 +00:00
continue ;
}
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
/* clear the just started flag */
tcp - > flags & = ~ TCB_STARTUP ;
/* set current output file */
outf = tcp - > outf ;
2009-10-27 16:27:13 +01:00
curcol = tcp - > curcol ;
1999-02-19 00:21:36 +00:00
if ( cflag ) {
struct timeval stime ;
2000-09-01 21:03:06 +00:00
# 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 ;
2002-12-16 20:40:39 +00:00
# else /* !FREEBSD */
1999-02-19 00:21:36 +00:00
stime . tv_sec = tcp - > status . pr_stime . tv_sec ;
stime . tv_usec = tcp - > status . pr_stime . tv_nsec / 1000 ;
2000-09-01 21:03:06 +00:00
# endif /* !FREEBSD */
1999-02-19 00:21:36 +00:00
tv_sub ( & tcp - > dtime , & stime , & tcp - > stime ) ;
tcp - > stime = stime ;
}
1999-08-29 23:15:07 +00:00
what = tcp - > status . PR_WHAT ;
switch ( tcp - > status . PR_WHY ) {
2000-09-01 21:03:06 +00:00
# ifndef FREEBSD
1999-02-19 00:21:36 +00:00
case PR_REQUESTED :
1999-08-29 23:15:07 +00:00
if ( tcp - > status . PR_FLAGS & PR_ASLEEP ) {
tcp - > status . PR_WHY = PR_SYSENTRY ;
1999-02-19 00:21:36 +00:00
if ( trace_syscall ( tcp ) < 0 ) {
fprintf ( stderr , " syscall trouble \n " ) ;
exit ( 1 ) ;
}
}
break ;
2000-09-01 21:03:06 +00:00
# endif /* !FREEBSD */
1999-02-19 00:21:36 +00:00
case PR_SYSENTRY :
1999-11-26 13:11:29 +00:00
# ifdef POLL_HACK
in_syscall = tcp ;
# endif
1999-02-19 00:21:36 +00:00
case PR_SYSEXIT :
if ( trace_syscall ( tcp ) < 0 ) {
fprintf ( stderr , " syscall trouble \n " ) ;
exit ( 1 ) ;
}
break ;
case PR_SIGNALLED :
2010-03-28 19:24:54 +00:00
if ( cflag ! = CFLAG_ONLY_STATS
& & ( qual_flags [ what ] & QUAL_SIGNAL ) ) {
1999-02-19 00:21:36 +00:00
printleader ( tcp ) ;
tprintf ( " --- %s (%s) --- " ,
1999-03-29 23:23:13 +00:00
signame ( what ) , strsignal ( what ) ) ;
2009-01-06 21:45:06 +00:00
printtrailer ( ) ;
2001-10-18 15:13:53 +00:00
# ifdef PR_INFO
if ( tcp - > status . PR_INFO . si_signo = = what ) {
printleader ( tcp ) ;
tprintf ( " siginfo= " ) ;
printsiginfo ( & tcp - > status . PR_INFO , 1 ) ;
2009-01-06 21:45:06 +00:00
printtrailer ( ) ;
2001-10-18 15:13:53 +00:00
}
# endif
1999-02-19 00:21:36 +00:00
}
break ;
case PR_FAULTED :
2010-03-28 19:24:54 +00:00
if ( cflag ! = CFLAGS_ONLY_STATS
& & ( qual_flags [ what ] & QUAL_FAULT ) ) {
1999-02-19 00:21:36 +00:00
printleader ( tcp ) ;
tprintf ( " === FAULT %d === " , what ) ;
2009-01-06 21:45:06 +00:00
printtrailer ( ) ;
1999-02-19 00:21:36 +00:00
}
break ;
2000-09-01 21:03:06 +00:00
# ifdef FREEBSD
case 0 : /* handle case we polled for nothing */
2009-02-27 20:32:52 +00:00
continue ;
2002-12-16 20:40:39 +00:00
# endif
1999-02-19 00:21:36 +00:00
default :
1999-08-29 23:15:07 +00:00
fprintf ( stderr , " odd stop %d \n " , tcp - > status . PR_WHY ) ;
1999-02-19 00:21:36 +00:00
exit ( 1 ) ;
break ;
}
2009-10-27 16:27:13 +01:00
/* Remember current print column before continuing. */
tcp - > curcol = curcol ;
1999-08-29 23:15:07 +00:00
arg = 0 ;
2002-12-16 20:40:39 +00:00
# ifndef FREEBSD
2011-06-07 12:13:24 +02:00
if ( IOCTL ( tcp - > pfd , PIOCRUN , & arg ) < 0 )
2002-12-16 20:40:39 +00:00
# else
2011-06-07 12:13:24 +02:00
if ( IOCTL ( tcp - > pfd , PIOCRUN , 0 ) < 0 )
2002-12-16 20:40:39 +00:00
# endif
2010-07-09 11:49:27 +02:00
{
1999-02-19 00:21:36 +00:00
perror ( " PIOCRUN " ) ;
exit ( 1 ) ;
}
}
return 0 ;
}
2000-09-01 21:03:06 +00:00
# else /* !USE_PROCFS */
1999-02-19 00:21:36 +00:00
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# 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 . */
2009-01-29 20:38:20 +00:00
struct tcb * leader = NULL ;
if ( tcp - > flags & TCB_CLONE_THREAD )
leader = tcp - > parent ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
if ( sig < 0 ) {
2009-01-29 20:38:20 +00:00
if ( leader ! = NULL & & leader ! = tcp
& & ! ( leader - > flags & TCB_GROUP_EXITING )
& & ! ( tcp - > flags & TCB_STARTUP )
) {
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
fprintf ( stderr ,
" PANIC: handle_group_exit: %d leader %d \n " ,
tcp - > pid , leader ? leader - > pid : - 1 ) ;
2009-01-29 20:38:20 +00:00
}
/* TCP no longer exists therefore you must not detach() it. */
2007-07-05 18:43:16 +00:00
droptcb ( tcp ) ; /* Already died. */
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
}
else {
2005-08-03 11:23:46 +00:00
/* Mark that we are taking the process down. */
tcp - > flags | = TCB_EXITING | TCB_GROUP_EXITING ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
if ( tcp - > flags & TCB_ATTACHED ) {
2007-07-11 08:35:11 +00:00
detach ( tcp , sig ) ;
2009-02-27 20:32:52 +00:00
if ( leader ! = NULL & & leader ! = tcp )
2007-08-03 10:02:00 +00:00
leader - > flags | = TCB_GROUP_EXITING ;
2008-12-17 19:21:59 +00:00
} else {
if ( ptrace_restart ( PTRACE_CONT , tcp , sig ) < 0 ) {
cleanup ( ) ;
return - 1 ;
}
if ( leader ! = NULL ) {
2004-10-20 01:00:27 +00:00
leader - > flags | = TCB_GROUP_EXITING ;
2008-12-17 19:21:59 +00:00
if ( leader ! = tcp )
droptcb ( tcp ) ;
}
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
/* The leader will report to us as parent now,
and then we ' ll get to the SIG = = - 1 case . */
return 0 ;
}
}
return 0 ;
}
# endif
2010-11-12 17:26:08 +08:00
# ifdef LINUX
static int
handle_ptrace_event ( int status , struct tcb * tcp )
{
if ( status > > 16 = = PTRACE_EVENT_VFORK | |
status > > 16 = = PTRACE_EVENT_CLONE | |
status > > 16 = = PTRACE_EVENT_FORK ) {
2011-02-19 21:33:50 +00:00
long childpid ;
2010-11-12 17:26:08 +08:00
if ( do_ptrace ( PTRACE_GETEVENTMSG , tcp , NULL , & childpid ) < 0 ) {
if ( errno ! = ESRCH ) {
fprintf ( stderr , " \
% s : handle_ptrace_event : ptrace cannot get new child ' s pid \ n " ,
progname ) ;
cleanup ( ) ;
exit ( 1 ) ;
}
return - 1 ;
}
return handle_new_child ( tcp , childpid , 0 ) ;
}
2011-05-23 21:29:03 +02:00
if ( status > > 16 = = PTRACE_EVENT_EXEC ) {
if ( debug )
fprintf ( stderr , " PTRACE_EVENT_EXEC on pid %d (ignored) \n " , tcp - > pid ) ;
return 0 ;
}
2011-05-27 14:36:01 +02:00
/* Some PTRACE_EVENT_foo we didn't ask for?! */
error_msg ( " Unexpected status %x on pid %d " , status , tcp - > pid ) ;
2010-11-12 17:26:08 +08:00
return 1 ;
}
# endif
2009-06-02 16:49:22 -07:00
static int
trace ( )
1999-02-19 00:21:36 +00:00
{
int pid ;
int wait_errno ;
int status ;
struct tcb * tcp ;
# ifdef LINUX
struct rusage ru ;
2001-03-28 14:40:14 +00:00
# ifdef __WALL
2009-06-02 16:49:22 -07:00
static int wait4_options = __WALL ;
2001-03-28 14:40:14 +00:00
# endif
1999-02-19 00:21:36 +00:00
# endif /* LINUX */
2009-06-02 16:49:22 -07:00
while ( nprocs ! = 0 ) {
2009-03-17 14:29:59 +00:00
if ( interrupted )
2009-06-02 16:49:22 -07:00
return 0 ;
if ( interactive )
sigprocmask ( SIG_SETMASK , & empty_set , NULL ) ;
1999-02-19 00:21:36 +00:00
# ifdef LINUX
2001-03-28 14:40:14 +00:00
# ifdef __WALL
2009-06-02 16:49:22 -07:00
pid = wait4 ( - 1 , & status , wait4_options , cflag ? & ru : NULL ) ;
2002-12-17 04:50:47 +00:00
if ( pid < 0 & & ( wait4_options & __WALL ) & & errno = = EINVAL ) {
2001-03-28 14:40:14 +00:00
/* this kernel does not support __WALL */
wait4_options & = ~ __WALL ;
errno = 0 ;
2009-06-02 16:49:22 -07:00
pid = wait4 ( - 1 , & status , wait4_options ,
cflag ? & ru : NULL ) ;
2001-03-28 14:40:14 +00:00
}
2002-12-17 04:50:47 +00:00
if ( pid < 0 & & ! ( wait4_options & __WALL ) & & errno = = ECHILD ) {
2001-03-28 14:40:14 +00:00
/* most likely a "cloned" process */
2009-06-02 16:49:22 -07:00
pid = wait4 ( - 1 , & status , __WCLONE ,
cflag ? & ru : NULL ) ;
if ( pid = = - 1 ) {
fprintf ( stderr , " strace: clone wait4 "
2001-03-28 14:40:14 +00:00
" failed: %s \n " , strerror ( errno ) ) ;
}
}
2009-06-02 16:49:22 -07:00
# else
pid = wait4 ( - 1 , & status , 0 , cflag ? & ru : NULL ) ;
# endif /* __WALL */
1999-02-19 00:21:36 +00:00
# endif /* LINUX */
# ifdef SUNOS4
pid = wait ( & status ) ;
# endif /* SUNOS4 */
wait_errno = errno ;
2009-06-02 16:49:22 -07:00
if ( interactive )
sigprocmask ( SIG_BLOCK , & blocked_set , NULL ) ;
1999-02-19 00:21:36 +00:00
if ( pid = = - 1 ) {
2009-06-02 16:49:22 -07:00
switch ( wait_errno ) {
case EINTR :
1999-02-19 00:21:36 +00:00
continue ;
2009-06-02 16:49:22 -07:00
case ECHILD :
1999-02-19 00:21:36 +00:00
/*
* We would like to verify this case
* but sometimes a race in Solbourne ' s
* version of SunOS sometimes reports
* ECHILD before sending us SIGCHILD .
*/
2009-06-02 16:49:22 -07:00
return 0 ;
default :
errno = wait_errno ;
perror ( " strace: wait " ) ;
return - 1 ;
1999-02-19 00:21:36 +00:00
}
}
2006-12-13 21:45:31 +00:00
if ( pid = = popen_pid ) {
if ( WIFEXITED ( status ) | | WIFSIGNALED ( status ) )
2011-06-22 13:03:56 +02:00
popen_pid = 0 ;
2006-12-13 21:45:31 +00:00
continue ;
}
1999-02-19 00:21:36 +00:00
if ( debug )
fprintf ( stderr , " [wait(%#x) = %u] \n " , status , pid ) ;
/* Look up `pid' in our table. */
if ( ( tcp = pid2tcb ( pid ) ) = = NULL ) {
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# ifdef LINUX
2008-07-18 00:25:10 +00:00
if ( followfork ) {
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
/* 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 . */
Two cleanups: tcb table expansion failure is not really a survivable
event, we do not have any viable way to continue. No wonder most
places where that is detected have FIXMEs.
It's way simpler to treat as fatal failure, and handle it inside
tcb table expansion finctions.
Second cleanup: tidy up haphazard locations of a few externs.
* defs.h: Change return type of expand_tcbtab() to void.
Declare change_syscall().
* process.c: Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(fork_tcb): Change return type to void - it can't fail now.
* strace.c: Move extern declarations out of function bodies.
Change all callsites of alloctcb(), alloc_tcb() and
fork_tcb(), removing now-redundant error checks.
(expand_tcbtab): Change return type to void - it can't fail now.
On failure to expand, print a message, clean up, and exit.
(alloc_tcb): On failure to expand, print a message, clean up, and exit.
* util.c (setbpt): Remove extern declaration from function body.
2009-01-17 01:52:54 +00:00
tcp = alloctcb ( pid ) ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
tcp - > flags | = TCB_ATTACHED | TCB_SUSPENDED ;
if ( ! qflag )
fprintf ( stderr , " \
Process % d attached ( waiting for parent ) \ n " ,
pid ) ;
2000-02-03 21:58:30 +00:00
}
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
else
/* This can happen if a clone call used
CLONE_PTRACE itself . */
2009-06-02 16:49:22 -07:00
# endif
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
{
fprintf ( stderr , " unknown pid: %u \n " , pid ) ;
if ( WIFSTOPPED ( status ) )
ptrace ( PTRACE_CONT , pid , ( char * ) 1 , 0 ) ;
exit ( 1 ) ;
}
1999-02-19 00:21:36 +00:00
}
2009-06-02 16:49:22 -07:00
/* set current output file */
outf = tcp - > outf ;
2009-10-27 16:27:13 +01:00
curcol = tcp - > curcol ;
2009-02-10 16:03:20 +00:00
if ( cflag ) {
2009-06-02 16:49:22 -07:00
# ifdef LINUX
1999-02-19 00:21:36 +00:00
tv_sub ( & tcp - > dtime , & ru . ru_stime , & tcp - > stime ) ;
tcp - > stime = ru . ru_stime ;
2009-06-02 16:49:22 -07:00
# endif /* !LINUX */
1999-02-19 00:21:36 +00:00
}
2009-06-02 16:49:22 -07:00
2009-01-17 00:21:31 +00:00
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 .
*/
continue ;
}
1999-02-19 00:21:36 +00:00
if ( WIFSIGNALED ( status ) ) {
2008-11-10 17:14:58 +00:00
if ( pid = = strace_child )
exit_code = 0x100 | WTERMSIG ( status ) ;
2010-03-28 19:24:54 +00:00
if ( cflag ! = CFLAG_ONLY_STATS
1999-02-19 00:21:36 +00:00
& & ( qual_flags [ WTERMSIG ( status ) ] & QUAL_SIGNAL ) ) {
printleader ( tcp ) ;
2004-01-13 09:59:45 +00:00
tprintf ( " +++ killed by %s %s+++ " ,
signame ( WTERMSIG ( status ) ) ,
# ifdef WCOREDUMP
WCOREDUMP ( status ) ? " (core dumped) " :
# endif
" " ) ;
2009-01-06 21:45:06 +00:00
printtrailer ( ) ;
1999-02-19 00:21:36 +00:00
}
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# ifdef TCB_GROUP_EXITING
handle_group_exit ( tcp , - 1 ) ;
# else
1999-02-19 00:21:36 +00:00
droptcb ( tcp ) ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# endif
1999-02-19 00:21:36 +00:00
continue ;
}
if ( WIFEXITED ( status ) ) {
2008-11-10 17:14:58 +00:00
if ( pid = = strace_child )
exit_code = WEXITSTATUS ( status ) ;
1999-02-19 00:21:36 +00:00
if ( debug )
2009-01-29 20:38:20 +00:00
fprintf ( stderr , " pid %u exited with %d \n " , pid , WEXITSTATUS ( status ) ) ;
if ( ( tcp - > flags & ( TCB_ATTACHED | TCB_STARTUP ) ) = = TCB_ATTACHED
2004-10-20 01:00:27 +00:00
# ifdef TCB_GROUP_EXITING
2009-01-29 20:38:20 +00:00
& & ! ( tcp - > parent & & ( tcp - > parent - > flags & TCB_GROUP_EXITING ) )
2007-08-03 10:02:00 +00:00
& & ! ( tcp - > flags & TCB_GROUP_EXITING )
2004-10-20 01:00:27 +00:00
# endif
2009-01-29 20:38:20 +00:00
) {
1999-02-19 00:21:36 +00:00
fprintf ( stderr ,
2009-01-29 20:38:20 +00:00
" PANIC: attached pid %u exited with %d \n " ,
pid , WEXITSTATUS ( status ) ) ;
}
2003-06-10 03:05:53 +00:00
if ( tcp = = tcp_last ) {
2009-01-29 20:38:20 +00:00
if ( ( tcp - > flags & ( TCB_INSYSCALL | TCB_REPRINT ) ) = = TCB_INSYSCALL )
2003-06-10 03:05:53 +00:00
tprintf ( " <unfinished ... exit status %d> \n " ,
WEXITSTATUS ( status ) ) ;
tcp_last = NULL ;
}
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# ifdef TCB_GROUP_EXITING
handle_group_exit ( tcp , - 1 ) ;
# else
1999-02-19 00:21:36 +00:00
droptcb ( tcp ) ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# endif
1999-02-19 00:21:36 +00:00
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 " ,
1999-03-29 23:23:13 +00:00
pid , signame ( WSTOPSIG ( status ) ) ) ;
1999-02-19 00:21:36 +00:00
2011-05-23 21:29:03 +02:00
if ( status > > 16 ) {
2010-11-12 17:26:08 +08:00
if ( handle_ptrace_event ( status , tcp ) ! = 1 )
goto tracing ;
}
2007-06-11 22:06:31 +00:00
/*
* Interestingly , the process may stop
* with STOPSIG equal to some other signal
2009-06-02 16:49:22 -07:00
* than SIGSTOP if we happend to attach
2007-06-11 22:06:31 +00:00
* just before the process takes a signal .
2009-10-07 20:41:29 -04:00
* A no - MMU vforked child won ' t send up a signal ,
* so skip the first ( lost ) execve notification .
2007-06-11 22:06:31 +00:00
*/
2009-10-07 20:41:29 -04:00
if ( ( tcp - > flags & TCB_STARTUP ) & &
( WSTOPSIG ( status ) = = SIGSTOP | | strace_vforked ) ) {
1999-02-19 00:21:36 +00:00
/*
* 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 ;
2007-06-11 22:06:31 +00:00
if ( tcp - > flags & TCB_BPTSET ) {
1999-02-19 00:21:36 +00:00
/*
2007-06-11 22:06:31 +00:00
* One example is a breakpoint inherited from
* parent through fork ( ) .
1999-02-19 00:21:36 +00:00
*/
if ( clearbpt ( tcp ) < 0 ) /* Pretty fatal */ {
droptcb ( tcp ) ;
cleanup ( ) ;
return - 1 ;
}
}
2010-11-12 17:26:08 +08:00
# ifdef LINUX
2011-06-21 14:34:10 +02:00
/* If options were not set for this tracee yet */
if ( tcp - > parent = = NULL ) {
if ( ptrace_setoptions ) {
if ( debug )
fprintf ( stderr , " setting opts %x on pid %d \n " , ptrace_setoptions , tcp - > pid ) ;
if ( ptrace ( PTRACE_SETOPTIONS , tcp - > pid , NULL , ptrace_setoptions ) < 0 ) {
if ( errno ! = ESRCH ) {
/* Should never happen, really */
perror_msg_and_die ( " PTRACE_SETOPTIONS " ) ;
}
2011-05-23 21:29:03 +02:00
}
}
}
2010-11-12 17:26:08 +08:00
# endif
1999-02-19 00:21:36 +00:00
goto tracing ;
}
2011-05-27 14:36:01 +02:00
if ( WSTOPSIG ( status ) ! = syscall_trap_sig ) {
1999-02-19 00:21:36 +00:00
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 ) ;
2008-12-17 19:21:59 +00:00
if ( ptrace_restart ( PTRACE_SYSCALL , tcp , 0 ) < 0 ) {
1999-02-19 00:21:36 +00:00
cleanup ( ) ;
return - 1 ;
}
continue ;
}
2010-03-28 19:24:54 +00:00
if ( cflag ! = CFLAG_ONLY_STATS
1999-02-19 00:21:36 +00:00
& & ( qual_flags [ WSTOPSIG ( status ) ] & QUAL_SIGNAL ) ) {
2011-03-10 14:44:45 +00:00
siginfo_t si ;
# if defined(PT_CR_IPSR) && defined(PT_CR_IIP)
2011-03-10 21:20:35 +00:00
long pc = 0 ;
long psr = 0 ;
2001-10-09 23:47:38 +00:00
2008-12-16 18:18:40 +00:00
upeek ( tcp , PT_CR_IPSR , & psr ) ;
upeek ( tcp , PT_CR_IIP , & pc ) ;
2001-10-09 23:47:38 +00:00
2011-03-10 21:20:35 +00:00
# define PSR_RI 41
2001-10-09 23:47:38 +00:00
pc + = ( psr > > PSR_RI ) & 0x3 ;
2011-03-10 21:20:35 +00:00
# define PC_FORMAT_STR " @ %lx"
# define PC_FORMAT_ARG pc
# else
# define PC_FORMAT_STR "%s"
# define PC_FORMAT_ARG ""
2001-10-09 23:47:38 +00:00
# endif
1999-02-19 00:21:36 +00:00
printleader ( tcp ) ;
2011-03-10 21:20:35 +00:00
if ( ptrace ( PTRACE_GETSIGINFO , pid , 0 , & si ) = = 0 ) {
tprintf ( " --- " ) ;
printsiginfo ( & si , verbose ( tcp ) ) ;
tprintf ( " (%s) " PC_FORMAT_STR " --- " ,
strsignal ( WSTOPSIG ( status ) ) ,
PC_FORMAT_ARG ) ;
} else
tprintf ( " --- %s by %s " PC_FORMAT_STR " --- " ,
strsignal ( WSTOPSIG ( status ) ) ,
signame ( WSTOPSIG ( status ) ) ,
PC_FORMAT_ARG ) ;
2009-01-06 21:45:06 +00:00
printtrailer ( ) ;
1999-02-19 00:21:36 +00:00
}
2004-10-20 01:00:27 +00:00
if ( ( ( tcp - > flags & TCB_ATTACHED ) | |
tcp - > nclone_threads > 0 ) & &
1999-02-19 00:21:36 +00:00
! sigishandled ( tcp , WSTOPSIG ( status ) ) ) {
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# ifdef TCB_GROUP_EXITING
handle_group_exit ( tcp , WSTOPSIG ( status ) ) ;
# else
1999-02-19 00:21:36 +00:00
detach ( tcp , WSTOPSIG ( status ) ) ;
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# endif
1999-02-19 00:21:36 +00:00
continue ;
}
2008-12-17 19:21:59 +00:00
if ( ptrace_restart ( PTRACE_SYSCALL , tcp , WSTOPSIG ( status ) ) < 0 ) {
1999-02-19 00:21:36 +00:00
cleanup ( ) ;
return - 1 ;
}
tcp - > flags & = ~ TCB_SUSPENDED ;
continue ;
}
2007-06-11 22:06:31 +00:00
/* we handled the STATUS, we are permitted to interrupt now. */
if ( interrupted )
return 0 ;
2009-06-02 16:49:22 -07:00
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 ( ) .
2009-01-06 21:45:06 +00:00
*/
if ( tcp - > flags & TCB_ATTACHED ) {
2009-06-02 16:49:22 -07:00
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 ( ) ;
}
1999-02-19 00:21:36 +00:00
detach ( tcp , 0 ) ;
2009-01-06 21:45:06 +00:00
} else {
2009-06-02 16:49:22 -07:00
ptrace ( PTRACE_KILL ,
tcp - > pid , ( char * ) 1 , SIGTERM ) ;
1999-02-19 00:21:36 +00:00
droptcb ( tcp ) ;
}
continue ;
}
if ( tcp - > flags & TCB_EXITING ) {
2003-01-08 Roland McGrath <roland@redhat.com>
Support for new Linux 2.5 thread features.
* defs.h [LINUX]: Define __NR_exit_group if not defined.
(struct tcb): New members nclone_threads, nclone_detached,
and nclone_waiting.
(TCB_CLONE_DETACHED, TCB_CLONE_THREAD, TCB_GROUP_EXITING): New macros.
(waiting_parent): Macro removed.
(pid2tcb): Declare it.
* process.c (internal_clone) [TCB_CLONE_THREAD]: Reparent the new
child to our parent if we are a CLONE_THREAD child ourselves.
Maintain TCB_CLONE_THREAD and TCB_CLONE_DETACHED flags and counts.
(internal_wait) [TCB_CLONE_THREAD]: Factor out detached children when
determining if we have any. If TCB_CLONE_THREAD is set, check
parent's children instead of our own, and bump nclone_waiting count.
(internal_exit) [__NR_exit_group]: Set the TCB_GROUP_EXITING flag if
the syscall was exit_group.
* syscall.c (internal_syscall): Use internal_exit for exit_group.
* strace.c (pid2tcb): No longer static.
(alloctcb) [TCB_CLONE_THREAD]: Initialize new fields.
(droptcb) [TCB_CLONE_THREAD]: Maintain new fields.
If we have thread children, set TCB_EXITING and don't clear the TCB.
(resume) [TCB_CLONE_THREAD]: Decrement parent's nclone_waiting.
(detach) [TCB_CLONE_THREAD]: When calling resume, check all thread
children of our parent that might be waiting for us too.
[TCB_GROUP_EXITING] (handle_group_exit): New function.
(trace) [TCB_GROUP_EXITING]: Use that in place of detach or droptcb.
Revamp -f support for Linux.
* util.c [LINUX] (setbpt, clearbpt): New implementations that tweak
the system call to be clone with CLONE_PTRACE set. Various new static
helper functions.
* process.c (internal_clone): Define also #ifdef SYS_clone2.
Initialize TCPCHILD->parent field.
[CLONE_PTRACE]: Don't do PTRACE_ATTACH here, because it's preattached.
Check in case the new child is in the tcb already.
(internal_fork) [LINUX]: Just call internal_clone.
* strace.c (trace) [LINUX]: Under -f/-F, grok an unknown pid
reporting to wait, put it in the TCB with TCB_ATTACHED|TCB_SUSPENDED.
2003-01-09 06:53:31 +00:00
# ifdef TCB_GROUP_EXITING
if ( tcp - > flags & TCB_GROUP_EXITING ) {
if ( handle_group_exit ( tcp , 0 ) < 0 )
return - 1 ;
continue ;
}
# endif
1999-02-19 00:21:36 +00:00
if ( tcp - > flags & TCB_ATTACHED )
detach ( tcp , 0 ) ;
2008-12-17 19:21:59 +00:00
else if ( ptrace_restart ( PTRACE_CONT , tcp , 0 ) < 0 ) {
1999-02-19 00:21:36 +00:00
cleanup ( ) ;
return - 1 ;
}
continue ;
}
if ( tcp - > flags & TCB_SUSPENDED ) {
if ( ! qflag )
fprintf ( stderr , " Process %u suspended \n " , pid ) ;
continue ;
}
tracing :
2009-10-27 16:27:13 +01:00
/* Remember current print column before continuing. */
tcp - > curcol = curcol ;
2008-12-17 19:21:59 +00:00
if ( ptrace_restart ( PTRACE_SYSCALL , tcp , 0 ) < 0 ) {
1999-02-19 00:21:36 +00:00
cleanup ( ) ;
return - 1 ;
}
}
return 0 ;
}
2000-09-01 21:03:06 +00:00
# endif /* !USE_PROCFS */
1999-02-19 00:21:36 +00:00
# include <stdarg.h>
void
tprintf ( const char * fmt , . . . )
{
va_list args ;
2009-10-27 16:56:43 +01:00
va_start ( args , fmt ) ;
2003-11-06 23:41:22 +00:00
if ( outf ) {
int n = vfprintf ( outf , fmt , args ) ;
2009-10-27 16:27:13 +01:00
if ( n < 0 ) {
if ( outf ! = stderr )
perror ( outfname = = NULL
? " <writing to pipe> " : outfname ) ;
} else
2003-11-06 23:41:22 +00:00
curcol + = n ;
}
1999-02-19 00:21:36 +00:00
va_end ( args ) ;
return ;
}
void
2011-05-30 14:00:14 +02:00
printleader ( struct tcb * tcp )
1999-02-19 00:21:36 +00:00
{
2008-12-17 19:21:59 +00:00
if ( tcp_last ) {
2009-06-02 16:49:22 -07:00
if ( tcp_last - > ptrace_errno ) {
2008-12-17 19:21:59 +00:00
if ( tcp_last - > flags & TCB_INSYSCALL ) {
2011-06-08 16:15:04 +02:00
tprintf ( " <unavailable>) " ) ;
2009-06-02 16:49:22 -07:00
tabto ( acolumn ) ;
2008-12-17 19:21:59 +00:00
}
2009-06-02 16:49:22 -07:00
tprintf ( " = ? <unavailable> \n " ) ;
tcp_last - > ptrace_errno = 0 ;
2008-12-17 19:21:59 +00:00
} else if ( ! outfname | | followfork < 2 | | tcp_last = = tcp ) {
2009-01-28 19:00:54 +00:00
tcp_last - > flags | = TCB_REPRINT ;
2009-06-02 16:49:22 -07:00
tprintf ( " <unfinished ...> \n " ) ;
2008-12-17 19:21:59 +00:00
}
1999-02-19 00:21:36 +00:00
}
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
2011-05-30 14:00:14 +02:00
tabto ( int col )
1999-02-19 00:21:36 +00:00
{
if ( curcol < col )
tprintf ( " %*s " , col - curcol , " " ) ;
}
void
2009-01-06 21:45:06 +00:00
printtrailer ( void )
1999-02-19 00:21:36 +00:00
{
tprintf ( " \n " ) ;
tcp_last = NULL ;
}
1999-08-29 23:15:07 +00:00
1999-11-29 15:34:02 +00:00
# ifdef HAVE_MP_PROCFS
1999-08-29 23:15:07 +00:00
2009-01-06 21:45:06 +00:00
int
mp_ioctl ( int fd , int cmd , void * arg , int size )
{
1999-08-29 23:15:07 +00:00
struct iovec iov [ 2 ] ;
int n = 1 ;
2002-12-16 20:40:39 +00:00
1999-08-29 23:15:07 +00:00
iov [ 0 ] . iov_base = & cmd ;
iov [ 0 ] . iov_len = sizeof cmd ;
if ( arg ) {
+ + n ;
iov [ 1 ] . iov_base = arg ;
iov [ 1 ] . iov_len = size ;
}
2002-12-16 20:40:39 +00:00
2009-01-06 21:45:06 +00:00
return writev ( fd , iov , n ) ;
1999-08-29 23:15:07 +00:00
}
# endif