sched/fair: Rewrite runnable load and utilization average tracking
The idea of runnable load average (let runnable time contribute to weight) was proposed by Paul Turner and Ben Segall, and it is still followed by this rewrite. This rewrite aims to solve the following issues: 1. cfs_rq's load average (namely runnable_load_avg and blocked_load_avg) is updated at the granularity of an entity at a time, which results in the cfs_rq's load average is stale or partially updated: at any time, only one entity is up to date, all other entities are effectively lagging behind. This is undesirable. To illustrate, if we have n runnable entities in the cfs_rq, as time elapses, they certainly become outdated: t0: cfs_rq { e1_old, e2_old, ..., en_old } and when we update: t1: update e1, then we have cfs_rq { e1_new, e2_old, ..., en_old } t2: update e2, then we have cfs_rq { e1_old, e2_new, ..., en_old } ... We solve this by combining all runnable entities' load averages together in cfs_rq's avg, and update the cfs_rq's avg as a whole. This is based on the fact that if we regard the update as a function, then: w * update(e) = update(w * e) and update(e1) + update(e2) = update(e1 + e2), then w1 * update(e1) + w2 * update(e2) = update(w1 * e1 + w2 * e2) therefore, by this rewrite, we have an entirely updated cfs_rq at the time we update it: t1: update cfs_rq { e1_new, e2_new, ..., en_new } t2: update cfs_rq { e1_new, e2_new, ..., en_new } ... 2. cfs_rq's load average is different between top rq->cfs_rq and other task_group's per CPU cfs_rqs in whether or not blocked_load_average contributes to the load. The basic idea behind runnable load average (the same for utilization) is that the blocked state is taken into account as opposed to only accounting for the currently runnable state. Therefore, the average should include both the runnable/running and blocked load averages. This rewrite does that. In addition, we also combine runnable/running and blocked averages of all entities into the cfs_rq's average, and update it together at once. This is based on the fact that: update(runnable) + update(blocked) = update(runnable + blocked) This significantly reduces the code as we don't need to separately maintain/update runnable/running load and blocked load. 3. How task_group entities' share is calculated is complex and imprecise. We reduce the complexity in this rewrite to allow a very simple rule: the task_group's load_avg is aggregated from its per CPU cfs_rqs's load_avgs. Then group entity's weight is simply proportional to its own cfs_rq's load_avg / task_group's load_avg. To illustrate, if a task_group has { cfs_rq1, cfs_rq2, ..., cfs_rqn }, then, task_group_avg = cfs_rq1_avg + cfs_rq2_avg + ... + cfs_rqn_avg, then cfs_rqx's entity's share = cfs_rqx_avg / task_group_avg * task_group's share To sum up, this rewrite in principle is equivalent to the current one, but fixes the issues described above. Turns out, it significantly reduces the code complexity and hence increases clarity and efficiency. In addition, the new averages are more smooth/continuous (no spurious spikes and valleys) and updated more consistently and quickly to reflect the load dynamics. As a result, we have less load tracking overhead, better performance, and especially better power efficiency due to more balanced load. Signed-off-by: Yuyang Du <yuyang.du@intel.com> Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Mike Galbraith <efault@gmx.de> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: arjan@linux.intel.com Cc: bsegall@google.com Cc: dietmar.eggemann@arm.com Cc: fengguang.wu@intel.com Cc: len.brown@intel.com Cc: morten.rasmussen@arm.com Cc: pjt@google.com Cc: rafael.j.wysocki@intel.com Cc: umgwanakikbuti@gmail.com Cc: vincent.guittot@linaro.org Link: http://lkml.kernel.org/r/1436918682-4971-3-git-send-email-yuyang.du@intel.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
This commit is contained in:
parent
cd126afe83
commit
9d89c257df
@ -1175,29 +1175,24 @@ struct load_weight {
|
|||||||
u32 inv_weight;
|
u32 inv_weight;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The load_avg/util_avg accumulates an infinite geometric series.
|
||||||
|
* 1) load_avg factors the amount of time that a sched_entity is
|
||||||
|
* runnable on a rq into its weight. For cfs_rq, it is the aggregated
|
||||||
|
* such weights of all runnable and blocked sched_entities.
|
||||||
|
* 2) util_avg factors frequency scaling into the amount of time
|
||||||
|
* that a sched_entity is running on a CPU, in the range [0..SCHED_LOAD_SCALE].
|
||||||
|
* For cfs_rq, it is the aggregated such times of all runnable and
|
||||||
|
* blocked sched_entities.
|
||||||
|
* The 64 bit load_sum can:
|
||||||
|
* 1) for cfs_rq, afford 4353082796 (=2^64/47742/88761) entities with
|
||||||
|
* the highest weight (=88761) always runnable, we should not overflow
|
||||||
|
* 2) for entity, support any load.weight always runnable
|
||||||
|
*/
|
||||||
struct sched_avg {
|
struct sched_avg {
|
||||||
u64 last_runnable_update;
|
u64 last_update_time, load_sum;
|
||||||
s64 decay_count;
|
u32 util_sum, period_contrib;
|
||||||
/*
|
unsigned long load_avg, util_avg;
|
||||||
* utilization_avg_contrib describes the amount of time that a
|
|
||||||
* sched_entity is running on a CPU. It is based on running_avg_sum
|
|
||||||
* and is scaled in the range [0..SCHED_LOAD_SCALE].
|
|
||||||
* load_avg_contrib described the amount of time that a sched_entity
|
|
||||||
* is runnable on a rq. It is based on both runnable_avg_sum and the
|
|
||||||
* weight of the task.
|
|
||||||
*/
|
|
||||||
unsigned long load_avg_contrib, utilization_avg_contrib;
|
|
||||||
/*
|
|
||||||
* These sums represent an infinite geometric series and so are bound
|
|
||||||
* above by 1024/(1-y). Thus we only need a u32 to store them for all
|
|
||||||
* choices of y < 1-2^(-32)*1024.
|
|
||||||
* running_avg_sum reflects the time that the sched_entity is
|
|
||||||
* effectively running on the CPU.
|
|
||||||
* runnable_avg_sum represents the amount of time a sched_entity is on
|
|
||||||
* a runqueue which includes the running time that is monitored by
|
|
||||||
* running_avg_sum.
|
|
||||||
*/
|
|
||||||
u32 runnable_avg_sum, avg_period, running_avg_sum;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_SCHEDSTATS
|
#ifdef CONFIG_SCHEDSTATS
|
||||||
@ -1263,7 +1258,7 @@ struct sched_entity {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/* Per-entity load-tracking */
|
/* Per entity load average tracking */
|
||||||
struct sched_avg avg;
|
struct sched_avg avg;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
@ -2020,9 +2020,6 @@ static void __sched_fork(unsigned long clone_flags, struct task_struct *p)
|
|||||||
p->se.prev_sum_exec_runtime = 0;
|
p->se.prev_sum_exec_runtime = 0;
|
||||||
p->se.nr_migrations = 0;
|
p->se.nr_migrations = 0;
|
||||||
p->se.vruntime = 0;
|
p->se.vruntime = 0;
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
p->se.avg.decay_count = 0;
|
|
||||||
#endif
|
|
||||||
INIT_LIST_HEAD(&p->se.group_node);
|
INIT_LIST_HEAD(&p->se.group_node);
|
||||||
|
|
||||||
#ifdef CONFIG_SCHEDSTATS
|
#ifdef CONFIG_SCHEDSTATS
|
||||||
|
@ -88,12 +88,8 @@ static void print_cfs_group_stats(struct seq_file *m, int cpu, struct task_group
|
|||||||
#endif
|
#endif
|
||||||
P(se->load.weight);
|
P(se->load.weight);
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
P(se->avg.runnable_avg_sum);
|
P(se->avg.load_avg);
|
||||||
P(se->avg.running_avg_sum);
|
P(se->avg.util_avg);
|
||||||
P(se->avg.avg_period);
|
|
||||||
P(se->avg.load_avg_contrib);
|
|
||||||
P(se->avg.utilization_avg_contrib);
|
|
||||||
P(se->avg.decay_count);
|
|
||||||
#endif
|
#endif
|
||||||
#undef PN
|
#undef PN
|
||||||
#undef P
|
#undef P
|
||||||
@ -209,21 +205,19 @@ void print_cfs_rq(struct seq_file *m, int cpu, struct cfs_rq *cfs_rq)
|
|||||||
SEQ_printf(m, " .%-30s: %d\n", "nr_running", cfs_rq->nr_running);
|
SEQ_printf(m, " .%-30s: %d\n", "nr_running", cfs_rq->nr_running);
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "load", cfs_rq->load.weight);
|
SEQ_printf(m, " .%-30s: %ld\n", "load", cfs_rq->load.weight);
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "runnable_load_avg",
|
SEQ_printf(m, " .%-30s: %lu\n", "load_avg",
|
||||||
cfs_rq->runnable_load_avg);
|
cfs_rq->avg.load_avg);
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "blocked_load_avg",
|
SEQ_printf(m, " .%-30s: %lu\n", "util_avg",
|
||||||
cfs_rq->blocked_load_avg);
|
cfs_rq->avg.util_avg);
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "utilization_load_avg",
|
SEQ_printf(m, " .%-30s: %ld\n", "removed_load_avg",
|
||||||
cfs_rq->utilization_load_avg);
|
atomic_long_read(&cfs_rq->removed_load_avg));
|
||||||
|
SEQ_printf(m, " .%-30s: %ld\n", "removed_util_avg",
|
||||||
|
atomic_long_read(&cfs_rq->removed_util_avg));
|
||||||
#ifdef CONFIG_FAIR_GROUP_SCHED
|
#ifdef CONFIG_FAIR_GROUP_SCHED
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "tg_load_contrib",
|
SEQ_printf(m, " .%-30s: %lu\n", "tg_load_avg_contrib",
|
||||||
cfs_rq->tg_load_contrib);
|
cfs_rq->tg_load_avg_contrib);
|
||||||
SEQ_printf(m, " .%-30s: %d\n", "tg_runnable_contrib",
|
|
||||||
cfs_rq->tg_runnable_contrib);
|
|
||||||
SEQ_printf(m, " .%-30s: %ld\n", "tg_load_avg",
|
SEQ_printf(m, " .%-30s: %ld\n", "tg_load_avg",
|
||||||
atomic_long_read(&cfs_rq->tg->load_avg));
|
atomic_long_read(&cfs_rq->tg->load_avg));
|
||||||
SEQ_printf(m, " .%-30s: %d\n", "tg->runnable_avg",
|
|
||||||
atomic_read(&cfs_rq->tg->runnable_avg));
|
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
#ifdef CONFIG_CFS_BANDWIDTH
|
#ifdef CONFIG_CFS_BANDWIDTH
|
||||||
@ -631,12 +625,11 @@ void proc_sched_show_task(struct task_struct *p, struct seq_file *m)
|
|||||||
|
|
||||||
P(se.load.weight);
|
P(se.load.weight);
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
P(se.avg.runnable_avg_sum);
|
P(se.avg.load_sum);
|
||||||
P(se.avg.running_avg_sum);
|
P(se.avg.util_sum);
|
||||||
P(se.avg.avg_period);
|
P(se.avg.load_avg);
|
||||||
P(se.avg.load_avg_contrib);
|
P(se.avg.util_avg);
|
||||||
P(se.avg.utilization_avg_contrib);
|
P(se.avg.last_update_time);
|
||||||
P(se.avg.decay_count);
|
|
||||||
#endif
|
#endif
|
||||||
P(policy);
|
P(policy);
|
||||||
P(prio);
|
P(prio);
|
||||||
|
@ -283,9 +283,6 @@ static inline struct cfs_rq *group_cfs_rq(struct sched_entity *grp)
|
|||||||
return grp->my_q;
|
return grp->my_q;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void update_cfs_rq_blocked_load(struct cfs_rq *cfs_rq,
|
|
||||||
int force_update);
|
|
||||||
|
|
||||||
static inline void list_add_leaf_cfs_rq(struct cfs_rq *cfs_rq)
|
static inline void list_add_leaf_cfs_rq(struct cfs_rq *cfs_rq)
|
||||||
{
|
{
|
||||||
if (!cfs_rq->on_list) {
|
if (!cfs_rq->on_list) {
|
||||||
@ -305,8 +302,6 @@ static inline void list_add_leaf_cfs_rq(struct cfs_rq *cfs_rq)
|
|||||||
}
|
}
|
||||||
|
|
||||||
cfs_rq->on_list = 1;
|
cfs_rq->on_list = 1;
|
||||||
/* We should have no load, but we need to update last_decay. */
|
|
||||||
update_cfs_rq_blocked_load(cfs_rq, 0);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -664,19 +659,31 @@ static u64 sched_vslice(struct cfs_rq *cfs_rq, struct sched_entity *se)
|
|||||||
static int select_idle_sibling(struct task_struct *p, int cpu);
|
static int select_idle_sibling(struct task_struct *p, int cpu);
|
||||||
static unsigned long task_h_load(struct task_struct *p);
|
static unsigned long task_h_load(struct task_struct *p);
|
||||||
|
|
||||||
static inline void __update_task_entity_contrib(struct sched_entity *se);
|
/*
|
||||||
static inline void __update_task_entity_utilization(struct sched_entity *se);
|
* We choose a half-life close to 1 scheduling period.
|
||||||
|
* Note: The tables below are dependent on this value.
|
||||||
|
*/
|
||||||
|
#define LOAD_AVG_PERIOD 32
|
||||||
|
#define LOAD_AVG_MAX 47742 /* maximum possible load avg */
|
||||||
|
#define LOAD_AVG_MAX_N 345 /* number of full periods to produce LOAD_MAX_AVG */
|
||||||
|
|
||||||
/* Give new task start runnable values to heavy its load in infant time */
|
/* Give new task start runnable values to heavy its load in infant time */
|
||||||
void init_task_runnable_average(struct task_struct *p)
|
void init_task_runnable_average(struct task_struct *p)
|
||||||
{
|
{
|
||||||
u32 slice;
|
struct sched_avg *sa = &p->se.avg;
|
||||||
|
|
||||||
slice = sched_slice(task_cfs_rq(p), &p->se) >> 10;
|
sa->last_update_time = 0;
|
||||||
p->se.avg.runnable_avg_sum = p->se.avg.running_avg_sum = slice;
|
/*
|
||||||
p->se.avg.avg_period = slice;
|
* sched_avg's period_contrib should be strictly less then 1024, so
|
||||||
__update_task_entity_contrib(&p->se);
|
* we give it 1023 to make sure it is almost a period (1024us), and
|
||||||
__update_task_entity_utilization(&p->se);
|
* will definitely be update (after enqueue).
|
||||||
|
*/
|
||||||
|
sa->period_contrib = 1023;
|
||||||
|
sa->load_avg = scale_load_down(p->se.load.weight);
|
||||||
|
sa->load_sum = sa->load_avg * LOAD_AVG_MAX;
|
||||||
|
sa->util_avg = scale_load_down(SCHED_LOAD_SCALE);
|
||||||
|
sa->util_sum = LOAD_AVG_MAX;
|
||||||
|
/* when this task enqueue'ed, it will contribute to its cfs_rq's load_avg */
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
void init_task_runnable_average(struct task_struct *p)
|
void init_task_runnable_average(struct task_struct *p)
|
||||||
@ -1698,8 +1705,8 @@ static u64 numa_get_avg_runtime(struct task_struct *p, u64 *period)
|
|||||||
delta = runtime - p->last_sum_exec_runtime;
|
delta = runtime - p->last_sum_exec_runtime;
|
||||||
*period = now - p->last_task_numa_placement;
|
*period = now - p->last_task_numa_placement;
|
||||||
} else {
|
} else {
|
||||||
delta = p->se.avg.runnable_avg_sum;
|
delta = p->se.avg.load_sum / p->se.load.weight;
|
||||||
*period = p->se.avg.avg_period;
|
*period = LOAD_AVG_MAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
p->last_sum_exec_runtime = runtime;
|
p->last_sum_exec_runtime = runtime;
|
||||||
@ -2347,13 +2354,13 @@ static inline long calc_tg_weight(struct task_group *tg, struct cfs_rq *cfs_rq)
|
|||||||
long tg_weight;
|
long tg_weight;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Use this CPU's actual weight instead of the last load_contribution
|
* Use this CPU's real-time load instead of the last load contribution
|
||||||
* to gain a more accurate current total weight. See
|
* as the updating of the contribution is delayed, and we will use the
|
||||||
* __update_cfs_rq_tg_load_contrib().
|
* the real-time load to calc the share. See update_tg_load_avg().
|
||||||
*/
|
*/
|
||||||
tg_weight = atomic_long_read(&tg->load_avg);
|
tg_weight = atomic_long_read(&tg->load_avg);
|
||||||
tg_weight -= cfs_rq->tg_load_contrib;
|
tg_weight -= cfs_rq->tg_load_avg_contrib;
|
||||||
tg_weight += cfs_rq->load.weight;
|
tg_weight += cfs_rq->avg.load_avg;
|
||||||
|
|
||||||
return tg_weight;
|
return tg_weight;
|
||||||
}
|
}
|
||||||
@ -2363,7 +2370,7 @@ static long calc_cfs_shares(struct cfs_rq *cfs_rq, struct task_group *tg)
|
|||||||
long tg_weight, load, shares;
|
long tg_weight, load, shares;
|
||||||
|
|
||||||
tg_weight = calc_tg_weight(tg, cfs_rq);
|
tg_weight = calc_tg_weight(tg, cfs_rq);
|
||||||
load = cfs_rq->load.weight;
|
load = cfs_rq->avg.load_avg;
|
||||||
|
|
||||||
shares = (tg->shares * load);
|
shares = (tg->shares * load);
|
||||||
if (tg_weight)
|
if (tg_weight)
|
||||||
@ -2425,14 +2432,6 @@ static inline void update_cfs_shares(struct cfs_rq *cfs_rq)
|
|||||||
#endif /* CONFIG_FAIR_GROUP_SCHED */
|
#endif /* CONFIG_FAIR_GROUP_SCHED */
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/*
|
|
||||||
* We choose a half-life close to 1 scheduling period.
|
|
||||||
* Note: The tables below are dependent on this value.
|
|
||||||
*/
|
|
||||||
#define LOAD_AVG_PERIOD 32
|
|
||||||
#define LOAD_AVG_MAX 47742 /* maximum possible load avg */
|
|
||||||
#define LOAD_AVG_MAX_N 345 /* number of full periods to produce LOAD_MAX_AVG */
|
|
||||||
|
|
||||||
/* Precomputed fixed inverse multiplies for multiplication by y^n */
|
/* Precomputed fixed inverse multiplies for multiplication by y^n */
|
||||||
static const u32 runnable_avg_yN_inv[] = {
|
static const u32 runnable_avg_yN_inv[] = {
|
||||||
0xffffffff, 0xfa83b2da, 0xf5257d14, 0xefe4b99a, 0xeac0c6e6, 0xe5b906e6,
|
0xffffffff, 0xfa83b2da, 0xf5257d14, 0xefe4b99a, 0xeac0c6e6, 0xe5b906e6,
|
||||||
@ -2481,9 +2480,8 @@ static __always_inline u64 decay_load(u64 val, u64 n)
|
|||||||
local_n %= LOAD_AVG_PERIOD;
|
local_n %= LOAD_AVG_PERIOD;
|
||||||
}
|
}
|
||||||
|
|
||||||
val *= runnable_avg_yN_inv[local_n];
|
val = mul_u64_u32_shr(val, runnable_avg_yN_inv[local_n], 32);
|
||||||
/* We don't use SRR here since we always want to round down. */
|
return val;
|
||||||
return val >> 32;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2542,23 +2540,22 @@ static u32 __compute_runnable_contrib(u64 n)
|
|||||||
* load_avg = u_0` + y*(u_0 + u_1*y + u_2*y^2 + ... )
|
* load_avg = u_0` + y*(u_0 + u_1*y + u_2*y^2 + ... )
|
||||||
* = u_0 + u_1*y + u_2*y^2 + ... [re-labeling u_i --> u_{i+1}]
|
* = u_0 + u_1*y + u_2*y^2 + ... [re-labeling u_i --> u_{i+1}]
|
||||||
*/
|
*/
|
||||||
static __always_inline int __update_entity_runnable_avg(u64 now, int cpu,
|
static __always_inline int
|
||||||
struct sched_avg *sa,
|
__update_load_avg(u64 now, int cpu, struct sched_avg *sa,
|
||||||
int runnable,
|
unsigned long weight, int running)
|
||||||
int running)
|
|
||||||
{
|
{
|
||||||
u64 delta, periods;
|
u64 delta, periods;
|
||||||
u32 runnable_contrib;
|
u32 contrib;
|
||||||
int delta_w, decayed = 0;
|
int delta_w, decayed = 0;
|
||||||
unsigned long scale_freq = arch_scale_freq_capacity(NULL, cpu);
|
unsigned long scale_freq = arch_scale_freq_capacity(NULL, cpu);
|
||||||
|
|
||||||
delta = now - sa->last_runnable_update;
|
delta = now - sa->last_update_time;
|
||||||
/*
|
/*
|
||||||
* This should only happen when time goes backwards, which it
|
* This should only happen when time goes backwards, which it
|
||||||
* unfortunately does during sched clock init when we swap over to TSC.
|
* unfortunately does during sched clock init when we swap over to TSC.
|
||||||
*/
|
*/
|
||||||
if ((s64)delta < 0) {
|
if ((s64)delta < 0) {
|
||||||
sa->last_runnable_update = now;
|
sa->last_update_time = now;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2569,26 +2566,26 @@ static __always_inline int __update_entity_runnable_avg(u64 now, int cpu,
|
|||||||
delta >>= 10;
|
delta >>= 10;
|
||||||
if (!delta)
|
if (!delta)
|
||||||
return 0;
|
return 0;
|
||||||
sa->last_runnable_update = now;
|
sa->last_update_time = now;
|
||||||
|
|
||||||
/* delta_w is the amount already accumulated against our next period */
|
/* delta_w is the amount already accumulated against our next period */
|
||||||
delta_w = sa->avg_period % 1024;
|
delta_w = sa->period_contrib;
|
||||||
if (delta + delta_w >= 1024) {
|
if (delta + delta_w >= 1024) {
|
||||||
/* period roll-over */
|
|
||||||
decayed = 1;
|
decayed = 1;
|
||||||
|
|
||||||
|
/* how much left for next period will start over, we don't know yet */
|
||||||
|
sa->period_contrib = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Now that we know we're crossing a period boundary, figure
|
* Now that we know we're crossing a period boundary, figure
|
||||||
* out how much from delta we need to complete the current
|
* out how much from delta we need to complete the current
|
||||||
* period and accrue it.
|
* period and accrue it.
|
||||||
*/
|
*/
|
||||||
delta_w = 1024 - delta_w;
|
delta_w = 1024 - delta_w;
|
||||||
if (runnable)
|
if (weight)
|
||||||
sa->runnable_avg_sum += delta_w;
|
sa->load_sum += weight * delta_w;
|
||||||
if (running)
|
if (running)
|
||||||
sa->running_avg_sum += delta_w * scale_freq
|
sa->util_sum += delta_w * scale_freq >> SCHED_CAPACITY_SHIFT;
|
||||||
>> SCHED_CAPACITY_SHIFT;
|
|
||||||
sa->avg_period += delta_w;
|
|
||||||
|
|
||||||
delta -= delta_w;
|
delta -= delta_w;
|
||||||
|
|
||||||
@ -2596,334 +2593,156 @@ static __always_inline int __update_entity_runnable_avg(u64 now, int cpu,
|
|||||||
periods = delta / 1024;
|
periods = delta / 1024;
|
||||||
delta %= 1024;
|
delta %= 1024;
|
||||||
|
|
||||||
sa->runnable_avg_sum = decay_load(sa->runnable_avg_sum,
|
sa->load_sum = decay_load(sa->load_sum, periods + 1);
|
||||||
periods + 1);
|
sa->util_sum = decay_load((u64)(sa->util_sum), periods + 1);
|
||||||
sa->running_avg_sum = decay_load(sa->running_avg_sum,
|
|
||||||
periods + 1);
|
|
||||||
sa->avg_period = decay_load(sa->avg_period,
|
|
||||||
periods + 1);
|
|
||||||
|
|
||||||
/* Efficiently calculate \sum (1..n_period) 1024*y^i */
|
/* Efficiently calculate \sum (1..n_period) 1024*y^i */
|
||||||
runnable_contrib = __compute_runnable_contrib(periods);
|
contrib = __compute_runnable_contrib(periods);
|
||||||
if (runnable)
|
if (weight)
|
||||||
sa->runnable_avg_sum += runnable_contrib;
|
sa->load_sum += weight * contrib;
|
||||||
if (running)
|
if (running)
|
||||||
sa->running_avg_sum += runnable_contrib * scale_freq
|
sa->util_sum += contrib * scale_freq >> SCHED_CAPACITY_SHIFT;
|
||||||
>> SCHED_CAPACITY_SHIFT;
|
|
||||||
sa->avg_period += runnable_contrib;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Remainder of delta accrued against u_0` */
|
/* Remainder of delta accrued against u_0` */
|
||||||
if (runnable)
|
if (weight)
|
||||||
sa->runnable_avg_sum += delta;
|
sa->load_sum += weight * delta;
|
||||||
if (running)
|
if (running)
|
||||||
sa->running_avg_sum += delta * scale_freq
|
sa->util_sum += delta * scale_freq >> SCHED_CAPACITY_SHIFT;
|
||||||
>> SCHED_CAPACITY_SHIFT;
|
|
||||||
sa->avg_period += delta;
|
sa->period_contrib += delta;
|
||||||
|
|
||||||
|
if (decayed) {
|
||||||
|
sa->load_avg = div_u64(sa->load_sum, LOAD_AVG_MAX);
|
||||||
|
sa->util_avg = (sa->util_sum << SCHED_LOAD_SHIFT) / LOAD_AVG_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
return decayed;
|
return decayed;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Synchronize an entity's decay with its parenting cfs_rq.*/
|
|
||||||
static inline u64 __synchronize_entity_decay(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
struct cfs_rq *cfs_rq = cfs_rq_of(se);
|
|
||||||
u64 decays = atomic64_read(&cfs_rq->decay_counter);
|
|
||||||
|
|
||||||
decays -= se->avg.decay_count;
|
|
||||||
se->avg.decay_count = 0;
|
|
||||||
if (!decays)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
se->avg.load_avg_contrib = decay_load(se->avg.load_avg_contrib, decays);
|
|
||||||
se->avg.utilization_avg_contrib =
|
|
||||||
decay_load(se->avg.utilization_avg_contrib, decays);
|
|
||||||
|
|
||||||
return decays;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef CONFIG_FAIR_GROUP_SCHED
|
#ifdef CONFIG_FAIR_GROUP_SCHED
|
||||||
static inline void __update_cfs_rq_tg_load_contrib(struct cfs_rq *cfs_rq,
|
|
||||||
int force_update)
|
|
||||||
{
|
|
||||||
struct task_group *tg = cfs_rq->tg;
|
|
||||||
long tg_contrib;
|
|
||||||
|
|
||||||
tg_contrib = cfs_rq->runnable_load_avg + cfs_rq->blocked_load_avg;
|
|
||||||
tg_contrib -= cfs_rq->tg_load_contrib;
|
|
||||||
|
|
||||||
if (!tg_contrib)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (force_update || abs(tg_contrib) > cfs_rq->tg_load_contrib / 8) {
|
|
||||||
atomic_long_add(tg_contrib, &tg->load_avg);
|
|
||||||
cfs_rq->tg_load_contrib += tg_contrib;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Aggregate cfs_rq runnable averages into an equivalent task_group
|
* Updating tg's load_avg is necessary before update_cfs_share (which is done)
|
||||||
* representation for computing load contributions.
|
* and effective_load (which is not done because it is too costly).
|
||||||
*/
|
*/
|
||||||
static inline void __update_tg_runnable_avg(struct sched_avg *sa,
|
static inline void update_tg_load_avg(struct cfs_rq *cfs_rq, int force)
|
||||||
struct cfs_rq *cfs_rq)
|
|
||||||
{
|
{
|
||||||
struct task_group *tg = cfs_rq->tg;
|
long delta = cfs_rq->avg.load_avg - cfs_rq->tg_load_avg_contrib;
|
||||||
long contrib;
|
|
||||||
|
|
||||||
/* The fraction of a cpu used by this cfs_rq */
|
if (force || abs(delta) > cfs_rq->tg_load_avg_contrib / 64) {
|
||||||
contrib = div_u64((u64)sa->runnable_avg_sum << NICE_0_SHIFT,
|
atomic_long_add(delta, &cfs_rq->tg->load_avg);
|
||||||
sa->avg_period + 1);
|
cfs_rq->tg_load_avg_contrib = cfs_rq->avg.load_avg;
|
||||||
contrib -= cfs_rq->tg_runnable_contrib;
|
|
||||||
|
|
||||||
if (abs(contrib) > cfs_rq->tg_runnable_contrib / 64) {
|
|
||||||
atomic_add(contrib, &tg->runnable_avg);
|
|
||||||
cfs_rq->tg_runnable_contrib += contrib;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void __update_group_entity_contrib(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
struct cfs_rq *cfs_rq = group_cfs_rq(se);
|
|
||||||
struct task_group *tg = cfs_rq->tg;
|
|
||||||
int runnable_avg;
|
|
||||||
|
|
||||||
u64 contrib;
|
|
||||||
|
|
||||||
contrib = cfs_rq->tg_load_contrib * tg->shares;
|
|
||||||
se->avg.load_avg_contrib = div_u64(contrib,
|
|
||||||
atomic_long_read(&tg->load_avg) + 1);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* For group entities we need to compute a correction term in the case
|
|
||||||
* that they are consuming <1 cpu so that we would contribute the same
|
|
||||||
* load as a task of equal weight.
|
|
||||||
*
|
|
||||||
* Explicitly co-ordinating this measurement would be expensive, but
|
|
||||||
* fortunately the sum of each cpus contribution forms a usable
|
|
||||||
* lower-bound on the true value.
|
|
||||||
*
|
|
||||||
* Consider the aggregate of 2 contributions. Either they are disjoint
|
|
||||||
* (and the sum represents true value) or they are disjoint and we are
|
|
||||||
* understating by the aggregate of their overlap.
|
|
||||||
*
|
|
||||||
* Extending this to N cpus, for a given overlap, the maximum amount we
|
|
||||||
* understand is then n_i(n_i+1)/2 * w_i where n_i is the number of
|
|
||||||
* cpus that overlap for this interval and w_i is the interval width.
|
|
||||||
*
|
|
||||||
* On a small machine; the first term is well-bounded which bounds the
|
|
||||||
* total error since w_i is a subset of the period. Whereas on a
|
|
||||||
* larger machine, while this first term can be larger, if w_i is the
|
|
||||||
* of consequential size guaranteed to see n_i*w_i quickly converge to
|
|
||||||
* our upper bound of 1-cpu.
|
|
||||||
*/
|
|
||||||
runnable_avg = atomic_read(&tg->runnable_avg);
|
|
||||||
if (runnable_avg < NICE_0_LOAD) {
|
|
||||||
se->avg.load_avg_contrib *= runnable_avg;
|
|
||||||
se->avg.load_avg_contrib >>= NICE_0_SHIFT;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* CONFIG_FAIR_GROUP_SCHED */
|
#else /* CONFIG_FAIR_GROUP_SCHED */
|
||||||
static inline void __update_cfs_rq_tg_load_contrib(struct cfs_rq *cfs_rq,
|
static inline void update_tg_load_avg(struct cfs_rq *cfs_rq, int force) {}
|
||||||
int force_update) {}
|
|
||||||
static inline void __update_tg_runnable_avg(struct sched_avg *sa,
|
|
||||||
struct cfs_rq *cfs_rq) {}
|
|
||||||
static inline void __update_group_entity_contrib(struct sched_entity *se) {}
|
|
||||||
#endif /* CONFIG_FAIR_GROUP_SCHED */
|
#endif /* CONFIG_FAIR_GROUP_SCHED */
|
||||||
|
|
||||||
static inline void __update_task_entity_contrib(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
u32 contrib;
|
|
||||||
|
|
||||||
/* avoid overflowing a 32-bit type w/ SCHED_LOAD_SCALE */
|
|
||||||
contrib = se->avg.runnable_avg_sum * scale_load_down(se->load.weight);
|
|
||||||
contrib /= (se->avg.avg_period + 1);
|
|
||||||
se->avg.load_avg_contrib = scale_load(contrib);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Compute the current contribution to load_avg by se, return any delta */
|
|
||||||
static long __update_entity_load_avg_contrib(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
long old_contrib = se->avg.load_avg_contrib;
|
|
||||||
|
|
||||||
if (entity_is_task(se)) {
|
|
||||||
__update_task_entity_contrib(se);
|
|
||||||
} else {
|
|
||||||
__update_tg_runnable_avg(&se->avg, group_cfs_rq(se));
|
|
||||||
__update_group_entity_contrib(se);
|
|
||||||
}
|
|
||||||
|
|
||||||
return se->avg.load_avg_contrib - old_contrib;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static inline void __update_task_entity_utilization(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
u32 contrib;
|
|
||||||
|
|
||||||
/* avoid overflowing a 32-bit type w/ SCHED_LOAD_SCALE */
|
|
||||||
contrib = se->avg.running_avg_sum * scale_load_down(SCHED_LOAD_SCALE);
|
|
||||||
contrib /= (se->avg.avg_period + 1);
|
|
||||||
se->avg.utilization_avg_contrib = scale_load(contrib);
|
|
||||||
}
|
|
||||||
|
|
||||||
static long __update_entity_utilization_avg_contrib(struct sched_entity *se)
|
|
||||||
{
|
|
||||||
long old_contrib = se->avg.utilization_avg_contrib;
|
|
||||||
|
|
||||||
if (entity_is_task(se))
|
|
||||||
__update_task_entity_utilization(se);
|
|
||||||
else
|
|
||||||
se->avg.utilization_avg_contrib =
|
|
||||||
group_cfs_rq(se)->utilization_load_avg;
|
|
||||||
|
|
||||||
return se->avg.utilization_avg_contrib - old_contrib;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void subtract_blocked_load_contrib(struct cfs_rq *cfs_rq,
|
|
||||||
long load_contrib)
|
|
||||||
{
|
|
||||||
if (likely(load_contrib < cfs_rq->blocked_load_avg))
|
|
||||||
cfs_rq->blocked_load_avg -= load_contrib;
|
|
||||||
else
|
|
||||||
cfs_rq->blocked_load_avg = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline u64 cfs_rq_clock_task(struct cfs_rq *cfs_rq);
|
static inline u64 cfs_rq_clock_task(struct cfs_rq *cfs_rq);
|
||||||
|
|
||||||
/* Update a sched_entity's runnable average */
|
/* Group cfs_rq's load_avg is used for task_h_load and update_cfs_share */
|
||||||
static inline void update_entity_load_avg(struct sched_entity *se,
|
static inline int update_cfs_rq_load_avg(u64 now, struct cfs_rq *cfs_rq)
|
||||||
int update_cfs_rq)
|
{
|
||||||
|
int decayed;
|
||||||
|
struct sched_avg *sa = &cfs_rq->avg;
|
||||||
|
|
||||||
|
if (atomic_long_read(&cfs_rq->removed_load_avg)) {
|
||||||
|
long r = atomic_long_xchg(&cfs_rq->removed_load_avg, 0);
|
||||||
|
sa->load_avg = max_t(long, sa->load_avg - r, 0);
|
||||||
|
sa->load_sum = max_t(s64, sa->load_sum - r * LOAD_AVG_MAX, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (atomic_long_read(&cfs_rq->removed_util_avg)) {
|
||||||
|
long r = atomic_long_xchg(&cfs_rq->removed_util_avg, 0);
|
||||||
|
sa->util_avg = max_t(long, sa->util_avg - r, 0);
|
||||||
|
sa->util_sum = max_t(s32, sa->util_sum -
|
||||||
|
((r * LOAD_AVG_MAX) >> SCHED_LOAD_SHIFT), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
decayed = __update_load_avg(now, cpu_of(rq_of(cfs_rq)), sa,
|
||||||
|
scale_load_down(cfs_rq->load.weight), cfs_rq->curr != NULL);
|
||||||
|
|
||||||
|
#ifndef CONFIG_64BIT
|
||||||
|
smp_wmb();
|
||||||
|
cfs_rq->load_last_update_time_copy = sa->last_update_time;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return decayed;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Update task and its cfs_rq load average */
|
||||||
|
static inline void update_load_avg(struct sched_entity *se, int update_tg)
|
||||||
{
|
{
|
||||||
struct cfs_rq *cfs_rq = cfs_rq_of(se);
|
struct cfs_rq *cfs_rq = cfs_rq_of(se);
|
||||||
long contrib_delta, utilization_delta;
|
|
||||||
int cpu = cpu_of(rq_of(cfs_rq));
|
int cpu = cpu_of(rq_of(cfs_rq));
|
||||||
u64 now;
|
u64 now = cfs_rq_clock_task(cfs_rq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For a group entity we need to use their owned cfs_rq_clock_task() in
|
* Track task load average for carrying it to new CPU after migrated, and
|
||||||
* case they are the parent of a throttled hierarchy.
|
* track group sched_entity load average for task_h_load calc in migration
|
||||||
*/
|
*/
|
||||||
if (entity_is_task(se))
|
__update_load_avg(now, cpu, &se->avg,
|
||||||
now = cfs_rq_clock_task(cfs_rq);
|
se->on_rq * scale_load_down(se->load.weight), cfs_rq->curr == se);
|
||||||
else
|
|
||||||
now = cfs_rq_clock_task(group_cfs_rq(se));
|
|
||||||
|
|
||||||
if (!__update_entity_runnable_avg(now, cpu, &se->avg, se->on_rq,
|
if (update_cfs_rq_load_avg(now, cfs_rq) && update_tg)
|
||||||
cfs_rq->curr == se))
|
update_tg_load_avg(cfs_rq, 0);
|
||||||
return;
|
}
|
||||||
|
|
||||||
contrib_delta = __update_entity_load_avg_contrib(se);
|
/* Add the load generated by se into cfs_rq's load average */
|
||||||
utilization_delta = __update_entity_utilization_avg_contrib(se);
|
static inline void
|
||||||
|
enqueue_entity_load_avg(struct cfs_rq *cfs_rq, struct sched_entity *se)
|
||||||
|
{
|
||||||
|
struct sched_avg *sa = &se->avg;
|
||||||
|
u64 now = cfs_rq_clock_task(cfs_rq);
|
||||||
|
int migrated = 0, decayed;
|
||||||
|
|
||||||
if (!update_cfs_rq)
|
if (sa->last_update_time == 0) {
|
||||||
return;
|
sa->last_update_time = now;
|
||||||
|
migrated = 1;
|
||||||
if (se->on_rq) {
|
|
||||||
cfs_rq->runnable_load_avg += contrib_delta;
|
|
||||||
cfs_rq->utilization_load_avg += utilization_delta;
|
|
||||||
} else {
|
|
||||||
subtract_blocked_load_contrib(cfs_rq, -contrib_delta);
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
__update_load_avg(now, cpu_of(rq_of(cfs_rq)), sa,
|
||||||
|
se->on_rq * scale_load_down(se->load.weight), cfs_rq->curr == se);
|
||||||
|
}
|
||||||
|
|
||||||
|
decayed = update_cfs_rq_load_avg(now, cfs_rq);
|
||||||
|
|
||||||
|
if (migrated) {
|
||||||
|
cfs_rq->avg.load_avg += sa->load_avg;
|
||||||
|
cfs_rq->avg.load_sum += sa->load_sum;
|
||||||
|
cfs_rq->avg.util_avg += sa->util_avg;
|
||||||
|
cfs_rq->avg.util_sum += sa->util_sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (decayed || migrated)
|
||||||
|
update_tg_load_avg(cfs_rq, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Decay the load contributed by all blocked children and account this so that
|
* Task first catches up with cfs_rq, and then subtract
|
||||||
* their contribution may appropriately discounted when they wake up.
|
* itself from the cfs_rq (task must be off the queue now).
|
||||||
*/
|
*/
|
||||||
static void update_cfs_rq_blocked_load(struct cfs_rq *cfs_rq, int force_update)
|
void remove_entity_load_avg(struct sched_entity *se)
|
||||||
{
|
{
|
||||||
u64 now = cfs_rq_clock_task(cfs_rq) >> 20;
|
struct cfs_rq *cfs_rq = cfs_rq_of(se);
|
||||||
u64 decays;
|
u64 last_update_time;
|
||||||
|
|
||||||
decays = now - cfs_rq->last_decay;
|
#ifndef CONFIG_64BIT
|
||||||
if (!decays && !force_update)
|
u64 last_update_time_copy;
|
||||||
return;
|
|
||||||
|
|
||||||
if (atomic_long_read(&cfs_rq->removed_load)) {
|
do {
|
||||||
unsigned long removed_load;
|
last_update_time_copy = cfs_rq->load_last_update_time_copy;
|
||||||
removed_load = atomic_long_xchg(&cfs_rq->removed_load, 0);
|
smp_rmb();
|
||||||
subtract_blocked_load_contrib(cfs_rq, removed_load);
|
last_update_time = cfs_rq->avg.last_update_time;
|
||||||
}
|
} while (last_update_time != last_update_time_copy);
|
||||||
|
#else
|
||||||
|
last_update_time = cfs_rq->avg.last_update_time;
|
||||||
|
#endif
|
||||||
|
|
||||||
if (decays) {
|
__update_load_avg(last_update_time, cpu_of(rq_of(cfs_rq)), &se->avg, 0, 0);
|
||||||
cfs_rq->blocked_load_avg = decay_load(cfs_rq->blocked_load_avg,
|
atomic_long_add(se->avg.load_avg, &cfs_rq->removed_load_avg);
|
||||||
decays);
|
atomic_long_add(se->avg.util_avg, &cfs_rq->removed_util_avg);
|
||||||
atomic64_add(decays, &cfs_rq->decay_counter);
|
|
||||||
cfs_rq->last_decay = now;
|
|
||||||
}
|
|
||||||
|
|
||||||
__update_cfs_rq_tg_load_contrib(cfs_rq, force_update);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Add the load generated by se into cfs_rq's child load-average */
|
|
||||||
static inline void enqueue_entity_load_avg(struct cfs_rq *cfs_rq,
|
|
||||||
struct sched_entity *se,
|
|
||||||
int wakeup)
|
|
||||||
{
|
|
||||||
/*
|
|
||||||
* We track migrations using entity decay_count <= 0, on a wake-up
|
|
||||||
* migration we use a negative decay count to track the remote decays
|
|
||||||
* accumulated while sleeping.
|
|
||||||
*
|
|
||||||
* Newly forked tasks are enqueued with se->avg.decay_count == 0, they
|
|
||||||
* are seen by enqueue_entity_load_avg() as a migration with an already
|
|
||||||
* constructed load_avg_contrib.
|
|
||||||
*/
|
|
||||||
if (unlikely(se->avg.decay_count <= 0)) {
|
|
||||||
se->avg.last_runnable_update = rq_clock_task(rq_of(cfs_rq));
|
|
||||||
if (se->avg.decay_count) {
|
|
||||||
/*
|
|
||||||
* In a wake-up migration we have to approximate the
|
|
||||||
* time sleeping. This is because we can't synchronize
|
|
||||||
* clock_task between the two cpus, and it is not
|
|
||||||
* guaranteed to be read-safe. Instead, we can
|
|
||||||
* approximate this using our carried decays, which are
|
|
||||||
* explicitly atomically readable.
|
|
||||||
*/
|
|
||||||
se->avg.last_runnable_update -= (-se->avg.decay_count)
|
|
||||||
<< 20;
|
|
||||||
update_entity_load_avg(se, 0);
|
|
||||||
/* Indicate that we're now synchronized and on-rq */
|
|
||||||
se->avg.decay_count = 0;
|
|
||||||
}
|
|
||||||
wakeup = 0;
|
|
||||||
} else {
|
|
||||||
__synchronize_entity_decay(se);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* migrated tasks did not contribute to our blocked load */
|
|
||||||
if (wakeup) {
|
|
||||||
subtract_blocked_load_contrib(cfs_rq, se->avg.load_avg_contrib);
|
|
||||||
update_entity_load_avg(se, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
cfs_rq->runnable_load_avg += se->avg.load_avg_contrib;
|
|
||||||
cfs_rq->utilization_load_avg += se->avg.utilization_avg_contrib;
|
|
||||||
/* we force update consideration on load-balancer moves */
|
|
||||||
update_cfs_rq_blocked_load(cfs_rq, !wakeup);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Remove se's load from this cfs_rq child load-average, if the entity is
|
|
||||||
* transitioning to a blocked state we track its projected decay using
|
|
||||||
* blocked_load_avg.
|
|
||||||
*/
|
|
||||||
static inline void dequeue_entity_load_avg(struct cfs_rq *cfs_rq,
|
|
||||||
struct sched_entity *se,
|
|
||||||
int sleep)
|
|
||||||
{
|
|
||||||
update_entity_load_avg(se, 1);
|
|
||||||
/* we force update consideration on load-balancer moves */
|
|
||||||
update_cfs_rq_blocked_load(cfs_rq, !sleep);
|
|
||||||
|
|
||||||
cfs_rq->runnable_load_avg -= se->avg.load_avg_contrib;
|
|
||||||
cfs_rq->utilization_load_avg -= se->avg.utilization_avg_contrib;
|
|
||||||
if (sleep) {
|
|
||||||
cfs_rq->blocked_load_avg += se->avg.load_avg_contrib;
|
|
||||||
se->avg.decay_count = atomic64_read(&cfs_rq->decay_counter);
|
|
||||||
} /* migrations, e.g. sleep=0 leave decay_count == 0 */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2948,16 +2767,10 @@ static int idle_balance(struct rq *this_rq);
|
|||||||
|
|
||||||
#else /* CONFIG_SMP */
|
#else /* CONFIG_SMP */
|
||||||
|
|
||||||
static inline void update_entity_load_avg(struct sched_entity *se,
|
static inline void update_load_avg(struct sched_entity *se, int update_tg) {}
|
||||||
int update_cfs_rq) {}
|
static inline void
|
||||||
static inline void enqueue_entity_load_avg(struct cfs_rq *cfs_rq,
|
enqueue_entity_load_avg(struct cfs_rq *cfs_rq, struct sched_entity *se) {}
|
||||||
struct sched_entity *se,
|
static inline void remove_entity_load_avg(struct sched_entity *se) {}
|
||||||
int wakeup) {}
|
|
||||||
static inline void dequeue_entity_load_avg(struct cfs_rq *cfs_rq,
|
|
||||||
struct sched_entity *se,
|
|
||||||
int sleep) {}
|
|
||||||
static inline void update_cfs_rq_blocked_load(struct cfs_rq *cfs_rq,
|
|
||||||
int force_update) {}
|
|
||||||
|
|
||||||
static inline int idle_balance(struct rq *rq)
|
static inline int idle_balance(struct rq *rq)
|
||||||
{
|
{
|
||||||
@ -3089,7 +2902,7 @@ enqueue_entity(struct cfs_rq *cfs_rq, struct sched_entity *se, int flags)
|
|||||||
* Update run-time statistics of the 'current'.
|
* Update run-time statistics of the 'current'.
|
||||||
*/
|
*/
|
||||||
update_curr(cfs_rq);
|
update_curr(cfs_rq);
|
||||||
enqueue_entity_load_avg(cfs_rq, se, flags & ENQUEUE_WAKEUP);
|
enqueue_entity_load_avg(cfs_rq, se);
|
||||||
account_entity_enqueue(cfs_rq, se);
|
account_entity_enqueue(cfs_rq, se);
|
||||||
update_cfs_shares(cfs_rq);
|
update_cfs_shares(cfs_rq);
|
||||||
|
|
||||||
@ -3164,7 +2977,7 @@ dequeue_entity(struct cfs_rq *cfs_rq, struct sched_entity *se, int flags)
|
|||||||
* Update run-time statistics of the 'current'.
|
* Update run-time statistics of the 'current'.
|
||||||
*/
|
*/
|
||||||
update_curr(cfs_rq);
|
update_curr(cfs_rq);
|
||||||
dequeue_entity_load_avg(cfs_rq, se, flags & DEQUEUE_SLEEP);
|
update_load_avg(se, 1);
|
||||||
|
|
||||||
update_stats_dequeue(cfs_rq, se);
|
update_stats_dequeue(cfs_rq, se);
|
||||||
if (flags & DEQUEUE_SLEEP) {
|
if (flags & DEQUEUE_SLEEP) {
|
||||||
@ -3254,7 +3067,7 @@ set_next_entity(struct cfs_rq *cfs_rq, struct sched_entity *se)
|
|||||||
*/
|
*/
|
||||||
update_stats_wait_end(cfs_rq, se);
|
update_stats_wait_end(cfs_rq, se);
|
||||||
__dequeue_entity(cfs_rq, se);
|
__dequeue_entity(cfs_rq, se);
|
||||||
update_entity_load_avg(se, 1);
|
update_load_avg(se, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
update_stats_curr_start(cfs_rq, se);
|
update_stats_curr_start(cfs_rq, se);
|
||||||
@ -3354,7 +3167,7 @@ static void put_prev_entity(struct cfs_rq *cfs_rq, struct sched_entity *prev)
|
|||||||
/* Put 'current' back into the tree. */
|
/* Put 'current' back into the tree. */
|
||||||
__enqueue_entity(cfs_rq, prev);
|
__enqueue_entity(cfs_rq, prev);
|
||||||
/* in !on_rq case, update occurred at dequeue */
|
/* in !on_rq case, update occurred at dequeue */
|
||||||
update_entity_load_avg(prev, 1);
|
update_load_avg(prev, 0);
|
||||||
}
|
}
|
||||||
cfs_rq->curr = NULL;
|
cfs_rq->curr = NULL;
|
||||||
}
|
}
|
||||||
@ -3370,8 +3183,7 @@ entity_tick(struct cfs_rq *cfs_rq, struct sched_entity *curr, int queued)
|
|||||||
/*
|
/*
|
||||||
* Ensure that runnable average is periodically updated.
|
* Ensure that runnable average is periodically updated.
|
||||||
*/
|
*/
|
||||||
update_entity_load_avg(curr, 1);
|
update_load_avg(curr, 1);
|
||||||
update_cfs_rq_blocked_load(cfs_rq, 1);
|
|
||||||
update_cfs_shares(cfs_rq);
|
update_cfs_shares(cfs_rq);
|
||||||
|
|
||||||
#ifdef CONFIG_SCHED_HRTICK
|
#ifdef CONFIG_SCHED_HRTICK
|
||||||
@ -4244,8 +4056,8 @@ enqueue_task_fair(struct rq *rq, struct task_struct *p, int flags)
|
|||||||
if (cfs_rq_throttled(cfs_rq))
|
if (cfs_rq_throttled(cfs_rq))
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
update_load_avg(se, 1);
|
||||||
update_cfs_shares(cfs_rq);
|
update_cfs_shares(cfs_rq);
|
||||||
update_entity_load_avg(se, 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!se)
|
if (!se)
|
||||||
@ -4304,8 +4116,8 @@ static void dequeue_task_fair(struct rq *rq, struct task_struct *p, int flags)
|
|||||||
if (cfs_rq_throttled(cfs_rq))
|
if (cfs_rq_throttled(cfs_rq))
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
update_load_avg(se, 1);
|
||||||
update_cfs_shares(cfs_rq);
|
update_cfs_shares(cfs_rq);
|
||||||
update_entity_load_avg(se, 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!se)
|
if (!se)
|
||||||
@ -4444,7 +4256,7 @@ static void __update_cpu_load(struct rq *this_rq, unsigned long this_load,
|
|||||||
static void update_idle_cpu_load(struct rq *this_rq)
|
static void update_idle_cpu_load(struct rq *this_rq)
|
||||||
{
|
{
|
||||||
unsigned long curr_jiffies = READ_ONCE(jiffies);
|
unsigned long curr_jiffies = READ_ONCE(jiffies);
|
||||||
unsigned long load = this_rq->cfs.runnable_load_avg;
|
unsigned long load = this_rq->cfs.avg.load_avg;
|
||||||
unsigned long pending_updates;
|
unsigned long pending_updates;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -4490,7 +4302,7 @@ void update_cpu_load_nohz(void)
|
|||||||
*/
|
*/
|
||||||
void update_cpu_load_active(struct rq *this_rq)
|
void update_cpu_load_active(struct rq *this_rq)
|
||||||
{
|
{
|
||||||
unsigned long load = this_rq->cfs.runnable_load_avg;
|
unsigned long load = this_rq->cfs.avg.load_avg;
|
||||||
/*
|
/*
|
||||||
* See the mess around update_idle_cpu_load() / update_cpu_load_nohz().
|
* See the mess around update_idle_cpu_load() / update_cpu_load_nohz().
|
||||||
*/
|
*/
|
||||||
@ -4501,7 +4313,7 @@ void update_cpu_load_active(struct rq *this_rq)
|
|||||||
/* Used instead of source_load when we know the type == 0 */
|
/* Used instead of source_load when we know the type == 0 */
|
||||||
static unsigned long weighted_cpuload(const int cpu)
|
static unsigned long weighted_cpuload(const int cpu)
|
||||||
{
|
{
|
||||||
return cpu_rq(cpu)->cfs.runnable_load_avg;
|
return cpu_rq(cpu)->cfs.avg.load_avg;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -4551,7 +4363,7 @@ static unsigned long cpu_avg_load_per_task(int cpu)
|
|||||||
{
|
{
|
||||||
struct rq *rq = cpu_rq(cpu);
|
struct rq *rq = cpu_rq(cpu);
|
||||||
unsigned long nr_running = READ_ONCE(rq->cfs.h_nr_running);
|
unsigned long nr_running = READ_ONCE(rq->cfs.h_nr_running);
|
||||||
unsigned long load_avg = rq->cfs.runnable_load_avg;
|
unsigned long load_avg = rq->cfs.avg.load_avg;
|
||||||
|
|
||||||
if (nr_running)
|
if (nr_running)
|
||||||
return load_avg / nr_running;
|
return load_avg / nr_running;
|
||||||
@ -4670,7 +4482,7 @@ static long effective_load(struct task_group *tg, int cpu, long wl, long wg)
|
|||||||
/*
|
/*
|
||||||
* w = rw_i + @wl
|
* w = rw_i + @wl
|
||||||
*/
|
*/
|
||||||
w = se->my_q->load.weight + wl;
|
w = se->my_q->avg.load_avg + wl;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* wl = S * s'_i; see (2)
|
* wl = S * s'_i; see (2)
|
||||||
@ -4691,7 +4503,7 @@ static long effective_load(struct task_group *tg, int cpu, long wl, long wg)
|
|||||||
/*
|
/*
|
||||||
* wl = dw_i = S * (s'_i - s_i); see (3)
|
* wl = dw_i = S * (s'_i - s_i); see (3)
|
||||||
*/
|
*/
|
||||||
wl -= se->load.weight;
|
wl -= se->avg.load_avg;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Recursively apply this logic to all parent groups to compute
|
* Recursively apply this logic to all parent groups to compute
|
||||||
@ -4761,14 +4573,14 @@ static int wake_affine(struct sched_domain *sd, struct task_struct *p, int sync)
|
|||||||
*/
|
*/
|
||||||
if (sync) {
|
if (sync) {
|
||||||
tg = task_group(current);
|
tg = task_group(current);
|
||||||
weight = current->se.load.weight;
|
weight = current->se.avg.load_avg;
|
||||||
|
|
||||||
this_load += effective_load(tg, this_cpu, -weight, -weight);
|
this_load += effective_load(tg, this_cpu, -weight, -weight);
|
||||||
load += effective_load(tg, prev_cpu, 0, -weight);
|
load += effective_load(tg, prev_cpu, 0, -weight);
|
||||||
}
|
}
|
||||||
|
|
||||||
tg = task_group(p);
|
tg = task_group(p);
|
||||||
weight = p->se.load.weight;
|
weight = p->se.avg.load_avg;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* In low-load situations, where prev_cpu is idle and this_cpu is idle
|
* In low-load situations, where prev_cpu is idle and this_cpu is idle
|
||||||
@ -4961,12 +4773,12 @@ done:
|
|||||||
* tasks. The unit of the return value must be the one of capacity so we can
|
* tasks. The unit of the return value must be the one of capacity so we can
|
||||||
* compare the usage with the capacity of the CPU that is available for CFS
|
* compare the usage with the capacity of the CPU that is available for CFS
|
||||||
* task (ie cpu_capacity).
|
* task (ie cpu_capacity).
|
||||||
* cfs.utilization_load_avg is the sum of running time of runnable tasks on a
|
* cfs.avg.util_avg is the sum of running time of runnable tasks on a
|
||||||
* CPU. It represents the amount of utilization of a CPU in the range
|
* CPU. It represents the amount of utilization of a CPU in the range
|
||||||
* [0..SCHED_LOAD_SCALE]. The usage of a CPU can't be higher than the full
|
* [0..SCHED_LOAD_SCALE]. The usage of a CPU can't be higher than the full
|
||||||
* capacity of the CPU because it's about the running time on this CPU.
|
* capacity of the CPU because it's about the running time on this CPU.
|
||||||
* Nevertheless, cfs.utilization_load_avg can be higher than SCHED_LOAD_SCALE
|
* Nevertheless, cfs.avg.util_avg can be higher than SCHED_LOAD_SCALE
|
||||||
* because of unfortunate rounding in avg_period and running_load_avg or just
|
* because of unfortunate rounding in util_avg or just
|
||||||
* after migrating tasks until the average stabilizes with the new running
|
* after migrating tasks until the average stabilizes with the new running
|
||||||
* time. So we need to check that the usage stays into the range
|
* time. So we need to check that the usage stays into the range
|
||||||
* [0..cpu_capacity_orig] and cap if necessary.
|
* [0..cpu_capacity_orig] and cap if necessary.
|
||||||
@ -4975,7 +4787,7 @@ done:
|
|||||||
*/
|
*/
|
||||||
static int get_cpu_usage(int cpu)
|
static int get_cpu_usage(int cpu)
|
||||||
{
|
{
|
||||||
unsigned long usage = cpu_rq(cpu)->cfs.utilization_load_avg;
|
unsigned long usage = cpu_rq(cpu)->cfs.avg.util_avg;
|
||||||
unsigned long capacity = capacity_orig_of(cpu);
|
unsigned long capacity = capacity_orig_of(cpu);
|
||||||
|
|
||||||
if (usage >= SCHED_LOAD_SCALE)
|
if (usage >= SCHED_LOAD_SCALE)
|
||||||
@ -5084,26 +4896,22 @@ select_task_rq_fair(struct task_struct *p, int prev_cpu, int sd_flag, int wake_f
|
|||||||
* previous cpu. However, the caller only guarantees p->pi_lock is held; no
|
* previous cpu. However, the caller only guarantees p->pi_lock is held; no
|
||||||
* other assumptions, including the state of rq->lock, should be made.
|
* other assumptions, including the state of rq->lock, should be made.
|
||||||
*/
|
*/
|
||||||
static void
|
static void migrate_task_rq_fair(struct task_struct *p, int next_cpu)
|
||||||
migrate_task_rq_fair(struct task_struct *p, int next_cpu)
|
|
||||||
{
|
{
|
||||||
struct sched_entity *se = &p->se;
|
|
||||||
struct cfs_rq *cfs_rq = cfs_rq_of(se);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Load tracking: accumulate removed load so that it can be processed
|
* We are supposed to update the task to "current" time, then its up to date
|
||||||
* when we next update owning cfs_rq under rq->lock. Tasks contribute
|
* and ready to go to new CPU/cfs_rq. But we have difficulty in getting
|
||||||
* to blocked load iff they have a positive decay-count. It can never
|
* what current time is, so simply throw away the out-of-date time. This
|
||||||
* be negative here since on-rq tasks have decay-count == 0.
|
* will result in the wakee task is less decayed, but giving the wakee more
|
||||||
|
* load sounds not bad.
|
||||||
*/
|
*/
|
||||||
if (se->avg.decay_count) {
|
remove_entity_load_avg(&p->se);
|
||||||
se->avg.decay_count = -__synchronize_entity_decay(se);
|
|
||||||
atomic_long_add(se->avg.load_avg_contrib,
|
/* Tell new CPU we are migrated */
|
||||||
&cfs_rq->removed_load);
|
p->se.avg.last_update_time = 0;
|
||||||
}
|
|
||||||
|
|
||||||
/* We have migrated, no longer consider this task hot */
|
/* We have migrated, no longer consider this task hot */
|
||||||
se->exec_start = 0;
|
p->se.exec_start = 0;
|
||||||
}
|
}
|
||||||
#endif /* CONFIG_SMP */
|
#endif /* CONFIG_SMP */
|
||||||
|
|
||||||
@ -5966,36 +5774,6 @@ static void attach_tasks(struct lb_env *env)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_FAIR_GROUP_SCHED
|
#ifdef CONFIG_FAIR_GROUP_SCHED
|
||||||
/*
|
|
||||||
* update tg->load_weight by folding this cpu's load_avg
|
|
||||||
*/
|
|
||||||
static void __update_blocked_averages_cpu(struct task_group *tg, int cpu)
|
|
||||||
{
|
|
||||||
struct sched_entity *se = tg->se[cpu];
|
|
||||||
struct cfs_rq *cfs_rq = tg->cfs_rq[cpu];
|
|
||||||
|
|
||||||
/* throttled entities do not contribute to load */
|
|
||||||
if (throttled_hierarchy(cfs_rq))
|
|
||||||
return;
|
|
||||||
|
|
||||||
update_cfs_rq_blocked_load(cfs_rq, 1);
|
|
||||||
|
|
||||||
if (se) {
|
|
||||||
update_entity_load_avg(se, 1);
|
|
||||||
/*
|
|
||||||
* We pivot on our runnable average having decayed to zero for
|
|
||||||
* list removal. This generally implies that all our children
|
|
||||||
* have also been removed (modulo rounding error or bandwidth
|
|
||||||
* control); however, such cases are rare and we can fix these
|
|
||||||
* at enqueue.
|
|
||||||
*
|
|
||||||
* TODO: fix up out-of-order children on enqueue.
|
|
||||||
*/
|
|
||||||
if (!se->avg.runnable_avg_sum && !cfs_rq->nr_running)
|
|
||||||
list_del_leaf_cfs_rq(cfs_rq);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void update_blocked_averages(int cpu)
|
static void update_blocked_averages(int cpu)
|
||||||
{
|
{
|
||||||
struct rq *rq = cpu_rq(cpu);
|
struct rq *rq = cpu_rq(cpu);
|
||||||
@ -6004,19 +5782,19 @@ static void update_blocked_averages(int cpu)
|
|||||||
|
|
||||||
raw_spin_lock_irqsave(&rq->lock, flags);
|
raw_spin_lock_irqsave(&rq->lock, flags);
|
||||||
update_rq_clock(rq);
|
update_rq_clock(rq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Iterates the task_group tree in a bottom up fashion, see
|
* Iterates the task_group tree in a bottom up fashion, see
|
||||||
* list_add_leaf_cfs_rq() for details.
|
* list_add_leaf_cfs_rq() for details.
|
||||||
*/
|
*/
|
||||||
for_each_leaf_cfs_rq(rq, cfs_rq) {
|
for_each_leaf_cfs_rq(rq, cfs_rq) {
|
||||||
/*
|
/* throttled entities do not contribute to load */
|
||||||
* Note: We may want to consider periodically releasing
|
if (throttled_hierarchy(cfs_rq))
|
||||||
* rq->lock about these updates so that creating many task
|
continue;
|
||||||
* groups does not result in continually extending hold time.
|
|
||||||
*/
|
|
||||||
__update_blocked_averages_cpu(cfs_rq->tg, rq->cpu);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (update_cfs_rq_load_avg(cfs_rq_clock_task(cfs_rq), cfs_rq))
|
||||||
|
update_tg_load_avg(cfs_rq, 0);
|
||||||
|
}
|
||||||
raw_spin_unlock_irqrestore(&rq->lock, flags);
|
raw_spin_unlock_irqrestore(&rq->lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6044,14 +5822,13 @@ static void update_cfs_rq_h_load(struct cfs_rq *cfs_rq)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!se) {
|
if (!se) {
|
||||||
cfs_rq->h_load = cfs_rq->runnable_load_avg;
|
cfs_rq->h_load = cfs_rq->avg.load_avg;
|
||||||
cfs_rq->last_h_load_update = now;
|
cfs_rq->last_h_load_update = now;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ((se = cfs_rq->h_load_next) != NULL) {
|
while ((se = cfs_rq->h_load_next) != NULL) {
|
||||||
load = cfs_rq->h_load;
|
load = cfs_rq->h_load;
|
||||||
load = div64_ul(load * se->avg.load_avg_contrib,
|
load = div64_ul(load * se->avg.load_avg, cfs_rq->avg.load_avg + 1);
|
||||||
cfs_rq->runnable_load_avg + 1);
|
|
||||||
cfs_rq = group_cfs_rq(se);
|
cfs_rq = group_cfs_rq(se);
|
||||||
cfs_rq->h_load = load;
|
cfs_rq->h_load = load;
|
||||||
cfs_rq->last_h_load_update = now;
|
cfs_rq->last_h_load_update = now;
|
||||||
@ -6063,8 +5840,8 @@ static unsigned long task_h_load(struct task_struct *p)
|
|||||||
struct cfs_rq *cfs_rq = task_cfs_rq(p);
|
struct cfs_rq *cfs_rq = task_cfs_rq(p);
|
||||||
|
|
||||||
update_cfs_rq_h_load(cfs_rq);
|
update_cfs_rq_h_load(cfs_rq);
|
||||||
return div64_ul(p->se.avg.load_avg_contrib * cfs_rq->h_load,
|
return div64_ul(p->se.avg.load_avg * cfs_rq->h_load,
|
||||||
cfs_rq->runnable_load_avg + 1);
|
cfs_rq->avg.load_avg + 1);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static inline void update_blocked_averages(int cpu)
|
static inline void update_blocked_averages(int cpu)
|
||||||
@ -6073,7 +5850,7 @@ static inline void update_blocked_averages(int cpu)
|
|||||||
|
|
||||||
static unsigned long task_h_load(struct task_struct *p)
|
static unsigned long task_h_load(struct task_struct *p)
|
||||||
{
|
{
|
||||||
return p->se.avg.load_avg_contrib;
|
return p->se.avg.load_avg;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -8071,15 +7848,18 @@ static void switched_from_fair(struct rq *rq, struct task_struct *p)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/*
|
/* Catch up with the cfs_rq and remove our load when we leave */
|
||||||
* Remove our load from contribution when we leave sched_fair
|
__update_load_avg(cfs_rq->avg.last_update_time, cpu_of(rq), &se->avg,
|
||||||
* and ensure we don't carry in an old decay_count if we
|
se->on_rq * scale_load_down(se->load.weight), cfs_rq->curr == se);
|
||||||
* switch back.
|
|
||||||
*/
|
cfs_rq->avg.load_avg =
|
||||||
if (se->avg.decay_count) {
|
max_t(long, cfs_rq->avg.load_avg - se->avg.load_avg, 0);
|
||||||
__synchronize_entity_decay(se);
|
cfs_rq->avg.load_sum =
|
||||||
subtract_blocked_load_contrib(cfs_rq, se->avg.load_avg_contrib);
|
max_t(s64, cfs_rq->avg.load_sum - se->avg.load_sum, 0);
|
||||||
}
|
cfs_rq->avg.util_avg =
|
||||||
|
max_t(long, cfs_rq->avg.util_avg - se->avg.util_avg, 0);
|
||||||
|
cfs_rq->avg.util_sum =
|
||||||
|
max_t(s32, cfs_rq->avg.util_sum - se->avg.util_sum, 0);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -8136,8 +7916,8 @@ void init_cfs_rq(struct cfs_rq *cfs_rq)
|
|||||||
cfs_rq->min_vruntime_copy = cfs_rq->min_vruntime;
|
cfs_rq->min_vruntime_copy = cfs_rq->min_vruntime;
|
||||||
#endif
|
#endif
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
atomic64_set(&cfs_rq->decay_counter, 1);
|
atomic_long_set(&cfs_rq->removed_load_avg, 0);
|
||||||
atomic_long_set(&cfs_rq->removed_load, 0);
|
atomic_long_set(&cfs_rq->removed_util_avg, 0);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -8182,14 +7962,14 @@ static void task_move_group_fair(struct task_struct *p, int queued)
|
|||||||
if (!queued) {
|
if (!queued) {
|
||||||
cfs_rq = cfs_rq_of(se);
|
cfs_rq = cfs_rq_of(se);
|
||||||
se->vruntime += cfs_rq->min_vruntime;
|
se->vruntime += cfs_rq->min_vruntime;
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/*
|
/* Virtually synchronize task with its new cfs_rq */
|
||||||
* migrate_task_rq_fair() will have removed our previous
|
p->se.avg.last_update_time = cfs_rq->avg.last_update_time;
|
||||||
* contribution, but we must synchronize for ongoing future
|
cfs_rq->avg.load_avg += p->se.avg.load_avg;
|
||||||
* decay.
|
cfs_rq->avg.load_sum += p->se.avg.load_sum;
|
||||||
*/
|
cfs_rq->avg.util_avg += p->se.avg.util_avg;
|
||||||
se->avg.decay_count = atomic64_read(&cfs_rq->decay_counter);
|
cfs_rq->avg.util_sum += p->se.avg.util_sum;
|
||||||
cfs_rq->blocked_load_avg += se->avg.load_avg_contrib;
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -245,7 +245,6 @@ struct task_group {
|
|||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
atomic_long_t load_avg;
|
atomic_long_t load_avg;
|
||||||
atomic_t runnable_avg;
|
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -366,27 +365,18 @@ struct cfs_rq {
|
|||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
/*
|
/*
|
||||||
* CFS Load tracking
|
* CFS load tracking
|
||||||
* Under CFS, load is tracked on a per-entity basis and aggregated up.
|
|
||||||
* This allows for the description of both thread and group usage (in
|
|
||||||
* the FAIR_GROUP_SCHED case).
|
|
||||||
* runnable_load_avg is the sum of the load_avg_contrib of the
|
|
||||||
* sched_entities on the rq.
|
|
||||||
* blocked_load_avg is similar to runnable_load_avg except that its
|
|
||||||
* the blocked sched_entities on the rq.
|
|
||||||
* utilization_load_avg is the sum of the average running time of the
|
|
||||||
* sched_entities on the rq.
|
|
||||||
*/
|
*/
|
||||||
unsigned long runnable_load_avg, blocked_load_avg, utilization_load_avg;
|
struct sched_avg avg;
|
||||||
atomic64_t decay_counter;
|
#ifdef CONFIG_FAIR_GROUP_SCHED
|
||||||
u64 last_decay;
|
unsigned long tg_load_avg_contrib;
|
||||||
atomic_long_t removed_load;
|
#endif
|
||||||
|
atomic_long_t removed_load_avg, removed_util_avg;
|
||||||
|
#ifndef CONFIG_64BIT
|
||||||
|
u64 load_last_update_time_copy;
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_FAIR_GROUP_SCHED
|
#ifdef CONFIG_FAIR_GROUP_SCHED
|
||||||
/* Required to track per-cpu representation of a task_group */
|
|
||||||
u32 tg_runnable_contrib;
|
|
||||||
unsigned long tg_load_contrib;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* h_load = weight * f(tg)
|
* h_load = weight * f(tg)
|
||||||
*
|
*
|
||||||
|
Loading…
x
Reference in New Issue
Block a user