| #ifndef _ASM_STACKTRACE_H |
| #define _ASM_STACKTRACE_H |
| |
| #include <asm/ptrace.h> |
| |
| #ifdef CONFIG_KALLSYMS |
| extern int raw_show_trace; |
| extern unsigned long unwind_stack(struct task_struct *task, unsigned long *sp, |
| unsigned long pc, unsigned long *ra); |
| extern unsigned long unwind_stack_by_address(unsigned long stack_page, |
| unsigned long *sp, |
| unsigned long pc, |
| unsigned long *ra); |
| #else |
| #define raw_show_trace 1 |
| static inline unsigned long unwind_stack(struct task_struct *task, |
| unsigned long *sp, unsigned long pc, unsigned long *ra) |
| { |
| return 0; |
| } |
| #endif |
| |
| static __always_inline void prepare_frametrace(struct pt_regs *regs) |
| { |
| #ifndef CONFIG_KALLSYMS |
| /* |
| * Remove any garbage that may be in regs (specially func |
| * addresses) to avoid show_raw_backtrace() to report them |
| */ |
| memset(regs, 0, sizeof(*regs)); |
| #endif |
| __asm__ __volatile__( |
| ".set push\n\t" |
| ".set noat\n\t" |
| #ifdef CONFIG_64BIT |
| "1: dla $1, 1b\n\t" |
| "sd $1, %0\n\t" |
| "sd $29, %1\n\t" |
| "sd $31, %2\n\t" |
| #else |
| "1: la $1, 1b\n\t" |
| "sw $1, %0\n\t" |
| "sw $29, %1\n\t" |
| "sw $31, %2\n\t" |
| #endif |
| ".set pop\n\t" |
| : "=m" (regs->cp0_epc), |
| "=m" (regs->regs[29]), "=m" (regs->regs[31]) |
| : : "memory"); |
| } |
| |
| #endif /* _ASM_STACKTRACE_H */ |