aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/sched.c
diff options
context:
space:
mode:
authorBenjamin Herrenschmidt <benh@kernel.crashing.org>2010-04-16 17:20:00 -0400
committerIngo Molnar <mingo@elte.hu>2010-04-23 05:00:28 -0400
commit4b402210486c6414fe5fbfd85934a0a22da56b04 (patch)
tree23e7052c710e0443ccd895a1000b2bdd7723ab51 /kernel/sched.c
parentd5a30458a90597915977f06e79406b664a41b8ac (diff)
mutex: Don't spin when the owner CPU is offline or other weird cases
Due to recent load-balancer changes that delay the task migration to the next wakeup, the adaptive mutex spinning ends up in a live lock when the owner's CPU gets offlined because the cpu_online() check lives before the owner running check. This patch changes mutex_spin_on_owner() to return 0 (don't spin) in any case where we aren't sure about the owner struct validity or CPU number, and if the said CPU is offline. There is no point going back & re-evaluate spinning in corner cases like that, let's just go to sleep. Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org> Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl> LKML-Reference: <1271212509.13059.135.camel@pasglop> Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'kernel/sched.c')
-rw-r--r--kernel/sched.c8
1 files changed, 4 insertions, 4 deletions
diff --git a/kernel/sched.c b/kernel/sched.c
index 6af210a7de70..de0bd26e520a 100644
--- a/kernel/sched.c
+++ b/kernel/sched.c
@@ -3780,7 +3780,7 @@ int mutex_spin_on_owner(struct mutex *lock, struct thread_info *owner)
3780 * the mutex owner just released it and exited. 3780 * the mutex owner just released it and exited.
3781 */ 3781 */
3782 if (probe_kernel_address(&owner->cpu, cpu)) 3782 if (probe_kernel_address(&owner->cpu, cpu))
3783 goto out; 3783 return 0;
3784#else 3784#else
3785 cpu = owner->cpu; 3785 cpu = owner->cpu;
3786#endif 3786#endif
@@ -3790,14 +3790,14 @@ int mutex_spin_on_owner(struct mutex *lock, struct thread_info *owner)
3790 * the cpu field may no longer be valid. 3790 * the cpu field may no longer be valid.
3791 */ 3791 */
3792 if (cpu >= nr_cpumask_bits) 3792 if (cpu >= nr_cpumask_bits)
3793 goto out; 3793 return 0;
3794 3794
3795 /* 3795 /*
3796 * We need to validate that we can do a 3796 * We need to validate that we can do a
3797 * get_cpu() and that we have the percpu area. 3797 * get_cpu() and that we have the percpu area.
3798 */ 3798 */
3799 if (!cpu_online(cpu)) 3799 if (!cpu_online(cpu))
3800 goto out; 3800 return 0;
3801 3801
3802 rq = cpu_rq(cpu); 3802 rq = cpu_rq(cpu);
3803 3803
@@ -3816,7 +3816,7 @@ int mutex_spin_on_owner(struct mutex *lock, struct thread_info *owner)
3816 3816
3817 cpu_relax(); 3817 cpu_relax();
3818 } 3818 }
3819out: 3819
3820 return 1; 3820 return 1;
3821} 3821}
3822#endif 3822#endif