Hi Morten,

On 07/07/2015 11:24 AM, Morten Rasmussen wrote:
> In mainline find_idlest_group() selects the wake-up target group purely
> based on group load which leads to suboptimal choices in low load
> scenarios. An idle group with reduced capacity (due to RT tasks or
> different cpu type) isn't necessarily a better target than a lightly
> loaded group with higher capacity.
> 
> The patch adds spare capacity as an additional group selection
> parameter. The target group is now selected based on the following
> criteria listed by highest priority first:
> 
> 1. If energy-aware scheduling is enabled the group with the lowest
> capacity containing a cpu with enough spare capacity to accommodate the
> task (with a bit to spare) is selected if such exists.
> 
> 2. Return the group with the cpu with most spare capacity and this
> capacity is significant if such group exists. Significant spare capacity
> is currently at least 20% to spare.
> 
> 3. Return the group with the lowest load, unless it is the local group
> in which case NULL is returned and the search is continued at the next
> (lower) level.
> 
> cc: Ingo Molnar <mi...@redhat.com>
> cc: Peter Zijlstra <pet...@infradead.org>
> 
> Signed-off-by: Morten Rasmussen <morten.rasmus...@arm.com>
> ---
>  kernel/sched/fair.c | 18 ++++++++++++++++--
>  1 file changed, 16 insertions(+), 2 deletions(-)
> 
> diff --git a/kernel/sched/fair.c b/kernel/sched/fair.c
> index b0294f0..0f7dbda4 100644
> --- a/kernel/sched/fair.c
> +++ b/kernel/sched/fair.c
> @@ -5247,9 +5247,10 @@ find_idlest_group(struct sched_domain *sd, struct 
> task_struct *p,
>                 int this_cpu, int sd_flag)
>  {
>       struct sched_group *idlest = NULL, *group = sd->groups;
> -     struct sched_group *fit_group = NULL;
> +     struct sched_group *fit_group = NULL, *spare_group = NULL;
>       unsigned long min_load = ULONG_MAX, this_load = 0;
>       unsigned long fit_capacity = ULONG_MAX;
> +     unsigned long max_spare_capacity = capacity_margin - SCHED_LOAD_SCALE;
>       int load_idx = sd->forkexec_idx;
>       int imbalance = 100 + (sd->imbalance_pct-100)/2;
>  
> @@ -5257,7 +5258,7 @@ find_idlest_group(struct sched_domain *sd, struct 
> task_struct *p,
>               load_idx = sd->wake_idx;
>  
>       do {
> -             unsigned long load, avg_load;
> +             unsigned long load, avg_load, spare_capacity;
>               int local_group;
>               int i;
>  
> @@ -5290,6 +5291,16 @@ find_idlest_group(struct sched_domain *sd, struct 
> task_struct *p,
>                               fit_capacity = capacity_of(i);
>                               fit_group = group;
>                       }
> +
> +                     /*
> +                      * Look for group which has most spare capacity on a
> +                      * single cpu.
> +                      */
> +                     spare_capacity = capacity_of(i) - get_cpu_usage(i);
> +                     if (spare_capacity > max_spare_capacity) {
> +                             max_spare_capacity = spare_capacity;
> +                             spare_group = group;
> +                     }

Another minor buglet: get_cpu_usage(i) here could be > capacity_of(i)
because usage is bounded by capacity_orig_of(i). Should it be bounded by
capacity_of() instead?

>               }
>  
>               /* Adjust by relative CPU capacity of the group */
> @@ -5306,6 +5317,9 @@ find_idlest_group(struct sched_domain *sd, struct 
> task_struct *p,
>       if (fit_group)
>               return fit_group;
>  
> +     if (spare_group)
> +             return spare_group;
> +
>       if (!idlest || 100*this_load < imbalance*min_load)
>               return NULL;
>       return idlest;
> 

Thanks,
-Sai
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to