aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/sched.c
diff options
context:
space:
mode:
authorThomas Gleixner <tglx@linutronix.de>2010-02-16 10:48:56 -0500
committerThomas Gleixner <tglx@linutronix.de>2010-02-16 10:48:56 -0500
commit6e40f5bbbc734231bc5809d3eb785e3c21f275d7 (patch)
tree8520fc84d9643502ebe86233cefcd774e2babf72 /kernel/sched.c
parent301ba0457f1ed853fc08e57785f8c87fe7e49c68 (diff)
parent0970d2992dfd7d5ec2c787417cf464f01eeaf42a (diff)
Merge branch 'sched/urgent' into sched/core
Conflicts: kernel/sched.c Necessary due to the urgent fixes which conflict with the code move from sched.c to sched_fair.c Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'kernel/sched.c')
-rw-r--r--kernel/sched.c73
1 files changed, 46 insertions, 27 deletions
diff --git a/kernel/sched.c b/kernel/sched.c
index bae6fcfe6d75..af5fa239804d 100644
--- a/kernel/sched.c
+++ b/kernel/sched.c
@@ -898,16 +898,33 @@ static inline void finish_lock_switch(struct rq *rq, struct task_struct *prev)
898#endif /* __ARCH_WANT_UNLOCKED_CTXSW */ 898#endif /* __ARCH_WANT_UNLOCKED_CTXSW */
899 899
900/* 900/*
901 * Check whether the task is waking, we use this to synchronize against
902 * ttwu() so that task_cpu() reports a stable number.
903 *
904 * We need to make an exception for PF_STARTING tasks because the fork
905 * path might require task_rq_lock() to work, eg. it can call
906 * set_cpus_allowed_ptr() from the cpuset clone_ns code.
907 */
908static inline int task_is_waking(struct task_struct *p)
909{
910 return unlikely((p->state == TASK_WAKING) && !(p->flags & PF_STARTING));
911}
912
913/*
901 * __task_rq_lock - lock the runqueue a given task resides on. 914 * __task_rq_lock - lock the runqueue a given task resides on.
902 * Must be called interrupts disabled. 915 * Must be called interrupts disabled.
903 */ 916 */
904static inline struct rq *__task_rq_lock(struct task_struct *p) 917static inline struct rq *__task_rq_lock(struct task_struct *p)
905 __acquires(rq->lock) 918 __acquires(rq->lock)
906{ 919{
920 struct rq *rq;
921
907 for (;;) { 922 for (;;) {
908 struct rq *rq = task_rq(p); 923 while (task_is_waking(p))
924 cpu_relax();
925 rq = task_rq(p);
909 raw_spin_lock(&rq->lock); 926 raw_spin_lock(&rq->lock);
910 if (likely(rq == task_rq(p))) 927 if (likely(rq == task_rq(p) && !task_is_waking(p)))
911 return rq; 928 return rq;
912 raw_spin_unlock(&rq->lock); 929 raw_spin_unlock(&rq->lock);
913 } 930 }
@@ -924,10 +941,12 @@ static struct rq *task_rq_lock(struct task_struct *p, unsigned long *flags)
924 struct rq *rq; 941 struct rq *rq;
925 942
926 for (;;) { 943 for (;;) {
944 while (task_is_waking(p))
945 cpu_relax();
927 local_irq_save(*flags); 946 local_irq_save(*flags);
928 rq = task_rq(p); 947 rq = task_rq(p);
929 raw_spin_lock(&rq->lock); 948 raw_spin_lock(&rq->lock);
930 if (likely(rq == task_rq(p))) 949 if (likely(rq == task_rq(p) && !task_is_waking(p)))
931 return rq; 950 return rq;
932 raw_spin_unlock_irqrestore(&rq->lock, *flags); 951 raw_spin_unlock_irqrestore(&rq->lock, *flags);
933 } 952 }
@@ -2374,14 +2393,27 @@ static int try_to_wake_up(struct task_struct *p, unsigned int state,
2374 __task_rq_unlock(rq); 2393 __task_rq_unlock(rq);
2375 2394
2376 cpu = select_task_rq(p, SD_BALANCE_WAKE, wake_flags); 2395 cpu = select_task_rq(p, SD_BALANCE_WAKE, wake_flags);
2377 if (cpu != orig_cpu) 2396 if (cpu != orig_cpu) {
2397 /*
2398 * Since we migrate the task without holding any rq->lock,
2399 * we need to be careful with task_rq_lock(), since that
2400 * might end up locking an invalid rq.
2401 */
2378 set_task_cpu(p, cpu); 2402 set_task_cpu(p, cpu);
2403 }
2379 2404
2380 rq = __task_rq_lock(p); 2405 rq = cpu_rq(cpu);
2406 raw_spin_lock(&rq->lock);
2381 update_rq_clock(rq); 2407 update_rq_clock(rq);
2382 2408
2409 /*
2410 * We migrated the task without holding either rq->lock, however
2411 * since the task is not on the task list itself, nobody else
2412 * will try and migrate the task, hence the rq should match the
2413 * cpu we just moved it to.
2414 */
2415 WARN_ON(task_cpu(p) != cpu);
2383 WARN_ON(p->state != TASK_WAKING); 2416 WARN_ON(p->state != TASK_WAKING);
2384 cpu = task_cpu(p);
2385 2417
2386#ifdef CONFIG_SCHEDSTATS 2418#ifdef CONFIG_SCHEDSTATS
2387 schedstat_inc(rq, ttwu_count); 2419 schedstat_inc(rq, ttwu_count);
@@ -2613,7 +2645,7 @@ void wake_up_new_task(struct task_struct *p, unsigned long clone_flags)
2613{ 2645{
2614 unsigned long flags; 2646 unsigned long flags;
2615 struct rq *rq; 2647 struct rq *rq;
2616 int cpu __maybe_unused = get_cpu(); 2648 int cpu = get_cpu();
2617 2649
2618#ifdef CONFIG_SMP 2650#ifdef CONFIG_SMP
2619 /* 2651 /*
@@ -2629,7 +2661,13 @@ void wake_up_new_task(struct task_struct *p, unsigned long clone_flags)
2629 set_task_cpu(p, cpu); 2661 set_task_cpu(p, cpu);
2630#endif 2662#endif
2631 2663
2632 rq = task_rq_lock(p, &flags); 2664 /*
2665 * Since the task is not on the rq and we still have TASK_WAKING set
2666 * nobody else will migrate this task.
2667 */
2668 rq = cpu_rq(cpu);
2669 raw_spin_lock_irqsave(&rq->lock, flags);
2670
2633 BUG_ON(p->state != TASK_WAKING); 2671 BUG_ON(p->state != TASK_WAKING);
2634 p->state = TASK_RUNNING; 2672 p->state = TASK_RUNNING;
2635 update_rq_clock(rq); 2673 update_rq_clock(rq);
@@ -5302,27 +5340,8 @@ int set_cpus_allowed_ptr(struct task_struct *p, const struct cpumask *new_mask)
5302 struct rq *rq; 5340 struct rq *rq;
5303 int ret = 0; 5341 int ret = 0;
5304 5342
5305 /*
5306 * Since we rely on wake-ups to migrate sleeping tasks, don't change
5307 * the ->cpus_allowed mask from under waking tasks, which would be
5308 * possible when we change rq->lock in ttwu(), so synchronize against
5309 * TASK_WAKING to avoid that.
5310 *
5311 * Make an exception for freshly cloned tasks, since cpuset namespaces
5312 * might move the task about, we have to validate the target in
5313 * wake_up_new_task() anyway since the cpu might have gone away.
5314 */
5315again:
5316 while (p->state == TASK_WAKING && !(p->flags & PF_STARTING))
5317 cpu_relax();
5318
5319 rq = task_rq_lock(p, &flags); 5343 rq = task_rq_lock(p, &flags);
5320 5344
5321 if (p->state == TASK_WAKING && !(p->flags & PF_STARTING)) {
5322 task_rq_unlock(rq, &flags);
5323 goto again;
5324 }
5325
5326 if (!cpumask_intersects(new_mask, cpu_active_mask)) { 5345 if (!cpumask_intersects(new_mask, cpu_active_mask)) {
5327 ret = -EINVAL; 5346 ret = -EINVAL;
5328 goto out; 5347 goto out;