mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-11 13:27:06 +09:00
745 lines
20 KiB
C
745 lines
20 KiB
C
/*
|
|
* CPUFreq hotplug governor
|
|
*
|
|
* Copyright (C) 2010 Texas Instruments, Inc.
|
|
* Mike Turquette <mturquette@ti.com>
|
|
* Santosh Shilimkar <santosh.shilimkar@ti.com>
|
|
*
|
|
* Based on ondemand governor
|
|
* Copyright (C) 2001 Russell King
|
|
* (C) 2003 Venkatesh Pallipadi <venkatesh.pallipadi@intel.com>,
|
|
* Jun Nakajima <jun.nakajima@intel.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/cpufreq.h>
|
|
#include <linux/cpu.h>
|
|
#include <linux/jiffies.h>
|
|
#include <linux/kernel_stat.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/hrtimer.h>
|
|
#include <linux/tick.h>
|
|
#include <linux/ktime.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/err.h>
|
|
#include <linux/slab.h>
|
|
|
|
/* greater than 80% avg load across online CPUs increases frequency */
|
|
#define DEFAULT_UP_FREQ_MIN_LOAD (80)
|
|
|
|
/* Keep 10% of idle under the up threshold when decreasing the frequency */
|
|
#define DEFAULT_FREQ_DOWN_DIFFERENTIAL (10)
|
|
|
|
/* less than 35% avg load across online CPUs decreases frequency */
|
|
#define DEFAULT_DOWN_FREQ_MAX_LOAD (35)
|
|
|
|
/* default sampling period (uSec) is bogus; 10x ondemand's default for x86 */
|
|
#define DEFAULT_SAMPLING_PERIOD (100000)
|
|
|
|
/* default number of sampling periods to average before hotplug-in decision */
|
|
#define DEFAULT_HOTPLUG_IN_SAMPLING_PERIODS (5)
|
|
|
|
/* default number of sampling periods to average before hotplug-out decision */
|
|
#define DEFAULT_HOTPLUG_OUT_SAMPLING_PERIODS (20)
|
|
|
|
static void do_dbs_timer(struct work_struct *work);
|
|
static int cpufreq_governor_dbs(struct cpufreq_policy *policy,
|
|
unsigned int event);
|
|
|
|
#ifndef CONFIG_CPU_FREQ_DEFAULT_GOV_HOTPLUG
|
|
static
|
|
#endif
|
|
struct cpufreq_governor cpufreq_gov_hotplug = {
|
|
.name = "hotplug",
|
|
.governor = cpufreq_governor_dbs,
|
|
.owner = THIS_MODULE,
|
|
};
|
|
|
|
struct cpu_dbs_info_s {
|
|
cputime64_t prev_cpu_idle;
|
|
cputime64_t prev_cpu_wall;
|
|
cputime64_t prev_cpu_nice;
|
|
struct cpufreq_policy *cur_policy;
|
|
struct delayed_work work;
|
|
struct cpufreq_frequency_table *freq_table;
|
|
int cpu;
|
|
/*
|
|
* percpu mutex that serializes governor limit change with
|
|
* do_dbs_timer invocation. We do not want do_dbs_timer to run
|
|
* when user is changing the governor or limits.
|
|
*/
|
|
struct mutex timer_mutex;
|
|
};
|
|
static DEFINE_PER_CPU(struct cpu_dbs_info_s, hp_cpu_dbs_info);
|
|
|
|
static unsigned int dbs_enable; /* number of CPUs using this policy */
|
|
|
|
/*
|
|
* dbs_mutex protects data in dbs_tuners_ins from concurrent changes on
|
|
* different CPUs. It protects dbs_enable in governor start/stop.
|
|
*/
|
|
static DEFINE_MUTEX(dbs_mutex);
|
|
|
|
static struct workqueue_struct *khotplug_wq;
|
|
|
|
static struct dbs_tuners {
|
|
unsigned int sampling_rate;
|
|
unsigned int up_threshold;
|
|
unsigned int down_differential;
|
|
unsigned int down_threshold;
|
|
unsigned int hotplug_in_sampling_periods;
|
|
unsigned int hotplug_out_sampling_periods;
|
|
unsigned int hotplug_load_index;
|
|
unsigned int *hotplug_load_history;
|
|
unsigned int ignore_nice;
|
|
unsigned int io_is_busy;
|
|
} dbs_tuners_ins = {
|
|
.sampling_rate = DEFAULT_SAMPLING_PERIOD,
|
|
.up_threshold = DEFAULT_UP_FREQ_MIN_LOAD,
|
|
.down_differential = DEFAULT_FREQ_DOWN_DIFFERENTIAL,
|
|
.down_threshold = DEFAULT_DOWN_FREQ_MAX_LOAD,
|
|
.hotplug_in_sampling_periods = DEFAULT_HOTPLUG_IN_SAMPLING_PERIODS,
|
|
.hotplug_out_sampling_periods = DEFAULT_HOTPLUG_OUT_SAMPLING_PERIODS,
|
|
.hotplug_load_index = 0,
|
|
.ignore_nice = 0,
|
|
.io_is_busy = 0,
|
|
};
|
|
|
|
/*
|
|
* A corner case exists when switching io_is_busy at run-time: comparing idle
|
|
* times from a non-io_is_busy period to an io_is_busy period (or vice-versa)
|
|
* will misrepresent the actual change in system idleness. We ignore this
|
|
* corner case: enabling io_is_busy might cause freq increase and disabling
|
|
* might cause freq decrease, which probably matches the original intent.
|
|
*/
|
|
static inline cputime64_t get_cpu_idle_time(unsigned int cpu, cputime64_t *wall)
|
|
{
|
|
u64 idle_time;
|
|
u64 iowait_time;
|
|
|
|
/* cpufreq-hotplug always assumes CONFIG_NO_HZ */
|
|
idle_time = get_cpu_idle_time_us(cpu, wall);
|
|
|
|
/* add time spent doing I/O to idle time */
|
|
if (dbs_tuners_ins.io_is_busy) {
|
|
iowait_time = get_cpu_iowait_time_us(cpu, wall);
|
|
/* cpufreq-hotplug always assumes CONFIG_NO_HZ */
|
|
if (iowait_time != -1ULL && idle_time >= iowait_time)
|
|
idle_time -= iowait_time;
|
|
}
|
|
|
|
return idle_time;
|
|
}
|
|
|
|
/************************** sysfs interface ************************/
|
|
|
|
/* XXX look at global sysfs macros in cpufreq.h, can those be used here? */
|
|
|
|
/* cpufreq_hotplug Governor Tunables */
|
|
#define show_one(file_name, object) \
|
|
static ssize_t show_##file_name \
|
|
(struct kobject *kobj, struct attribute *attr, char *buf) \
|
|
{ \
|
|
return sprintf(buf, "%u\n", dbs_tuners_ins.object); \
|
|
}
|
|
show_one(sampling_rate, sampling_rate);
|
|
show_one(up_threshold, up_threshold);
|
|
show_one(down_differential, down_differential);
|
|
show_one(down_threshold, down_threshold);
|
|
show_one(hotplug_in_sampling_periods, hotplug_in_sampling_periods);
|
|
show_one(hotplug_out_sampling_periods, hotplug_out_sampling_periods);
|
|
show_one(ignore_nice_load, ignore_nice);
|
|
show_one(io_is_busy, io_is_busy);
|
|
|
|
static ssize_t store_sampling_rate(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
if (ret != 1)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_tuners_ins.sampling_rate = input;
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static ssize_t store_up_threshold(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
|
|
if (ret != 1 || input <= dbs_tuners_ins.down_threshold) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_tuners_ins.up_threshold = input;
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static ssize_t store_down_differential(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
|
|
if (ret != 1 || input >= dbs_tuners_ins.up_threshold)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_tuners_ins.down_differential = input;
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static ssize_t store_down_threshold(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
|
|
if (ret != 1 || input >= dbs_tuners_ins.up_threshold) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_tuners_ins.down_threshold = input;
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static ssize_t store_hotplug_in_sampling_periods(struct kobject *a,
|
|
struct attribute *b, const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
unsigned int *temp;
|
|
unsigned int max_windows;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
|
|
if (ret != 1)
|
|
return -EINVAL;
|
|
|
|
/* already using this value, bail out */
|
|
if (input == dbs_tuners_ins.hotplug_in_sampling_periods)
|
|
return count;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
ret = count;
|
|
max_windows = max(dbs_tuners_ins.hotplug_in_sampling_periods,
|
|
dbs_tuners_ins.hotplug_out_sampling_periods);
|
|
|
|
/* no need to resize array */
|
|
if (input <= max_windows) {
|
|
dbs_tuners_ins.hotplug_in_sampling_periods = input;
|
|
goto out;
|
|
}
|
|
|
|
/* resize array */
|
|
temp = kmalloc((sizeof(unsigned int) * input), GFP_KERNEL);
|
|
|
|
if (!temp || IS_ERR(temp)) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
memcpy(temp, dbs_tuners_ins.hotplug_load_history,
|
|
(max_windows * sizeof(unsigned int)));
|
|
kfree(dbs_tuners_ins.hotplug_load_history);
|
|
|
|
/* replace old buffer, old number of sampling periods & old index */
|
|
dbs_tuners_ins.hotplug_load_history = temp;
|
|
dbs_tuners_ins.hotplug_in_sampling_periods = input;
|
|
dbs_tuners_ins.hotplug_load_index = max_windows;
|
|
out:
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t store_hotplug_out_sampling_periods(struct kobject *a,
|
|
struct attribute *b, const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
unsigned int *temp;
|
|
unsigned int max_windows;
|
|
int ret;
|
|
ret = sscanf(buf, "%u", &input);
|
|
|
|
if (ret != 1)
|
|
return -EINVAL;
|
|
|
|
/* already using this value, bail out */
|
|
if (input == dbs_tuners_ins.hotplug_out_sampling_periods)
|
|
return count;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
ret = count;
|
|
max_windows = max(dbs_tuners_ins.hotplug_in_sampling_periods,
|
|
dbs_tuners_ins.hotplug_out_sampling_periods);
|
|
|
|
/* no need to resize array */
|
|
if (input <= max_windows) {
|
|
dbs_tuners_ins.hotplug_out_sampling_periods = input;
|
|
goto out;
|
|
}
|
|
|
|
/* resize array */
|
|
temp = kmalloc((sizeof(unsigned int) * input), GFP_KERNEL);
|
|
|
|
if (!temp || IS_ERR(temp)) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
memcpy(temp, dbs_tuners_ins.hotplug_load_history,
|
|
(max_windows * sizeof(unsigned int)));
|
|
kfree(dbs_tuners_ins.hotplug_load_history);
|
|
|
|
/* replace old buffer, old number of sampling periods & old index */
|
|
dbs_tuners_ins.hotplug_load_history = temp;
|
|
dbs_tuners_ins.hotplug_out_sampling_periods = input;
|
|
dbs_tuners_ins.hotplug_load_index = max_windows;
|
|
out:
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t store_ignore_nice_load(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
|
|
unsigned int j;
|
|
|
|
ret = sscanf(buf, "%u", &input);
|
|
if (ret != 1)
|
|
return -EINVAL;
|
|
|
|
if (input > 1)
|
|
input = 1;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
if (input == dbs_tuners_ins.ignore_nice) { /* nothing to do */
|
|
mutex_unlock(&dbs_mutex);
|
|
return count;
|
|
}
|
|
dbs_tuners_ins.ignore_nice = input;
|
|
|
|
/* we need to re-evaluate prev_cpu_idle */
|
|
for_each_online_cpu(j) {
|
|
struct cpu_dbs_info_s *dbs_info;
|
|
dbs_info = &per_cpu(hp_cpu_dbs_info, j);
|
|
dbs_info->prev_cpu_idle = get_cpu_idle_time(j,
|
|
&dbs_info->prev_cpu_wall);
|
|
if (dbs_tuners_ins.ignore_nice)
|
|
dbs_info->prev_cpu_nice = kstat_cpu(j).cpustat.nice;
|
|
|
|
}
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static ssize_t store_io_is_busy(struct kobject *a, struct attribute *b,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int input;
|
|
int ret;
|
|
|
|
ret = sscanf(buf, "%u", &input);
|
|
if (ret != 1)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_tuners_ins.io_is_busy = !!input;
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
define_one_global_rw(sampling_rate);
|
|
define_one_global_rw(up_threshold);
|
|
define_one_global_rw(down_differential);
|
|
define_one_global_rw(down_threshold);
|
|
define_one_global_rw(hotplug_in_sampling_periods);
|
|
define_one_global_rw(hotplug_out_sampling_periods);
|
|
define_one_global_rw(ignore_nice_load);
|
|
define_one_global_rw(io_is_busy);
|
|
|
|
static struct attribute *dbs_attributes[] = {
|
|
&sampling_rate.attr,
|
|
&up_threshold.attr,
|
|
&down_differential.attr,
|
|
&down_threshold.attr,
|
|
&hotplug_in_sampling_periods.attr,
|
|
&hotplug_out_sampling_periods.attr,
|
|
&ignore_nice_load.attr,
|
|
&io_is_busy.attr,
|
|
NULL
|
|
};
|
|
|
|
static struct attribute_group dbs_attr_group = {
|
|
.attrs = dbs_attributes,
|
|
.name = "hotplug",
|
|
};
|
|
|
|
/************************** sysfs end ************************/
|
|
|
|
static void dbs_check_cpu(struct cpu_dbs_info_s *this_dbs_info)
|
|
{
|
|
/* combined load of all enabled CPUs */
|
|
unsigned int total_load = 0;
|
|
/* single largest CPU load percentage*/
|
|
unsigned int max_load = 0;
|
|
/* largest CPU load in terms of frequency */
|
|
unsigned int max_load_freq = 0;
|
|
/* average load across all enabled CPUs */
|
|
unsigned int avg_load = 0;
|
|
/* average load across multiple sampling periods for hotplug events */
|
|
unsigned int hotplug_in_avg_load = 0;
|
|
unsigned int hotplug_out_avg_load = 0;
|
|
/* number of sampling periods averaged for hotplug decisions */
|
|
unsigned int periods;
|
|
|
|
struct cpufreq_policy *policy;
|
|
unsigned int i, j;
|
|
|
|
policy = this_dbs_info->cur_policy;
|
|
|
|
/*
|
|
* cpu load accounting
|
|
* get highest load, total load and average load across all CPUs
|
|
*/
|
|
for_each_cpu(j, policy->cpus) {
|
|
unsigned int load;
|
|
unsigned int idle_time, wall_time;
|
|
cputime64_t cur_wall_time, cur_idle_time;
|
|
struct cpu_dbs_info_s *j_dbs_info;
|
|
|
|
j_dbs_info = &per_cpu(hp_cpu_dbs_info, j);
|
|
|
|
/* update both cur_idle_time and cur_wall_time */
|
|
cur_idle_time = get_cpu_idle_time(j, &cur_wall_time);
|
|
|
|
/* how much wall time has passed since last iteration? */
|
|
wall_time = (unsigned int) cputime64_sub(cur_wall_time,
|
|
j_dbs_info->prev_cpu_wall);
|
|
j_dbs_info->prev_cpu_wall = cur_wall_time;
|
|
|
|
/* how much idle time has passed since last iteration? */
|
|
idle_time = (unsigned int) cputime64_sub(cur_idle_time,
|
|
j_dbs_info->prev_cpu_idle);
|
|
j_dbs_info->prev_cpu_idle = cur_idle_time;
|
|
|
|
if (unlikely(!wall_time || wall_time < idle_time))
|
|
continue;
|
|
|
|
/* load is the percentage of time not spent in idle */
|
|
load = 100 * (wall_time - idle_time) / wall_time;
|
|
|
|
/* keep track of combined load across all CPUs */
|
|
total_load += load;
|
|
|
|
/* keep track of highest single load across all CPUs */
|
|
if (load > max_load)
|
|
max_load = load;
|
|
}
|
|
|
|
/* use the max load in the OPP freq change policy */
|
|
max_load_freq = max_load * policy->cur;
|
|
|
|
/* calculate the average load across all related CPUs */
|
|
avg_load = total_load / num_online_cpus();
|
|
|
|
|
|
/*
|
|
* hotplug load accounting
|
|
* average load over multiple sampling periods
|
|
*/
|
|
|
|
/* how many sampling periods do we use for hotplug decisions? */
|
|
periods = max(dbs_tuners_ins.hotplug_in_sampling_periods,
|
|
dbs_tuners_ins.hotplug_out_sampling_periods);
|
|
|
|
/* store avg_load in the circular buffer */
|
|
dbs_tuners_ins.hotplug_load_history[dbs_tuners_ins.hotplug_load_index]
|
|
= avg_load;
|
|
|
|
/* compute average load across in & out sampling periods */
|
|
for (i = 0, j = dbs_tuners_ins.hotplug_load_index;
|
|
i < periods; i++, j--) {
|
|
if (i < dbs_tuners_ins.hotplug_in_sampling_periods)
|
|
hotplug_in_avg_load +=
|
|
dbs_tuners_ins.hotplug_load_history[j];
|
|
if (i < dbs_tuners_ins.hotplug_out_sampling_periods)
|
|
hotplug_out_avg_load +=
|
|
dbs_tuners_ins.hotplug_load_history[j];
|
|
|
|
if (j == 0)
|
|
j = periods;
|
|
}
|
|
|
|
hotplug_in_avg_load = hotplug_in_avg_load /
|
|
dbs_tuners_ins.hotplug_in_sampling_periods;
|
|
|
|
hotplug_out_avg_load = hotplug_out_avg_load /
|
|
dbs_tuners_ins.hotplug_out_sampling_periods;
|
|
|
|
/* return to first element if we're at the circular buffer's end */
|
|
if (++dbs_tuners_ins.hotplug_load_index == periods)
|
|
dbs_tuners_ins.hotplug_load_index = 0;
|
|
|
|
/* check if auxiliary CPU is needed based on avg_load */
|
|
if (avg_load > dbs_tuners_ins.up_threshold) {
|
|
/* should we enable auxillary CPUs? */
|
|
if (num_online_cpus() < 2 && hotplug_in_avg_load >
|
|
dbs_tuners_ins.up_threshold) {
|
|
/* hotplug with cpufreq is nasty
|
|
* a call to cpufreq_governor_dbs may cause a lockup.
|
|
* wq is not running here so its safe.
|
|
*/
|
|
mutex_unlock(&this_dbs_info->timer_mutex);
|
|
cpu_up(1);
|
|
mutex_lock(&this_dbs_info->timer_mutex);
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* check for frequency increase based on max_load */
|
|
if (max_load > dbs_tuners_ins.up_threshold) {
|
|
/* increase to highest frequency supported */
|
|
if (policy->cur < policy->max)
|
|
__cpufreq_driver_target(policy, policy->max,
|
|
CPUFREQ_RELATION_H);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/* check for frequency decrease */
|
|
if (avg_load < dbs_tuners_ins.down_threshold) {
|
|
/* are we at the minimum frequency already? */
|
|
if (policy->cur == policy->min) {
|
|
/* should we disable auxillary CPUs? */
|
|
if (num_online_cpus() > 1 && hotplug_out_avg_load <
|
|
dbs_tuners_ins.down_threshold) {
|
|
mutex_unlock(&this_dbs_info->timer_mutex);
|
|
cpu_down(1);
|
|
mutex_lock(&this_dbs_info->timer_mutex);
|
|
}
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* go down to the lowest frequency which can sustain the load by
|
|
* keeping 30% of idle in order to not cross the up_threshold
|
|
*/
|
|
if ((max_load_freq <
|
|
(dbs_tuners_ins.up_threshold - dbs_tuners_ins.down_differential) *
|
|
policy->cur) && (policy->cur > policy->min)) {
|
|
unsigned int freq_next;
|
|
freq_next = max_load_freq /
|
|
(dbs_tuners_ins.up_threshold -
|
|
dbs_tuners_ins.down_differential);
|
|
|
|
if (freq_next < policy->min)
|
|
freq_next = policy->min;
|
|
|
|
__cpufreq_driver_target(policy, freq_next,
|
|
CPUFREQ_RELATION_L);
|
|
}
|
|
out:
|
|
return;
|
|
}
|
|
|
|
static void do_dbs_timer(struct work_struct *work)
|
|
{
|
|
struct cpu_dbs_info_s *dbs_info =
|
|
container_of(work, struct cpu_dbs_info_s, work.work);
|
|
unsigned int cpu = dbs_info->cpu;
|
|
|
|
/* We want all related CPUs to do sampling nearly on same jiffy */
|
|
int delay = usecs_to_jiffies(dbs_tuners_ins.sampling_rate);
|
|
|
|
mutex_lock(&dbs_info->timer_mutex);
|
|
dbs_check_cpu(dbs_info);
|
|
queue_delayed_work_on(cpu, khotplug_wq, &dbs_info->work, delay);
|
|
mutex_unlock(&dbs_info->timer_mutex);
|
|
}
|
|
|
|
static inline void dbs_timer_init(struct cpu_dbs_info_s *dbs_info)
|
|
{
|
|
/* We want all related CPUs to do sampling nearly on same jiffy */
|
|
int delay = usecs_to_jiffies(dbs_tuners_ins.sampling_rate);
|
|
delay -= jiffies % delay;
|
|
|
|
INIT_DELAYED_WORK_DEFERRABLE(&dbs_info->work, do_dbs_timer);
|
|
queue_delayed_work_on(dbs_info->cpu, khotplug_wq, &dbs_info->work,
|
|
delay);
|
|
}
|
|
|
|
static inline void dbs_timer_exit(struct cpu_dbs_info_s *dbs_info)
|
|
{
|
|
cancel_delayed_work_sync(&dbs_info->work);
|
|
}
|
|
|
|
static int cpufreq_governor_dbs(struct cpufreq_policy *policy,
|
|
unsigned int event)
|
|
{
|
|
unsigned int cpu = policy->cpu;
|
|
struct cpu_dbs_info_s *this_dbs_info;
|
|
unsigned int i, j, max_periods;
|
|
int rc;
|
|
|
|
this_dbs_info = &per_cpu(hp_cpu_dbs_info, cpu);
|
|
|
|
switch (event) {
|
|
case CPUFREQ_GOV_START:
|
|
if ((!cpu_online(cpu)) || (!policy->cur))
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
dbs_enable++;
|
|
for_each_cpu(j, policy->cpus) {
|
|
struct cpu_dbs_info_s *j_dbs_info;
|
|
j_dbs_info = &per_cpu(hp_cpu_dbs_info, j);
|
|
j_dbs_info->cur_policy = policy;
|
|
|
|
j_dbs_info->prev_cpu_idle = get_cpu_idle_time(j,
|
|
&j_dbs_info->prev_cpu_wall);
|
|
if (dbs_tuners_ins.ignore_nice) {
|
|
j_dbs_info->prev_cpu_nice =
|
|
kstat_cpu(j).cpustat.nice;
|
|
}
|
|
|
|
max_periods = max(DEFAULT_HOTPLUG_IN_SAMPLING_PERIODS,
|
|
DEFAULT_HOTPLUG_OUT_SAMPLING_PERIODS);
|
|
dbs_tuners_ins.hotplug_load_history = kmalloc(
|
|
(sizeof(unsigned int) * max_periods),
|
|
GFP_KERNEL);
|
|
if (!dbs_tuners_ins.hotplug_load_history) {
|
|
WARN_ON(1);
|
|
return -ENOMEM;
|
|
}
|
|
for (i = 0; i < max_periods; i++)
|
|
dbs_tuners_ins.hotplug_load_history[i] = 50;
|
|
}
|
|
this_dbs_info->cpu = cpu;
|
|
this_dbs_info->freq_table = cpufreq_frequency_get_table(cpu);
|
|
/*
|
|
* Start the timerschedule work, when this governor
|
|
* is used for first time
|
|
*/
|
|
if (dbs_enable == 1) {
|
|
rc = sysfs_create_group(cpufreq_global_kobject,
|
|
&dbs_attr_group);
|
|
if (rc) {
|
|
mutex_unlock(&dbs_mutex);
|
|
return rc;
|
|
}
|
|
}
|
|
mutex_unlock(&dbs_mutex);
|
|
|
|
mutex_init(&this_dbs_info->timer_mutex);
|
|
dbs_timer_init(this_dbs_info);
|
|
break;
|
|
|
|
case CPUFREQ_GOV_STOP:
|
|
dbs_timer_exit(this_dbs_info);
|
|
|
|
mutex_lock(&dbs_mutex);
|
|
mutex_destroy(&this_dbs_info->timer_mutex);
|
|
dbs_enable--;
|
|
mutex_unlock(&dbs_mutex);
|
|
if (!dbs_enable)
|
|
sysfs_remove_group(cpufreq_global_kobject,
|
|
&dbs_attr_group);
|
|
kfree(dbs_tuners_ins.hotplug_load_history);
|
|
/*
|
|
* XXX BIG CAVEAT: Stopping the governor with CPU1 offline
|
|
* will result in it remaining offline until the user onlines
|
|
* it again. It is up to the user to do this (for now).
|
|
*/
|
|
break;
|
|
|
|
case CPUFREQ_GOV_LIMITS:
|
|
mutex_lock(&this_dbs_info->timer_mutex);
|
|
if (policy->max < this_dbs_info->cur_policy->cur)
|
|
__cpufreq_driver_target(this_dbs_info->cur_policy,
|
|
policy->max, CPUFREQ_RELATION_H);
|
|
else if (policy->min > this_dbs_info->cur_policy->cur)
|
|
__cpufreq_driver_target(this_dbs_info->cur_policy,
|
|
policy->min, CPUFREQ_RELATION_L);
|
|
mutex_unlock(&this_dbs_info->timer_mutex);
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int __init cpufreq_gov_dbs_init(void)
|
|
{
|
|
int err;
|
|
cputime64_t wall;
|
|
u64 idle_time;
|
|
int cpu = get_cpu();
|
|
|
|
idle_time = get_cpu_idle_time_us(cpu, &wall);
|
|
put_cpu();
|
|
if (idle_time != -1ULL) {
|
|
dbs_tuners_ins.up_threshold = DEFAULT_UP_FREQ_MIN_LOAD;
|
|
} else {
|
|
pr_err("cpufreq-hotplug: %s: assumes CONFIG_NO_HZ\n",
|
|
__func__);
|
|
return -EINVAL;
|
|
}
|
|
|
|
khotplug_wq = create_workqueue("khotplug");
|
|
if (!khotplug_wq) {
|
|
pr_err("Creation of khotplug failed\n");
|
|
return -EFAULT;
|
|
}
|
|
err = cpufreq_register_governor(&cpufreq_gov_hotplug);
|
|
if (err)
|
|
destroy_workqueue(khotplug_wq);
|
|
|
|
return err;
|
|
}
|
|
|
|
static void __exit cpufreq_gov_dbs_exit(void)
|
|
{
|
|
cpufreq_unregister_governor(&cpufreq_gov_hotplug);
|
|
destroy_workqueue(khotplug_wq);
|
|
}
|
|
|
|
MODULE_AUTHOR("Mike Turquette <mturquette@ti.com>");
|
|
MODULE_DESCRIPTION("'cpufreq_hotplug' - cpufreq governor for dynamic frequency scaling and CPU hotplugging");
|
|
MODULE_LICENSE("GPL");
|
|
|
|
#ifdef CONFIG_CPU_FREQ_DEFAULT_GOV_HOTPLUG
|
|
fs_initcall(cpufreq_gov_dbs_init);
|
|
#else
|
|
module_init(cpufreq_gov_dbs_init);
|
|
#endif
|
|
module_exit(cpufreq_gov_dbs_exit);
|