mirror of https://gitee.com/openkylin/linux.git
48 lines
1.0 KiB
C
48 lines
1.0 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
// Copyright (C) 2005-2017 Andes Technology Corporation
|
|
|
|
#include <linux/sched/debug.h>
|
|
#include <linux/sched/task_stack.h>
|
|
#include <linux/stacktrace.h>
|
|
|
|
void save_stack_trace(struct stack_trace *trace)
|
|
{
|
|
save_stack_trace_tsk(current, trace);
|
|
}
|
|
|
|
void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
|
|
{
|
|
unsigned long *fpn;
|
|
int skip = trace->skip;
|
|
int savesched;
|
|
|
|
if (tsk == current) {
|
|
__asm__ __volatile__("\tori\t%0, $fp, #0\n":"=r"(fpn));
|
|
savesched = 1;
|
|
} else {
|
|
fpn = (unsigned long *)thread_saved_fp(tsk);
|
|
savesched = 0;
|
|
}
|
|
|
|
while (!kstack_end(fpn) && !((unsigned long)fpn & 0x3)
|
|
&& (fpn >= (unsigned long *)TASK_SIZE)) {
|
|
unsigned long lpp, fpp;
|
|
|
|
lpp = fpn[-1];
|
|
fpp = fpn[FP_OFFSET];
|
|
if (!__kernel_text_address(lpp))
|
|
break;
|
|
|
|
if (savesched || !in_sched_functions(lpp)) {
|
|
if (skip) {
|
|
skip--;
|
|
} else {
|
|
trace->entries[trace->nr_entries++] = lpp;
|
|
if (trace->nr_entries >= trace->max_entries)
|
|
break;
|
|
}
|
|
}
|
|
fpn = (unsigned long *)fpp;
|
|
}
|
|
}
|