aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorArnaldo Carvalho de Melo <acme@redhat.com>2016-05-10 17:08:32 -0400
committerArnaldo Carvalho de Melo <acme@redhat.com>2016-05-16 22:11:51 -0400
commit3b1fff08038bd0792b1aa1e9703b2dd0512a3fd0 (patch)
tree894e513ec904c0b4249917b64adac9d5bda162b5
parentcfbcf468454ab4b20f0b4b62da51920b99fdb19e (diff)
perf core: Add a 'nr' field to perf_event_callchain_context
We will use it to count how many addresses are in the entry->ip[] array, excluding PERF_CONTEXT_{KERNEL,USER,etc} entries, so that we can really return the number of entries specified by the user via the relevant sysctl, kernel.perf_event_max_contexts, or via the per event perf_event_attr.sample_max_stack knob. This way we keep the perf_sample->ip_callchain->nr meaning, that is the number of entries, be it real addresses or PERF_CONTEXT_ entries, while honouring the max_stack knobs, i.e. the end result will be max_stack entries if we have at least that many entries in a given stack trace. Cc: David Ahern <dsahern@gmail.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Jiri Olsa <jolsa@redhat.com> Cc: Namhyung Kim <namhyung@kernel.org> Cc: Peter Zijlstra <peterz@infradead.org> Link: http://lkml.kernel.org/n/tip-s8teto51tdqvlfhefndtat9r@git.kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
-rw-r--r--arch/arm/kernel/perf_callchain.c2
-rw-r--r--arch/arm64/kernel/perf_callchain.c4
-rw-r--r--arch/metag/kernel/perf_callchain.c2
-rw-r--r--arch/mips/kernel/perf_event.c4
-rw-r--r--arch/powerpc/perf/callchain.c4
-rw-r--r--arch/sparc/kernel/perf_event.c6
-rw-r--r--arch/x86/events/core.c4
-rw-r--r--include/linux/perf_event.h6
-rw-r--r--kernel/events/callchain.c3
9 files changed, 18 insertions, 17 deletions
diff --git a/arch/arm/kernel/perf_callchain.c b/arch/arm/kernel/perf_callchain.c
index bc552e813e7b..22bf1f64d99a 100644
--- a/arch/arm/kernel/perf_callchain.c
+++ b/arch/arm/kernel/perf_callchain.c
@@ -75,7 +75,7 @@ perf_callchain_user(struct perf_callchain_entry_ctx *entry, struct pt_regs *regs
75 75
76 tail = (struct frame_tail __user *)regs->ARM_fp - 1; 76 tail = (struct frame_tail __user *)regs->ARM_fp - 1;
77 77
78 while ((entry->entry->nr < entry->max_stack) && 78 while ((entry->nr < entry->max_stack) &&
79 tail && !((unsigned long)tail & 0x3)) 79 tail && !((unsigned long)tail & 0x3))
80 tail = user_backtrace(tail, entry); 80 tail = user_backtrace(tail, entry);
81} 81}
diff --git a/arch/arm64/kernel/perf_callchain.c b/arch/arm64/kernel/perf_callchain.c
index 0d60150057cf..713ca824f266 100644
--- a/arch/arm64/kernel/perf_callchain.c
+++ b/arch/arm64/kernel/perf_callchain.c
@@ -122,7 +122,7 @@ void perf_callchain_user(struct perf_callchain_entry_ctx *entry,
122 122
123 tail = (struct frame_tail __user *)regs->regs[29]; 123 tail = (struct frame_tail __user *)regs->regs[29];
124 124
125 while (entry->entry->nr < entry->max_stack && 125 while (entry->nr < entry->max_stack &&
126 tail && !((unsigned long)tail & 0xf)) 126 tail && !((unsigned long)tail & 0xf))
127 tail = user_backtrace(tail, entry); 127 tail = user_backtrace(tail, entry);
128 } else { 128 } else {
@@ -132,7 +132,7 @@ void perf_callchain_user(struct perf_callchain_entry_ctx *entry,
132 132
133 tail = (struct compat_frame_tail __user *)regs->compat_fp - 1; 133 tail = (struct compat_frame_tail __user *)regs->compat_fp - 1;
134 134
135 while ((entry->entry->nr < entry->max_stack) && 135 while ((entry->nr < entry->max_stack) &&
136 tail && !((unsigned long)tail & 0x3)) 136 tail && !((unsigned long)tail & 0x3))
137 tail = compat_user_backtrace(tail, entry); 137 tail = compat_user_backtrace(tail, entry);
138#endif 138#endif
diff --git a/arch/metag/kernel/perf_callchain.c b/arch/metag/kernel/perf_callchain.c
index b3261a98b15b..3e8e048040df 100644
--- a/arch/metag/kernel/perf_callchain.c
+++ b/arch/metag/kernel/perf_callchain.c
@@ -65,7 +65,7 @@ perf_callchain_user(struct perf_callchain_entry_ctx *entry, struct pt_regs *regs
65 65
66 --frame; 66 --frame;
67 67
68 while ((entry->entry->nr < entry->max_stack) && frame) 68 while ((entry->nr < entry->max_stack) && frame)
69 frame = user_backtrace(frame, entry); 69 frame = user_backtrace(frame, entry);
70} 70}
71 71
diff --git a/arch/mips/kernel/perf_event.c b/arch/mips/kernel/perf_event.c
index 22395c7d7030..d64056e0bb56 100644
--- a/arch/mips/kernel/perf_event.c
+++ b/arch/mips/kernel/perf_event.c
@@ -35,7 +35,7 @@ static void save_raw_perf_callchain(struct perf_callchain_entry_ctx *entry,
35 addr = *sp++; 35 addr = *sp++;
36 if (__kernel_text_address(addr)) { 36 if (__kernel_text_address(addr)) {
37 perf_callchain_store(entry, addr); 37 perf_callchain_store(entry, addr);
38 if (entry->entry->nr >= entry->max_stack) 38 if (entry->nr >= entry->max_stack)
39 break; 39 break;
40 } 40 }
41 } 41 }
@@ -59,7 +59,7 @@ void perf_callchain_kernel(struct perf_callchain_entry_ctx *entry,
59 } 59 }
60 do { 60 do {
61 perf_callchain_store(entry, pc); 61 perf_callchain_store(entry, pc);
62 if (entry->entry->nr >= entry->max_stack) 62 if (entry->nr >= entry->max_stack)
63 break; 63 break;
64 pc = unwind_stack(current, &sp, pc, &ra); 64 pc = unwind_stack(current, &sp, pc, &ra);
65 } while (pc); 65 } while (pc);
diff --git a/arch/powerpc/perf/callchain.c b/arch/powerpc/perf/callchain.c
index c9260c1dfdbc..f68f213dc36c 100644
--- a/arch/powerpc/perf/callchain.c
+++ b/arch/powerpc/perf/callchain.c
@@ -247,7 +247,7 @@ static void perf_callchain_user_64(struct perf_callchain_entry_ctx *entry,
247 sp = regs->gpr[1]; 247 sp = regs->gpr[1];
248 perf_callchain_store(entry, next_ip); 248 perf_callchain_store(entry, next_ip);
249 249
250 while (entry->entry->nr < entry->max_stack) { 250 while (entry->nr < entry->max_stack) {
251 fp = (unsigned long __user *) sp; 251 fp = (unsigned long __user *) sp;
252 if (!valid_user_sp(sp, 1) || read_user_stack_64(fp, &next_sp)) 252 if (!valid_user_sp(sp, 1) || read_user_stack_64(fp, &next_sp))
253 return; 253 return;
@@ -453,7 +453,7 @@ static void perf_callchain_user_32(struct perf_callchain_entry_ctx *entry,
453 sp = regs->gpr[1]; 453 sp = regs->gpr[1];
454 perf_callchain_store(entry, next_ip); 454 perf_callchain_store(entry, next_ip);
455 455
456 while (entry->entry->nr < entry->max_stack) { 456 while (entry->nr < entry->max_stack) {
457 fp = (unsigned int __user *) (unsigned long) sp; 457 fp = (unsigned int __user *) (unsigned long) sp;
458 if (!valid_user_sp(sp, 0) || read_user_stack_32(fp, &next_sp)) 458 if (!valid_user_sp(sp, 0) || read_user_stack_32(fp, &next_sp))
459 return; 459 return;
diff --git a/arch/sparc/kernel/perf_event.c b/arch/sparc/kernel/perf_event.c
index bcc5376db74b..710f3278d448 100644
--- a/arch/sparc/kernel/perf_event.c
+++ b/arch/sparc/kernel/perf_event.c
@@ -1756,7 +1756,7 @@ void perf_callchain_kernel(struct perf_callchain_entry_ctx *entry,
1756 } 1756 }
1757 } 1757 }
1758#endif 1758#endif
1759 } while (entry->entry->nr < entry->max_stack); 1759 } while (entry->nr < entry->max_stack);
1760} 1760}
1761 1761
1762static inline int 1762static inline int
@@ -1790,7 +1790,7 @@ static void perf_callchain_user_64(struct perf_callchain_entry_ctx *entry,
1790 pc = sf.callers_pc; 1790 pc = sf.callers_pc;
1791 ufp = (unsigned long)sf.fp + STACK_BIAS; 1791 ufp = (unsigned long)sf.fp + STACK_BIAS;
1792 perf_callchain_store(entry, pc); 1792 perf_callchain_store(entry, pc);
1793 } while (entry->entry->nr < entry->max_stack); 1793 } while (entry->nr < entry->max_stack);
1794} 1794}
1795 1795
1796static void perf_callchain_user_32(struct perf_callchain_entry_ctx *entry, 1796static void perf_callchain_user_32(struct perf_callchain_entry_ctx *entry,
@@ -1822,7 +1822,7 @@ static void perf_callchain_user_32(struct perf_callchain_entry_ctx *entry,
1822 ufp = (unsigned long)sf.fp; 1822 ufp = (unsigned long)sf.fp;
1823 } 1823 }
1824 perf_callchain_store(entry, pc); 1824 perf_callchain_store(entry, pc);
1825 } while (entry->entry->nr < entry->max_stack); 1825 } while (entry->nr < entry->max_stack);
1826} 1826}
1827 1827
1828void 1828void
diff --git a/arch/x86/events/core.c b/arch/x86/events/core.c
index 07f2b01cfb72..5de96a18cd9c 100644
--- a/arch/x86/events/core.c
+++ b/arch/x86/events/core.c
@@ -2283,7 +2283,7 @@ perf_callchain_user32(struct pt_regs *regs, struct perf_callchain_entry_ctx *ent
2283 2283
2284 fp = compat_ptr(ss_base + regs->bp); 2284 fp = compat_ptr(ss_base + regs->bp);
2285 pagefault_disable(); 2285 pagefault_disable();
2286 while (entry->entry->nr < entry->max_stack) { 2286 while (entry->nr < entry->max_stack) {
2287 unsigned long bytes; 2287 unsigned long bytes;
2288 frame.next_frame = 0; 2288 frame.next_frame = 0;
2289 frame.return_address = 0; 2289 frame.return_address = 0;
@@ -2343,7 +2343,7 @@ perf_callchain_user(struct perf_callchain_entry_ctx *entry, struct pt_regs *regs
2343 return; 2343 return;
2344 2344
2345 pagefault_disable(); 2345 pagefault_disable();
2346 while (entry->entry->nr < entry->max_stack) { 2346 while (entry->nr < entry->max_stack) {
2347 unsigned long bytes; 2347 unsigned long bytes;
2348 frame.next_frame = NULL; 2348 frame.next_frame = NULL;
2349 frame.return_address = 0; 2349 frame.return_address = 0;
diff --git a/include/linux/perf_event.h b/include/linux/perf_event.h
index dbd18246b36e..3803bb1a862b 100644
--- a/include/linux/perf_event.h
+++ b/include/linux/perf_event.h
@@ -64,6 +64,7 @@ struct perf_callchain_entry {
64struct perf_callchain_entry_ctx { 64struct perf_callchain_entry_ctx {
65 struct perf_callchain_entry *entry; 65 struct perf_callchain_entry *entry;
66 u32 max_stack; 66 u32 max_stack;
67 u32 nr;
67}; 68};
68 69
69struct perf_raw_record { 70struct perf_raw_record {
@@ -1080,9 +1081,10 @@ extern int sysctl_perf_event_max_stack;
1080 1081
1081static inline int perf_callchain_store(struct perf_callchain_entry_ctx *ctx, u64 ip) 1082static inline int perf_callchain_store(struct perf_callchain_entry_ctx *ctx, u64 ip)
1082{ 1083{
1083 struct perf_callchain_entry *entry = ctx->entry; 1084 if (ctx->nr < ctx->max_stack) {
1084 if (entry->nr < ctx->max_stack) { 1085 struct perf_callchain_entry *entry = ctx->entry;
1085 entry->ip[entry->nr++] = ip; 1086 entry->ip[entry->nr++] = ip;
1087 ++ctx->nr;
1086 return 0; 1088 return 0;
1087 } else { 1089 } else {
1088 return -1; /* no more room, stop walking the stack */ 1090 return -1; /* no more room, stop walking the stack */
diff --git a/kernel/events/callchain.c b/kernel/events/callchain.c
index af95ad92893a..8774ff86debb 100644
--- a/kernel/events/callchain.c
+++ b/kernel/events/callchain.c
@@ -196,8 +196,7 @@ get_perf_callchain(struct pt_regs *regs, u32 init_nr, bool kernel, bool user,
196 196
197 ctx.entry = entry; 197 ctx.entry = entry;
198 ctx.max_stack = max_stack; 198 ctx.max_stack = max_stack;
199 199 ctx.nr = entry->nr = init_nr;
200 entry->nr = init_nr;
201 200
202 if (kernel && !user_mode(regs)) { 201 if (kernel && !user_mode(regs)) {
203 if (add_mark) 202 if (add_mark)