aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGregory Haskins <ghaskins@novell.com>2008-12-29 09:39:52 -0500
committerGregory Haskins <ghaskins@novell.com>2008-12-29 09:39:52 -0500
commit967fc04671feea4dbf780c9e55a0bc8fcf68a14e (patch)
tree223f2bb8c59138cc70fbb0e438ae27819ebe1a92
parent8f45e2b516201d1bf681e6026fa5276385def565 (diff)
sched: add sched_class->needs_post_schedule() member
We currently run class->post_schedule() outside of the rq->lock, which means that we need to test for the need to post_schedule outside of the lock to avoid a forced reacquistion. This is currently not a problem as we only look at rq->rt.overloaded. However, we want to enhance this going forward to look at more state to reduce the need to post_schedule to a bare minimum set. Therefore, we introduce a new member-func called needs_post_schedule() which tests for the post_schedule condtion without actually performing the work. Therefore it is safe to call this function before the rq->lock is released, because we are guaranteed not to drop the lock at an intermediate point (such as what post_schedule() may do). We will use this later in the series [ rostedt: removed paranoid BUG_ON ] Signed-off-by: Gregory Haskins <ghaskins@novell.com>
-rw-r--r--include/linux/sched.h1
-rw-r--r--kernel/sched.c8
-rw-r--r--kernel/sched_rt.c24
3 files changed, 22 insertions, 11 deletions
diff --git a/include/linux/sched.h b/include/linux/sched.h
index e5f928a079e8..836a86c32a65 100644
--- a/include/linux/sched.h
+++ b/include/linux/sched.h
@@ -1012,6 +1012,7 @@ struct sched_class {
1012 struct rq *busiest, struct sched_domain *sd, 1012 struct rq *busiest, struct sched_domain *sd,
1013 enum cpu_idle_type idle); 1013 enum cpu_idle_type idle);
1014 void (*pre_schedule) (struct rq *this_rq, struct task_struct *task); 1014 void (*pre_schedule) (struct rq *this_rq, struct task_struct *task);
1015 int (*needs_post_schedule) (struct rq *this_rq);
1015 void (*post_schedule) (struct rq *this_rq); 1016 void (*post_schedule) (struct rq *this_rq);
1016 void (*task_wake_up) (struct rq *this_rq, struct task_struct *task); 1017 void (*task_wake_up) (struct rq *this_rq, struct task_struct *task);
1017 1018
diff --git a/kernel/sched.c b/kernel/sched.c
index 8fca364f3593..3acbad8991a2 100644
--- a/kernel/sched.c
+++ b/kernel/sched.c
@@ -2621,6 +2621,12 @@ static void finish_task_switch(struct rq *rq, struct task_struct *prev)
2621{ 2621{
2622 struct mm_struct *mm = rq->prev_mm; 2622 struct mm_struct *mm = rq->prev_mm;
2623 long prev_state; 2623 long prev_state;
2624#ifdef CONFIG_SMP
2625 int post_schedule = 0;
2626
2627 if (current->sched_class->needs_post_schedule)
2628 post_schedule = current->sched_class->needs_post_schedule(rq);
2629#endif
2624 2630
2625 rq->prev_mm = NULL; 2631 rq->prev_mm = NULL;
2626 2632
@@ -2639,7 +2645,7 @@ static void finish_task_switch(struct rq *rq, struct task_struct *prev)
2639 finish_arch_switch(prev); 2645 finish_arch_switch(prev);
2640 finish_lock_switch(rq, prev); 2646 finish_lock_switch(rq, prev);
2641#ifdef CONFIG_SMP 2647#ifdef CONFIG_SMP
2642 if (current->sched_class->post_schedule) 2648 if (post_schedule)
2643 current->sched_class->post_schedule(rq); 2649 current->sched_class->post_schedule(rq);
2644#endif 2650#endif
2645 2651
diff --git a/kernel/sched_rt.c b/kernel/sched_rt.c
index 8d33843cb2c4..b0b6ea4ed674 100644
--- a/kernel/sched_rt.c
+++ b/kernel/sched_rt.c
@@ -1290,20 +1290,23 @@ static void pre_schedule_rt(struct rq *rq, struct task_struct *prev)
1290 pull_rt_task(rq); 1290 pull_rt_task(rq);
1291} 1291}
1292 1292
1293/*
1294 * assumes rq->lock is held
1295 */
1296static int needs_post_schedule_rt(struct rq *rq)
1297{
1298 return rq->rt.overloaded ? 1 : 0;
1299}
1300
1293static void post_schedule_rt(struct rq *rq) 1301static void post_schedule_rt(struct rq *rq)
1294{ 1302{
1295 /* 1303 /*
1296 * If we have more than one rt_task queued, then 1304 * This is only called if needs_post_schedule_rt() indicates that
1297 * see if we can push the other rt_tasks off to other CPUS. 1305 * we need to push tasks away
1298 * Note we may release the rq lock, and since
1299 * the lock was owned by prev, we need to release it
1300 * first via finish_lock_switch and then reaquire it here.
1301 */ 1306 */
1302 if (unlikely(rq->rt.overloaded)) { 1307 spin_lock_irq(&rq->lock);
1303 spin_lock_irq(&rq->lock); 1308 push_rt_tasks(rq);
1304 push_rt_tasks(rq); 1309 spin_unlock_irq(&rq->lock);
1305 spin_unlock_irq(&rq->lock);
1306 }
1307} 1310}
1308 1311
1309/* 1312/*
@@ -1557,6 +1560,7 @@ static const struct sched_class rt_sched_class = {
1557 .rq_online = rq_online_rt, 1560 .rq_online = rq_online_rt,
1558 .rq_offline = rq_offline_rt, 1561 .rq_offline = rq_offline_rt,
1559 .pre_schedule = pre_schedule_rt, 1562 .pre_schedule = pre_schedule_rt,
1563 .needs_post_schedule = needs_post_schedule_rt,
1560 .post_schedule = post_schedule_rt, 1564 .post_schedule = post_schedule_rt,
1561 .task_wake_up = task_wake_up_rt, 1565 .task_wake_up = task_wake_up_rt,
1562 .switched_from = switched_from_rt, 1566 .switched_from = switched_from_rt,