diff options
author | Peter Zijlstra <peterz@infradead.org> | 2015-02-25 09:56:04 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@kernel.org> | 2015-03-27 04:49:45 -0400 |
commit | ccd41c86ad4d464d0ed4e48d80759ff85c2115b0 (patch) | |
tree | b53b1e7507761a9144f900eff6f97cd4bb6d15c5 | |
parent | 9332d250b4b4f67c633894b311e022e3cf943bd5 (diff) |
perf: Fix racy group access
While looking at some fuzzer output I noticed that we do not hold any
locks on leader->ctx and therefore the sibling_list iteration is
unsafe.
Acquire the relevant ctx->mutex before calling into the pmu specific
code.
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Cc: Vince Weaver <vincent.weaver@maine.edu>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Sasha Levin <sasha.levin@oracle.com>
Link: http://lkml.kernel.org/r/20150225151639.GL5029@twins.programming.kicks-ass.net
Signed-off-by: Ingo Molnar <mingo@kernel.org>
-rw-r--r-- | kernel/events/core.c | 11 |
1 files changed, 11 insertions, 0 deletions
diff --git a/kernel/events/core.c b/kernel/events/core.c index b01dfb602db1..bb1a7c36e794 100644 --- a/kernel/events/core.c +++ b/kernel/events/core.c | |||
@@ -7036,12 +7036,23 @@ EXPORT_SYMBOL_GPL(perf_pmu_unregister); | |||
7036 | 7036 | ||
7037 | static int perf_try_init_event(struct pmu *pmu, struct perf_event *event) | 7037 | static int perf_try_init_event(struct pmu *pmu, struct perf_event *event) |
7038 | { | 7038 | { |
7039 | struct perf_event_context *ctx = NULL; | ||
7039 | int ret; | 7040 | int ret; |
7040 | 7041 | ||
7041 | if (!try_module_get(pmu->module)) | 7042 | if (!try_module_get(pmu->module)) |
7042 | return -ENODEV; | 7043 | return -ENODEV; |
7044 | |||
7045 | if (event->group_leader != event) { | ||
7046 | ctx = perf_event_ctx_lock(event->group_leader); | ||
7047 | BUG_ON(!ctx); | ||
7048 | } | ||
7049 | |||
7043 | event->pmu = pmu; | 7050 | event->pmu = pmu; |
7044 | ret = pmu->event_init(event); | 7051 | ret = pmu->event_init(event); |
7052 | |||
7053 | if (ctx) | ||
7054 | perf_event_ctx_unlock(event->group_leader, ctx); | ||
7055 | |||
7045 | if (ret) | 7056 | if (ret) |
7046 | module_put(pmu->module); | 7057 | module_put(pmu->module); |
7047 | 7058 | ||