diff options
author | Peter Zijlstra <a.p.zijlstra@chello.nl> | 2009-03-23 18:22:07 +0100 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2009-04-06 09:30:25 +0200 |
commit | 96f6d4444302bb2ea2cf409529eef816462f6ce0 (patch) | |
tree | 34ae6944b4a8e6db89c50dba5a3a8b9da3315122 /kernel/perf_counter.c | |
parent | f4a2deb4860497f4332cf6a1acddab3dd628ddf0 (diff) | |
download | op-kernel-dev-96f6d4444302bb2ea2cf409529eef816462f6ce0.zip op-kernel-dev-96f6d4444302bb2ea2cf409529eef816462f6ce0.tar.gz |
perf_counter: avoid recursion
Tracepoint events like lock_acquire and software counters like
pagefaults can recurse into the perf counter code again, avoid that.
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
Orig-LKML-Reference: <20090323172417.152096433@chello.nl>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'kernel/perf_counter.c')
-rw-r--r-- | kernel/perf_counter.c | 26 |
1 files changed, 26 insertions, 0 deletions
diff --git a/kernel/perf_counter.c b/kernel/perf_counter.c index ca14fc4..ce34bff 100644 --- a/kernel/perf_counter.c +++ b/kernel/perf_counter.c @@ -23,6 +23,7 @@ #include <linux/mm.h> #include <linux/vmstat.h> #include <linux/rculist.h> +#include <linux/hardirq.h> #include <asm/irq_regs.h> @@ -1532,10 +1533,31 @@ static void perf_swcounter_ctx_event(struct perf_counter_context *ctx, rcu_read_unlock(); } +static int *perf_swcounter_recursion_context(struct perf_cpu_context *cpuctx) +{ + if (in_nmi()) + return &cpuctx->recursion[3]; + + if (in_irq()) + return &cpuctx->recursion[2]; + + if (in_softirq()) + return &cpuctx->recursion[1]; + + return &cpuctx->recursion[0]; +} + static void __perf_swcounter_event(enum perf_event_types type, u32 event, u64 nr, int nmi, struct pt_regs *regs) { struct perf_cpu_context *cpuctx = &get_cpu_var(perf_cpu_context); + int *recursion = perf_swcounter_recursion_context(cpuctx); + + if (*recursion) + goto out; + + (*recursion)++; + barrier(); perf_swcounter_ctx_event(&cpuctx->ctx, type, event, nr, nmi, regs); if (cpuctx->task_ctx) { @@ -1543,6 +1565,10 @@ static void __perf_swcounter_event(enum perf_event_types type, u32 event, nr, nmi, regs); } + barrier(); + (*recursion)--; + +out: put_cpu_var(perf_cpu_context); } |