2005-04-16 15:20:36 -07:00
/* By Ross Biro 1/23/92 */
/*
* Pentium III FXSR , SSE support
* Gareth Hughes < gareth @ valinux . com > , May 2000
2008-01-30 13:31:09 +01:00
*
* BTS tracing
* Markus Metzger < markus . t . metzger @ intel . com > , Dec 2007
2005-04-16 15:20:36 -07:00
*/
# include <linux/kernel.h>
# include <linux/sched.h>
# include <linux/mm.h>
# include <linux/smp.h>
# include <linux/errno.h>
# include <linux/ptrace.h>
2008-01-30 13:31:52 +01:00
# include <linux/regset.h>
2008-03-16 23:36:28 -07:00
# include <linux/tracehook.h>
2005-04-16 15:20:36 -07:00
# include <linux/user.h>
2008-01-30 13:31:53 +01:00
# include <linux/elf.h>
2005-04-16 15:20:36 -07:00
# include <linux/security.h>
# include <linux/audit.h>
# include <linux/seccomp.h>
2005-05-01 08:59:14 -07:00
# include <linux/signal.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
# include <asm/pgtable.h>
# include <asm/system.h>
# include <asm/processor.h>
# include <asm/i387.h>
# include <asm/debugreg.h>
# include <asm/ldt.h>
# include <asm/desc.h>
2008-01-30 13:31:01 +01:00
# include <asm/prctl.h>
# include <asm/proto.h>
2008-01-30 13:31:09 +01:00
# include <asm/ds.h>
2009-04-08 20:40:59 +02:00
# include <trace/syscall.h>
2008-01-30 13:31:53 +01:00
# include "tls.h"
enum x86_regset {
REGSET_GENERAL ,
REGSET_FP ,
REGSET_XFP ,
2008-08-08 15:58:39 -07:00
REGSET_IOPERM64 = REGSET_XFP ,
2008-01-30 13:31:53 +01:00
REGSET_TLS ,
2008-08-08 15:58:39 -07:00
REGSET_IOPERM32 ,
2008-01-30 13:31:53 +01:00
} ;
2008-01-30 13:31:09 +01:00
2005-04-16 15:20:36 -07:00
/*
* does not yet catch signals sent when the child dies .
* in exit . c or in signal . c .
*/
2006-01-05 23:11:29 -05:00
/*
* Determines which flags the user has access to [ 1 = access , 0 = no access ] .
*/
2008-01-30 13:31:01 +01:00
# define FLAG_MASK_32 ((unsigned long) \
( X86_EFLAGS_CF | X86_EFLAGS_PF | \
X86_EFLAGS_AF | X86_EFLAGS_ZF | \
X86_EFLAGS_SF | X86_EFLAGS_TF | \
X86_EFLAGS_DF | X86_EFLAGS_OF | \
X86_EFLAGS_RF | X86_EFLAGS_AC ) )
2008-01-30 13:31:01 +01:00
/*
* Determines whether a value may be installed in a segment register .
*/
static inline bool invalid_selector ( u16 value )
{
return unlikely ( value ! = 0 & & ( value & SEGMENT_RPL_MASK ) ! = USER_RPL ) ;
}
# ifdef CONFIG_X86_32
2008-01-30 13:31:01 +01:00
# define FLAG_MASK FLAG_MASK_32
2005-04-16 15:20:36 -07:00
2008-07-25 10:19:27 +05:30
static unsigned long * pt_regs_access ( struct pt_regs * regs , unsigned long regno )
2005-04-16 15:20:36 -07:00
{
2008-01-30 13:30:56 +01:00
BUILD_BUG_ON ( offsetof ( struct pt_regs , bx ) ! = 0 ) ;
2009-02-09 22:17:40 +09:00
return & regs - > bx + ( regno > > 2 ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:31:01 +01:00
static u16 get_segment_reg ( struct task_struct * task , unsigned long offset )
2005-04-16 15:20:36 -07:00
{
2008-01-30 13:31:01 +01:00
/*
* Returning the value truncates it to 16 bits .
*/
unsigned int retval ;
if ( offset ! = offsetof ( struct user_regs_struct , gs ) )
retval = * pt_regs_access ( task_pt_regs ( task ) , offset ) ;
else {
if ( task = = current )
2009-02-09 22:17:40 +09:00
retval = get_user_gs ( task_pt_regs ( task ) ) ;
else
retval = task_user_gs ( task ) ;
2008-01-30 13:31:01 +01:00
}
return retval ;
}
static int set_segment_reg ( struct task_struct * task ,
unsigned long offset , u16 value )
{
/*
* The value argument was already truncated to 16 bits .
*/
2008-01-30 13:31:01 +01:00
if ( invalid_selector ( value ) )
2008-01-30 13:31:01 +01:00
return - EIO ;
2008-02-06 22:39:44 +01:00
/*
* For % cs and % ss we cannot permit a null selector .
* We can permit a bogus selector as long as it has USER_RPL .
* Null selectors are fine for other segment registers , but
* we will never get back to user mode with invalid % cs or % ss
* and will take the trap in iret instead . Much code relies
* on user_mode ( ) to distinguish a user trap frame ( which can
* safely use invalid selectors ) from a kernel trap frame .
*/
switch ( offset ) {
case offsetof ( struct user_regs_struct , cs ) :
case offsetof ( struct user_regs_struct , ss ) :
if ( unlikely ( value = = 0 ) )
return - EIO ;
default :
2008-01-30 13:31:01 +01:00
* pt_regs_access ( task_pt_regs ( task ) , offset ) = value ;
2008-02-06 22:39:44 +01:00
break ;
case offsetof ( struct user_regs_struct , gs ) :
2008-01-30 13:31:01 +01:00
if ( task = = current )
2009-02-09 22:17:40 +09:00
set_user_gs ( task_pt_regs ( task ) , value ) ;
else
task_user_gs ( task ) = value ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:31:01 +01:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-01-30 13:31:01 +01:00
static unsigned long debugreg_addr_limit ( struct task_struct * task )
{
return TASK_SIZE - 3 ;
}
# else /* CONFIG_X86_64 */
# define FLAG_MASK (FLAG_MASK_32 | X86_EFLAGS_NT)
static unsigned long * pt_regs_access ( struct pt_regs * regs , unsigned long offset )
{
BUILD_BUG_ON ( offsetof ( struct pt_regs , r15 ) ! = 0 ) ;
return & regs - > r15 + ( offset / sizeof ( regs - > r15 ) ) ;
}
static u16 get_segment_reg ( struct task_struct * task , unsigned long offset )
{
/*
* Returning the value truncates it to 16 bits .
*/
unsigned int seg ;
switch ( offset ) {
case offsetof ( struct user_regs_struct , fs ) :
if ( task = = current ) {
/* Older gas can't assemble movq %?s,%r?? */
asm ( " movl %%fs,%0 " : " =r " ( seg ) ) ;
return seg ;
}
return task - > thread . fsindex ;
case offsetof ( struct user_regs_struct , gs ) :
if ( task = = current ) {
asm ( " movl %%gs,%0 " : " =r " ( seg ) ) ;
return seg ;
}
return task - > thread . gsindex ;
case offsetof ( struct user_regs_struct , ds ) :
if ( task = = current ) {
asm ( " movl %%ds,%0 " : " =r " ( seg ) ) ;
return seg ;
}
return task - > thread . ds ;
case offsetof ( struct user_regs_struct , es ) :
if ( task = = current ) {
asm ( " movl %%es,%0 " : " =r " ( seg ) ) ;
return seg ;
}
return task - > thread . es ;
case offsetof ( struct user_regs_struct , cs ) :
case offsetof ( struct user_regs_struct , ss ) :
break ;
}
return * pt_regs_access ( task_pt_regs ( task ) , offset ) ;
}
static int set_segment_reg ( struct task_struct * task ,
unsigned long offset , u16 value )
{
/*
* The value argument was already truncated to 16 bits .
*/
if ( invalid_selector ( value ) )
return - EIO ;
switch ( offset ) {
case offsetof ( struct user_regs_struct , fs ) :
/*
* If this is setting fs as for normal 64 - bit use but
* setting fs_base has implicitly changed it , leave it .
*/
if ( ( value = = FS_TLS_SEL & & task - > thread . fsindex = = 0 & &
task - > thread . fs ! = 0 ) | |
( value = = 0 & & task - > thread . fsindex = = FS_TLS_SEL & &
task - > thread . fs = = 0 ) )
break ;
task - > thread . fsindex = value ;
if ( task = = current )
loadsegment ( fs , task - > thread . fsindex ) ;
break ;
case offsetof ( struct user_regs_struct , gs ) :
/*
* If this is setting gs as for normal 64 - bit use but
* setting gs_base has implicitly changed it , leave it .
*/
if ( ( value = = GS_TLS_SEL & & task - > thread . gsindex = = 0 & &
task - > thread . gs ! = 0 ) | |
( value = = 0 & & task - > thread . gsindex = = GS_TLS_SEL & &
task - > thread . gs = = 0 ) )
break ;
task - > thread . gsindex = value ;
if ( task = = current )
load_gs_index ( task - > thread . gsindex ) ;
break ;
case offsetof ( struct user_regs_struct , ds ) :
task - > thread . ds = value ;
if ( task = = current )
loadsegment ( ds , task - > thread . ds ) ;
break ;
case offsetof ( struct user_regs_struct , es ) :
task - > thread . es = value ;
if ( task = = current )
loadsegment ( es , task - > thread . es ) ;
break ;
/*
* Can ' t actually change these in 64 - bit mode .
*/
case offsetof ( struct user_regs_struct , cs ) :
2008-02-06 22:39:44 +01:00
if ( unlikely ( value = = 0 ) )
return - EIO ;
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_IA32_EMULATION
if ( test_tsk_thread_flag ( task , TIF_IA32 ) )
task_pt_regs ( task ) - > cs = value ;
# endif
2008-01-30 13:31:01 +01:00
break ;
2008-01-30 13:31:01 +01:00
case offsetof ( struct user_regs_struct , ss ) :
2008-02-06 22:39:44 +01:00
if ( unlikely ( value = = 0 ) )
return - EIO ;
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_IA32_EMULATION
if ( test_tsk_thread_flag ( task , TIF_IA32 ) )
task_pt_regs ( task ) - > ss = value ;
# endif
2008-01-30 13:31:01 +01:00
break ;
2008-01-30 13:31:01 +01:00
}
return 0 ;
}
static unsigned long debugreg_addr_limit ( struct task_struct * task )
{
# ifdef CONFIG_IA32_EMULATION
if ( test_tsk_thread_flag ( task , TIF_IA32 ) )
return IA32_PAGE_OFFSET - 3 ;
# endif
2009-02-20 23:32:28 +01:00
return TASK_SIZE_MAX - 7 ;
2008-01-30 13:31:01 +01:00
}
# endif /* CONFIG_X86_32 */
2008-01-30 13:31:01 +01:00
static unsigned long get_flags ( struct task_struct * task )
2005-04-16 15:20:36 -07:00
{
2008-01-30 13:31:01 +01:00
unsigned long retval = task_pt_regs ( task ) - > flags ;
/*
* If the debugger set TF , hide it from the readout .
*/
if ( test_tsk_thread_flag ( task , TIF_FORCED_TF ) )
retval & = ~ X86_EFLAGS_TF ;
2005-04-16 15:20:36 -07:00
return retval ;
}
2008-01-30 13:31:01 +01:00
static int set_flags ( struct task_struct * task , unsigned long value )
{
struct pt_regs * regs = task_pt_regs ( task ) ;
/*
* If the user value contains TF , mark that
* it was not " us " ( the debugger ) that set it .
* If not , make sure it stays set if we had .
*/
if ( value & X86_EFLAGS_TF )
clear_tsk_thread_flag ( task , TIF_FORCED_TF ) ;
else if ( test_tsk_thread_flag ( task , TIF_FORCED_TF ) )
value | = X86_EFLAGS_TF ;
regs - > flags = ( regs - > flags & ~ FLAG_MASK ) | ( value & FLAG_MASK ) ;
return 0 ;
}
static int putreg ( struct task_struct * child ,
unsigned long offset , unsigned long value )
{
switch ( offset ) {
case offsetof ( struct user_regs_struct , cs ) :
case offsetof ( struct user_regs_struct , ds ) :
case offsetof ( struct user_regs_struct , es ) :
case offsetof ( struct user_regs_struct , fs ) :
case offsetof ( struct user_regs_struct , gs ) :
case offsetof ( struct user_regs_struct , ss ) :
return set_segment_reg ( child , offset , value ) ;
case offsetof ( struct user_regs_struct , flags ) :
return set_flags ( child , value ) ;
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_X86_64
2008-03-07 14:56:02 -08:00
/*
* Orig_ax is really just a flag with small positive and
* negative values , so make sure to always sign - extend it
* from 32 bits so that it works correctly regardless of
* whether we come from a 32 - bit environment or not .
*/
case offsetof ( struct user_regs_struct , orig_ax ) :
value = ( long ) ( s32 ) value ;
break ;
2008-01-30 13:31:01 +01:00
case offsetof ( struct user_regs_struct , fs_base ) :
if ( value > = TASK_SIZE_OF ( child ) )
return - EIO ;
/*
* When changing the segment base , use do_arch_prctl
* to set either thread . fs or thread . fsindex and the
* corresponding GDT slot .
*/
if ( child - > thread . fs ! = value )
return do_arch_prctl ( child , ARCH_SET_FS , value ) ;
return 0 ;
case offsetof ( struct user_regs_struct , gs_base ) :
/*
* Exactly the same here as the % fs handling above .
*/
if ( value > = TASK_SIZE_OF ( child ) )
return - EIO ;
if ( child - > thread . gs ! = value )
return do_arch_prctl ( child , ARCH_SET_GS , value ) ;
return 0 ;
# endif
2008-01-30 13:31:01 +01:00
}
* pt_regs_access ( task_pt_regs ( child ) , offset ) = value ;
return 0 ;
}
static unsigned long getreg ( struct task_struct * task , unsigned long offset )
{
switch ( offset ) {
case offsetof ( struct user_regs_struct , cs ) :
case offsetof ( struct user_regs_struct , ds ) :
case offsetof ( struct user_regs_struct , es ) :
case offsetof ( struct user_regs_struct , fs ) :
case offsetof ( struct user_regs_struct , gs ) :
case offsetof ( struct user_regs_struct , ss ) :
return get_segment_reg ( task , offset ) ;
case offsetof ( struct user_regs_struct , flags ) :
return get_flags ( task ) ;
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_X86_64
case offsetof ( struct user_regs_struct , fs_base ) : {
/*
* do_arch_prctl may have used a GDT slot instead of
* the MSR . To userland , it appears the same either
* way , except the % fs segment selector might not be 0.
*/
unsigned int seg = task - > thread . fsindex ;
if ( task - > thread . fs ! = 0 )
return task - > thread . fs ;
if ( task = = current )
asm ( " movl %%fs,%0 " : " =r " ( seg ) ) ;
if ( seg ! = FS_TLS_SEL )
return 0 ;
return get_desc_base ( & task - > thread . tls_array [ FS_TLS ] ) ;
}
case offsetof ( struct user_regs_struct , gs_base ) : {
/*
* Exactly the same here as the % fs handling above .
*/
unsigned int seg = task - > thread . gsindex ;
if ( task - > thread . gs ! = 0 )
return task - > thread . gs ;
if ( task = = current )
asm ( " movl %%gs,%0 " : " =r " ( seg ) ) ;
if ( seg ! = GS_TLS_SEL )
return 0 ;
return get_desc_base ( & task - > thread . tls_array [ GS_TLS ] ) ;
}
# endif
2008-01-30 13:31:01 +01:00
}
return * pt_regs_access ( task_pt_regs ( task ) , offset ) ;
}
2008-01-30 13:31:52 +01:00
static int genregs_get ( struct task_struct * target ,
const struct user_regset * regset ,
unsigned int pos , unsigned int count ,
void * kbuf , void __user * ubuf )
{
if ( kbuf ) {
unsigned long * k = kbuf ;
while ( count > 0 ) {
* k + + = getreg ( target , pos ) ;
count - = sizeof ( * k ) ;
pos + = sizeof ( * k ) ;
}
} else {
unsigned long __user * u = ubuf ;
while ( count > 0 ) {
if ( __put_user ( getreg ( target , pos ) , u + + ) )
return - EFAULT ;
count - = sizeof ( * u ) ;
pos + = sizeof ( * u ) ;
}
}
return 0 ;
}
static int genregs_set ( struct task_struct * target ,
const struct user_regset * regset ,
unsigned int pos , unsigned int count ,
const void * kbuf , const void __user * ubuf )
{
int ret = 0 ;
if ( kbuf ) {
const unsigned long * k = kbuf ;
while ( count > 0 & & ! ret ) {
ret = putreg ( target , pos , * k + + ) ;
count - = sizeof ( * k ) ;
pos + = sizeof ( * k ) ;
}
} else {
const unsigned long __user * u = ubuf ;
while ( count > 0 & & ! ret ) {
unsigned long word ;
ret = __get_user ( word , u + + ) ;
if ( ret )
break ;
ret = putreg ( target , pos , word ) ;
count - = sizeof ( * u ) ;
pos + = sizeof ( * u ) ;
}
}
return ret ;
}
2008-01-30 13:30:52 +01:00
/*
* This function is trivial and will be inlined by the compiler .
* Having it separates the implementation details of debug
* registers from the interface details of ptrace .
*/
static unsigned long ptrace_get_debugreg ( struct task_struct * child , int n )
{
2008-01-30 13:30:59 +01:00
switch ( n ) {
case 0 : return child - > thread . debugreg0 ;
case 1 : return child - > thread . debugreg1 ;
case 2 : return child - > thread . debugreg2 ;
case 3 : return child - > thread . debugreg3 ;
case 6 : return child - > thread . debugreg6 ;
case 7 : return child - > thread . debugreg7 ;
}
return 0 ;
2008-01-30 13:30:52 +01:00
}
static int ptrace_set_debugreg ( struct task_struct * child ,
int n , unsigned long data )
{
2008-01-30 13:30:59 +01:00
int i ;
2008-01-30 13:30:52 +01:00
if ( unlikely ( n = = 4 | | n = = 5 ) )
return - EIO ;
2008-01-30 13:31:01 +01:00
if ( n < 4 & & unlikely ( data > = debugreg_addr_limit ( child ) ) )
2008-01-30 13:30:52 +01:00
return - EIO ;
2008-01-30 13:30:59 +01:00
switch ( n ) {
case 0 : child - > thread . debugreg0 = data ; break ;
case 1 : child - > thread . debugreg1 = data ; break ;
case 2 : child - > thread . debugreg2 = data ; break ;
case 3 : child - > thread . debugreg3 = data ; break ;
case 6 :
2008-01-30 13:31:01 +01:00
if ( ( data & ~ 0xffffffffUL ) ! = 0 )
return - EIO ;
2008-01-30 13:30:59 +01:00
child - > thread . debugreg6 = data ;
break ;
case 7 :
2008-01-30 13:30:52 +01:00
/*
* Sanity - check data . Take one half - byte at once with
* check = ( val > > ( 16 + 4 * i ) ) & 0xf . It contains the
* R / Wi and LENi bits ; bits 0 and 1 are R / Wi , and bits
* 2 and 3 are LENi . Given a list of invalid values ,
* we do mask | = 1 < < invalid_value , so that
* ( mask > > check ) & 1 is a correct test for invalid
* values .
*
* R / Wi contains the type of the breakpoint /
* watchpoint , LENi contains the length of the watched
* data in the watchpoint case .
*
* The invalid values are :
2008-01-30 13:31:01 +01:00
* - LENi = = 0x10 ( undefined ) , so mask | = 0x0f00 . [ 32 - bit ]
2008-01-30 13:30:52 +01:00
* - R / Wi = = 0x10 ( break on I / O reads or writes ) , so
* mask | = 0x4444 .
* - R / Wi = = 0x00 & & LENi ! = 0x00 , so we have mask | =
* 0x1110 .
*
* Finally , mask = 0x0f00 | 0x4444 | 0x1110 = = 0x5f54 .
*
* See the Intel Manual " System Programming Guide " ,
* 15.2 .4
*
* Note that LENi = = 0x10 is defined on x86_64 in long
* mode ( i . e . even for 32 - bit userspace software , but
* 64 - bit kernel ) , so the x86_64 mask value is 0x5454 .
* See the AMD manual no . 24593 ( AMD64 System Programming )
*/
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_X86_32
# define DR7_MASK 0x5f54
# else
# define DR7_MASK 0x5554
# endif
2008-01-30 13:30:52 +01:00
data & = ~ DR_CONTROL_RESERVED ;
for ( i = 0 ; i < 4 ; i + + )
2008-01-30 13:31:01 +01:00
if ( ( DR7_MASK > > ( ( data > > ( 16 + 4 * i ) ) & 0xf ) ) & 1 )
2008-01-30 13:30:52 +01:00
return - EIO ;
2008-01-30 13:30:59 +01:00
child - > thread . debugreg7 = data ;
2008-01-30 13:30:52 +01:00
if ( data )
set_tsk_thread_flag ( child , TIF_DEBUG ) ;
else
clear_tsk_thread_flag ( child , TIF_DEBUG ) ;
2008-01-30 13:30:59 +01:00
break ;
2008-01-30 13:30:52 +01:00
}
return 0 ;
}
2008-08-08 15:58:39 -07:00
/*
* These access the current or another ( stopped ) task ' s io permission
* bitmap for debugging or core dump .
*/
static int ioperm_active ( struct task_struct * target ,
const struct user_regset * regset )
{
return target - > thread . io_bitmap_max / regset - > size ;
}
2008-02-26 09:40:27 +01:00
2008-08-08 15:58:39 -07:00
static int ioperm_get ( struct task_struct * target ,
const struct user_regset * regset ,
unsigned int pos , unsigned int count ,
void * kbuf , void __user * ubuf )
2008-01-30 13:31:09 +01:00
{
2008-08-08 15:58:39 -07:00
if ( ! target - > thread . io_bitmap_ptr )
2008-01-30 13:31:09 +01:00
return - ENXIO ;
2008-08-08 15:58:39 -07:00
return user_regset_copyout ( & pos , & count , & kbuf , & ubuf ,
target - > thread . io_bitmap_ptr ,
0 , IO_BITMAP_BYTES ) ;
}
2008-04-08 11:01:58 +02:00
# ifdef CONFIG_X86_PTRACE_BTS
static int ptrace_bts_read_record ( struct task_struct * child , size_t index ,
2008-01-30 13:31:09 +01:00
struct bts_struct __user * out )
{
2008-12-11 13:49:59 +01:00
const struct bts_trace * trace ;
struct bts_struct bts ;
const unsigned char * at ;
2008-04-08 11:01:58 +02:00
int error ;
2008-01-30 13:31:09 +01:00
2008-12-11 13:49:59 +01:00
trace = ds_read_bts ( child - > bts ) ;
if ( ! trace )
return - EPERM ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
at = trace - > ds . top - ( ( index + 1 ) * trace - > ds . size ) ;
if ( ( void * ) at < trace - > ds . begin )
at + = ( trace - > ds . n * trace - > ds . size ) ;
2008-04-08 11:01:58 +02:00
2008-12-11 13:49:59 +01:00
if ( ! trace - > read )
return - EOPNOTSUPP ;
2008-04-08 11:01:58 +02:00
2008-12-11 13:49:59 +01:00
error = trace - > read ( child - > bts , at , & bts ) ;
2008-04-08 11:01:58 +02:00
if ( error < 0 )
return error ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
if ( copy_to_user ( out , & bts , sizeof ( bts ) ) )
2008-01-30 13:31:09 +01:00
return - EFAULT ;
2008-12-11 13:49:59 +01:00
return sizeof ( bts ) ;
2008-01-30 13:31:09 +01:00
}
2008-01-30 13:31:20 +01:00
static int ptrace_bts_drain ( struct task_struct * child ,
2008-01-30 13:32:03 +01:00
long size ,
2008-01-30 13:31:20 +01:00
struct bts_struct __user * out )
{
2008-12-11 13:49:59 +01:00
const struct bts_trace * trace ;
const unsigned char * at ;
int error , drained = 0 ;
2008-01-30 13:31:09 +01:00
2008-12-11 13:49:59 +01:00
trace = ds_read_bts ( child - > bts ) ;
if ( ! trace )
return - EPERM ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
if ( ! trace - > read )
return - EOPNOTSUPP ;
if ( size < ( trace - > ds . top - trace - > ds . begin ) )
2008-01-30 13:32:03 +01:00
return - EIO ;
2008-12-11 13:49:59 +01:00
for ( at = trace - > ds . begin ; ( void * ) at < trace - > ds . top ;
out + + , drained + + , at + = trace - > ds . size ) {
struct bts_struct bts ;
int error ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
error = trace - > read ( child - > bts , at , & bts ) ;
if ( error < 0 )
return error ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
if ( copy_to_user ( out , & bts , sizeof ( bts ) ) )
2008-01-30 13:31:20 +01:00
return - EFAULT ;
}
2008-12-11 13:49:59 +01:00
memset ( trace - > ds . begin , 0 , trace - > ds . n * trace - > ds . size ) ;
error = ds_reset_bts ( child - > bts ) ;
2008-04-08 11:01:58 +02:00
if ( error < 0 )
return error ;
2008-01-30 13:31:20 +01:00
2008-12-11 13:49:59 +01:00
return drained ;
2008-01-30 13:31:20 +01:00
}
2008-12-19 15:17:02 +01:00
static int ptrace_bts_allocate_buffer ( struct task_struct * child , size_t size )
{
child - > bts_buffer = alloc_locked_buffer ( size ) ;
if ( ! child - > bts_buffer )
return - ENOMEM ;
child - > bts_size = size ;
return 0 ;
}
static void ptrace_bts_free_buffer ( struct task_struct * child )
{
free_locked_buffer ( child - > bts_buffer , child - > bts_size ) ;
child - > bts_buffer = NULL ;
child - > bts_size = 0 ;
}
2008-01-30 13:31:20 +01:00
static int ptrace_bts_config ( struct task_struct * child ,
2008-01-30 13:32:03 +01:00
long cfg_size ,
2008-01-30 13:31:20 +01:00
const struct ptrace_bts_config __user * ucfg )
{
struct ptrace_bts_config cfg ;
2008-12-11 13:49:59 +01:00
unsigned int flags = 0 ;
2008-01-30 13:31:20 +01:00
2008-01-30 13:32:03 +01:00
if ( cfg_size < sizeof ( cfg ) )
2008-12-11 13:49:59 +01:00
return - EIO ;
2008-01-30 13:32:03 +01:00
2008-01-30 13:31:20 +01:00
if ( copy_from_user ( & cfg , ucfg , sizeof ( cfg ) ) )
2008-12-11 13:49:59 +01:00
return - EFAULT ;
2008-11-25 09:05:27 +01:00
2008-12-11 13:49:59 +01:00
if ( child - > bts ) {
ds_release_bts ( child - > bts ) ;
child - > bts = NULL ;
}
2008-04-08 11:01:58 +02:00
2008-12-11 13:49:59 +01:00
if ( cfg . flags & PTRACE_BTS_O_SIGNAL ) {
if ( ! cfg . signal )
return - EINVAL ;
2008-11-25 09:01:25 +01:00
2008-12-11 13:49:59 +01:00
child - > thread . bts_ovfl_signal = cfg . signal ;
2009-03-13 15:56:58 +08:00
return - EOPNOTSUPP ;
2008-12-11 13:49:59 +01:00
}
2008-11-25 09:05:27 +01:00
2008-12-11 13:49:59 +01:00
if ( ( cfg . flags & PTRACE_BTS_O_ALLOC ) & &
( cfg . size ! = child - > bts_size ) ) {
2008-12-19 15:17:02 +01:00
int error ;
2008-11-25 09:05:27 +01:00
2008-12-19 15:17:02 +01:00
ptrace_bts_free_buffer ( child ) ;
error = ptrace_bts_allocate_buffer ( child , cfg . size ) ;
if ( error < 0 )
return error ;
2008-01-30 13:31:20 +01:00
}
2008-01-30 13:32:03 +01:00
if ( cfg . flags & PTRACE_BTS_O_TRACE )
2008-12-11 13:49:59 +01:00
flags | = BTS_USER ;
2008-01-30 13:31:09 +01:00
2008-01-30 13:32:03 +01:00
if ( cfg . flags & PTRACE_BTS_O_SCHED )
2008-12-11 13:49:59 +01:00
flags | = BTS_TIMESTAMPS ;
2008-01-30 13:31:09 +01:00
2008-12-11 13:49:59 +01:00
child - > bts = ds_request_bts ( child , child - > bts_buffer , child - > bts_size ,
/* ovfl = */ NULL , /* th = */ ( size_t ) - 1 ,
flags ) ;
if ( IS_ERR ( child - > bts ) ) {
int error = PTR_ERR ( child - > bts ) ;
2008-01-30 13:32:03 +01:00
2008-12-19 15:17:02 +01:00
ptrace_bts_free_buffer ( child ) ;
2008-12-11 13:49:59 +01:00
child - > bts = NULL ;
2008-01-30 13:32:03 +01:00
2008-12-11 13:49:59 +01:00
return error ;
}
2008-01-30 13:32:03 +01:00
2008-12-11 13:49:59 +01:00
return sizeof ( cfg ) ;
2008-01-30 13:31:09 +01:00
}
2008-01-30 13:31:20 +01:00
static int ptrace_bts_status ( struct task_struct * child ,
2008-01-30 13:32:03 +01:00
long cfg_size ,
2008-01-30 13:31:20 +01:00
struct ptrace_bts_config __user * ucfg )
2008-01-30 13:31:09 +01:00
{
2008-12-11 13:49:59 +01:00
const struct bts_trace * trace ;
2008-01-30 13:31:20 +01:00
struct ptrace_bts_config cfg ;
2008-01-30 13:31:09 +01:00
2008-01-30 13:32:03 +01:00
if ( cfg_size < sizeof ( cfg ) )
return - EIO ;
2008-12-11 13:49:59 +01:00
trace = ds_read_bts ( child - > bts ) ;
if ( ! trace )
return - EPERM ;
2008-01-30 13:31:09 +01:00
2008-04-08 11:01:58 +02:00
memset ( & cfg , 0 , sizeof ( cfg ) ) ;
2008-12-11 13:49:59 +01:00
cfg . size = trace - > ds . end - trace - > ds . begin ;
2008-04-08 11:01:58 +02:00
cfg . signal = child - > thread . bts_ovfl_signal ;
cfg . bts_size = sizeof ( struct bts_struct ) ;
2008-01-30 13:31:09 +01:00
2008-04-08 11:01:58 +02:00
if ( cfg . signal )
cfg . flags | = PTRACE_BTS_O_SIGNAL ;
2008-01-30 13:31:09 +01:00
2008-12-11 13:49:59 +01:00
if ( trace - > ds . flags & BTS_USER )
2008-04-08 11:01:58 +02:00
cfg . flags | = PTRACE_BTS_O_TRACE ;
2008-12-11 13:49:59 +01:00
if ( trace - > ds . flags & BTS_TIMESTAMPS )
2008-04-08 11:01:58 +02:00
cfg . flags | = PTRACE_BTS_O_SCHED ;
2008-01-30 13:32:54 +01:00
2008-01-30 13:31:20 +01:00
if ( copy_to_user ( ucfg , & cfg , sizeof ( cfg ) ) )
return - EFAULT ;
2008-01-30 13:31:09 +01:00
2008-01-30 13:31:20 +01:00
return sizeof ( cfg ) ;
2008-01-30 13:31:09 +01:00
}
2008-12-11 13:49:59 +01:00
static int ptrace_bts_clear ( struct task_struct * child )
2008-03-04 15:05:39 -08:00
{
2008-12-11 13:49:59 +01:00
const struct bts_trace * trace ;
2008-03-04 15:05:39 -08:00
2008-12-11 13:49:59 +01:00
trace = ds_read_bts ( child - > bts ) ;
if ( ! trace )
return - EPERM ;
2008-03-04 15:05:39 -08:00
2008-12-11 13:49:59 +01:00
memset ( trace - > ds . begin , 0 , trace - > ds . n * trace - > ds . size ) ;
2008-03-04 15:05:39 -08:00
2008-12-11 13:49:59 +01:00
return ds_reset_bts ( child - > bts ) ;
2008-03-04 15:05:39 -08:00
}
2008-12-11 13:49:59 +01:00
static int ptrace_bts_size ( struct task_struct * child )
2008-01-30 13:31:09 +01:00
{
2008-12-11 13:49:59 +01:00
const struct bts_trace * trace ;
2008-04-08 11:01:58 +02:00
2008-12-11 13:49:59 +01:00
trace = ds_read_bts ( child - > bts ) ;
if ( ! trace )
return - EPERM ;
2008-04-08 11:01:58 +02:00
2008-12-11 13:49:59 +01:00
return ( trace - > ds . top - trace - > ds . begin ) / trace - > ds . size ;
2008-04-08 11:01:58 +02:00
}
2008-12-19 15:10:24 +01:00
static void ptrace_bts_fork ( struct task_struct * tsk )
{
tsk - > bts = NULL ;
tsk - > bts_buffer = NULL ;
tsk - > bts_size = 0 ;
tsk - > thread . bts_ovfl_signal = 0 ;
}
static void ptrace_bts_untrace ( struct task_struct * child )
{
if ( unlikely ( child - > bts ) ) {
ds_release_bts ( child - > bts ) ;
child - > bts = NULL ;
2008-12-19 15:17:02 +01:00
/* We cannot update total_vm and locked_vm since
child ' s mm is already gone . But we can reclaim the
memory . */
2008-12-19 15:10:24 +01:00
kfree ( child - > bts_buffer ) ;
child - > bts_buffer = NULL ;
child - > bts_size = 0 ;
}
}
static void ptrace_bts_detach ( struct task_struct * child )
{
2009-02-11 15:10:27 +01:00
/*
* Ptrace_detach ( ) races with ptrace_untrace ( ) in case
* the child dies and is reaped by another thread .
*
* We only do the memory accounting at this point and
* leave the buffer deallocation and the bts tracer
* release to ptrace_bts_untrace ( ) which will be called
* later on with tasklist_lock held .
*/
release_locked_buffer ( child - > bts_buffer , child - > bts_size ) ;
2008-12-19 15:10:24 +01:00
}
# else
static inline void ptrace_bts_fork ( struct task_struct * tsk ) { }
static inline void ptrace_bts_detach ( struct task_struct * child ) { }
static inline void ptrace_bts_untrace ( struct task_struct * child ) { }
2008-04-08 11:01:58 +02:00
# endif /* CONFIG_X86_PTRACE_BTS */
2008-01-30 13:31:09 +01:00
2008-12-19 15:10:24 +01:00
void x86_ptrace_fork ( struct task_struct * child , unsigned long clone_flags )
{
ptrace_bts_fork ( child ) ;
}
void x86_ptrace_untrace ( struct task_struct * child )
{
ptrace_bts_untrace ( child ) ;
}
2005-04-16 15:20:36 -07:00
/*
* Called by kernel / ptrace . c when detaching . .
*
* Make sure the single step bit is not set .
*/
void ptrace_disable ( struct task_struct * child )
2008-01-30 13:30:58 +01:00
{
2008-01-30 13:30:48 +01:00
user_disable_single_step ( child ) ;
2008-01-30 13:31:01 +01:00
# ifdef TIF_SYSCALL_EMU
2005-09-03 15:57:21 -07:00
clear_tsk_thread_flag ( child , TIF_SYSCALL_EMU ) ;
2008-01-30 13:31:01 +01:00
# endif
2008-12-19 15:10:24 +01:00
ptrace_bts_detach ( child ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:31:54 +01:00
# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
static const struct user_regset_view user_x86_32_view ; /* Initialized below. */
# endif
2005-11-07 00:59:47 -08:00
long arch_ptrace ( struct task_struct * child , long request , long addr , long data )
2005-04-16 15:20:36 -07:00
{
2008-01-30 13:31:54 +01:00
int ret ;
2005-04-16 15:20:36 -07:00
unsigned long __user * datap = ( unsigned long __user * ) data ;
switch ( request ) {
/* read the word at location addr in the USER area. */
case PTRACE_PEEKUSR : {
unsigned long tmp ;
ret = - EIO ;
2008-01-30 13:31:01 +01:00
if ( ( addr & ( sizeof ( data ) - 1 ) ) | | addr < 0 | |
addr > = sizeof ( struct user ) )
2005-04-16 15:20:36 -07:00
break ;
tmp = 0 ; /* Default return condition */
2008-01-30 13:31:01 +01:00
if ( addr < sizeof ( struct user_regs_struct ) )
2005-04-16 15:20:36 -07:00
tmp = getreg ( child , addr ) ;
2008-01-30 13:31:01 +01:00
else if ( addr > = offsetof ( struct user , u_debugreg [ 0 ] ) & &
addr < = offsetof ( struct user , u_debugreg [ 7 ] ) ) {
addr - = offsetof ( struct user , u_debugreg [ 0 ] ) ;
tmp = ptrace_get_debugreg ( child , addr / sizeof ( data ) ) ;
2005-04-16 15:20:36 -07:00
}
ret = put_user ( tmp , datap ) ;
break ;
}
case PTRACE_POKEUSR : /* write the word at location addr in the USER area */
ret = - EIO ;
2008-01-30 13:31:01 +01:00
if ( ( addr & ( sizeof ( data ) - 1 ) ) | | addr < 0 | |
addr > = sizeof ( struct user ) )
2005-04-16 15:20:36 -07:00
break ;
2008-01-30 13:31:01 +01:00
if ( addr < sizeof ( struct user_regs_struct ) )
2005-04-16 15:20:36 -07:00
ret = putreg ( child , addr , data ) ;
2008-01-30 13:31:01 +01:00
else if ( addr > = offsetof ( struct user , u_debugreg [ 0 ] ) & &
addr < = offsetof ( struct user , u_debugreg [ 7 ] ) ) {
addr - = offsetof ( struct user , u_debugreg [ 0 ] ) ;
ret = ptrace_set_debugreg ( child ,
addr / sizeof ( data ) , data ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:31:01 +01:00
break ;
2005-04-16 15:20:36 -07:00
2008-01-30 13:31:54 +01:00
case PTRACE_GETREGS : /* Get all gp regs from the child. */
return copy_regset_to_user ( child ,
task_user_regset_view ( current ) ,
REGSET_GENERAL ,
0 , sizeof ( struct user_regs_struct ) ,
datap ) ;
case PTRACE_SETREGS : /* Set all gp regs in the child. */
return copy_regset_from_user ( child ,
task_user_regset_view ( current ) ,
REGSET_GENERAL ,
0 , sizeof ( struct user_regs_struct ) ,
datap ) ;
case PTRACE_GETFPREGS : /* Get the child FPU state. */
return copy_regset_to_user ( child ,
task_user_regset_view ( current ) ,
REGSET_FP ,
0 , sizeof ( struct user_i387_struct ) ,
datap ) ;
case PTRACE_SETFPREGS : /* Set the child FPU state. */
return copy_regset_from_user ( child ,
task_user_regset_view ( current ) ,
REGSET_FP ,
0 , sizeof ( struct user_i387_struct ) ,
datap ) ;
2005-04-16 15:20:36 -07:00
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_X86_32
2008-01-30 13:31:54 +01:00
case PTRACE_GETFPXREGS : /* Get the child extended FPU state. */
return copy_regset_to_user ( child , & user_x86_32_view ,
REGSET_XFP ,
0 , sizeof ( struct user_fxsr_struct ) ,
2008-06-30 14:02:41 -07:00
datap ) ? - EIO : 0 ;
2008-01-30 13:31:54 +01:00
case PTRACE_SETFPXREGS : /* Set the child extended FPU state. */
return copy_regset_from_user ( child , & user_x86_32_view ,
REGSET_XFP ,
0 , sizeof ( struct user_fxsr_struct ) ,
2008-06-30 14:02:41 -07:00
datap ) ? - EIO : 0 ;
2008-01-30 13:31:01 +01:00
# endif
2005-04-16 15:20:36 -07:00
2008-01-30 13:31:01 +01:00
# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
2005-04-16 15:20:36 -07:00
case PTRACE_GET_THREAD_AREA :
2008-01-30 13:30:46 +01:00
if ( addr < 0 )
return - EIO ;
ret = do_get_thread_area ( child , addr ,
( struct user_desc __user * ) data ) ;
2005-04-16 15:20:36 -07:00
break ;
case PTRACE_SET_THREAD_AREA :
2008-01-30 13:30:46 +01:00
if ( addr < 0 )
return - EIO ;
ret = do_set_thread_area ( child , addr ,
( struct user_desc __user * ) data , 0 ) ;
2005-04-16 15:20:36 -07:00
break ;
2008-01-30 13:31:01 +01:00
# endif
# ifdef CONFIG_X86_64
/* normal 64bit interface to access TLS data.
Works just like arch_prctl , except that the arguments
are reversed . */
case PTRACE_ARCH_PRCTL :
ret = do_arch_prctl ( child , data , addr ) ;
break ;
# endif
2005-04-16 15:20:36 -07:00
2008-02-26 09:40:27 +01:00
/*
* These bits need more cooking - not enabled yet :
*/
2008-04-08 11:01:58 +02:00
# ifdef CONFIG_X86_PTRACE_BTS
2008-01-30 13:31:20 +01:00
case PTRACE_BTS_CONFIG :
ret = ptrace_bts_config
2008-01-30 13:32:03 +01:00
( child , data , ( struct ptrace_bts_config __user * ) addr ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-01-30 13:31:20 +01:00
case PTRACE_BTS_STATUS :
ret = ptrace_bts_status
2008-01-30 13:32:03 +01:00
( child , data , ( struct ptrace_bts_config __user * ) addr ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-12-11 13:49:59 +01:00
case PTRACE_BTS_SIZE :
ret = ptrace_bts_size ( child ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-01-30 13:31:20 +01:00
case PTRACE_BTS_GET :
2008-01-30 13:31:09 +01:00
ret = ptrace_bts_read_record
2008-01-30 13:31:20 +01:00
( child , data , ( struct bts_struct __user * ) addr ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-01-30 13:31:20 +01:00
case PTRACE_BTS_CLEAR :
2008-12-11 13:49:59 +01:00
ret = ptrace_bts_clear ( child ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-01-30 13:31:20 +01:00
case PTRACE_BTS_DRAIN :
ret = ptrace_bts_drain
2008-01-30 13:32:03 +01:00
( child , data , ( struct bts_struct __user * ) addr ) ;
2008-01-30 13:31:09 +01:00
break ;
2008-04-08 11:01:58 +02:00
# endif /* CONFIG_X86_PTRACE_BTS */
2008-01-30 13:31:09 +01:00
2005-04-16 15:20:36 -07:00
default :
ret = ptrace_request ( child , request , addr , data ) ;
break ;
}
2008-01-30 13:30:52 +01:00
2005-04-16 15:20:36 -07:00
return ret ;
}
2008-01-30 13:31:01 +01:00
# ifdef CONFIG_IA32_EMULATION
2008-01-30 13:31:01 +01:00
# include <linux/compat.h>
# include <linux/syscalls.h>
# include <asm/ia32.h>
2008-01-30 13:31:01 +01:00
# include <asm/user32.h>
# define R32(l,q) \
case offsetof ( struct user32 , regs . l ) : \
regs - > q = value ; break
# define SEG32(rs) \
case offsetof ( struct user32 , regs . rs ) : \
return set_segment_reg ( child , \
offsetof ( struct user_regs_struct , rs ) , \
value ) ; \
break
static int putreg32 ( struct task_struct * child , unsigned regno , u32 value )
{
struct pt_regs * regs = task_pt_regs ( child ) ;
switch ( regno ) {
SEG32 ( cs ) ;
SEG32 ( ds ) ;
SEG32 ( es ) ;
SEG32 ( fs ) ;
SEG32 ( gs ) ;
SEG32 ( ss ) ;
R32 ( ebx , bx ) ;
R32 ( ecx , cx ) ;
R32 ( edx , dx ) ;
R32 ( edi , di ) ;
R32 ( esi , si ) ;
R32 ( ebp , bp ) ;
R32 ( eax , ax ) ;
R32 ( eip , ip ) ;
R32 ( esp , sp ) ;
2008-02-28 19:57:07 -08:00
case offsetof ( struct user32 , regs . orig_eax ) :
/*
* Sign - extend the value so that orig_eax = - 1
* causes ( long ) orig_ax < 0 tests to fire correctly .
*/
regs - > orig_ax = ( long ) ( s32 ) value ;
break ;
2008-01-30 13:31:01 +01:00
case offsetof ( struct user32 , regs . eflags ) :
return set_flags ( child , value ) ;
case offsetof ( struct user32 , u_debugreg [ 0 ] ) . . .
offsetof ( struct user32 , u_debugreg [ 7 ] ) :
regno - = offsetof ( struct user32 , u_debugreg [ 0 ] ) ;
return ptrace_set_debugreg ( child , regno / 4 , value ) ;
default :
if ( regno > sizeof ( struct user32 ) | | ( regno & 3 ) )
return - EIO ;
/*
* Other dummy fields in the virtual user structure
* are ignored
*/
break ;
}
return 0 ;
}
# undef R32
# undef SEG32
# define R32(l,q) \
case offsetof ( struct user32 , regs . l ) : \
* val = regs - > q ; break
# define SEG32(rs) \
case offsetof ( struct user32 , regs . rs ) : \
* val = get_segment_reg ( child , \
offsetof ( struct user_regs_struct , rs ) ) ; \
break
static int getreg32 ( struct task_struct * child , unsigned regno , u32 * val )
{
struct pt_regs * regs = task_pt_regs ( child ) ;
switch ( regno ) {
SEG32 ( ds ) ;
SEG32 ( es ) ;
SEG32 ( fs ) ;
SEG32 ( gs ) ;
R32 ( cs , cs ) ;
R32 ( ss , ss ) ;
R32 ( ebx , bx ) ;
R32 ( ecx , cx ) ;
R32 ( edx , dx ) ;
R32 ( edi , di ) ;
R32 ( esi , si ) ;
R32 ( ebp , bp ) ;
R32 ( eax , ax ) ;
R32 ( orig_eax , orig_ax ) ;
R32 ( eip , ip ) ;
R32 ( esp , sp ) ;
case offsetof ( struct user32 , regs . eflags ) :
* val = get_flags ( child ) ;
break ;
case offsetof ( struct user32 , u_debugreg [ 0 ] ) . . .
offsetof ( struct user32 , u_debugreg [ 7 ] ) :
regno - = offsetof ( struct user32 , u_debugreg [ 0 ] ) ;
* val = ptrace_get_debugreg ( child , regno / 4 ) ;
break ;
default :
if ( regno > sizeof ( struct user32 ) | | ( regno & 3 ) )
return - EIO ;
/*
* Other dummy fields in the virtual user structure
* are ignored
*/
* val = 0 ;
break ;
}
return 0 ;
}
# undef R32
# undef SEG32
2008-01-30 13:31:52 +01:00
static int genregs32_get ( struct task_struct * target ,
const struct user_regset * regset ,
unsigned int pos , unsigned int count ,
void * kbuf , void __user * ubuf )
{
if ( kbuf ) {
compat_ulong_t * k = kbuf ;
while ( count > 0 ) {
getreg32 ( target , pos , k + + ) ;
count - = sizeof ( * k ) ;
pos + = sizeof ( * k ) ;
}
} else {
compat_ulong_t __user * u = ubuf ;
while ( count > 0 ) {
compat_ulong_t word ;
getreg32 ( target , pos , & word ) ;
if ( __put_user ( word , u + + ) )
return - EFAULT ;
count - = sizeof ( * u ) ;
pos + = sizeof ( * u ) ;
}
}
return 0 ;
}
static int genregs32_set ( struct task_struct * target ,
const struct user_regset * regset ,
unsigned int pos , unsigned int count ,
const void * kbuf , const void __user * ubuf )
{
int ret = 0 ;
if ( kbuf ) {
const compat_ulong_t * k = kbuf ;
while ( count > 0 & & ! ret ) {
2008-02-21 20:37:24 -08:00
ret = putreg32 ( target , pos , * k + + ) ;
2008-01-30 13:31:52 +01:00
count - = sizeof ( * k ) ;
pos + = sizeof ( * k ) ;
}
} else {
const compat_ulong_t __user * u = ubuf ;
while ( count > 0 & & ! ret ) {
compat_ulong_t word ;
ret = __get_user ( word , u + + ) ;
if ( ret )
break ;
2008-02-21 20:37:24 -08:00
ret = putreg32 ( target , pos , word ) ;
2008-01-30 13:31:52 +01:00
count - = sizeof ( * u ) ;
pos + = sizeof ( * u ) ;
}
}
return ret ;
}
2008-04-22 12:21:25 -07:00
long compat_arch_ptrace ( struct task_struct * child , compat_long_t request ,
compat_ulong_t caddr , compat_ulong_t cdata )
2008-01-30 13:31:01 +01:00
{
2008-04-22 12:21:25 -07:00
unsigned long addr = caddr ;
unsigned long data = cdata ;
2008-01-30 13:31:01 +01:00
void __user * datap = compat_ptr ( data ) ;
int ret ;
__u32 val ;
switch ( request ) {
case PTRACE_PEEKUSR :
ret = getreg32 ( child , addr , & val ) ;
if ( ret = = 0 )
ret = put_user ( val , ( __u32 __user * ) datap ) ;
break ;
case PTRACE_POKEUSR :
ret = putreg32 ( child , addr , data ) ;
break ;
2008-01-30 13:31:54 +01:00
case PTRACE_GETREGS : /* Get all gp regs from the child. */
return copy_regset_to_user ( child , & user_x86_32_view ,
REGSET_GENERAL ,
0 , sizeof ( struct user_regs_struct32 ) ,
datap ) ;
case PTRACE_SETREGS : /* Set all gp regs in the child. */
return copy_regset_from_user ( child , & user_x86_32_view ,
REGSET_GENERAL , 0 ,
sizeof ( struct user_regs_struct32 ) ,
datap ) ;
case PTRACE_GETFPREGS : /* Get the child FPU state. */
return copy_regset_to_user ( child , & user_x86_32_view ,
REGSET_FP , 0 ,
sizeof ( struct user_i387_ia32_struct ) ,
datap ) ;
case PTRACE_SETFPREGS : /* Set the child FPU state. */
return copy_regset_from_user (
child , & user_x86_32_view , REGSET_FP ,
0 , sizeof ( struct user_i387_ia32_struct ) , datap ) ;
case PTRACE_GETFPXREGS : /* Get the child extended FPU state. */
return copy_regset_to_user ( child , & user_x86_32_view ,
REGSET_XFP , 0 ,
sizeof ( struct user32_fxsr_struct ) ,
datap ) ;
case PTRACE_SETFPXREGS : /* Set the child extended FPU state. */
return copy_regset_from_user ( child , & user_x86_32_view ,
REGSET_XFP , 0 ,
sizeof ( struct user32_fxsr_struct ) ,
datap ) ;
2008-01-30 13:31:01 +01:00
2008-04-22 12:21:25 -07:00
case PTRACE_GET_THREAD_AREA :
case PTRACE_SET_THREAD_AREA :
2008-12-11 13:49:59 +01:00
# ifdef CONFIG_X86_PTRACE_BTS
case PTRACE_BTS_CONFIG :
case PTRACE_BTS_STATUS :
case PTRACE_BTS_SIZE :
case PTRACE_BTS_GET :
case PTRACE_BTS_CLEAR :
case PTRACE_BTS_DRAIN :
# endif /* CONFIG_X86_PTRACE_BTS */
2008-04-22 12:21:25 -07:00
return arch_ptrace ( child , request , addr , data ) ;
2008-01-30 13:31:01 +01:00
default :
2008-01-30 13:31:56 +01:00
return compat_ptrace_request ( child , request , addr , data ) ;
2008-01-30 13:31:01 +01:00
}
return ret ;
}
2008-01-30 13:31:01 +01:00
# endif /* CONFIG_IA32_EMULATION */
2008-01-30 13:31:53 +01:00
# ifdef CONFIG_X86_64
static const struct user_regset x86_64_regsets [ ] = {
[ REGSET_GENERAL ] = {
. core_note_type = NT_PRSTATUS ,
. n = sizeof ( struct user_regs_struct ) / sizeof ( long ) ,
. size = sizeof ( long ) , . align = sizeof ( long ) ,
. get = genregs_get , . set = genregs_set
} ,
[ REGSET_FP ] = {
. core_note_type = NT_PRFPREG ,
. n = sizeof ( struct user_i387_struct ) / sizeof ( long ) ,
. size = sizeof ( long ) , . align = sizeof ( long ) ,
. active = xfpregs_active , . get = xfpregs_get , . set = xfpregs_set
} ,
2008-08-08 15:58:39 -07:00
[ REGSET_IOPERM64 ] = {
. core_note_type = NT_386_IOPERM ,
. n = IO_BITMAP_LONGS ,
. size = sizeof ( long ) , . align = sizeof ( long ) ,
. active = ioperm_active , . get = ioperm_get
} ,
2008-01-30 13:31:53 +01:00
} ;
static const struct user_regset_view user_x86_64_view = {
. name = " x86_64 " , . e_machine = EM_X86_64 ,
. regsets = x86_64_regsets , . n = ARRAY_SIZE ( x86_64_regsets )
} ;
# else /* CONFIG_X86_32 */
# define user_regs_struct32 user_regs_struct
# define genregs32_get genregs_get
# define genregs32_set genregs_set
2008-05-09 15:43:44 -07:00
# define user_i387_ia32_struct user_i387_struct
# define user32_fxsr_struct user_fxsr_struct
2008-01-30 13:31:53 +01:00
# endif /* CONFIG_X86_64 */
# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
static const struct user_regset x86_32_regsets [ ] = {
[ REGSET_GENERAL ] = {
. core_note_type = NT_PRSTATUS ,
. n = sizeof ( struct user_regs_struct32 ) / sizeof ( u32 ) ,
. size = sizeof ( u32 ) , . align = sizeof ( u32 ) ,
. get = genregs32_get , . set = genregs32_set
} ,
[ REGSET_FP ] = {
. core_note_type = NT_PRFPREG ,
2008-05-09 15:43:44 -07:00
. n = sizeof ( struct user_i387_ia32_struct ) / sizeof ( u32 ) ,
2008-01-30 13:31:53 +01:00
. size = sizeof ( u32 ) , . align = sizeof ( u32 ) ,
. active = fpregs_active , . get = fpregs_get , . set = fpregs_set
} ,
[ REGSET_XFP ] = {
. core_note_type = NT_PRXFPREG ,
2008-05-09 15:43:44 -07:00
. n = sizeof ( struct user32_fxsr_struct ) / sizeof ( u32 ) ,
2008-01-30 13:31:53 +01:00
. size = sizeof ( u32 ) , . align = sizeof ( u32 ) ,
. active = xfpregs_active , . get = xfpregs_get , . set = xfpregs_set
} ,
[ REGSET_TLS ] = {
2008-01-30 13:31:56 +01:00
. core_note_type = NT_386_TLS ,
2008-01-30 13:31:53 +01:00
. n = GDT_ENTRY_TLS_ENTRIES , . bias = GDT_ENTRY_TLS_MIN ,
. size = sizeof ( struct user_desc ) ,
. align = sizeof ( struct user_desc ) ,
. active = regset_tls_active ,
. get = regset_tls_get , . set = regset_tls_set
} ,
2008-08-08 15:58:39 -07:00
[ REGSET_IOPERM32 ] = {
. core_note_type = NT_386_IOPERM ,
. n = IO_BITMAP_BYTES / sizeof ( u32 ) ,
. size = sizeof ( u32 ) , . align = sizeof ( u32 ) ,
. active = ioperm_active , . get = ioperm_get
} ,
2008-01-30 13:31:53 +01:00
} ;
static const struct user_regset_view user_x86_32_view = {
. name = " i386 " , . e_machine = EM_386 ,
. regsets = x86_32_regsets , . n = ARRAY_SIZE ( x86_32_regsets )
} ;
# endif
const struct user_regset_view * task_user_regset_view ( struct task_struct * task )
{
# ifdef CONFIG_IA32_EMULATION
if ( test_tsk_thread_flag ( task , TIF_IA32 ) )
# endif
# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
return & user_x86_32_view ;
# endif
# ifdef CONFIG_X86_64
return & user_x86_64_view ;
# endif
}
2008-09-23 15:23:52 +05:30
void send_sigtrap ( struct task_struct * tsk , struct pt_regs * regs ,
int error_code , int si_code )
2005-04-16 15:20:36 -07:00
{
struct siginfo info ;
tsk - > thread . trap_no = 1 ;
tsk - > thread . error_code = error_code ;
memset ( & info , 0 , sizeof ( info ) ) ;
info . si_signo = SIGTRAP ;
2008-09-23 15:23:52 +05:30
info . si_code = si_code ;
2005-04-16 15:20:36 -07:00
2008-01-30 13:30:56 +01:00
/* User-mode ip? */
info . si_addr = user_mode_vm ( regs ) ? ( void __user * ) regs - > ip : NULL ;
2005-04-16 15:20:36 -07:00
2007-10-20 01:13:56 +02:00
/* Send us the fake SIGTRAP */
2005-04-16 15:20:36 -07:00
force_sig_info ( SIGTRAP , & info , tsk ) ;
}
2008-01-30 13:31:01 +01:00
2008-07-09 02:38:07 -07:00
# ifdef CONFIG_X86_32
# define IS_IA32 1
# elif defined CONFIG_IA32_EMULATION
2009-02-27 19:03:24 -08:00
# define IS_IA32 is_compat_task()
2008-07-09 02:38:07 -07:00
# else
# define IS_IA32 0
# endif
/*
* We must return the syscall number to actually look up in the table .
* This can be - 1L to skip running any syscall at all .
*/
asmregparm long syscall_trace_enter ( struct pt_regs * regs )
2008-01-30 13:31:01 +01:00
{
2008-07-09 02:38:07 -07:00
long ret = 0 ;
2008-07-09 02:39:29 -07:00
/*
* If we stepped into a sysenter / syscall insn , it trapped in
* kernel mode ; do_debug ( ) cleared TF and set TIF_SINGLESTEP .
* If user - mode had set TF itself , then it ' s still clear from
* do_debug ( ) and we need to set it again to restore the user
* state . If we entered on the slow path , TF was already set .
*/
if ( test_thread_flag ( TIF_SINGLESTEP ) )
regs - > flags | = X86_EFLAGS_TF ;
2008-01-30 13:31:01 +01:00
/* do the secure computing check first */
secure_computing ( regs - > orig_ax ) ;
2008-07-09 02:38:07 -07:00
if ( unlikely ( test_thread_flag ( TIF_SYSCALL_EMU ) ) )
ret = - 1L ;
2008-03-16 23:36:28 -07:00
if ( ( ret | | test_thread_flag ( TIF_SYSCALL_TRACE ) ) & &
tracehook_report_syscall_entry ( regs ) )
ret = - 1L ;
2008-01-30 13:31:01 +01:00
2009-03-07 05:53:00 +01:00
if ( unlikely ( test_thread_flag ( TIF_SYSCALL_FTRACE ) ) )
ftrace_syscall_enter ( regs ) ;
2008-01-30 13:31:01 +01:00
if ( unlikely ( current - > audit_context ) ) {
2008-07-09 02:38:07 -07:00
if ( IS_IA32 )
2008-01-30 13:31:01 +01:00
audit_syscall_entry ( AUDIT_ARCH_I386 ,
regs - > orig_ax ,
regs - > bx , regs - > cx ,
regs - > dx , regs - > si ) ;
2008-07-09 02:38:07 -07:00
# ifdef CONFIG_X86_64
else
2008-01-30 13:31:01 +01:00
audit_syscall_entry ( AUDIT_ARCH_X86_64 ,
regs - > orig_ax ,
regs - > di , regs - > si ,
regs - > dx , regs - > r10 ) ;
2008-07-09 02:38:07 -07:00
# endif
2008-01-30 13:31:01 +01:00
}
2008-07-09 02:38:07 -07:00
return ret ? : regs - > orig_ax ;
2008-01-30 13:31:01 +01:00
}
2008-07-09 02:38:07 -07:00
asmregparm void syscall_trace_leave ( struct pt_regs * regs )
2008-01-30 13:31:01 +01:00
{
if ( unlikely ( current - > audit_context ) )
audit_syscall_exit ( AUDITSC_RESULT ( regs - > ax ) , regs - > ax ) ;
2009-03-07 05:53:00 +01:00
if ( unlikely ( test_thread_flag ( TIF_SYSCALL_FTRACE ) ) )
ftrace_syscall_exit ( regs ) ;
2008-07-09 02:38:07 -07:00
if ( test_thread_flag ( TIF_SYSCALL_TRACE ) )
2008-03-16 23:36:28 -07:00
tracehook_report_syscall_exit ( regs , 0 ) ;
2008-01-30 13:31:01 +01:00
2008-07-09 02:38:07 -07:00
/*
* If TIF_SYSCALL_EMU is set , we only get here because of
* TIF_SINGLESTEP ( i . e . this is PTRACE_SYSEMU_SINGLESTEP ) .
* We already reported this syscall instruction in
* syscall_trace_enter ( ) , so don ' t do any more now .
*/
if ( unlikely ( test_thread_flag ( TIF_SYSCALL_EMU ) ) )
return ;
/*
* If we are single - stepping , synthesize a trap to follow the
* system call instruction .
*/
if ( test_thread_flag ( TIF_SINGLESTEP ) & &
2009-04-02 16:58:00 -07:00
tracehook_consider_fatal_signal ( current , SIGTRAP ) )
2008-09-23 15:23:52 +05:30
send_sigtrap ( current , regs , 0 , TRAP_BRKPT ) ;
2008-07-09 02:38:07 -07:00
}