2005-04-17 02:20:36 +04:00
/*
* linux / kernel / ptrace . c
*
* ( C ) Copyright 1999 Linus Torvalds
*
* Common interfaces for " ptrace() " which we do not want
* to continually duplicate across every architecture .
*/
2006-01-11 23:17:46 +03:00
# include <linux/capability.h>
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/errno.h>
# include <linux/mm.h>
# include <linux/highmem.h>
# include <linux/pagemap.h>
# include <linux/smp_lock.h>
# include <linux/ptrace.h>
# include <linux/security.h>
2005-05-01 19:59:14 +04:00
# include <linux/signal.h>
2007-03-20 20:58:35 +03:00
# include <linux/audit.h>
2007-10-19 10:40:14 +04:00
# include <linux/pid_namespace.h>
2008-02-06 12:36:44 +03:00
# include <linux/syscalls.h>
2005-04-17 02:20:36 +04:00
# include <asm/pgtable.h>
# include <asm/uaccess.h>
/*
* ptrace a task : make the debugger its new parent and
* move it to the ptrace list .
*
* Must be called with the tasklist lock write - held .
*/
2006-07-03 11:25:41 +04:00
void __ptrace_link ( struct task_struct * child , struct task_struct * new_parent )
2005-04-17 02:20:36 +04:00
{
2006-04-02 15:43:40 +04:00
BUG_ON ( ! list_empty ( & child - > ptrace_list ) ) ;
2005-04-17 02:20:36 +04:00
if ( child - > parent = = new_parent )
return ;
list_add ( & child - > ptrace_list , & child - > parent - > ptrace_children ) ;
2006-03-29 04:11:05 +04:00
remove_parent ( child ) ;
2005-04-17 02:20:36 +04:00
child - > parent = new_parent ;
2006-03-29 04:11:05 +04:00
add_parent ( child ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Turn a tracing stop into a normal stop now , since with no tracer there
* would be no way to wake it up with SIGCONT or SIGKILL . If there was a
* signal sent that would resume the child , but didn ' t because it was in
* TASK_TRACED , resume it now .
* Requires that irqs be disabled .
*/
2006-07-03 11:25:41 +04:00
void ptrace_untrace ( struct task_struct * child )
2005-04-17 02:20:36 +04:00
{
spin_lock ( & child - > sighand - > siglock ) ;
2007-12-06 19:06:16 +03:00
if ( task_is_traced ( child ) ) {
2005-04-17 02:20:36 +04:00
if ( child - > signal - > flags & SIGNAL_STOP_STOPPED ) {
2008-02-06 12:36:13 +03:00
__set_task_state ( child , TASK_STOPPED ) ;
2005-04-17 02:20:36 +04:00
} else {
signal_wake_up ( child , 1 ) ;
}
}
spin_unlock ( & child - > sighand - > siglock ) ;
}
/*
* unptrace a task : move it back to its original parent and
* remove it from the ptrace list .
*
* Must be called with the tasklist lock write - held .
*/
2006-07-03 11:25:41 +04:00
void __ptrace_unlink ( struct task_struct * child )
2005-04-17 02:20:36 +04:00
{
2006-02-15 22:50:10 +03:00
BUG_ON ( ! child - > ptrace ) ;
2005-04-17 02:20:36 +04:00
child - > ptrace = 0 ;
2008-04-30 11:53:14 +04:00
if ( ptrace_reparented ( child ) ) {
2005-04-17 02:20:36 +04:00
list_del_init ( & child - > ptrace_list ) ;
2006-03-29 04:11:05 +04:00
remove_parent ( child ) ;
2005-04-17 02:20:36 +04:00
child - > parent = child - > real_parent ;
2006-03-29 04:11:05 +04:00
add_parent ( child ) ;
2005-04-17 02:20:36 +04:00
}
2007-12-06 19:06:16 +03:00
if ( task_is_traced ( child ) )
2006-04-13 03:30:20 +04:00
ptrace_untrace ( child ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Check that we have indeed attached to the thing . .
*/
int ptrace_check_attach ( struct task_struct * child , int kill )
{
int ret = - ESRCH ;
/*
* We take the read lock around doing both checks to close a
* possible race where someone else was tracing our child and
* detached between these two checks . After this locked check ,
* we are sure that this is our traced child and that can only
* be changed by us so it ' s not changing right after this .
*/
read_lock ( & tasklist_lock ) ;
2008-02-08 15:19:00 +03:00
if ( ( child - > ptrace & PT_PTRACED ) & & child - > parent = = current ) {
2005-04-17 02:20:36 +04:00
ret = 0 ;
2008-02-08 15:19:00 +03:00
/*
* child - > sighand can ' t be NULL , release_task ( )
* does ptrace_unlink ( ) before __exit_signal ( ) .
*/
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & child - > sighand - > siglock ) ;
2008-02-06 12:36:13 +03:00
if ( task_is_stopped ( child ) )
2005-04-17 02:20:36 +04:00
child - > state = TASK_TRACED ;
2008-02-06 12:36:13 +03:00
else if ( ! task_is_traced ( child ) & & ! kill )
2005-04-17 02:20:36 +04:00
ret = - ESRCH ;
spin_unlock_irq ( & child - > sighand - > siglock ) ;
}
read_unlock ( & tasklist_lock ) ;
2008-02-06 12:36:13 +03:00
if ( ! ret & & ! kill )
2005-04-17 02:20:36 +04:00
wait_task_inactive ( child ) ;
/* All systems go.. */
return ret ;
}
2008-01-02 17:09:57 +03:00
int __ptrace_may_attach ( struct task_struct * task )
2005-09-07 02:18:24 +04:00
{
2006-06-26 11:25:59 +04:00
/* May we inspect the given task?
* This check is used both for attaching with ptrace
* and for allowing access to sensitive information in / proc .
*
* ptrace_attach denies several cases that / proc allows
* because setting up the necessary parent / child relationship
* or halting the specified task is impossible .
*/
int dumpable = 0 ;
/* Don't let security modules deny introspection */
if ( task = = current )
return 0 ;
2005-09-07 02:18:24 +04:00
if ( ( ( current - > uid ! = task - > euid ) | |
( current - > uid ! = task - > suid ) | |
( current - > uid ! = task - > uid ) | |
( current - > gid ! = task - > egid ) | |
( current - > gid ! = task - > sgid ) | |
( current - > gid ! = task - > gid ) ) & & ! capable ( CAP_SYS_PTRACE ) )
return - EPERM ;
smp_rmb ( ) ;
2006-06-26 11:25:59 +04:00
if ( task - > mm )
2007-07-19 12:48:27 +04:00
dumpable = get_dumpable ( task - > mm ) ;
2006-06-26 11:25:59 +04:00
if ( ! dumpable & & ! capable ( CAP_SYS_PTRACE ) )
2005-09-07 02:18:24 +04:00
return - EPERM ;
return security_ptrace ( current , task ) ;
}
int ptrace_may_attach ( struct task_struct * task )
{
int err ;
task_lock ( task ) ;
2008-01-02 17:09:57 +03:00
err = __ptrace_may_attach ( task ) ;
2005-09-07 02:18:24 +04:00
task_unlock ( task ) ;
return ! err ;
}
2005-04-17 02:20:36 +04:00
int ptrace_attach ( struct task_struct * task )
{
int retval ;
2007-07-16 10:39:26 +04:00
unsigned long flags ;
2006-05-07 21:49:33 +04:00
2007-03-20 20:58:35 +03:00
audit_ptrace ( task ) ;
2005-04-17 02:20:36 +04:00
retval = - EPERM ;
2007-10-19 10:40:18 +04:00
if ( same_thread_group ( task , current ) )
2006-05-07 21:49:33 +04:00
goto out ;
2006-05-11 22:08:49 +04:00
repeat :
/*
* Nasty , nasty .
*
* We want to hold both the task - lock and the
* tasklist_lock for writing at the same time .
* But that ' s against the rules ( tasklist_lock
* is taken for reading by interrupts on other
* cpu ' s that may have task_lock ) .
*/
2006-05-07 21:49:33 +04:00
task_lock ( task ) ;
2007-07-16 10:39:26 +04:00
if ( ! write_trylock_irqsave ( & tasklist_lock , flags ) ) {
2006-05-11 22:08:49 +04:00
task_unlock ( task ) ;
do {
cpu_relax ( ) ;
} while ( ! write_can_lock ( & tasklist_lock ) ) ;
goto repeat ;
}
2006-05-07 21:49:33 +04:00
2006-06-26 11:25:59 +04:00
if ( ! task - > mm )
goto bad ;
2005-04-17 02:20:36 +04:00
/* the same process cannot be attached many times */
if ( task - > ptrace & PT_PTRACED )
goto bad ;
2008-01-03 00:48:27 +03:00
retval = __ptrace_may_attach ( task ) ;
2005-04-17 02:20:36 +04:00
if ( retval )
goto bad ;
/* Go */
2008-02-08 15:18:58 +03:00
task - > ptrace | = PT_PTRACED ;
2005-04-17 02:20:36 +04:00
if ( capable ( CAP_SYS_PTRACE ) )
task - > ptrace | = PT_PTRACE_CAP ;
__ptrace_link ( task , current ) ;
2008-04-30 11:53:14 +04:00
send_sig_info ( SIGSTOP , SEND_SIG_FORCED , task ) ;
2005-04-17 02:20:36 +04:00
bad :
2007-07-16 10:39:26 +04:00
write_unlock_irqrestore ( & tasklist_lock , flags ) ;
2005-04-17 02:20:36 +04:00
task_unlock ( task ) ;
2006-05-07 21:49:33 +04:00
out :
2005-04-17 02:20:36 +04:00
return retval ;
}
2006-06-26 11:26:07 +04:00
static inline void __ptrace_detach ( struct task_struct * child , unsigned int data )
2006-02-15 22:50:10 +03:00
{
child - > exit_code = data ;
/* .. re-parent .. */
__ptrace_unlink ( child ) ;
/* .. and wake it up. */
if ( child - > exit_state ! = EXIT_ZOMBIE )
wake_up_process ( child ) ;
}
2005-04-17 02:20:36 +04:00
int ptrace_detach ( struct task_struct * child , unsigned int data )
{
2005-05-01 19:59:14 +04:00
if ( ! valid_signal ( data ) )
2006-02-15 22:50:10 +03:00
return - EIO ;
2005-04-17 02:20:36 +04:00
/* Architecture-specific hardware disable .. */
ptrace_disable ( child ) ;
2007-09-05 14:05:56 +04:00
clear_tsk_thread_flag ( child , TIF_SYSCALL_TRACE ) ;
2005-04-17 02:20:36 +04:00
write_lock_irq ( & tasklist_lock ) ;
2006-06-26 11:26:07 +04:00
/* protect against de_thread()->release_task() */
2006-02-15 22:50:10 +03:00
if ( child - > ptrace )
__ptrace_detach ( child , data ) ;
2005-04-17 02:20:36 +04:00
write_unlock_irq ( & tasklist_lock ) ;
return 0 ;
}
int ptrace_readdata ( struct task_struct * tsk , unsigned long src , char __user * dst , int len )
{
int copied = 0 ;
while ( len > 0 ) {
char buf [ 128 ] ;
int this_len , retval ;
this_len = ( len > sizeof ( buf ) ) ? sizeof ( buf ) : len ;
retval = access_process_vm ( tsk , src , buf , this_len , 0 ) ;
if ( ! retval ) {
if ( copied )
break ;
return - EIO ;
}
if ( copy_to_user ( dst , buf , retval ) )
return - EFAULT ;
copied + = retval ;
src + = retval ;
dst + = retval ;
len - = retval ;
}
return copied ;
}
int ptrace_writedata ( struct task_struct * tsk , char __user * src , unsigned long dst , int len )
{
int copied = 0 ;
while ( len > 0 ) {
char buf [ 128 ] ;
int this_len , retval ;
this_len = ( len > sizeof ( buf ) ) ? sizeof ( buf ) : len ;
if ( copy_from_user ( buf , src , this_len ) )
return - EFAULT ;
retval = access_process_vm ( tsk , dst , buf , this_len , 1 ) ;
if ( ! retval ) {
if ( copied )
break ;
return - EIO ;
}
copied + = retval ;
src + = retval ;
dst + = retval ;
len - = retval ;
}
return copied ;
}
static int ptrace_setoptions ( struct task_struct * child , long data )
{
child - > ptrace & = ~ PT_TRACE_MASK ;
if ( data & PTRACE_O_TRACESYSGOOD )
child - > ptrace | = PT_TRACESYSGOOD ;
if ( data & PTRACE_O_TRACEFORK )
child - > ptrace | = PT_TRACE_FORK ;
if ( data & PTRACE_O_TRACEVFORK )
child - > ptrace | = PT_TRACE_VFORK ;
if ( data & PTRACE_O_TRACECLONE )
child - > ptrace | = PT_TRACE_CLONE ;
if ( data & PTRACE_O_TRACEEXEC )
child - > ptrace | = PT_TRACE_EXEC ;
if ( data & PTRACE_O_TRACEVFORKDONE )
child - > ptrace | = PT_TRACE_VFORK_DONE ;
if ( data & PTRACE_O_TRACEEXIT )
child - > ptrace | = PT_TRACE_EXIT ;
return ( data & ~ PTRACE_O_MASK ) ? - EINVAL : 0 ;
}
2008-04-21 00:10:12 +04:00
static int ptrace_getsiginfo ( struct task_struct * child , siginfo_t * info )
2005-04-17 02:20:36 +04:00
{
int error = - ESRCH ;
read_lock ( & tasklist_lock ) ;
if ( likely ( child - > sighand ! = NULL ) ) {
error = - EINVAL ;
spin_lock_irq ( & child - > sighand - > siglock ) ;
if ( likely ( child - > last_siginfo ! = NULL ) ) {
2008-04-21 00:10:12 +04:00
* info = * child - > last_siginfo ;
2005-04-17 02:20:36 +04:00
error = 0 ;
}
spin_unlock_irq ( & child - > sighand - > siglock ) ;
}
read_unlock ( & tasklist_lock ) ;
return error ;
}
2008-04-21 00:10:12 +04:00
static int ptrace_setsiginfo ( struct task_struct * child , const siginfo_t * info )
2005-04-17 02:20:36 +04:00
{
int error = - ESRCH ;
read_lock ( & tasklist_lock ) ;
if ( likely ( child - > sighand ! = NULL ) ) {
error = - EINVAL ;
spin_lock_irq ( & child - > sighand - > siglock ) ;
if ( likely ( child - > last_siginfo ! = NULL ) ) {
2008-04-21 00:10:12 +04:00
* child - > last_siginfo = * info ;
2005-04-17 02:20:36 +04:00
error = 0 ;
}
spin_unlock_irq ( & child - > sighand - > siglock ) ;
}
read_unlock ( & tasklist_lock ) ;
return error ;
}
2008-01-30 15:30:51 +03:00
# ifdef PTRACE_SINGLESTEP
# define is_singlestep(request) ((request) == PTRACE_SINGLESTEP)
# else
# define is_singlestep(request) 0
# endif
2008-01-30 15:30:53 +03:00
# ifdef PTRACE_SINGLEBLOCK
# define is_singleblock(request) ((request) == PTRACE_SINGLEBLOCK)
# else
# define is_singleblock(request) 0
# endif
2008-01-30 15:30:51 +03:00
# ifdef PTRACE_SYSEMU
# define is_sysemu_singlestep(request) ((request) == PTRACE_SYSEMU_SINGLESTEP)
# else
# define is_sysemu_singlestep(request) 0
# endif
static int ptrace_resume ( struct task_struct * child , long request , long data )
{
if ( ! valid_signal ( data ) )
return - EIO ;
if ( request = = PTRACE_SYSCALL )
set_tsk_thread_flag ( child , TIF_SYSCALL_TRACE ) ;
else
clear_tsk_thread_flag ( child , TIF_SYSCALL_TRACE ) ;
# ifdef TIF_SYSCALL_EMU
if ( request = = PTRACE_SYSEMU | | request = = PTRACE_SYSEMU_SINGLESTEP )
set_tsk_thread_flag ( child , TIF_SYSCALL_EMU ) ;
else
clear_tsk_thread_flag ( child , TIF_SYSCALL_EMU ) ;
# endif
2008-01-30 15:30:53 +03:00
if ( is_singleblock ( request ) ) {
if ( unlikely ( ! arch_has_block_step ( ) ) )
return - EIO ;
user_enable_block_step ( child ) ;
} else if ( is_singlestep ( request ) | | is_sysemu_singlestep ( request ) ) {
2008-01-30 15:30:51 +03:00
if ( unlikely ( ! arch_has_single_step ( ) ) )
return - EIO ;
user_enable_single_step ( child ) ;
}
else
user_disable_single_step ( child ) ;
child - > exit_code = data ;
wake_up_process ( child ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
int ptrace_request ( struct task_struct * child , long request ,
long addr , long data )
{
int ret = - EIO ;
2008-04-21 00:10:12 +04:00
siginfo_t siginfo ;
2005-04-17 02:20:36 +04:00
switch ( request ) {
2008-01-30 15:31:47 +03:00
case PTRACE_PEEKTEXT :
case PTRACE_PEEKDATA :
return generic_ptrace_peekdata ( child , addr , data ) ;
case PTRACE_POKETEXT :
case PTRACE_POKEDATA :
return generic_ptrace_pokedata ( child , addr , data ) ;
2005-04-17 02:20:36 +04:00
# ifdef PTRACE_OLDSETOPTIONS
case PTRACE_OLDSETOPTIONS :
# endif
case PTRACE_SETOPTIONS :
ret = ptrace_setoptions ( child , data ) ;
break ;
case PTRACE_GETEVENTMSG :
ret = put_user ( child - > ptrace_message , ( unsigned long __user * ) data ) ;
break ;
2008-04-21 00:10:12 +04:00
2005-04-17 02:20:36 +04:00
case PTRACE_GETSIGINFO :
2008-04-21 00:10:12 +04:00
ret = ptrace_getsiginfo ( child , & siginfo ) ;
if ( ! ret )
ret = copy_siginfo_to_user ( ( siginfo_t __user * ) data ,
& siginfo ) ;
2005-04-17 02:20:36 +04:00
break ;
2008-04-21 00:10:12 +04:00
2005-04-17 02:20:36 +04:00
case PTRACE_SETSIGINFO :
2008-04-21 00:10:12 +04:00
if ( copy_from_user ( & siginfo , ( siginfo_t __user * ) data ,
sizeof siginfo ) )
ret = - EFAULT ;
else
ret = ptrace_setsiginfo ( child , & siginfo ) ;
2005-04-17 02:20:36 +04:00
break ;
2008-04-21 00:10:12 +04:00
2007-10-16 12:23:45 +04:00
case PTRACE_DETACH : /* detach a process that was attached. */
ret = ptrace_detach ( child , data ) ;
break ;
2008-01-30 15:30:51 +03:00
# ifdef PTRACE_SINGLESTEP
case PTRACE_SINGLESTEP :
# endif
2008-01-30 15:30:53 +03:00
# ifdef PTRACE_SINGLEBLOCK
case PTRACE_SINGLEBLOCK :
# endif
2008-01-30 15:30:51 +03:00
# ifdef PTRACE_SYSEMU
case PTRACE_SYSEMU :
case PTRACE_SYSEMU_SINGLESTEP :
# endif
case PTRACE_SYSCALL :
case PTRACE_CONT :
return ptrace_resume ( child , request , data ) ;
case PTRACE_KILL :
if ( child - > exit_state ) /* already dead */
return 0 ;
return ptrace_resume ( child , request , SIGKILL ) ;
2005-04-17 02:20:36 +04:00
default :
break ;
}
return ret ;
}
2005-11-07 11:59:47 +03:00
2006-01-08 12:02:33 +03:00
/**
* ptrace_traceme - - helper for PTRACE_TRACEME
*
* Performs checks and sets PT_PTRACED .
* Should be used by all ptrace implementations for PTRACE_TRACEME .
*/
int ptrace_traceme ( void )
2005-11-07 11:59:47 +03:00
{
2006-05-07 21:49:33 +04:00
int ret = - EPERM ;
2005-11-07 11:59:47 +03:00
/*
2006-01-08 12:02:33 +03:00
* Are we already being traced ?
*/
2006-05-07 21:49:33 +04:00
task_lock ( current ) ;
if ( ! ( current - > ptrace & PT_PTRACED ) ) {
ret = security_ptrace ( current - > parent , current ) ;
/*
* Set the ptrace bit in the process ptrace flags .
*/
if ( ! ret )
current - > ptrace | = PT_PTRACED ;
}
task_unlock ( current ) ;
return ret ;
2006-01-08 12:02:33 +03:00
}
2005-11-07 11:59:47 +03:00
2006-01-08 12:02:33 +03:00
/**
* ptrace_get_task_struct - - grab a task struct reference for ptrace
* @ pid : process id to grab a task_struct reference of
*
* This function is a helper for ptrace implementations . It checks
* permissions and then grabs a task struct for use of the actual
* ptrace implementation .
*
* Returns the task_struct for @ pid or an ERR_PTR ( ) on failure .
*/
struct task_struct * ptrace_get_task_struct ( pid_t pid )
{
struct task_struct * child ;
2005-11-07 11:59:47 +03:00
read_lock ( & tasklist_lock ) ;
2007-10-19 10:40:16 +04:00
child = find_task_by_vpid ( pid ) ;
2005-11-07 11:59:47 +03:00
if ( child )
get_task_struct ( child ) ;
2006-09-29 13:00:07 +04:00
2005-11-07 11:59:47 +03:00
read_unlock ( & tasklist_lock ) ;
if ( ! child )
2006-01-08 12:02:33 +03:00
return ERR_PTR ( - ESRCH ) ;
return child ;
2005-11-07 11:59:47 +03:00
}
2007-10-16 12:26:37 +04:00
# ifndef arch_ptrace_attach
# define arch_ptrace_attach(child) do { } while (0)
# endif
2006-01-08 12:02:33 +03:00
# ifndef __ARCH_SYS_PTRACE
2005-11-07 11:59:47 +03:00
asmlinkage long sys_ptrace ( long request , long pid , long addr , long data )
{
struct task_struct * child ;
long ret ;
/*
* This lock_kernel fixes a subtle race with suid exec
*/
lock_kernel ( ) ;
2006-01-08 12:02:33 +03:00
if ( request = = PTRACE_TRACEME ) {
ret = ptrace_traceme ( ) ;
2007-11-27 15:02:40 +03:00
if ( ! ret )
arch_ptrace_attach ( current ) ;
2005-11-07 11:59:47 +03:00
goto out ;
2006-01-08 12:02:33 +03:00
}
child = ptrace_get_task_struct ( pid ) ;
if ( IS_ERR ( child ) ) {
ret = PTR_ERR ( child ) ;
goto out ;
}
2005-11-07 11:59:47 +03:00
if ( request = = PTRACE_ATTACH ) {
ret = ptrace_attach ( child ) ;
2007-10-16 12:26:37 +04:00
/*
* Some architectures need to do book - keeping after
* a ptrace attach .
*/
if ( ! ret )
arch_ptrace_attach ( child ) ;
2005-11-14 03:06:33 +03:00
goto out_put_task_struct ;
2005-11-07 11:59:47 +03:00
}
ret = ptrace_check_attach ( child , request = = PTRACE_KILL ) ;
if ( ret < 0 )
goto out_put_task_struct ;
ret = arch_ptrace ( child , request , addr , data ) ;
if ( ret < 0 )
goto out_put_task_struct ;
out_put_task_struct :
put_task_struct ( child ) ;
out :
unlock_kernel ( ) ;
return ret ;
}
# endif /* __ARCH_SYS_PTRACE */
2007-07-17 15:03:43 +04:00
int generic_ptrace_peekdata ( struct task_struct * tsk , long addr , long data )
{
unsigned long tmp ;
int copied ;
copied = access_process_vm ( tsk , addr , & tmp , sizeof ( tmp ) , 0 ) ;
if ( copied ! = sizeof ( tmp ) )
return - EIO ;
return put_user ( tmp , ( unsigned long __user * ) data ) ;
}
2007-07-17 15:03:44 +04:00
int generic_ptrace_pokedata ( struct task_struct * tsk , long addr , long data )
{
int copied ;
copied = access_process_vm ( tsk , addr , & data , sizeof ( data ) , 1 ) ;
return ( copied = = sizeof ( data ) ) ? 0 : - EIO ;
}
2008-01-30 15:31:47 +03:00
2008-04-29 00:57:19 +04:00
# if defined CONFIG_COMPAT && defined __ARCH_WANT_COMPAT_SYS_PTRACE
2008-01-30 15:31:47 +03:00
# include <linux/compat.h>
int compat_ptrace_request ( struct task_struct * child , compat_long_t request ,
compat_ulong_t addr , compat_ulong_t data )
{
compat_ulong_t __user * datap = compat_ptr ( data ) ;
compat_ulong_t word ;
2008-04-21 00:10:12 +04:00
siginfo_t siginfo ;
2008-01-30 15:31:47 +03:00
int ret ;
switch ( request ) {
case PTRACE_PEEKTEXT :
case PTRACE_PEEKDATA :
ret = access_process_vm ( child , addr , & word , sizeof ( word ) , 0 ) ;
if ( ret ! = sizeof ( word ) )
ret = - EIO ;
else
ret = put_user ( word , datap ) ;
break ;
case PTRACE_POKETEXT :
case PTRACE_POKEDATA :
ret = access_process_vm ( child , addr , & data , sizeof ( data ) , 1 ) ;
ret = ( ret ! = sizeof ( data ) ? - EIO : 0 ) ;
break ;
case PTRACE_GETEVENTMSG :
ret = put_user ( ( compat_ulong_t ) child - > ptrace_message , datap ) ;
break ;
2008-04-21 00:10:12 +04:00
case PTRACE_GETSIGINFO :
ret = ptrace_getsiginfo ( child , & siginfo ) ;
if ( ! ret )
ret = copy_siginfo_to_user32 (
( struct compat_siginfo __user * ) datap ,
& siginfo ) ;
break ;
case PTRACE_SETSIGINFO :
memset ( & siginfo , 0 , sizeof siginfo ) ;
if ( copy_siginfo_from_user32 (
& siginfo , ( struct compat_siginfo __user * ) datap ) )
ret = - EFAULT ;
else
ret = ptrace_setsiginfo ( child , & siginfo ) ;
break ;
2008-01-30 15:31:47 +03:00
default :
ret = ptrace_request ( child , request , addr , data ) ;
}
return ret ;
}
2008-01-30 15:31:48 +03:00
asmlinkage long compat_sys_ptrace ( compat_long_t request , compat_long_t pid ,
compat_long_t addr , compat_long_t data )
{
struct task_struct * child ;
long ret ;
/*
* This lock_kernel fixes a subtle race with suid exec
*/
lock_kernel ( ) ;
if ( request = = PTRACE_TRACEME ) {
ret = ptrace_traceme ( ) ;
goto out ;
}
child = ptrace_get_task_struct ( pid ) ;
if ( IS_ERR ( child ) ) {
ret = PTR_ERR ( child ) ;
goto out ;
}
if ( request = = PTRACE_ATTACH ) {
ret = ptrace_attach ( child ) ;
/*
* Some architectures need to do book - keeping after
* a ptrace attach .
*/
if ( ! ret )
arch_ptrace_attach ( child ) ;
goto out_put_task_struct ;
}
ret = ptrace_check_attach ( child , request = = PTRACE_KILL ) ;
if ( ! ret )
ret = compat_arch_ptrace ( child , request , addr , data ) ;
out_put_task_struct :
put_task_struct ( child ) ;
out :
unlock_kernel ( ) ;
return ret ;
}
2008-04-29 00:57:19 +04:00
# endif /* CONFIG_COMPAT && __ARCH_WANT_COMPAT_SYS_PTRACE */