2008-12-16 09:35:40 +09:00
/*
* SH specific backtracing code for oprofile
*
* Copyright 2007 STMicroelectronics Ltd .
*
* Author : Dave Peverley < dpeverley @ mpc - data . co . uk >
*
* Based on ARM oprofile backtrace code by Richard Purdie and in turn , i386
* oprofile backtrace code by John Levon , David Smith
*
* 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 .
*
*/
# include <linux/oprofile.h>
# include <linux/sched.h>
# include <linux/kallsyms.h>
# include <linux/mm.h>
2009-08-11 22:43:20 +01:00
# include <asm/unwinder.h>
2008-12-16 09:35:40 +09:00
# include <asm/ptrace.h>
# include <asm/uaccess.h>
# include <asm/sections.h>
2009-08-07 16:11:19 +01:00
# include <asm/stacktrace.h>
static void backtrace_warning_symbol ( void * data , char * msg ,
unsigned long symbol )
{
/* Ignore warnings */
}
static void backtrace_warning ( void * data , char * msg )
{
/* Ignore warnings */
}
static int backtrace_stack ( void * data , char * name )
{
/* Yes, we want all stacks */
return 0 ;
}
static void backtrace_address ( void * data , unsigned long addr , int reliable )
{
unsigned int * depth = data ;
if ( ( * depth ) - - )
oprofile_add_trace ( addr ) ;
}
static struct stacktrace_ops backtrace_ops = {
. warning = backtrace_warning ,
. warning_symbol = backtrace_warning_symbol ,
. stack = backtrace_stack ,
. address = backtrace_address ,
} ;
2008-12-16 09:35:40 +09:00
/* Limit to stop backtracing too far. */
static int backtrace_limit = 20 ;
static unsigned long *
user_backtrace ( unsigned long * stackaddr , struct pt_regs * regs )
{
unsigned long buf_stack ;
/* Also check accessibility of address */
if ( ! access_ok ( VERIFY_READ , stackaddr , sizeof ( unsigned long ) ) )
return NULL ;
if ( __copy_from_user_inatomic ( & buf_stack , stackaddr , sizeof ( unsigned long ) ) )
return NULL ;
/* Quick paranoia check */
if ( buf_stack & 3 )
return NULL ;
oprofile_add_trace ( buf_stack ) ;
stackaddr + + ;
return stackaddr ;
}
void sh_backtrace ( struct pt_regs * const regs , unsigned int depth )
{
unsigned long * stackaddr ;
/*
* Paranoia - clip max depth as we could get lost in the weeds .
*/
if ( depth > backtrace_limit )
depth = backtrace_limit ;
stackaddr = ( unsigned long * ) regs - > regs [ 15 ] ;
if ( ! user_mode ( regs ) ) {
2009-08-07 16:11:19 +01:00
if ( depth )
2009-08-11 22:43:20 +01:00
unwind_stack ( NULL , regs , stackaddr ,
& backtrace_ops , & depth ) ;
2008-12-16 09:35:40 +09:00
return ;
}
while ( depth - - & & ( stackaddr ! = NULL ) )
stackaddr = user_backtrace ( stackaddr , regs ) ;
}