2005-04-17 02:20:36 +04:00
/*
2008-08-02 13:55:55 +04:00
* arch / arm / include / asm / ptrace . h
2005-04-17 02:20:36 +04:00
*
* Copyright ( C ) 1996 - 2003 Russell King
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# ifndef __ASM_ARM_PTRACE_H
# define __ASM_ARM_PTRACE_H
2007-07-11 14:29:39 +04:00
# include <asm/hwcap.h>
2005-04-17 02:20:36 +04:00
# define PTRACE_GETREGS 12
# define PTRACE_SETREGS 13
# define PTRACE_GETFPREGS 14
# define PTRACE_SETFPREGS 15
2007-05-06 17:49:56 +04:00
/* PTRACE_ATTACH is 16 */
/* PTRACE_DETACH is 17 */
2005-04-17 02:20:36 +04:00
# define PTRACE_GETWMMXREGS 18
# define PTRACE_SETWMMXREGS 19
2007-05-06 17:49:56 +04:00
/* 20 is unused */
2005-04-17 02:20:36 +04:00
# define PTRACE_OLDSETOPTIONS 21
# define PTRACE_GET_THREAD_AREA 22
2006-01-14 22:30:04 +03:00
# define PTRACE_SET_SYSCALL 23
2006-06-28 01:56:19 +04:00
/* PTRACE_SYSCALL is 24 */
# define PTRACE_GETCRUNCHREGS 25
# define PTRACE_SETCRUNCHREGS 26
2009-02-11 15:12:56 +03:00
# define PTRACE_GETVFPREGS 27
# define PTRACE_SETVFPREGS 28
2010-09-03 13:42:55 +04:00
# define PTRACE_GETHBPREGS 29
# define PTRACE_SETHBPREGS 30
2006-06-28 01:56:19 +04:00
2005-04-17 02:20:36 +04:00
/*
* PSR bits
*/
# define USR26_MODE 0x00000000
# define FIQ26_MODE 0x00000001
# define IRQ26_MODE 0x00000002
# define SVC26_MODE 0x00000003
# define USR_MODE 0x00000010
# define FIQ_MODE 0x00000011
# define IRQ_MODE 0x00000012
# define SVC_MODE 0x00000013
# define ABT_MODE 0x00000017
# define UND_MODE 0x0000001b
# define SYSTEM_MODE 0x0000001f
# define MODE32_BIT 0x00000010
# define MODE_MASK 0x0000001f
# define PSR_T_BIT 0x00000020
# define PSR_F_BIT 0x00000040
# define PSR_I_BIT 0x00000080
2007-07-11 14:29:39 +04:00
# define PSR_A_BIT 0x00000100
2009-05-30 17:00:18 +04:00
# define PSR_E_BIT 0x00000200
2005-04-17 02:20:36 +04:00
# define PSR_J_BIT 0x01000000
# define PSR_Q_BIT 0x08000000
# define PSR_V_BIT 0x10000000
# define PSR_C_BIT 0x20000000
# define PSR_Z_BIT 0x40000000
# define PSR_N_BIT 0x80000000
/*
* Groups of PSR bits
*/
# define PSR_f 0xff000000 /* Flags */
# define PSR_s 0x00ff0000 /* Status */
# define PSR_x 0x0000ff00 /* Extension */
# define PSR_c 0x000000ff /* Control */
2009-05-30 17:00:15 +04:00
/*
* ARMv7 groups of APSR bits
*/
# define PSR_ISET_MASK 0x01000010 /* ISA state (J, T) mask */
# define PSR_IT_MASK 0x0600fc00 /* If-Then execution state mask */
# define PSR_ENDIAN_MASK 0x00000200 /* Endianness state mask */
2009-05-30 17:00:18 +04:00
/*
* Default endianness state
*/
# ifdef CONFIG_CPU_ENDIAN_BE8
# define PSR_ENDSTATE PSR_E_BIT
# else
# define PSR_ENDSTATE 0
# endif
2009-07-24 15:34:58 +04:00
/*
* These are ' magic ' values for PTRACE_PEEKUSR that return info about where a
* process is located in memory .
*/
# define PT_TEXT_ADDR 0x10000
# define PT_DATA_ADDR 0x10004
# define PT_TEXT_END_ADDR 0x10008
2005-04-17 02:20:36 +04:00
# ifndef __ASSEMBLY__
2006-01-14 19:18:08 +03:00
/*
* This struct defines the way the registers are stored on the
* stack during a system call . Note that sizeof ( struct pt_regs )
* has to be a multiple of 8.
*/
2010-01-06 12:50:08 +03:00
# ifndef __KERNEL__
2005-04-17 02:20:36 +04:00
struct pt_regs {
long uregs [ 18 ] ;
} ;
2010-01-06 12:50:08 +03:00
# else /* __KERNEL__ */
struct pt_regs {
unsigned long uregs [ 18 ] ;
} ;
# endif /* __KERNEL__ */
2005-04-17 02:20:36 +04:00
# define ARM_cpsr uregs[16]
# define ARM_pc uregs[15]
# define ARM_lr uregs[14]
# define ARM_sp uregs[13]
# define ARM_ip uregs[12]
# define ARM_fp uregs[11]
# define ARM_r10 uregs[10]
# define ARM_r9 uregs[9]
# define ARM_r8 uregs[8]
# define ARM_r7 uregs[7]
# define ARM_r6 uregs[6]
# define ARM_r5 uregs[5]
# define ARM_r4 uregs[4]
# define ARM_r3 uregs[3]
# define ARM_r2 uregs[2]
# define ARM_r1 uregs[1]
# define ARM_r0 uregs[0]
# define ARM_ORIG_r0 uregs[17]
# ifdef __KERNEL__
2010-03-11 02:22:50 +03:00
# define arch_has_single_step() (1)
2005-04-17 02:20:36 +04:00
# define user_mode(regs) \
( ( ( regs ) - > ARM_cpsr & 0xf ) = = 0 )
# ifdef CONFIG_ARM_THUMB
# define thumb_mode(regs) \
( ( ( regs ) - > ARM_cpsr & PSR_T_BIT ) )
# else
# define thumb_mode(regs) (0)
# endif
2007-06-26 04:38:27 +04:00
# define isa_mode(regs) \
( ( ( ( regs ) - > ARM_cpsr & PSR_J_BIT ) > > 23 ) | \
( ( ( regs ) - > ARM_cpsr & PSR_T_BIT ) > > 5 ) )
2005-04-17 02:20:36 +04:00
# define processor_mode(regs) \
( ( regs ) - > ARM_cpsr & MODE_MASK )
# define interrupts_enabled(regs) \
( ! ( ( regs ) - > ARM_cpsr & PSR_I_BIT ) )
# define fast_interrupts_enabled(regs) \
( ! ( ( regs ) - > ARM_cpsr & PSR_F_BIT ) )
/* Are the current registers suitable for user mode?
* ( used to maintain security in signal handlers )
*/
static inline int valid_user_regs ( struct pt_regs * regs )
{
2010-08-14 02:33:46 +04:00
unsigned long mode = regs - > ARM_cpsr & MODE_MASK ;
/*
* Always clear the F ( FIQ ) and A ( delayed abort ) bits
*/
regs - > ARM_cpsr & = ~ ( PSR_F_BIT | PSR_A_BIT ) ;
if ( ( regs - > ARM_cpsr & PSR_I_BIT ) = = 0 ) {
if ( mode = = USR_MODE )
return 1 ;
if ( elf_hwcap & HWCAP_26BIT & & mode = = USR26_MODE )
return 1 ;
2007-07-11 14:29:39 +04:00
}
2005-04-17 02:20:36 +04:00
/*
* Force CPSR to something logical . . .
*/
2010-08-14 02:33:46 +04:00
regs - > ARM_cpsr & = PSR_f | PSR_s | PSR_x | PSR_T_BIT | MODE32_BIT ;
2007-07-11 14:29:39 +04:00
if ( ! ( elf_hwcap & HWCAP_26BIT ) )
regs - > ARM_cpsr | = USR_MODE ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2008-09-06 13:14:24 +04:00
# define instruction_pointer(regs) (regs)->ARM_pc
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
extern unsigned long profile_pc ( struct pt_regs * regs ) ;
# else
# define profile_pc(regs) instruction_pointer(regs)
# endif
2005-04-17 18:50:36 +04:00
# define predicate(x) ((x) & 0xf0000000)
2005-04-17 02:20:36 +04:00
# define PREDICATE_ALWAYS 0xe0000000
2008-07-25 12:47:34 +04:00
2010-06-25 15:24:53 +04:00
/*
* kprobe - based event tracer support
*/
# include <linux/stddef.h>
# include <linux/types.h>
# define MAX_REG_OFFSET (offsetof(struct pt_regs, ARM_ORIG_r0))
extern int regs_query_register_offset ( const char * name ) ;
extern const char * regs_query_register_name ( unsigned int offset ) ;
extern bool regs_within_kernel_stack ( struct pt_regs * regs , unsigned long addr ) ;
extern unsigned long regs_get_kernel_stack_nth ( struct pt_regs * regs ,
unsigned int n ) ;
/**
* regs_get_register ( ) - get register value from its offset
* @ regs : pt_regs from which register value is gotten
* @ offset : offset number of the register .
*
* regs_get_register returns the value of a register whose offset from @ regs .
* The @ offset is the offset of the register in struct pt_regs .
* If @ offset is bigger than MAX_REG_OFFSET , this returns 0.
*/
static inline unsigned long regs_get_register ( struct pt_regs * regs ,
unsigned int offset )
{
if ( unlikely ( offset > MAX_REG_OFFSET ) )
return 0 ;
return * ( unsigned long * ) ( ( unsigned long ) regs + offset ) ;
}
/* Valid only for Kernel mode traps. */
static inline unsigned long kernel_stack_pointer ( struct pt_regs * regs )
{
return regs - > ARM_sp ;
}
2008-07-25 12:47:34 +04:00
# endif /* __KERNEL__ */
2005-04-17 02:20:36 +04:00
# endif /* __ASSEMBLY__ */
# endif