2006-09-29 02:01:35 -07:00
/* Copyright 2005 Rusty Russell rusty@rustcorp.com.au IBM Corporation.
* GPL v2 and any later version .
*/
2005-04-16 15:20:36 -07:00
# include <linux/cpu.h>
# include <linux/err.h>
2007-05-08 00:25:08 -07:00
# include <linux/kthread.h>
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/stop_machine.h>
2005-04-16 15:20:36 -07:00
# include <linux/syscalls.h>
2007-05-10 22:22:47 -07:00
# include <linux/interrupt.h>
2005-04-16 15:20:36 -07:00
# include <asm/atomic.h>
# include <asm/uaccess.h>
/* Since we effect priority and affinity (both of which are visible
* to , and settable by outside processes ) we do indirection via a
* kthread . */
/* Thread to stop each CPU in user context. */
enum stopmachine_state {
STOPMACHINE_WAIT ,
STOPMACHINE_PREPARE ,
STOPMACHINE_DISABLE_IRQ ,
STOPMACHINE_EXIT ,
} ;
static enum stopmachine_state stopmachine_state ;
static unsigned int stopmachine_num_threads ;
static atomic_t stopmachine_thread_ack ;
2006-07-03 17:32:22 -07:00
static int stopmachine ( void * cpu )
2005-04-16 15:20:36 -07:00
{
int irqs_disabled = 0 ;
int prepared = 0 ;
2008-04-04 18:11:06 -07:00
set_cpus_allowed_ptr ( current , & cpumask_of_cpu ( ( int ) ( long ) cpu ) ) ;
2006-07-03 17:32:22 -07:00
2005-04-16 15:20:36 -07:00
/* Ack: we are alive */
2005-05-01 08:58:47 -07:00
smp_mb ( ) ; /* Theoretically the ack = 0 might not be on this CPU yet. */
2005-04-16 15:20:36 -07:00
atomic_inc ( & stopmachine_thread_ack ) ;
/* Simple state machine */
while ( stopmachine_state ! = STOPMACHINE_EXIT ) {
if ( stopmachine_state = = STOPMACHINE_DISABLE_IRQ
& & ! irqs_disabled ) {
local_irq_disable ( ) ;
2007-05-10 22:22:47 -07:00
hard_irq_disable ( ) ;
2005-04-16 15:20:36 -07:00
irqs_disabled = 1 ;
/* Ack: irqs disabled. */
2005-05-01 08:58:47 -07:00
smp_mb ( ) ; /* Must read state first. */
2005-04-16 15:20:36 -07:00
atomic_inc ( & stopmachine_thread_ack ) ;
} else if ( stopmachine_state = = STOPMACHINE_PREPARE
& & ! prepared ) {
/* Everyone is in place, hold CPU. */
preempt_disable ( ) ;
prepared = 1 ;
2005-05-01 08:58:47 -07:00
smp_mb ( ) ; /* Must read state first. */
2005-04-16 15:20:36 -07:00
atomic_inc ( & stopmachine_thread_ack ) ;
}
/* Yield in first stage: migration threads need to
* help our sisters onto their CPUs . */
if ( ! prepared & & ! irqs_disabled )
yield ( ) ;
2008-05-08 15:20:38 +02:00
cpu_relax ( ) ;
2005-04-16 15:20:36 -07:00
}
/* Ack: we are exiting. */
2005-05-01 08:58:47 -07:00
smp_mb ( ) ; /* Must read state first. */
2005-04-16 15:20:36 -07:00
atomic_inc ( & stopmachine_thread_ack ) ;
if ( irqs_disabled )
local_irq_enable ( ) ;
if ( prepared )
preempt_enable ( ) ;
return 0 ;
}
/* Change the thread state */
static void stopmachine_set_state ( enum stopmachine_state state )
{
atomic_set ( & stopmachine_thread_ack , 0 ) ;
2005-05-01 08:58:47 -07:00
smp_wmb ( ) ;
2005-04-16 15:20:36 -07:00
stopmachine_state = state ;
while ( atomic_read ( & stopmachine_thread_ack ) ! = stopmachine_num_threads )
cpu_relax ( ) ;
}
static int stop_machine ( void )
{
2006-07-03 17:32:22 -07:00
int i , ret = 0 ;
2005-04-16 15:20:36 -07:00
atomic_set ( & stopmachine_thread_ack , 0 ) ;
stopmachine_num_threads = 0 ;
stopmachine_state = STOPMACHINE_WAIT ;
for_each_online_cpu ( i ) {
2005-06-21 17:14:34 -07:00
if ( i = = raw_smp_processor_id ( ) )
2005-04-16 15:20:36 -07:00
continue ;
2006-07-03 17:32:22 -07:00
ret = kernel_thread ( stopmachine , ( void * ) ( long ) i , CLONE_KERNEL ) ;
if ( ret < 0 )
2005-04-16 15:20:36 -07:00
break ;
stopmachine_num_threads + + ;
}
/* Wait for them all to come to life. */
2008-05-08 15:20:38 +02:00
while ( atomic_read ( & stopmachine_thread_ack ) ! = stopmachine_num_threads ) {
2005-04-16 15:20:36 -07:00
yield ( ) ;
2008-05-08 15:20:38 +02:00
cpu_relax ( ) ;
}
2005-04-16 15:20:36 -07:00
/* If some failed, kill them all. */
if ( ret < 0 ) {
stopmachine_set_state ( STOPMACHINE_EXIT ) ;
return ret ;
}
/* Now they are all started, make them hold the CPUs, ready. */
2005-11-13 16:07:30 -08:00
preempt_disable ( ) ;
2005-04-16 15:20:36 -07:00
stopmachine_set_state ( STOPMACHINE_PREPARE ) ;
/* Make them disable irqs. */
2005-11-13 16:07:30 -08:00
local_irq_disable ( ) ;
2007-05-10 22:22:47 -07:00
hard_irq_disable ( ) ;
2005-04-16 15:20:36 -07:00
stopmachine_set_state ( STOPMACHINE_DISABLE_IRQ ) ;
return 0 ;
}
static void restart_machine ( void )
{
stopmachine_set_state ( STOPMACHINE_EXIT ) ;
local_irq_enable ( ) ;
2005-11-13 16:07:30 -08:00
preempt_enable_no_resched ( ) ;
2005-04-16 15:20:36 -07:00
}
2008-04-21 22:15:06 +00:00
struct stop_machine_data {
2005-04-16 15:20:36 -07:00
int ( * fn ) ( void * ) ;
void * data ;
struct completion done ;
} ;
static int do_stop ( void * _smdata )
{
struct stop_machine_data * smdata = _smdata ;
int ret ;
ret = stop_machine ( ) ;
if ( ret = = 0 ) {
ret = smdata - > fn ( smdata - > data ) ;
restart_machine ( ) ;
}
/* We're done: you can kthread_stop us now */
complete ( & smdata - > done ) ;
/* Wait for kthread_stop */
set_current_state ( TASK_INTERRUPTIBLE ) ;
while ( ! kthread_should_stop ( ) ) {
schedule ( ) ;
set_current_state ( TASK_INTERRUPTIBLE ) ;
}
__set_current_state ( TASK_RUNNING ) ;
return ret ;
}
struct task_struct * __stop_machine_run ( int ( * fn ) ( void * ) , void * data ,
unsigned int cpu )
{
2008-02-06 01:37:41 -08:00
static DEFINE_MUTEX ( stopmachine_mutex ) ;
2005-04-16 15:20:36 -07:00
struct stop_machine_data smdata ;
struct task_struct * p ;
smdata . fn = fn ;
smdata . data = data ;
init_completion ( & smdata . done ) ;
2008-02-06 01:37:41 -08:00
mutex_lock ( & stopmachine_mutex ) ;
2005-04-16 15:20:36 -07:00
/* If they don't care which CPU fn runs on, bind to any online one. */
if ( cpu = = NR_CPUS )
2005-06-21 17:14:34 -07:00
cpu = raw_smp_processor_id ( ) ;
2005-04-16 15:20:36 -07:00
p = kthread_create ( do_stop , & smdata , " kstopmachine " ) ;
if ( ! IS_ERR ( p ) ) {
2007-07-15 23:39:47 -07:00
struct sched_param param = { . sched_priority = MAX_RT_PRIO - 1 } ;
/* One high-prio thread per cpu. We'll do this one. */
sched_setscheduler ( p , SCHED_FIFO , & param ) ;
2005-04-16 15:20:36 -07:00
kthread_bind ( p , cpu ) ;
wake_up_process ( p ) ;
wait_for_completion ( & smdata . done ) ;
}
2008-02-06 01:37:41 -08:00
mutex_unlock ( & stopmachine_mutex ) ;
2005-04-16 15:20:36 -07:00
return p ;
}
int stop_machine_run ( int ( * fn ) ( void * ) , void * data , unsigned int cpu )
{
struct task_struct * p ;
int ret ;
/* No CPUs can come up or down during this. */
2008-01-25 21:08:02 +01:00
get_online_cpus ( ) ;
2005-04-16 15:20:36 -07:00
p = __stop_machine_run ( fn , data , cpu ) ;
if ( ! IS_ERR ( p ) )
ret = kthread_stop ( p ) ;
else
ret = PTR_ERR ( p ) ;
2008-01-25 21:08:02 +01:00
put_online_cpus ( ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
2007-05-08 00:25:08 -07:00
EXPORT_SYMBOL_GPL ( stop_machine_run ) ;