aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/perf_counter.c36
1 files changed, 17 insertions, 19 deletions
diff --git a/kernel/perf_counter.c b/kernel/perf_counter.c
index f7a8ab9576e4..5c6fae4f43d8 100644
--- a/kernel/perf_counter.c
+++ b/kernel/perf_counter.c
@@ -146,6 +146,14 @@ static void put_ctx(struct perf_counter_context *ctx)
146 } 146 }
147} 147}
148 148
149static void unclone_ctx(struct perf_counter_context *ctx)
150{
151 if (ctx->parent_ctx) {
152 put_ctx(ctx->parent_ctx);
153 ctx->parent_ctx = NULL;
154 }
155}
156
149/* 157/*
150 * Get the perf_counter_context for a task and lock it. 158 * Get the perf_counter_context for a task and lock it.
151 * This has to cope with with the fact that until it is locked, 159 * This has to cope with with the fact that until it is locked,
@@ -1463,10 +1471,8 @@ static void perf_counter_enable_on_exec(struct task_struct *task)
1463 /* 1471 /*
1464 * Unclone this context if we enabled any counter. 1472 * Unclone this context if we enabled any counter.
1465 */ 1473 */
1466 if (enabled && ctx->parent_ctx) { 1474 if (enabled)
1467 put_ctx(ctx->parent_ctx); 1475 unclone_ctx(ctx);
1468 ctx->parent_ctx = NULL;
1469 }
1470 1476
1471 spin_unlock(&ctx->lock); 1477 spin_unlock(&ctx->lock);
1472 1478
@@ -1526,7 +1532,6 @@ __perf_counter_init_context(struct perf_counter_context *ctx,
1526 1532
1527static struct perf_counter_context *find_get_context(pid_t pid, int cpu) 1533static struct perf_counter_context *find_get_context(pid_t pid, int cpu)
1528{ 1534{
1529 struct perf_counter_context *parent_ctx;
1530 struct perf_counter_context *ctx; 1535 struct perf_counter_context *ctx;
1531 struct perf_cpu_context *cpuctx; 1536 struct perf_cpu_context *cpuctx;
1532 struct task_struct *task; 1537 struct task_struct *task;
@@ -1586,11 +1591,7 @@ static struct perf_counter_context *find_get_context(pid_t pid, int cpu)
1586 retry: 1591 retry:
1587 ctx = perf_lock_task_context(task, &flags); 1592 ctx = perf_lock_task_context(task, &flags);
1588 if (ctx) { 1593 if (ctx) {
1589 parent_ctx = ctx->parent_ctx; 1594 unclone_ctx(ctx);
1590 if (parent_ctx) {
1591 put_ctx(parent_ctx);
1592 ctx->parent_ctx = NULL; /* no longer a clone */
1593 }
1594 spin_unlock_irqrestore(&ctx->lock, flags); 1595 spin_unlock_irqrestore(&ctx->lock, flags);
1595 } 1596 }
1596 1597
@@ -4262,15 +4263,12 @@ void perf_counter_exit_task(struct task_struct *child)
4262 */ 4263 */
4263 spin_lock(&child_ctx->lock); 4264 spin_lock(&child_ctx->lock);
4264 child->perf_counter_ctxp = NULL; 4265 child->perf_counter_ctxp = NULL;
4265 if (child_ctx->parent_ctx) { 4266 /*
4266 /* 4267 * If this context is a clone; unclone it so it can't get
4267 * This context is a clone; unclone it so it can't get 4268 * swapped to another process while we're removing all
4268 * swapped to another process while we're removing all 4269 * the counters from it.
4269 * the counters from it. 4270 */
4270 */ 4271 unclone_ctx(child_ctx);
4271 put_ctx(child_ctx->parent_ctx);
4272 child_ctx->parent_ctx = NULL;
4273 }
4274 spin_unlock(&child_ctx->lock); 4272 spin_unlock(&child_ctx->lock);
4275 local_irq_restore(flags); 4273 local_irq_restore(flags);
4276 4274