On 2 March 2016 at 23:49, Rafael J. Wysocki <rafael@xxxxxxxxxx> wrote: > On Wed, Mar 2, 2016 at 6:58 PM, Rafael J. Wysocki <rafael@xxxxxxxxxx> wrote: >> On Wed, Mar 2, 2016 at 6:10 PM, Vincent Guittot >> <vincent.guittot@xxxxxxxxxx> wrote: >>> Hi Rafael, >>> >>> >>> On 2 March 2016 at 03:27, Rafael J. Wysocki <rjw@xxxxxxxxxxxxx> wrote: >>>> From: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx> >>>> >>>> Add a new cpufreq scaling governor, called "schedutil", that uses >>>> scheduler-provided CPU utilization information as input for making >>>> its decisions. >>>> >>>> Doing that is possible after commit fe7034338ba0 (cpufreq: Add >>>> mechanism for registering utilization update callbacks) that >>>> introduced cpufreq_update_util() called by the scheduler on >>>> utilization changes (from CFS) and RT/DL task status updates. >>>> In particular, CPU frequency scaling decisions may be based on >>>> the the utilization data passed to cpufreq_update_util() by CFS. >>>> >>>> The new governor is relatively simple. >>>> >>>> The frequency selection formula used by it is essentially the same >>>> as the one used by the "ondemand" governor, although it doesn't use >>>> the additional up_threshold parameter, but instead of computing the >>>> load as the "non-idle CPU time" to "total CPU time" ratio, it takes >>>> the utilization data provided by CFS as input. More specifically, >>>> it represents "load" as the util/max ratio, where util and max >>>> are the utilization and CPU capacity coming from CFS. >>>> >>> >>> [snip] >>> >>>> + >>>> +static void sugov_update_commit(struct sugov_policy *sg_policy, u64 time, >>>> + unsigned long util, unsigned long max, >>>> + unsigned int next_freq) >>>> +{ >>>> + struct cpufreq_policy *policy = sg_policy->policy; >>>> + unsigned int rel; >>>> + >>>> + if (next_freq > policy->max) >>>> + next_freq = policy->max; >>>> + else if (next_freq < policy->min) >>>> + next_freq = policy->min; >>>> + >>>> + sg_policy->last_freq_update_time = time; >>>> + if (sg_policy->next_freq == next_freq) >>>> + return; >>>> + >>>> + sg_policy->next_freq = next_freq; >>>> + /* >>>> + * If utilization is less than max / 4, use RELATION_C to allow the >>>> + * minimum frequency to be selected more often in case the distance from >>>> + * it to the next available frequency in the table is significant. >>>> + */ >>>> + rel = util < (max >> 2) ? CPUFREQ_RELATION_C : CPUFREQ_RELATION_L; >>>> + if (policy->fast_switch_possible) { >>>> + cpufreq_driver_fast_switch(policy, next_freq, rel); >>>> + } else { >>>> + sg_policy->relation = rel; >>>> + sg_policy->work_in_progress = true; >>>> + irq_work_queue(&sg_policy->irq_work); >>>> + } >>>> +} >>>> + >>>> +static void sugov_update_single(struct update_util_data *data, u64 time, >>>> + unsigned long util, unsigned long max) >>>> +{ >>>> + struct sugov_cpu *sg_cpu = container_of(data, struct sugov_cpu, update_util); >>>> + struct sugov_policy *sg_policy = sg_cpu->sg_policy; >>>> + unsigned int min_f, max_f, next_f; >>>> + >>>> + if (!sugov_should_update_freq(sg_policy, time)) >>>> + return; >>>> + >>>> + min_f = sg_policy->policy->cpuinfo.min_freq; >>>> + max_f = sg_policy->policy->cpuinfo.max_freq; >>>> + next_f = util > max ? max_f : min_f + util * (max_f - min_f) / max; >>> >>> I think it has been pointed out in another email's thread but you >>> should change the way the next_f is computed. util reflects the >>> utilization of a CPU from 0 to its max compute capacity whereas >>> ondemand was using the load at the current frequency during the last >>> time window. I have understood that you want to keep same formula than >>> ondemand as a starting point but you use a different input to >>> calculate the next frequency so i don't see the rational of keeping >>> this formula. >> >> It is a formula that causes the entire available frequency range to be >> utilized proportionally to the utilization as reported by the >> scheduler (modulo the policy->min/max limits). Its (significant IMO) >> advantage is that it doesn't require any additional factors that would >> need to be determined somehow. > > In case a more formal derivation of this formula is needed, it is > based on the following 3 assumptions: > > (1) Performance is a linear function of frequency. > (2) Required performance is a linear function of the utilization ratio > x = util/max as provided by the scheduler (0 <= x <= 1). Just to mention that the utilization that you are using, varies with the frequency which add another variable in your equation > (3) The minimum possible frequency (min_freq) corresponds to x = 0 and > the maximum possible frequency (max_freq) corresponds to x = 1. > > (1) and (2) combined imply that > > f = a * x + b > > (f - frequency, a, b - constants to be determined) and then (3) quite > trivially leads to b = min_freq and a = max_freq - min_freq. > > Now, of course, the linearity assumptions may be questioned, but then > it's just the first approximation. If you go any further, though, you > end up with an expansion series like this: > > f(x) = c_0 + c_1 * x + c_2 * x^2 + c_3 * x^3 + ... > > where all of the c_j need to be determined in principle. With luck, > if you can guess what kind of a function f(x) may be, it may be > possible to reduce the number of coefficients to determine, but > question is whether or not that is going to work universally for all > systems. > > Thanks, > Rafael -- To unsubscribe from this list: send the line "unsubscribe linux-acpi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html