sched: RT-balance, optimize cpu search

This patch removes several cpumask operations by keeping track
of the first of the CPUS that is of the lowest priority. When
the search for the lowest priority runqueue is completed, all
the bits up to the first CPU with the lowest priority runqueue
is cleared.

Signed-off-by: Steven Rostedt <srostedt@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
This commit is contained in:
Steven Rostedt 2008-01-25 21:08:13 +01:00 committed by Ingo Molnar
parent 06f90dbd76
commit 610bf05645

View File

@ -296,29 +296,36 @@ static struct task_struct *pick_next_highest_task_rt(struct rq *rq,
} }
static DEFINE_PER_CPU(cpumask_t, local_cpu_mask); static DEFINE_PER_CPU(cpumask_t, local_cpu_mask);
static DEFINE_PER_CPU(cpumask_t, valid_cpu_mask);
static int find_lowest_cpus(struct task_struct *task, cpumask_t *lowest_mask) static int find_lowest_cpus(struct task_struct *task, cpumask_t *lowest_mask)
{ {
int cpu;
cpumask_t *valid_mask = &__get_cpu_var(valid_cpu_mask);
int lowest_prio = -1; int lowest_prio = -1;
int lowest_cpu = -1;
int count = 0; int count = 0;
int cpu;
cpus_clear(*lowest_mask); cpus_and(*lowest_mask, cpu_online_map, task->cpus_allowed);
cpus_and(*valid_mask, cpu_online_map, task->cpus_allowed);
/* /*
* Scan each rq for the lowest prio. * Scan each rq for the lowest prio.
*/ */
for_each_cpu_mask(cpu, *valid_mask) { for_each_cpu_mask(cpu, *lowest_mask) {
struct rq *rq = cpu_rq(cpu); struct rq *rq = cpu_rq(cpu);
/* We look for lowest RT prio or non-rt CPU */ /* We look for lowest RT prio or non-rt CPU */
if (rq->rt.highest_prio >= MAX_RT_PRIO) { if (rq->rt.highest_prio >= MAX_RT_PRIO) {
if (count) /*
* if we already found a low RT queue
* and now we found this non-rt queue
* clear the mask and set our bit.
* Otherwise just return the queue as is
* and the count==1 will cause the algorithm
* to use the first bit found.
*/
if (lowest_cpu != -1) {
cpus_clear(*lowest_mask); cpus_clear(*lowest_mask);
cpu_set(rq->cpu, *lowest_mask); cpu_set(rq->cpu, *lowest_mask);
}
return 1; return 1;
} }
@ -328,13 +335,29 @@ static int find_lowest_cpus(struct task_struct *task, cpumask_t *lowest_mask)
if (rq->rt.highest_prio > lowest_prio) { if (rq->rt.highest_prio > lowest_prio) {
/* new low - clear old data */ /* new low - clear old data */
lowest_prio = rq->rt.highest_prio; lowest_prio = rq->rt.highest_prio;
if (count) { lowest_cpu = cpu;
cpus_clear(*lowest_mask);
count = 0; count = 0;
} }
}
cpu_set(rq->cpu, *lowest_mask);
count++; count++;
} else
cpu_clear(cpu, *lowest_mask);
}
/*
* Clear out all the set bits that represent
* runqueues that were of higher prio than
* the lowest_prio.
*/
if (lowest_cpu > 0) {
/*
* Perhaps we could add another cpumask op to
* zero out bits. Like cpu_zero_bits(cpumask, nrbits);
* Then that could be optimized to use memset and such.
*/
for_each_cpu_mask(cpu, *lowest_mask) {
if (cpu >= lowest_cpu)
break;
cpu_clear(cpu, *lowest_mask);
} }
} }