diff options
| author | Haavard Skinnemoen <haavard.skinnemoen@atmel.com> | 2009-12-14 19:23:03 +0100 | 
|---|---|---|
| committer | Haavard Skinnemoen <haavard.skinnemoen@atmel.com> | 2009-12-14 19:23:03 +0100 | 
| commit | 559300bc0ef7ccd541656f1189d38e7088389559 (patch) | |
| tree | cd1682881ca4246e9c5a1e8632be4bdd9d9706c6 /kernel/workqueue.c | |
| parent | 5416bf33f92a4104dbcd6062bf377c8421ca3cfd (diff) | |
| parent | 22763c5cf3690a681551162c15d34d935308c8d7 (diff) | |
Merge commit 'v2.6.32'
Conflicts:
	arch/avr32/mach-at32ap/include/mach/cpu.h
Diffstat (limited to 'kernel/workqueue.c')
| -rw-r--r-- | kernel/workqueue.c | 35 | 
1 files changed, 34 insertions, 1 deletions
| diff --git a/kernel/workqueue.c b/kernel/workqueue.c index addfe2df93b..67e526b6ae8 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c @@ -640,6 +640,24 @@ int schedule_delayed_work(struct delayed_work *dwork,  EXPORT_SYMBOL(schedule_delayed_work);  /** + * flush_delayed_work - block until a dwork_struct's callback has terminated + * @dwork: the delayed work which is to be flushed + * + * Any timeout is cancelled, and any pending work is run immediately. + */ +void flush_delayed_work(struct delayed_work *dwork) +{ +	if (del_timer_sync(&dwork->timer)) { +		struct cpu_workqueue_struct *cwq; +		cwq = wq_per_cpu(keventd_wq, get_cpu()); +		__queue_work(cwq, &dwork->work); +		put_cpu(); +	} +	flush_work(&dwork->work); +} +EXPORT_SYMBOL(flush_delayed_work); + +/**   * schedule_delayed_work_on - queue work in global workqueue on CPU after delay   * @cpu: cpu to use   * @dwork: job to be done @@ -667,6 +685,7 @@ EXPORT_SYMBOL(schedule_delayed_work_on);  int schedule_on_each_cpu(work_func_t func)  {  	int cpu; +	int orig = -1;  	struct work_struct *works;  	works = alloc_percpu(struct work_struct); @@ -674,14 +693,28 @@ int schedule_on_each_cpu(work_func_t func)  		return -ENOMEM;  	get_online_cpus(); + +	/* +	 * When running in keventd don't schedule a work item on +	 * itself.  Can just call directly because the work queue is +	 * already bound.  This also is faster. +	 */ +	if (current_is_keventd()) +		orig = raw_smp_processor_id(); +  	for_each_online_cpu(cpu) {  		struct work_struct *work = per_cpu_ptr(works, cpu);  		INIT_WORK(work, func); -		schedule_work_on(cpu, work); +		if (cpu != orig) +			schedule_work_on(cpu, work);  	} +	if (orig >= 0) +		func(per_cpu_ptr(works, orig)); +  	for_each_online_cpu(cpu)  		flush_work(per_cpu_ptr(works, cpu)); +  	put_online_cpus();  	free_percpu(works);  	return 0; | 
