diff options
Diffstat (limited to 'kernel/stop_machine.c')
| -rw-r--r-- | kernel/stop_machine.c | 29 | 
1 files changed, 24 insertions, 5 deletions
| diff --git a/kernel/stop_machine.c b/kernel/stop_machine.c index f89014a2c238..067cb83f37ea 100644 --- a/kernel/stop_machine.c +++ b/kernel/stop_machine.c @@ -81,6 +81,7 @@ static bool cpu_stop_queue_work(unsigned int cpu, struct cpu_stop_work *work)  	unsigned long flags;  	bool enabled; +	preempt_disable();  	raw_spin_lock_irqsave(&stopper->lock, flags);  	enabled = stopper->enabled;  	if (enabled) @@ -90,6 +91,7 @@ static bool cpu_stop_queue_work(unsigned int cpu, struct cpu_stop_work *work)  	raw_spin_unlock_irqrestore(&stopper->lock, flags);  	wake_up_q(&wakeq); +	preempt_enable();  	return enabled;  } @@ -236,13 +238,24 @@ static int cpu_stop_queue_two_works(int cpu1, struct cpu_stop_work *work1,  	struct cpu_stopper *stopper2 = per_cpu_ptr(&cpu_stopper, cpu2);  	DEFINE_WAKE_Q(wakeq);  	int err; +  retry: +	/* +	 * The waking up of stopper threads has to happen in the same +	 * scheduling context as the queueing.  Otherwise, there is a +	 * possibility of one of the above stoppers being woken up by another +	 * CPU, and preempting us. This will cause us to not wake up the other +	 * stopper forever. +	 */ +	preempt_disable();  	raw_spin_lock_irq(&stopper1->lock);  	raw_spin_lock_nested(&stopper2->lock, SINGLE_DEPTH_NESTING); -	err = -ENOENT; -	if (!stopper1->enabled || !stopper2->enabled) +	if (!stopper1->enabled || !stopper2->enabled) { +		err = -ENOENT;  		goto unlock; +	} +  	/*  	 * Ensure that if we race with __stop_cpus() the stoppers won't get  	 * queued up in reverse order leading to system deadlock. @@ -253,24 +266,30 @@ retry:  	 * It can be falsely true but it is safe to spin until it is cleared,  	 * queue_stop_cpus_work() does everything under preempt_disable().  	 */ -	err = -EDEADLK; -	if (unlikely(stop_cpus_in_progress)) -			goto unlock; +	if (unlikely(stop_cpus_in_progress)) { +		err = -EDEADLK; +		goto unlock; +	}  	err = 0;  	__cpu_stop_queue_work(stopper1, work1, &wakeq);  	__cpu_stop_queue_work(stopper2, work2, &wakeq); +  unlock:  	raw_spin_unlock(&stopper2->lock);  	raw_spin_unlock_irq(&stopper1->lock);  	if (unlikely(err == -EDEADLK)) { +		preempt_enable(); +  		while (stop_cpus_in_progress)  			cpu_relax(); +  		goto retry;  	}  	wake_up_q(&wakeq); +	preempt_enable();  	return err;  } | 
