2008-04-29 13:38:50 +04:00
/* kgdb.c: KGDB support for 32-bit sparc.
*
* Copyright ( C ) 2008 David S . Miller < davem @ davemloft . net >
*/
# include <linux/kgdb.h>
# include <linux/kdebug.h>
# include <asm/kdebug.h>
# include <asm/ptrace.h>
# include <asm/irq.h>
extern unsigned long trapbase ;
void pt_regs_to_gdb_regs ( unsigned long * gdb_regs , struct pt_regs * regs )
{
2009-01-03 06:32:59 +03:00
struct reg_window32 * win ;
2008-04-29 13:38:50 +04:00
int i ;
gdb_regs [ GDB_G0 ] = 0 ;
for ( i = 0 ; i < 15 ; i + + )
gdb_regs [ GDB_G1 + i ] = regs - > u_regs [ UREG_G1 + i ] ;
2009-01-03 06:32:59 +03:00
win = ( struct reg_window32 * ) regs - > u_regs [ UREG_FP ] ;
2008-04-29 13:38:50 +04:00
for ( i = 0 ; i < 8 ; i + + )
gdb_regs [ GDB_L0 + i ] = win - > locals [ i ] ;
for ( i = 0 ; i < 8 ; i + + )
gdb_regs [ GDB_I0 + i ] = win - > ins [ i ] ;
for ( i = GDB_F0 ; i < = GDB_F31 ; i + + )
gdb_regs [ i ] = 0 ;
gdb_regs [ GDB_Y ] = regs - > y ;
gdb_regs [ GDB_PSR ] = regs - > psr ;
gdb_regs [ GDB_WIM ] = 0 ;
gdb_regs [ GDB_TBR ] = ( unsigned long ) & trapbase ;
gdb_regs [ GDB_PC ] = regs - > pc ;
gdb_regs [ GDB_NPC ] = regs - > npc ;
gdb_regs [ GDB_FSR ] = 0 ;
gdb_regs [ GDB_CSR ] = 0 ;
}
void sleeping_thread_to_gdb_regs ( unsigned long * gdb_regs , struct task_struct * p )
{
struct thread_info * t = task_thread_info ( p ) ;
2009-01-03 06:32:59 +03:00
struct reg_window32 * win ;
2008-04-29 13:38:50 +04:00
int i ;
for ( i = GDB_G0 ; i < GDB_G6 ; i + + )
gdb_regs [ i ] = 0 ;
gdb_regs [ GDB_G6 ] = ( unsigned long ) t ;
gdb_regs [ GDB_G7 ] = 0 ;
for ( i = GDB_O0 ; i < GDB_SP ; i + + )
gdb_regs [ i ] = 0 ;
gdb_regs [ GDB_SP ] = t - > ksp ;
gdb_regs [ GDB_O7 ] = 0 ;
2009-01-03 06:32:59 +03:00
win = ( struct reg_window32 * ) t - > ksp ;
2008-04-29 13:38:50 +04:00
for ( i = 0 ; i < 8 ; i + + )
gdb_regs [ GDB_L0 + i ] = win - > locals [ i ] ;
for ( i = 0 ; i < 8 ; i + + )
gdb_regs [ GDB_I0 + i ] = win - > ins [ i ] ;
for ( i = GDB_F0 ; i < = GDB_F31 ; i + + )
gdb_regs [ i ] = 0 ;
gdb_regs [ GDB_Y ] = 0 ;
gdb_regs [ GDB_PSR ] = t - > kpsr ;
gdb_regs [ GDB_WIM ] = t - > kwim ;
gdb_regs [ GDB_TBR ] = ( unsigned long ) & trapbase ;
gdb_regs [ GDB_PC ] = t - > kpc ;
gdb_regs [ GDB_NPC ] = t - > kpc + 4 ;
gdb_regs [ GDB_FSR ] = 0 ;
gdb_regs [ GDB_CSR ] = 0 ;
}
void gdb_regs_to_pt_regs ( unsigned long * gdb_regs , struct pt_regs * regs )
{
2009-01-03 06:32:59 +03:00
struct reg_window32 * win ;
2008-04-29 13:38:50 +04:00
int i ;
for ( i = 0 ; i < 15 ; i + + )
regs - > u_regs [ UREG_G1 + i ] = gdb_regs [ GDB_G1 + i ] ;
/* If the PSR register is changing, we have to preserve
* the CWP field , otherwise window save / restore explodes .
*/
if ( regs - > psr ! = gdb_regs [ GDB_PSR ] ) {
unsigned long cwp = regs - > psr & PSR_CWP ;
regs - > psr = ( gdb_regs [ GDB_PSR ] & ~ PSR_CWP ) | cwp ;
}
regs - > pc = gdb_regs [ GDB_PC ] ;
regs - > npc = gdb_regs [ GDB_NPC ] ;
regs - > y = gdb_regs [ GDB_Y ] ;
2009-01-03 06:32:59 +03:00
win = ( struct reg_window32 * ) regs - > u_regs [ UREG_FP ] ;
2008-04-29 13:38:50 +04:00
for ( i = 0 ; i < 8 ; i + + )
win - > locals [ i ] = gdb_regs [ GDB_L0 + i ] ;
for ( i = 0 ; i < 8 ; i + + )
win - > ins [ i ] = gdb_regs [ GDB_I0 + i ] ;
}
int kgdb_arch_handle_exception ( int e_vector , int signo , int err_code ,
char * remcomInBuffer , char * remcomOutBuffer ,
struct pt_regs * linux_regs )
{
unsigned long addr ;
char * ptr ;
switch ( remcomInBuffer [ 0 ] ) {
case ' c ' :
/* try to read optional parameter, pc unchanged if no parm */
ptr = & remcomInBuffer [ 1 ] ;
if ( kgdb_hex2long ( & ptr , & addr ) ) {
linux_regs - > pc = addr ;
linux_regs - > npc = addr + 4 ;
}
/* fallthru */
case ' D ' :
case ' k ' :
if ( linux_regs - > pc = = ( unsigned long ) arch_kgdb_breakpoint ) {
linux_regs - > pc = linux_regs - > npc ;
linux_regs - > npc + = 4 ;
}
return 0 ;
}
return - 1 ;
}
extern void do_hw_interrupt ( struct pt_regs * regs , unsigned long type ) ;
asmlinkage void kgdb_trap ( struct pt_regs * regs )
{
unsigned long flags ;
if ( user_mode ( regs ) ) {
do_hw_interrupt ( regs , 0xfd ) ;
return ;
}
flushw_all ( ) ;
local_irq_save ( flags ) ;
kgdb_handle_exception ( 0x172 , SIGTRAP , 0 , regs ) ;
local_irq_restore ( flags ) ;
}
int kgdb_arch_init ( void )
{
return 0 ;
}
void kgdb_arch_exit ( void )
{
}
2010-05-21 06:04:19 +04:00
void kgdb_arch_set_pc ( struct pt_regs * regs , unsigned long ip )
{
regs - > pc = ip ;
regs - > npc = regs - > pc + 4 ;
}
2008-04-29 13:38:50 +04:00
struct kgdb_arch arch_kgdb_ops = {
/* Breakpoint instruction: ta 0x7d */
. gdb_bpt_instr = { 0x91 , 0xd0 , 0x20 , 0x7d } ,
} ;