diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2010-08-06 12:30:52 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2010-08-06 12:30:52 -0400 |
| commit | 4aed2fd8e3181fea7c09ba79cf64e7e3f4413bf9 (patch) | |
| tree | 1f69733e5daab4915a76a41de0e4d1dc61e12cfb /arch/x86/kernel/stacktrace.c | |
| parent | 3a3527b6461b1298cc53ce72f336346739297ac8 (diff) | |
| parent | fc9ea5a1e53ee54f681e226d735008e2a6f8f470 (diff) | |
Merge branch 'perf-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip
* 'perf-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip: (162 commits)
tracing/kprobes: unregister_trace_probe needs to be called under mutex
perf: expose event__process function
perf events: Fix mmap offset determination
perf, powerpc: fsl_emb: Restore setting perf_sample_data.period
perf, powerpc: Convert the FSL driver to use local64_t
perf tools: Don't keep unreferenced maps when unmaps are detected
perf session: Invalidate last_match when removing threads from rb_tree
perf session: Free the ref_reloc_sym memory at the right place
x86,mmiotrace: Add support for tracing STOS instruction
perf, sched migration: Librarize task states and event headers helpers
perf, sched migration: Librarize the GUI class
perf, sched migration: Make the GUI class client agnostic
perf, sched migration: Make it vertically scrollable
perf, sched migration: Parameterize cpu height and spacing
perf, sched migration: Fix key bindings
perf, sched migration: Ignore unhandled task states
perf, sched migration: Handle ignored migrate out events
perf: New migration tool overview
tracing: Drop cpparg() macro
perf: Use tracepoint_synchronize_unregister() to flush any pending tracepoint call
...
Fix up trivial conflicts in Makefile and drivers/cpufreq/cpufreq.c
Diffstat (limited to 'arch/x86/kernel/stacktrace.c')
| -rw-r--r-- | arch/x86/kernel/stacktrace.c | 31 |
1 files changed, 16 insertions, 15 deletions
diff --git a/arch/x86/kernel/stacktrace.c b/arch/x86/kernel/stacktrace.c index 922eefbb3f6..b53c525368a 100644 --- a/arch/x86/kernel/stacktrace.c +++ b/arch/x86/kernel/stacktrace.c | |||
| @@ -23,11 +23,16 @@ static int save_stack_stack(void *data, char *name) | |||
| 23 | return 0; | 23 | return 0; |
| 24 | } | 24 | } |
| 25 | 25 | ||
| 26 | static void save_stack_address(void *data, unsigned long addr, int reliable) | 26 | static void |
| 27 | __save_stack_address(void *data, unsigned long addr, bool reliable, bool nosched) | ||
| 27 | { | 28 | { |
| 28 | struct stack_trace *trace = data; | 29 | struct stack_trace *trace = data; |
| 30 | #ifdef CONFIG_FRAME_POINTER | ||
| 29 | if (!reliable) | 31 | if (!reliable) |
| 30 | return; | 32 | return; |
| 33 | #endif | ||
| 34 | if (nosched && in_sched_functions(addr)) | ||
| 35 | return; | ||
| 31 | if (trace->skip > 0) { | 36 | if (trace->skip > 0) { |
| 32 | trace->skip--; | 37 | trace->skip--; |
| 33 | return; | 38 | return; |
| @@ -36,20 +41,15 @@ static void save_stack_address(void *data, unsigned long addr, int reliable) | |||
| 36 | trace->entries[trace->nr_entries++] = addr; | 41 | trace->entries[trace->nr_entries++] = addr; |
| 37 | } | 42 | } |
| 38 | 43 | ||
| 44 | static void save_stack_address(void *data, unsigned long addr, int reliable) | ||
| 45 | { | ||
| 46 | return __save_stack_address(data, addr, reliable, false); | ||
| 47 | } | ||
| 48 | |||
| 39 | static void | 49 | static void |
| 40 | save_stack_address_nosched(void *data, unsigned long addr, int reliable) | 50 | save_stack_address_nosched(void *data, unsigned long addr, int reliable) |
| 41 | { | 51 | { |
| 42 | struct stack_trace *trace = (struct stack_trace *)data; | 52 | return __save_stack_address(data, addr, reliable, true); |
| 43 | if (!reliable) | ||
| 44 | return; | ||
| 45 | if (in_sched_functions(addr)) | ||
| 46 | return; | ||
| 47 | if (trace->skip > 0) { | ||
| 48 | trace->skip--; | ||
| 49 | return; | ||
| 50 | } | ||
| 51 | if (trace->nr_entries < trace->max_entries) | ||
| 52 | trace->entries[trace->nr_entries++] = addr; | ||
| 53 | } | 53 | } |
| 54 | 54 | ||
| 55 | static const struct stacktrace_ops save_stack_ops = { | 55 | static const struct stacktrace_ops save_stack_ops = { |
| @@ -96,12 +96,13 @@ EXPORT_SYMBOL_GPL(save_stack_trace_tsk); | |||
| 96 | 96 | ||
| 97 | /* Userspace stacktrace - based on kernel/trace/trace_sysprof.c */ | 97 | /* Userspace stacktrace - based on kernel/trace/trace_sysprof.c */ |
| 98 | 98 | ||
| 99 | struct stack_frame { | 99 | struct stack_frame_user { |
| 100 | const void __user *next_fp; | 100 | const void __user *next_fp; |
| 101 | unsigned long ret_addr; | 101 | unsigned long ret_addr; |
| 102 | }; | 102 | }; |
| 103 | 103 | ||
| 104 | static int copy_stack_frame(const void __user *fp, struct stack_frame *frame) | 104 | static int |
| 105 | copy_stack_frame(const void __user *fp, struct stack_frame_user *frame) | ||
| 105 | { | 106 | { |
| 106 | int ret; | 107 | int ret; |
| 107 | 108 | ||
| @@ -126,7 +127,7 @@ static inline void __save_stack_trace_user(struct stack_trace *trace) | |||
| 126 | trace->entries[trace->nr_entries++] = regs->ip; | 127 | trace->entries[trace->nr_entries++] = regs->ip; |
| 127 | 128 | ||
| 128 | while (trace->nr_entries < trace->max_entries) { | 129 | while (trace->nr_entries < trace->max_entries) { |
| 129 | struct stack_frame frame; | 130 | struct stack_frame_user frame; |
| 130 | 131 | ||
| 131 | frame.next_fp = NULL; | 132 | frame.next_fp = NULL; |
| 132 | frame.ret_addr = 0; | 133 | frame.ret_addr = 0; |
