diff options
Diffstat (limited to 'kernel/sched/cpufreq_schedutil.c')
| -rw-r--r-- | kernel/sched/cpufreq_schedutil.c | 95 | 
1 files changed, 59 insertions, 36 deletions
| diff --git a/kernel/sched/cpufreq_schedutil.c b/kernel/sched/cpufreq_schedutil.c index 2f52ec0f1539..dd062a1c8cf0 100644 --- a/kernel/sched/cpufreq_schedutil.c +++ b/kernel/sched/cpufreq_schedutil.c @@ -60,7 +60,8 @@ struct sugov_cpu {  	u64 last_update;  	/* The fields below are only needed when sharing a policy. */ -	unsigned long util; +	unsigned long util_cfs; +	unsigned long util_dl;  	unsigned long max;  	unsigned int flags; @@ -176,21 +177,28 @@ static unsigned int get_next_freq(struct sugov_policy *sg_policy,  	return cpufreq_driver_resolve_freq(policy, freq);  } -static void sugov_get_util(unsigned long *util, unsigned long *max, int cpu) +static void sugov_get_util(struct sugov_cpu *sg_cpu)  { -	struct rq *rq = cpu_rq(cpu); -	unsigned long cfs_max; +	struct rq *rq = cpu_rq(sg_cpu->cpu); -	cfs_max = arch_scale_cpu_capacity(NULL, cpu); +	sg_cpu->max = arch_scale_cpu_capacity(NULL, sg_cpu->cpu); +	sg_cpu->util_cfs = cpu_util_cfs(rq); +	sg_cpu->util_dl  = cpu_util_dl(rq); +} -	*util = min(rq->cfs.avg.util_avg, cfs_max); -	*max = cfs_max; +static unsigned long sugov_aggregate_util(struct sugov_cpu *sg_cpu) +{ +	/* +	 * Ideally we would like to set util_dl as min/guaranteed freq and +	 * util_cfs + util_dl as requested freq. However, cpufreq is not yet +	 * ready for such an interface. So, we only do the latter for now. +	 */ +	return min(sg_cpu->util_cfs + sg_cpu->util_dl, sg_cpu->max);  } -static void sugov_set_iowait_boost(struct sugov_cpu *sg_cpu, u64 time, -				   unsigned int flags) +static void sugov_set_iowait_boost(struct sugov_cpu *sg_cpu, u64 time)  { -	if (flags & SCHED_CPUFREQ_IOWAIT) { +	if (sg_cpu->flags & SCHED_CPUFREQ_IOWAIT) {  		if (sg_cpu->iowait_boost_pending)  			return; @@ -244,7 +252,7 @@ static void sugov_iowait_boost(struct sugov_cpu *sg_cpu, unsigned long *util,  #ifdef CONFIG_NO_HZ_COMMON  static bool sugov_cpu_is_busy(struct sugov_cpu *sg_cpu)  { -	unsigned long idle_calls = tick_nohz_get_idle_calls(); +	unsigned long idle_calls = tick_nohz_get_idle_calls_cpu(sg_cpu->cpu);  	bool ret = idle_calls == sg_cpu->saved_idle_calls;  	sg_cpu->saved_idle_calls = idle_calls; @@ -264,7 +272,7 @@ static void sugov_update_single(struct update_util_data *hook, u64 time,  	unsigned int next_f;  	bool busy; -	sugov_set_iowait_boost(sg_cpu, time, flags); +	sugov_set_iowait_boost(sg_cpu, time);  	sg_cpu->last_update = time;  	if (!sugov_should_update_freq(sg_policy, time)) @@ -272,10 +280,12 @@ static void sugov_update_single(struct update_util_data *hook, u64 time,  	busy = sugov_cpu_is_busy(sg_cpu); -	if (flags & SCHED_CPUFREQ_RT_DL) { +	if (flags & SCHED_CPUFREQ_RT) {  		next_f = policy->cpuinfo.max_freq;  	} else { -		sugov_get_util(&util, &max, sg_cpu->cpu); +		sugov_get_util(sg_cpu); +		max = sg_cpu->max; +		util = sugov_aggregate_util(sg_cpu);  		sugov_iowait_boost(sg_cpu, &util, &max);  		next_f = get_next_freq(sg_policy, util, max);  		/* @@ -305,23 +315,27 @@ static unsigned int sugov_next_freq_shared(struct sugov_cpu *sg_cpu, u64 time)  		s64 delta_ns;  		/* -		 * If the CPU utilization was last updated before the previous -		 * frequency update and the time elapsed between the last update -		 * of the CPU utilization and the last frequency update is long -		 * enough, don't take the CPU into account as it probably is -		 * idle now (and clear iowait_boost for it). +		 * If the CFS CPU utilization was last updated before the +		 * previous frequency update and the time elapsed between the +		 * last update of the CPU utilization and the last frequency +		 * update is long enough, reset iowait_boost and util_cfs, as +		 * they are now probably stale. However, still consider the +		 * CPU contribution if it has some DEADLINE utilization +		 * (util_dl).  		 */  		delta_ns = time - j_sg_cpu->last_update;  		if (delta_ns > TICK_NSEC) {  			j_sg_cpu->iowait_boost = 0;  			j_sg_cpu->iowait_boost_pending = false; -			continue; +			j_sg_cpu->util_cfs = 0; +			if (j_sg_cpu->util_dl == 0) +				continue;  		} -		if (j_sg_cpu->flags & SCHED_CPUFREQ_RT_DL) +		if (j_sg_cpu->flags & SCHED_CPUFREQ_RT)  			return policy->cpuinfo.max_freq; -		j_util = j_sg_cpu->util;  		j_max = j_sg_cpu->max; +		j_util = sugov_aggregate_util(j_sg_cpu);  		if (j_util * max > j_max * util) {  			util = j_util;  			max = j_max; @@ -338,22 +352,18 @@ static void sugov_update_shared(struct update_util_data *hook, u64 time,  {  	struct sugov_cpu *sg_cpu = container_of(hook, struct sugov_cpu, update_util);  	struct sugov_policy *sg_policy = sg_cpu->sg_policy; -	unsigned long util, max;  	unsigned int next_f; -	sugov_get_util(&util, &max, sg_cpu->cpu); -  	raw_spin_lock(&sg_policy->update_lock); -	sg_cpu->util = util; -	sg_cpu->max = max; +	sugov_get_util(sg_cpu);  	sg_cpu->flags = flags; -	sugov_set_iowait_boost(sg_cpu, time, flags); +	sugov_set_iowait_boost(sg_cpu, time);  	sg_cpu->last_update = time;  	if (sugov_should_update_freq(sg_policy, time)) { -		if (flags & SCHED_CPUFREQ_RT_DL) +		if (flags & SCHED_CPUFREQ_RT)  			next_f = sg_policy->policy->cpuinfo.max_freq;  		else  			next_f = sugov_next_freq_shared(sg_cpu, time); @@ -383,9 +393,9 @@ static void sugov_irq_work(struct irq_work *irq_work)  	sg_policy = container_of(irq_work, struct sugov_policy, irq_work);  	/* -	 * For RT and deadline tasks, the schedutil governor shoots the -	 * frequency to maximum. Special care must be taken to ensure that this -	 * kthread doesn't result in the same behavior. +	 * For RT tasks, the schedutil governor shoots the frequency to maximum. +	 * Special care must be taken to ensure that this kthread doesn't result +	 * in the same behavior.  	 *  	 * This is (mostly) guaranteed by the work_in_progress flag. The flag is  	 * updated only at the end of the sugov_work() function and before that @@ -470,7 +480,20 @@ static void sugov_policy_free(struct sugov_policy *sg_policy)  static int sugov_kthread_create(struct sugov_policy *sg_policy)  {  	struct task_struct *thread; -	struct sched_param param = { .sched_priority = MAX_USER_RT_PRIO / 2 }; +	struct sched_attr attr = { +		.size = sizeof(struct sched_attr), +		.sched_policy = SCHED_DEADLINE, +		.sched_flags = SCHED_FLAG_SUGOV, +		.sched_nice = 0, +		.sched_priority = 0, +		/* +		 * Fake (unused) bandwidth; workaround to "fix" +		 * priority inheritance. +		 */ +		.sched_runtime	=  1000000, +		.sched_deadline = 10000000, +		.sched_period	= 10000000, +	};  	struct cpufreq_policy *policy = sg_policy->policy;  	int ret; @@ -488,10 +511,10 @@ static int sugov_kthread_create(struct sugov_policy *sg_policy)  		return PTR_ERR(thread);  	} -	ret = sched_setscheduler_nocheck(thread, SCHED_FIFO, ¶m); +	ret = sched_setattr_nocheck(thread, &attr);  	if (ret) {  		kthread_stop(thread); -		pr_warn("%s: failed to set SCHED_FIFO\n", __func__); +		pr_warn("%s: failed to set SCHED_DEADLINE\n", __func__);  		return ret;  	} @@ -655,7 +678,7 @@ static int sugov_start(struct cpufreq_policy *policy)  		memset(sg_cpu, 0, sizeof(*sg_cpu));  		sg_cpu->cpu = cpu;  		sg_cpu->sg_policy = sg_policy; -		sg_cpu->flags = SCHED_CPUFREQ_RT; +		sg_cpu->flags = 0;  		sg_cpu->iowait_boost_max = policy->cpuinfo.max_freq;  	} | 
