2011-04-08 00:25:40 +04:00
/*
* Copyright ( c ) 2011 , Comtrol Corp .
* 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 .
*
*/
# include "defs.h"
2011-08-30 20:05:26 +04:00
# include <sys/param.h>
2011-04-08 00:25:40 +04:00
# ifdef HAVE_POLL_H
2012-03-16 15:02:22 +04:00
# include <poll.h>
2011-04-08 00:25:40 +04:00
# endif
# ifdef HAVE_SYS_POLL_H
2012-03-16 15:02:22 +04:00
# include <sys / poll.h>
2011-04-08 00:25:40 +04:00
# endif
# include "syscall.h"
# define MAXSELECTED 256 /* max number of "selected" paths */
static const char * selected [ MAXSELECTED ] ; /* paths selected for tracing */
/*
* Return true if specified path matches one that we ' re tracing .
*/
static int
pathmatch ( const char * path )
{
2011-06-14 01:58:43 +04:00
unsigned int i ;
2011-04-08 00:25:40 +04:00
2011-06-22 16:32:43 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( selected ) ; + + i ) {
2011-04-08 00:25:40 +04:00
if ( selected [ i ] = = NULL )
return 0 ;
2012-03-15 21:03:56 +04:00
if ( strcmp ( path , selected [ i ] ) = = 0 )
2011-04-08 00:25:40 +04:00
return 1 ;
}
return 0 ;
}
/*
* Return true if specified path ( in user - space ) matches .
*/
static int
upathmatch ( struct tcb * tcp , unsigned long upath )
{
2011-06-22 16:32:43 +04:00
char path [ PATH_MAX + 1 ] ;
2011-04-08 00:25:40 +04:00
2012-01-20 14:56:00 +04:00
return umovestr ( tcp , upath , sizeof path , path ) > = 0 & &
2011-04-08 00:25:40 +04:00
pathmatch ( path ) ;
}
/*
* Return true if specified fd maps to a path we ' re tracing .
*/
static int
fdmatch ( struct tcb * tcp , int fd )
{
const char * path = getfdpath ( tcp , fd ) ;
return path & & pathmatch ( path ) ;
}
/*
* Add a path to the set we ' re tracing .
* Secifying NULL will delete all paths .
*/
static int
storepath ( const char * path )
{
2011-06-14 01:58:43 +04:00
unsigned int i ;
2011-04-08 00:25:40 +04:00
2012-03-15 21:03:56 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( selected ) ; + + i ) {
2011-06-22 16:32:43 +04:00
if ( ! selected [ i ] ) {
2011-04-08 00:25:40 +04:00
selected [ i ] = path ;
return 0 ;
}
2012-03-15 21:03:56 +04:00
}
2011-04-08 00:25:40 +04:00
2011-06-14 01:58:43 +04:00
fprintf ( stderr , " Max trace paths exceeded, only using first %u \n " ,
( unsigned int ) ARRAY_SIZE ( selected ) ) ;
2011-04-08 00:25:40 +04:00
return - 1 ;
}
/*
* Get path associated with fd .
*/
2012-03-15 21:03:56 +04:00
const char *
getfdpath ( struct tcb * tcp , int fd )
2011-04-08 00:25:40 +04:00
{
static char path [ PATH_MAX + 1 ] ;
2012-03-15 21:11:51 +04:00
char linkpath [ sizeof ( " /proc/%u/fd/%u " ) + 2 * sizeof ( int ) * 3 ] ;
2011-04-08 00:25:40 +04:00
ssize_t n ;
if ( fd < 0 )
return NULL ;
2012-03-15 21:03:56 +04:00
sprintf ( linkpath , " /proc/%u/fd/%u " , tcp - > pid , fd ) ;
2011-04-08 00:25:40 +04:00
n = readlink ( linkpath , path , ( sizeof path ) - 1 ) ;
if ( n < = 0 )
return NULL ;
path [ n ] = ' \0 ' ;
return path ;
}
/*
* Add a path to the set we ' re tracing . Also add the canonicalized
* version of the path . Secifying NULL will delete all paths .
*/
int
pathtrace_select ( const char * path )
{
2012-03-15 21:03:56 +04:00
char * rpath ;
2011-04-08 00:25:40 +04:00
if ( storepath ( path ) )
return - 1 ;
rpath = realpath ( path , NULL ) ;
if ( rpath = = NULL )
return 0 ;
/* if realpath and specified path are same, we're done */
2012-03-15 21:03:56 +04:00
if ( strcmp ( path , rpath ) = = 0 ) {
2011-04-08 00:25:40 +04:00
free ( rpath ) ;
return 0 ;
}
fprintf ( stderr , " Requested path '%s' resolved into '%s' \n " ,
path , rpath ) ;
return storepath ( rpath ) ;
}
/*
* Return true if syscall accesses a selected path
* ( or if no paths have been specified for tracing ) .
*/
int
pathtrace_match ( struct tcb * tcp )
{
const struct sysent * s ;
if ( selected [ 0 ] = = NULL )
return 1 ;
2013-02-16 16:22:38 +04:00
if ( ! SCNO_IS_VALID ( tcp - > scno ) )
2012-02-06 21:13:59 +04:00
return 0 ;
2011-04-08 00:25:40 +04:00
s = & sysent [ tcp - > scno ] ;
if ( ! ( s - > sys_flags & ( TRACE_FILE | TRACE_DESC ) ) )
return 0 ;
/*
* Check for special cases where we need to do something
* other than test arg [ 0 ] .
*/
if ( s - > sys_func = = sys_dup2 | |
s - > sys_func = = sys_dup3 | |
s - > sys_func = = sys_sendfile | |
s - > sys_func = = sys_sendfile64 | |
Remove redundant parsers
* desc.c (sys_dup): Remove.
* file.c (sys_pivotroot, sys_rmdir, sys_fchdir, sys_chroot, sys_fchroot,
sys_unlink, sys_symlink, sys_rename): Remove.
* linux/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_pivotroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
* linux/dummy.h: Add aliases for sys_chroot, sys_dup, sys_pivotroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink.
* pathtrace.c (pathtrace_match): Update.
* sunos4/dummy.h: Add aliases for sys_chroot, sys_dup, sys_fchdir,
sys_fchroot, sys_rename, sys_rmdir, sys_symlink, sys_unlink.
* svr4/dummy.h: Likewise.
* sunos4/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_fchroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
* svr4/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_fchroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
2011-11-29 02:48:53 +04:00
s - > sys_func = = sys_tee )
2011-04-08 00:25:40 +04:00
{
/* fd, fd */
return fdmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
fdmatch ( tcp , tcp - > u_arg [ 1 ] ) ;
}
if ( s - > sys_func = = sys_inotify_add_watch | |
s - > sys_func = = sys_faccessat | |
s - > sys_func = = sys_fchmodat | |
s - > sys_func = = sys_futimesat | |
s - > sys_func = = sys_mkdirat | |
s - > sys_func = = sys_unlinkat | |
s - > sys_func = = sys_newfstatat | |
s - > sys_func = = sys_mknodat | |
s - > sys_func = = sys_openat | |
s - > sys_func = = sys_readlinkat | |
s - > sys_func = = sys_utimensat | |
s - > sys_func = = sys_fchownat | |
2011-06-07 14:13:24 +04:00
s - > sys_func = = sys_pipe2 )
2011-04-08 00:25:40 +04:00
{
/* fd, path */
return fdmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
upathmatch ( tcp , tcp - > u_arg [ 1 ] ) ;
}
if ( s - > sys_func = = sys_link | |
2011-06-07 14:13:24 +04:00
s - > sys_func = = sys_mount )
2011-04-08 00:25:40 +04:00
{
/* path, path */
return upathmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
upathmatch ( tcp , tcp - > u_arg [ 1 ] ) ;
}
2012-10-27 03:43:13 +04:00
if ( s - > sys_func = = sys_quotactl )
{
/* x, path */
return upathmatch ( tcp , tcp - > u_arg [ 1 ] ) ;
}
2011-04-08 00:25:40 +04:00
if ( s - > sys_func = = sys_renameat | |
2011-06-07 14:13:24 +04:00
s - > sys_func = = sys_linkat )
2011-04-08 00:25:40 +04:00
{
/* fd, path, fd, path */
return fdmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
fdmatch ( tcp , tcp - > u_arg [ 2 ] ) | |
upathmatch ( tcp , tcp - > u_arg [ 1 ] ) | |
upathmatch ( tcp , tcp - > u_arg [ 3 ] ) ;
}
Add x32 support to strace
X32 support is added to Linux kernel 3.4. In a nutshell, x32 is x86-64 with
32bit pointers. At system call level, x32 is also identical to x86-64,
as shown by many changes like "defined(X86_64) || defined(X32)". The
main differerence bewteen x32 and x86-64 is off_t in x32 is long long
instead of long.
This patch adds x32 support to strace. Tested on Linux/x32.
* configure.ac: Support X32.
* defs.h: Set SUPPORTED_PERSONALITIES to 3 for X86_64,
Set PERSONALITY2_WORDSIZE to 4 for X86_64.
Add tcb::ext_arg for X32.
* file.c (stat): New for X32.
(sys_lseek): Use 64-bit version for X32.
(printstat64): Check current_personality != 1 for X86_64.
* ipc.c (indirect_ipccall): Check current_personality == 1
for X86_64.
* mem.c (sys_mmap64): Also use tcp->u_arg for X32. Print NULL
for zero address. Call printllval for offset for X32.
* pathtrace.c (pathtrace_match): Don't check sys_old_mmap for
X32.
* process.c (ARG_FLAGS): Defined for X32.
(ARG_STACK): Likewise.
(ARG_PTID): Likewise.
(change_syscall): Handle X32.
(struct_user_offsets): Support X32.
(sys_arch_prctl): Likewise.
* signal.c: Include <asm/sigcontext.h> for X32.
(SA_RESTORER): Also define for X32.
* syscall.c (update_personality): Support X32 for X86_64.
(is_restart_error): Likewise.
(syscall_fixup_on_sysenter): Likewise.
(get_syscall_args): Likewise.
(get_syscall_result): Likewise.
(get_error): Likewise.
(__X32_SYSCALL_BIT): Define if not defined.
(__X32_SYSCALL_MASK): Likewise.
(get_scno): Check DS register value for X32. Use
__X32_SYSCALL_MASK on X32 system calls.
* util.c (printllval): Use ext_arg for X32.
(printcall): Support X32.
(change_syscall): Likewise.
(arg0_offset): Likewise.
(arg1_offset): Likewise.
* Makefile.am (EXTRA_DIST): Add linux/x32/errnoent.h,
linux/x32/ioctlent.h.in, linux/x32/signalent.h,
linux/x32/syscallent.h, linux/x86_64/errnoent2.h,
linux/x86_64/ioctlent2.h, linux/x86_64/signalent2.h and
linux/x86_64/syscallent2.h.
* linux/x32/errnoent.h: New.
* linux/x32/ioctlent.h.in: Likewise.
* linux/x32/signalent.h: Likewise.
* linux/x32/syscallent.h: Likewise.
* linux/x86_64/errnoent2.h: Likewise.
* linux/x86_64/ioctlent2.h: Likewise.
* linux/x86_64/signalent2.h: Likewise.
* linux/x86_64/syscallent2.h: Likewise.
Signed-off-by: H.J. Lu <hongjiu.lu@intel.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2012-04-16 15:00:01 +04:00
if (
s - > sys_func = = sys_old_mmap | |
s - > sys_func = = sys_mmap ) {
2011-04-08 00:25:40 +04:00
/* x, x, x, x, fd */
return fdmatch ( tcp , tcp - > u_arg [ 4 ] ) ;
}
2011-06-22 16:32:43 +04:00
if ( s - > sys_func = = sys_symlinkat ) {
2011-04-08 00:25:40 +04:00
/* path, fd, path */
return fdmatch ( tcp , tcp - > u_arg [ 1 ] ) | |
upathmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
2011-06-07 14:13:24 +04:00
upathmatch ( tcp , tcp - > u_arg [ 2 ] ) ;
2011-04-08 00:25:40 +04:00
}
Remove redundant parsers
* desc.c (sys_dup): Remove.
* file.c (sys_pivotroot, sys_rmdir, sys_fchdir, sys_chroot, sys_fchroot,
sys_unlink, sys_symlink, sys_rename): Remove.
* linux/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_pivotroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
* linux/dummy.h: Add aliases for sys_chroot, sys_dup, sys_pivotroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink.
* pathtrace.c (pathtrace_match): Update.
* sunos4/dummy.h: Add aliases for sys_chroot, sys_dup, sys_fchdir,
sys_fchroot, sys_rename, sys_rmdir, sys_symlink, sys_unlink.
* svr4/dummy.h: Likewise.
* sunos4/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_fchroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
* svr4/syscall.h (sys_chroot, sys_dup, sys_fchdir, sys_fchroot,
sys_rename, sys_rmdir, sys_symlink, sys_unlink): Remove.
2011-11-29 02:48:53 +04:00
if ( s - > sys_func = = sys_splice ) {
2011-04-08 00:25:40 +04:00
/* fd, x, fd, x, x */
return fdmatch ( tcp , tcp - > u_arg [ 0 ] ) | |
fdmatch ( tcp , tcp - > u_arg [ 2 ] ) ;
}
2011-06-22 16:32:43 +04:00
if ( s - > sys_func = = sys_epoll_ctl ) {
2011-04-08 00:25:40 +04:00
/* x, x, fd, x */
return fdmatch ( tcp , tcp - > u_arg [ 2 ] ) ;
}
if ( s - > sys_func = = sys_select | |
s - > sys_func = = sys_oldselect | |
2011-06-07 14:13:24 +04:00
s - > sys_func = = sys_pselect6 )
2011-04-08 00:25:40 +04:00
{
2011-09-01 18:35:44 +04:00
int i , j ;
unsigned nfds ;
2011-04-08 00:25:40 +04:00
long * args , oldargs [ 5 ] ;
unsigned fdsize ;
fd_set * fds ;
2011-06-22 16:32:43 +04:00
if ( s - > sys_func = = sys_oldselect ) {
2011-04-08 00:25:40 +04:00
if ( umoven ( tcp , tcp - > u_arg [ 0 ] , sizeof oldargs ,
( char * ) oldargs ) < 0 )
{
fprintf ( stderr , " umoven() failed \n " ) ;
return 0 ;
}
args = oldargs ;
} else
args = tcp - > u_arg ;
nfds = args [ 0 ] ;
2011-09-01 18:35:44 +04:00
/* Beware of select(2^31-1, NULL, NULL, NULL) and similar... */
if ( args [ 0 ] > 1024 * 1024 )
nfds = 1024 * 1024 ;
if ( args [ 0 ] < 0 )
nfds = 0 ;
2011-04-08 00:25:40 +04:00
fdsize = ( ( ( ( nfds + 7 ) / 8 ) + sizeof ( long ) - 1 )
& - sizeof ( long ) ) ;
fds = malloc ( fdsize ) ;
2011-08-31 16:00:02 +04:00
if ( ! fds )
die_out_of_memory ( ) ;
2011-04-08 00:25:40 +04:00
2011-06-22 16:32:43 +04:00
for ( i = 1 ; i < = 3 ; + + i ) {
2011-04-08 00:25:40 +04:00
if ( args [ i ] = = 0 )
continue ;
2011-06-22 16:32:43 +04:00
if ( umoven ( tcp , args [ i ] , fdsize , ( char * ) fds ) < 0 ) {
2011-04-08 00:25:40 +04:00
fprintf ( stderr , " umoven() failed \n " ) ;
continue ;
}
for ( j = 0 ; j < nfds ; + + j )
2011-06-22 16:32:43 +04:00
if ( FD_ISSET ( j , fds ) & & fdmatch ( tcp , j ) ) {
2011-04-08 00:25:40 +04:00
free ( fds ) ;
return 1 ;
}
}
free ( fds ) ;
return 0 ;
}
if ( s - > sys_func = = sys_poll | |
2011-06-07 14:13:24 +04:00
s - > sys_func = = sys_ppoll )
2011-04-08 00:25:40 +04:00
{
struct pollfd fds ;
unsigned nfds ;
unsigned long start , cur , end ;
start = tcp - > u_arg [ 0 ] ;
nfds = tcp - > u_arg [ 1 ] ;
end = start + sizeof ( fds ) * nfds ;
if ( nfds = = 0 | | end < start )
return 0 ;
for ( cur = start ; cur < end ; cur + = sizeof ( fds ) )
if ( ( umoven ( tcp , cur , sizeof fds , ( char * ) & fds ) = = 0 )
& & fdmatch ( tcp , fds . fd ) )
return 1 ;
return 0 ;
}
if ( s - > sys_func = = printargs | |
s - > sys_func = = sys_pipe | |
s - > sys_func = = sys_pipe2 | |
s - > sys_func = = sys_eventfd2 | |
s - > sys_func = = sys_eventfd | |
s - > sys_func = = sys_inotify_init1 | |
s - > sys_func = = sys_timerfd_create | |
s - > sys_func = = sys_timerfd_settime | |
s - > sys_func = = sys_timerfd_gettime | |
s - > sys_func = = sys_epoll_create | |
2012-03-15 21:03:56 +04:00
strcmp ( s - > sys_name , " fanotify_init " ) = = 0 )
2011-04-08 00:25:40 +04:00
{
/*
* These have TRACE_FILE or TRACE_DESCRIPTOR set , but they
* don ' t have any file descriptor or path args to test .
*/
return 0 ;
}
/*
* Our fallback position for calls that haven ' t already
* been handled is to just check arg [ 0 ] .
*/
if ( s - > sys_flags & TRACE_FILE )
return upathmatch ( tcp , tcp - > u_arg [ 0 ] ) ;
if ( s - > sys_flags & TRACE_DESC )
return fdmatch ( tcp , tcp - > u_arg [ 0 ] ) ;
return 0 ;
}