2006-11-16 13:38:57 -08:00
# include <linux/sched.h>
# include <linux/stacktrace.h>
# include <linux/thread_info.h>
2008-07-08 22:15:40 +10:00
# include <linux/module.h>
2006-11-16 13:38:57 -08:00
# include <asm/ptrace.h>
2008-03-24 20:06:24 -07:00
# include <asm/stacktrace.h>
2006-11-16 13:38:57 -08:00
2008-08-12 18:33:56 -07:00
# include "kstack.h"
2007-05-08 00:23:29 -07:00
void save_stack_trace ( struct stack_trace * trace )
2006-11-16 13:38:57 -08:00
{
2007-05-08 00:23:29 -07:00
struct thread_info * tp = task_thread_info ( current ) ;
2008-08-13 17:17:52 -07:00
unsigned long ksp , fp ;
2006-11-16 13:38:57 -08:00
2008-03-24 20:06:24 -07:00
stack_trace_flush ( ) ;
2007-05-08 00:23:29 -07:00
__asm__ __volatile__ (
" mov %%fp, %0 "
: " =r " ( ksp )
) ;
2006-11-16 13:38:57 -08:00
fp = ksp + STACK_BIAS ;
do {
2008-05-21 18:15:53 -07:00
struct sparc_stackf * sf ;
2008-04-24 03:28:52 -07:00
struct pt_regs * regs ;
unsigned long pc ;
2006-11-16 13:38:57 -08:00
2008-08-12 18:33:56 -07:00
if ( ! kstack_valid ( tp , fp ) )
2006-11-16 13:38:57 -08:00
break ;
2008-05-21 18:15:53 -07:00
sf = ( struct sparc_stackf * ) fp ;
regs = ( struct pt_regs * ) ( sf + 1 ) ;
2008-04-24 03:28:52 -07:00
2008-08-12 18:33:56 -07:00
if ( kstack_is_trap_frame ( tp , regs ) ) {
2008-05-21 18:15:53 -07:00
if ( ! ( regs - > tstate & TSTATE_PRIV ) )
break ;
2008-04-24 03:28:52 -07:00
pc = regs - > tpc ;
fp = regs - > u_regs [ UREG_I6 ] + STACK_BIAS ;
} else {
2008-05-21 18:15:53 -07:00
pc = sf - > callers_pc ;
fp = ( unsigned long ) sf - > fp + STACK_BIAS ;
2008-04-24 03:28:52 -07:00
}
2006-11-16 13:38:57 -08:00
if ( trace - > skip > 0 )
trace - > skip - - ;
else
2008-04-24 03:28:52 -07:00
trace - > entries [ trace - > nr_entries + + ] = pc ;
2006-11-16 13:38:57 -08:00
} while ( trace - > nr_entries < trace - > max_entries ) ;
}
2008-07-03 09:17:55 +02:00
EXPORT_SYMBOL_GPL ( save_stack_trace ) ;