aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kernel/stacktrace.c
diff options
context:
space:
mode:
authorRussell King <rmk+kernel@arm.linux.org.uk>2014-05-03 06:03:28 -0400
committerRussell King <rmk+kernel@arm.linux.org.uk>2014-05-22 11:33:19 -0400
commit3683f44c42e991d313dc301504ee0fca1aeb8580 (patch)
tree8be709a8e9f72c0aff6fbc25222eac14146233a8 /arch/arm/kernel/stacktrace.c
parentdeace4a6b440f2e05f3e073338b28901d02a15c9 (diff)
ARM: stacktrace: avoid listing stacktrace functions in stacktrace
While debugging the FEC ethernet driver using stacktrace, it was noticed that the stacktraces always begin as follows: [<c00117b4>] save_stack_trace_tsk+0x0/0x98 [<c0011870>] save_stack_trace+0x24/0x28 ... This is because the stack trace code includes the stack frames for itself. This is incorrect behaviour, and also leads to "skip" doing the wrong thing (which is the number of stack frames to avoid recording.) Perversely, it does the right thing when passed a non-current thread. Fix this by ensuring that we have a known constant number of frames above the main stack trace function, and always skip these. Cc: <stable@vger.kernel.org> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'arch/arm/kernel/stacktrace.c')
-rw-r--r--arch/arm/kernel/stacktrace.c18
1 files changed, 13 insertions, 5 deletions
diff --git a/arch/arm/kernel/stacktrace.c b/arch/arm/kernel/stacktrace.c
index af4e8c8a5422..6582c4adc182 100644
--- a/arch/arm/kernel/stacktrace.c
+++ b/arch/arm/kernel/stacktrace.c
@@ -83,13 +83,16 @@ static int save_trace(struct stackframe *frame, void *d)
83 return trace->nr_entries >= trace->max_entries; 83 return trace->nr_entries >= trace->max_entries;
84} 84}
85 85
86void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace) 86/* This must be noinline to so that our skip calculation works correctly */
87static noinline void __save_stack_trace(struct task_struct *tsk,
88 struct stack_trace *trace, unsigned int nosched)
87{ 89{
88 struct stack_trace_data data; 90 struct stack_trace_data data;
89 struct stackframe frame; 91 struct stackframe frame;
90 92
91 data.trace = trace; 93 data.trace = trace;
92 data.skip = trace->skip; 94 data.skip = trace->skip;
95 data.no_sched_functions = nosched;
93 96
94 if (tsk != current) { 97 if (tsk != current) {
95#ifdef CONFIG_SMP 98#ifdef CONFIG_SMP
@@ -102,7 +105,6 @@ void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
102 trace->entries[trace->nr_entries++] = ULONG_MAX; 105 trace->entries[trace->nr_entries++] = ULONG_MAX;
103 return; 106 return;
104#else 107#else
105 data.no_sched_functions = 1;
106 frame.fp = thread_saved_fp(tsk); 108 frame.fp = thread_saved_fp(tsk);
107 frame.sp = thread_saved_sp(tsk); 109 frame.sp = thread_saved_sp(tsk);
108 frame.lr = 0; /* recovered from the stack */ 110 frame.lr = 0; /* recovered from the stack */
@@ -111,11 +113,12 @@ void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
111 } else { 113 } else {
112 register unsigned long current_sp asm ("sp"); 114 register unsigned long current_sp asm ("sp");
113 115
114 data.no_sched_functions = 0; 116 /* We don't want this function nor the caller */
117 data.skip += 2;
115 frame.fp = (unsigned long)__builtin_frame_address(0); 118 frame.fp = (unsigned long)__builtin_frame_address(0);
116 frame.sp = current_sp; 119 frame.sp = current_sp;
117 frame.lr = (unsigned long)__builtin_return_address(0); 120 frame.lr = (unsigned long)__builtin_return_address(0);
118 frame.pc = (unsigned long)save_stack_trace_tsk; 121 frame.pc = (unsigned long)__save_stack_trace;
119 } 122 }
120 123
121 walk_stackframe(&frame, save_trace, &data); 124 walk_stackframe(&frame, save_trace, &data);
@@ -123,9 +126,14 @@ void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
123 trace->entries[trace->nr_entries++] = ULONG_MAX; 126 trace->entries[trace->nr_entries++] = ULONG_MAX;
124} 127}
125 128
129void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
130{
131 __save_stack_trace(tsk, trace, 1);
132}
133
126void save_stack_trace(struct stack_trace *trace) 134void save_stack_trace(struct stack_trace *trace)
127{ 135{
128 save_stack_trace_tsk(current, trace); 136 __save_stack_trace(current, trace, 0);
129} 137}
130EXPORT_SYMBOL_GPL(save_stack_trace); 138EXPORT_SYMBOL_GPL(save_stack_trace);
131#endif 139#endif