Signed-off-by: John Kacur Index: linux-2.6.24.7-rt14/include/linux/sched.h =================================================================== --- linux-2.6.24.7-rt14.orig/include/linux/sched.h +++ linux-2.6.24.7-rt14/include/linux/sched.h @@ -897,6 +897,11 @@ struct uts_namespace; struct rq; struct sched_domain; +#define ENQUEUE_WAKEUP 0x01 +#define ENQUEUE_HEAD 0x02 + +#define DEQUEUE_SLEEP 0x01 + struct sched_class { const struct sched_class *next; Index: linux-2.6.24.7-rt14/kernel/sched.c =================================================================== --- linux-2.6.24.7-rt14.orig/kernel/sched.c +++ linux-2.6.24.7-rt14/kernel/sched.c @@ -1759,7 +1759,7 @@ out_activate: else schedstat_inc(p, se.nr_wakeups_remote); update_rq_clock(rq); - activate_task(rq, p, 1); + activate_task(rq, p, ENQUEUE_WAKEUP); check_preempt_curr(rq, p); success = 1; @@ -3968,7 +3968,7 @@ asmlinkage void __sched __schedule(void) prev->state = TASK_RUNNING; } else { touch_softlockup_watchdog(); - deactivate_task(rq, prev, 1); + deactivate_task(rq, prev, DEQUEUE_SLEEP); } switch_count = &prev->nvcsw; } @@ -4431,7 +4431,7 @@ EXPORT_SYMBOL(sleep_on_timeout); void task_setprio(struct task_struct *p, int prio) { unsigned long flags; - int oldprio, prev_resched, on_rq, running; + int oldprio, prev_resched, on_rq, running, down; struct rq *rq; const struct sched_class *prev_class = p->sched_class; @@ -4472,6 +4472,7 @@ void task_setprio(struct task_struct *p, else p->sched_class = &fair_sched_class; + down = (prio > p->prio) ? ENQUEUE_HEAD : 0; p->prio = prio; // trace_special_pid(p->pid, __PRIO(oldprio), PRIO(p)); @@ -4480,7 +4481,7 @@ void task_setprio(struct task_struct *p, if (running) p->sched_class->set_curr_task(rq); if (on_rq) { - enqueue_task(rq, p, 0); + enqueue_task(rq, p, down); check_class_changed(rq, p, prev_class, oldprio, running); } // trace_special(prev_resched, _need_resched(), 0); Index: linux-2.6.24.7-rt14/kernel/sched_fair.c =================================================================== --- linux-2.6.24.7-rt14.orig/kernel/sched_fair.c +++ linux-2.6.24.7-rt14/kernel/sched_fair.c @@ -764,6 +764,7 @@ static void enqueue_task_fair(struct rq { struct cfs_rq *cfs_rq; struct sched_entity *se = &p->se; + wakeup &= ENQUEUE_WAKEUP; for_each_sched_entity(se) { if (se->on_rq) @@ -783,6 +784,7 @@ static void dequeue_task_fair(struct rq { struct cfs_rq *cfs_rq; struct sched_entity *se = &p->se; + sleep &= DEQUEUE_SLEEP; for_each_sched_entity(se) { cfs_rq = cfs_rq_of(se); Index: linux-2.6.24.7-rt14/kernel/sched_rt.c =================================================================== --- linux-2.6.24.7-rt14.orig/kernel/sched_rt.c +++ linux-2.6.24.7-rt14/kernel/sched_rt.c @@ -185,7 +185,11 @@ static void enqueue_task_rt(struct rq *r { struct rt_prio_array *array = &rq->rt.active; - list_add_tail(&p->run_list, array->queue + p->prio); + if (unlikely(wakeup & ENQUEUE_HEAD)) + list_add(&p->run_list, array->queue + p->prio); + else + list_add_tail(&p->run_list, array->queue + p->prio); + __set_bit(p->prio, array->bitmap); inc_rt_tasks(p, rq);