提交 c61037e9 编写于 作者: P Peter Zijlstra 提交者: Ingo Molnar

sched/fair: Fix the group_capacity computation

Do away with 'phantom' cores due to N*frac(smt_power) >= 1 by limiting
the capacity to the actual number of cores.

The assumption of 1 < smt_power < 2 is an actual requirement because
of what SMT is so this should work regardless of the SMT
implementation.

It can still be defeated by creative use of cpu hotplug, but if you're
one of those freaks, you get to live with it.
Signed-off-by: NPeter Zijlstra <peterz@infradead.org>
Acked-by: NVincent Guittot <vincent.guitto@linaro.org>
Link: http://lkml.kernel.org/n/tip-dczmbi8tfgixacg1ji2av1un@git.kernel.orgSigned-off-by: NIngo Molnar <mingo@kernel.org>
上级 b37d9316
...@@ -4556,18 +4556,24 @@ static inline int sg_imbalanced(struct sched_group *group) ...@@ -4556,18 +4556,24 @@ static inline int sg_imbalanced(struct sched_group *group)
/* /*
* Compute the group capacity. * Compute the group capacity.
* *
* For now the capacity is simply the number of power units in the group_power. * Avoid the issue where N*frac(smt_power) >= 1 creates 'phantom' cores by
* A power unit represents a full core. * first dividing out the smt factor and computing the actual number of cores
* * and limit power unit capacity with that.
* This has an issue where N*frac(smt_power) >= 1, in that case we'll see extra
* 'cores' that aren't actually there.
*/ */
static inline int sg_capacity(struct lb_env *env, struct sched_group *group) static inline int sg_capacity(struct lb_env *env, struct sched_group *group)
{ {
unsigned int capacity, smt, cpus;
unsigned int power, power_orig;
power = group->sgp->power;
power_orig = group->sgp->power_orig;
cpus = group->group_weight;
unsigned int power = group->sgp->power; /* smt := ceil(cpus / power), assumes: 1 < smt_power < 2 */
unsigned int capacity = DIV_ROUND_CLOSEST(power, SCHED_POWER_SCALE); smt = DIV_ROUND_UP(SCHED_POWER_SCALE * cpus, power_orig);
capacity = cpus / smt; /* cores */
capacity = min_t(unsigned, capacity, DIV_ROUND_CLOSEST(power, SCHED_POWER_SCALE));
if (!capacity) if (!capacity)
capacity = fix_small_capacity(env->sd, group); capacity = fix_small_capacity(env->sd, group);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册