mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-10 23:20:05 +00:00
ARM: 9259/1: stacktrace: Convert stacktrace to generic ARCH_STACKWALK
Historically architectures have had duplicated code in their stack trace implementations for filtering what gets traced. In order to avoid this duplication some generic code has been provided using a new interface arch_stack_walk(), enabled by selecting ARCH_STACKWALK in Kconfig, which factors all this out into the generic stack trace code. Convert ARM to use this common infrastructure. When initializing the stack frame of the current task, arm64 uses __builtin_frame_address(1) to initialize the frame pointer, skipping arch_stack_walk(), see the commit c607ab4f916d ("arm64: stacktrace: don't trace arch_stack_walk()"). Since __builtin_frame_address(1) does not work on ARM, unwind_frame() is used to unwind the stack one layer forward before calling walk_stackframe(). Signed-off-by: Li Huafei <lihuafei1@huawei.com> Reviewed-by: Linus Walleij <linus.walleij@linaro.org> Signed-off-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
This commit is contained in:
parent
70ccc7c066
commit
9fbed16c3f
@ -17,6 +17,7 @@ config ARM
|
|||||||
select ARCH_HAS_PTE_SPECIAL if ARM_LPAE
|
select ARCH_HAS_PTE_SPECIAL if ARM_LPAE
|
||||||
select ARCH_HAS_SETUP_DMA_OPS
|
select ARCH_HAS_SETUP_DMA_OPS
|
||||||
select ARCH_HAS_SET_MEMORY
|
select ARCH_HAS_SET_MEMORY
|
||||||
|
select ARCH_STACKWALK
|
||||||
select ARCH_HAS_STRICT_KERNEL_RWX if MMU && !XIP_KERNEL
|
select ARCH_HAS_STRICT_KERNEL_RWX if MMU && !XIP_KERNEL
|
||||||
select ARCH_HAS_STRICT_MODULE_RWX if MMU
|
select ARCH_HAS_STRICT_MODULE_RWX if MMU
|
||||||
select ARCH_HAS_SYNC_DMA_FOR_DEVICE
|
select ARCH_HAS_SYNC_DMA_FOR_DEVICE
|
||||||
|
@ -142,40 +142,32 @@ void notrace walk_stackframe(struct stackframe *frame,
|
|||||||
EXPORT_SYMBOL(walk_stackframe);
|
EXPORT_SYMBOL(walk_stackframe);
|
||||||
|
|
||||||
#ifdef CONFIG_STACKTRACE
|
#ifdef CONFIG_STACKTRACE
|
||||||
struct stack_trace_data {
|
static void start_stack_trace(struct stackframe *frame, struct task_struct *task,
|
||||||
struct stack_trace *trace;
|
unsigned long fp, unsigned long sp,
|
||||||
unsigned int no_sched_functions;
|
unsigned long lr, unsigned long pc)
|
||||||
unsigned int skip;
|
|
||||||
};
|
|
||||||
|
|
||||||
static bool save_trace(void *d, unsigned long addr)
|
|
||||||
{
|
{
|
||||||
struct stack_trace_data *data = d;
|
frame->fp = fp;
|
||||||
struct stack_trace *trace = data->trace;
|
frame->sp = sp;
|
||||||
|
frame->lr = lr;
|
||||||
if (data->no_sched_functions && in_sched_functions(addr))
|
frame->pc = pc;
|
||||||
return true;
|
#ifdef CONFIG_KRETPROBES
|
||||||
if (data->skip) {
|
frame->kr_cur = NULL;
|
||||||
data->skip--;
|
frame->tsk = task;
|
||||||
return true;
|
#endif
|
||||||
}
|
#ifdef CONFIG_UNWINDER_FRAME_POINTER
|
||||||
|
frame->ex_frame = in_entry_text(frame->pc);
|
||||||
trace->entries[trace->nr_entries++] = addr;
|
#endif
|
||||||
return trace->nr_entries < trace->max_entries;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This must be noinline to so that our skip calculation works correctly */
|
void arch_stack_walk(stack_trace_consume_fn consume_entry, void *cookie,
|
||||||
static noinline void __save_stack_trace(struct task_struct *tsk,
|
struct task_struct *task, struct pt_regs *regs)
|
||||||
struct stack_trace *trace, unsigned int nosched)
|
|
||||||
{
|
{
|
||||||
struct stack_trace_data data;
|
|
||||||
struct stackframe frame;
|
struct stackframe frame;
|
||||||
|
|
||||||
data.trace = trace;
|
if (regs) {
|
||||||
data.skip = trace->skip;
|
start_stack_trace(&frame, NULL, regs->ARM_fp, regs->ARM_sp,
|
||||||
data.no_sched_functions = nosched;
|
regs->ARM_lr, regs->ARM_pc);
|
||||||
|
} else if (task != current) {
|
||||||
if (tsk != current) {
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/*
|
/*
|
||||||
* What guarantees do we have here that 'tsk' is not
|
* What guarantees do we have here that 'tsk' is not
|
||||||
@ -184,64 +176,22 @@ static noinline void __save_stack_trace(struct task_struct *tsk,
|
|||||||
*/
|
*/
|
||||||
return;
|
return;
|
||||||
#else
|
#else
|
||||||
frame.fp = thread_saved_fp(tsk);
|
start_stack_trace(&frame, task, thread_saved_fp(task),
|
||||||
frame.sp = thread_saved_sp(tsk);
|
thread_saved_sp(task), 0,
|
||||||
frame.lr = 0; /* recovered from the stack */
|
thread_saved_pc(task));
|
||||||
frame.pc = thread_saved_pc(tsk);
|
|
||||||
#endif
|
#endif
|
||||||
} else {
|
} else {
|
||||||
/* We don't want this function nor the caller */
|
|
||||||
data.skip += 2;
|
|
||||||
frame.fp = (unsigned long)__builtin_frame_address(0);
|
|
||||||
frame.sp = current_stack_pointer;
|
|
||||||
frame.lr = (unsigned long)__builtin_return_address(0);
|
|
||||||
here:
|
here:
|
||||||
frame.pc = (unsigned long)&&here;
|
start_stack_trace(&frame, task,
|
||||||
|
(unsigned long)__builtin_frame_address(0),
|
||||||
|
current_stack_pointer,
|
||||||
|
(unsigned long)__builtin_return_address(0),
|
||||||
|
(unsigned long)&&here);
|
||||||
|
/* skip this function */
|
||||||
|
if (unwind_frame(&frame))
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
#ifdef CONFIG_KRETPROBES
|
|
||||||
frame.kr_cur = NULL;
|
|
||||||
frame.tsk = tsk;
|
|
||||||
#endif
|
|
||||||
#ifdef CONFIG_UNWINDER_FRAME_POINTER
|
|
||||||
frame.ex_frame = false;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
walk_stackframe(&frame, save_trace, &data);
|
walk_stackframe(&frame, consume_entry, cookie);
|
||||||
}
|
}
|
||||||
|
|
||||||
void save_stack_trace_regs(struct pt_regs *regs, struct stack_trace *trace)
|
|
||||||
{
|
|
||||||
struct stack_trace_data data;
|
|
||||||
struct stackframe frame;
|
|
||||||
|
|
||||||
data.trace = trace;
|
|
||||||
data.skip = trace->skip;
|
|
||||||
data.no_sched_functions = 0;
|
|
||||||
|
|
||||||
frame.fp = regs->ARM_fp;
|
|
||||||
frame.sp = regs->ARM_sp;
|
|
||||||
frame.lr = regs->ARM_lr;
|
|
||||||
frame.pc = regs->ARM_pc;
|
|
||||||
#ifdef CONFIG_KRETPROBES
|
|
||||||
frame.kr_cur = NULL;
|
|
||||||
frame.tsk = current;
|
|
||||||
#endif
|
|
||||||
#ifdef CONFIG_UNWINDER_FRAME_POINTER
|
|
||||||
frame.ex_frame = in_entry_text(frame.pc);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
walk_stackframe(&frame, save_trace, &data);
|
|
||||||
}
|
|
||||||
|
|
||||||
void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
|
|
||||||
{
|
|
||||||
__save_stack_trace(tsk, trace, 1);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(save_stack_trace_tsk);
|
|
||||||
|
|
||||||
void save_stack_trace(struct stack_trace *trace)
|
|
||||||
{
|
|
||||||
__save_stack_trace(current, trace, 0);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(save_stack_trace);
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user