diff options
| author | Jens Axboe <jaxboe@fusionio.com> | 2010-10-19 09:13:04 +0200 | 
|---|---|---|
| committer | Jens Axboe <jaxboe@fusionio.com> | 2010-10-19 09:13:04 +0200 | 
| commit | fa251f89903d73989e2f63e13d0eaed1e07ce0da (patch) | |
| tree | 3f7fe779941e3b6d67754dd7c44a32f48ea47c74 /kernel/smp.c | |
| parent | dd3932eddf428571762596e17b65f5dc92ca361b (diff) | |
| parent | cd07202cc8262e1669edff0d97715f3dd9260917 (diff) | |
Merge branch 'v2.6.36-rc8' into for-2.6.37/barrier
Conflicts:
	block/blk-core.c
	drivers/block/loop.c
	mm/swapfile.c
Signed-off-by: Jens Axboe <jaxboe@fusionio.com>
Diffstat (limited to 'kernel/smp.c')
| -rw-r--r-- | kernel/smp.c | 17 | 
1 files changed, 14 insertions, 3 deletions
diff --git a/kernel/smp.c b/kernel/smp.c index 75c970c715d..ed6aacfcb7e 100644 --- a/kernel/smp.c +++ b/kernel/smp.c @@ -365,9 +365,10 @@ call:  EXPORT_SYMBOL_GPL(smp_call_function_any);  /** - * __smp_call_function_single(): Run a function on another CPU + * __smp_call_function_single(): Run a function on a specific CPU   * @cpu: The CPU to run on.   * @data: Pre-allocated and setup data structure + * @wait: If true, wait until function has completed on specified CPU.   *   * Like smp_call_function_single(), but allow caller to pass in a   * pre-allocated data structure. Useful for embedding @data inside @@ -376,8 +377,10 @@ EXPORT_SYMBOL_GPL(smp_call_function_any);  void __smp_call_function_single(int cpu, struct call_single_data *data,  				int wait)  { -	csd_lock(data); +	unsigned int this_cpu; +	unsigned long flags; +	this_cpu = get_cpu();  	/*  	 * Can deadlock when called with interrupts disabled.  	 * We allow cpu's that are not yet online though, as no one else can @@ -387,7 +390,15 @@ void __smp_call_function_single(int cpu, struct call_single_data *data,  	WARN_ON_ONCE(cpu_online(smp_processor_id()) && wait && irqs_disabled()  		     && !oops_in_progress); -	generic_exec_single(cpu, data, wait); +	if (cpu == this_cpu) { +		local_irq_save(flags); +		data->func(data->info); +		local_irq_restore(flags); +	} else { +		csd_lock(data); +		generic_exec_single(cpu, data, wait); +	} +	put_cpu();  }  /**  | 
