diff options
| -rw-r--r-- | include/linux/perf_event.h | 2 | ||||
| -rw-r--r-- | kernel/events/core.c | 81 | 
2 files changed, 30 insertions, 53 deletions
| diff --git a/include/linux/perf_event.h b/include/linux/perf_event.h index 216653466a67..5cad0e6f3552 100644 --- a/include/linux/perf_event.h +++ b/include/linux/perf_event.h @@ -469,6 +469,7 @@ struct perf_event_context {  	 */  	struct mutex			mutex; +	struct list_head		active_ctx_list;  	struct list_head		pinned_groups;  	struct list_head		flexible_groups;  	struct list_head		event_list; @@ -519,7 +520,6 @@ struct perf_cpu_context {  	int				exclusive;  	struct hrtimer			hrtimer;  	ktime_t				hrtimer_interval; -	struct list_head		rotation_list;  	struct pmu			*unique_pmu;  	struct perf_cgroup		*cgrp;  }; diff --git a/kernel/events/core.c b/kernel/events/core.c index 37cc20e8aa3b..7f2fbb8b5069 100644 --- a/kernel/events/core.c +++ b/kernel/events/core.c @@ -872,22 +872,32 @@ void perf_pmu_enable(struct pmu *pmu)  		pmu->pmu_enable(pmu);  } -static DEFINE_PER_CPU(struct list_head, rotation_list); +static DEFINE_PER_CPU(struct list_head, active_ctx_list);  /* - * perf_pmu_rotate_start() and perf_rotate_context() are fully serialized - * because they're strictly cpu affine and rotate_start is called with IRQs - * disabled, while rotate_context is called from IRQ context. + * perf_event_ctx_activate(), perf_event_ctx_deactivate(), and + * perf_event_task_tick() are fully serialized because they're strictly cpu + * affine and perf_event_ctx{activate,deactivate} are called with IRQs + * disabled, while perf_event_task_tick is called from IRQ context.   */ -static void perf_pmu_rotate_start(struct pmu *pmu) +static void perf_event_ctx_activate(struct perf_event_context *ctx)  { -	struct perf_cpu_context *cpuctx = this_cpu_ptr(pmu->pmu_cpu_context); -	struct list_head *head = this_cpu_ptr(&rotation_list); +	struct list_head *head = this_cpu_ptr(&active_ctx_list);  	WARN_ON(!irqs_disabled()); -	if (list_empty(&cpuctx->rotation_list)) -		list_add(&cpuctx->rotation_list, head); +	WARN_ON(!list_empty(&ctx->active_ctx_list)); + +	list_add(&ctx->active_ctx_list, head); +} + +static void perf_event_ctx_deactivate(struct perf_event_context *ctx) +{ +	WARN_ON(!irqs_disabled()); + +	WARN_ON(list_empty(&ctx->active_ctx_list)); + +	list_del_init(&ctx->active_ctx_list);  }  static void get_ctx(struct perf_event_context *ctx) @@ -1233,8 +1243,6 @@ list_add_event(struct perf_event *event, struct perf_event_context *ctx)  		ctx->nr_branch_stack++;  	list_add_rcu(&event->event_entry, &ctx->event_list); -	if (!ctx->nr_events) -		perf_pmu_rotate_start(ctx->pmu);  	ctx->nr_events++;  	if (event->attr.inherit_stat)  		ctx->nr_stat++; @@ -1561,7 +1569,8 @@ event_sched_out(struct perf_event *event,  	if (!is_software_event(event))  		cpuctx->active_oncpu--; -	ctx->nr_active--; +	if (!--ctx->nr_active) +		perf_event_ctx_deactivate(ctx);  	if (event->attr.freq && event->attr.sample_freq)  		ctx->nr_freq--;  	if (event->attr.exclusive || !cpuctx->active_oncpu) @@ -1885,7 +1894,8 @@ event_sched_in(struct perf_event *event,  	if (!is_software_event(event))  		cpuctx->active_oncpu++; -	ctx->nr_active++; +	if (!ctx->nr_active++) +		perf_event_ctx_activate(ctx);  	if (event->attr.freq && event->attr.sample_freq)  		ctx->nr_freq++; @@ -2742,12 +2752,6 @@ static void perf_event_context_sched_in(struct perf_event_context *ctx,  	perf_pmu_enable(ctx->pmu);  	perf_ctx_unlock(cpuctx, ctx); - -	/* -	 * Since these rotations are per-cpu, we need to ensure the -	 * cpu-context we got scheduled on is actually rotating. -	 */ -	perf_pmu_rotate_start(ctx->pmu);  }  /* @@ -3035,25 +3039,18 @@ static void rotate_ctx(struct perf_event_context *ctx)  		list_rotate_left(&ctx->flexible_groups);  } -/* - * perf_pmu_rotate_start() and perf_rotate_context() are fully serialized - * because they're strictly cpu affine and rotate_start is called with IRQs - * disabled, while rotate_context is called from IRQ context. - */  static int perf_rotate_context(struct perf_cpu_context *cpuctx)  {  	struct perf_event_context *ctx = NULL; -	int rotate = 0, remove = 1; +	int rotate = 0;  	if (cpuctx->ctx.nr_events) { -		remove = 0;  		if (cpuctx->ctx.nr_events != cpuctx->ctx.nr_active)  			rotate = 1;  	}  	ctx = cpuctx->task_ctx;  	if (ctx && ctx->nr_events) { -		remove = 0;  		if (ctx->nr_events != ctx->nr_active)  			rotate = 1;  	} @@ -3077,8 +3074,6 @@ static int perf_rotate_context(struct perf_cpu_context *cpuctx)  	perf_pmu_enable(cpuctx->ctx.pmu);  	perf_ctx_unlock(cpuctx, cpuctx->task_ctx);  done: -	if (remove) -		list_del_init(&cpuctx->rotation_list);  	return rotate;  } @@ -3096,9 +3091,8 @@ bool perf_event_can_stop_tick(void)  void perf_event_task_tick(void)  { -	struct list_head *head = this_cpu_ptr(&rotation_list); -	struct perf_cpu_context *cpuctx, *tmp; -	struct perf_event_context *ctx; +	struct list_head *head = this_cpu_ptr(&active_ctx_list); +	struct perf_event_context *ctx, *tmp;  	int throttled;  	WARN_ON(!irqs_disabled()); @@ -3106,14 +3100,8 @@ void perf_event_task_tick(void)  	__this_cpu_inc(perf_throttled_seq);  	throttled = __this_cpu_xchg(perf_throttled_count, 0); -	list_for_each_entry_safe(cpuctx, tmp, head, rotation_list) { -		ctx = &cpuctx->ctx; +	list_for_each_entry_safe(ctx, tmp, head, active_ctx_list)  		perf_adjust_freq_unthr_context(ctx, throttled); - -		ctx = cpuctx->task_ctx; -		if (ctx) -			perf_adjust_freq_unthr_context(ctx, throttled); -	}  }  static int event_enable_on_exec(struct perf_event *event, @@ -3272,6 +3260,7 @@ static void __perf_event_init_context(struct perf_event_context *ctx)  {  	raw_spin_lock_init(&ctx->lock);  	mutex_init(&ctx->mutex); +	INIT_LIST_HEAD(&ctx->active_ctx_list);  	INIT_LIST_HEAD(&ctx->pinned_groups);  	INIT_LIST_HEAD(&ctx->flexible_groups);  	INIT_LIST_HEAD(&ctx->event_list); @@ -6954,7 +6943,6 @@ skip_type:  		__perf_cpu_hrtimer_init(cpuctx, cpu); -		INIT_LIST_HEAD(&cpuctx->rotation_list);  		cpuctx->unique_pmu = pmu;  	} @@ -8384,7 +8372,7 @@ static void __init perf_event_init_all_cpus(void)  	for_each_possible_cpu(cpu) {  		swhash = &per_cpu(swevent_htable, cpu);  		mutex_init(&swhash->hlist_mutex); -		INIT_LIST_HEAD(&per_cpu(rotation_list, cpu)); +		INIT_LIST_HEAD(&per_cpu(active_ctx_list, cpu));  	}  } @@ -8405,22 +8393,11 @@ static void perf_event_init_cpu(int cpu)  }  #if defined CONFIG_HOTPLUG_CPU || defined CONFIG_KEXEC -static void perf_pmu_rotate_stop(struct pmu *pmu) -{ -	struct perf_cpu_context *cpuctx = this_cpu_ptr(pmu->pmu_cpu_context); - -	WARN_ON(!irqs_disabled()); - -	list_del_init(&cpuctx->rotation_list); -} -  static void __perf_event_exit_context(void *__info)  {  	struct remove_event re = { .detach_group = true };  	struct perf_event_context *ctx = __info; -	perf_pmu_rotate_stop(ctx->pmu); -  	rcu_read_lock();  	list_for_each_entry_rcu(re.event, &ctx->event_list, event_entry)  		__perf_remove_from_context(&re); | 
