ce26c5bb95
Four cpufreq-like governors are provided as examples. powersave: use the lowest frequency possible. The user (device) should set the polling_ms as 0 because polling is useless for this governor. performance: use the highest freqeuncy possible. The user (device) should set the polling_ms as 0 because polling is useless for this governor. userspace: use the user specified frequency stored at devfreq.user_set_freq. With sysfs support in the following patch, a user may set the value with the sysfs interface. simple_ondemand: simplified version of cpufreq's ondemand governor. When a user updates OPP entries (enable/disable/add), OPP framework automatically notifies devfreq to update operating frequency accordingly. Thus, devfreq users (device drivers) do not need to update devfreq manually with OPP entry updates or set polling_ms for powersave , performance, userspace, or any other "static" governors. Note that these are given only as basic examples for governors and any devices with devfreq may implement their own governors with the drivers and use them. Signed-off-by: MyungJoo Ham <myungjoo.ham@samsung.com> Signed-off-by: Kyungmin Park <kyungmin.park@samsung.com> Reviewed-by: Mike Turquette <mturquette@ti.com> Acked-by: Kevin Hilman <khilman@ti.com> Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
117 lines
2.6 KiB
C
117 lines
2.6 KiB
C
/*
|
|
* linux/drivers/devfreq/governor_simpleondemand.c
|
|
*
|
|
* Copyright (C) 2011 Samsung Electronics
|
|
* MyungJoo Ham <myungjoo.ham@samsung.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/slab.h>
|
|
#include <linux/device.h>
|
|
#include <linux/devfreq.h>
|
|
#include <linux/pm.h>
|
|
#include <linux/mutex.h>
|
|
#include "governor.h"
|
|
|
|
struct userspace_data {
|
|
unsigned long user_frequency;
|
|
bool valid;
|
|
};
|
|
|
|
static int devfreq_userspace_func(struct devfreq *df, unsigned long *freq)
|
|
{
|
|
struct userspace_data *data = df->data;
|
|
|
|
if (!data->valid)
|
|
*freq = df->previous_freq; /* No user freq specified yet */
|
|
else
|
|
*freq = data->user_frequency;
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t store_freq(struct device *dev, struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
struct devfreq *devfreq = to_devfreq(dev);
|
|
struct userspace_data *data;
|
|
unsigned long wanted;
|
|
int err = 0;
|
|
|
|
|
|
mutex_lock(&devfreq->lock);
|
|
data = devfreq->data;
|
|
|
|
sscanf(buf, "%lu", &wanted);
|
|
data->user_frequency = wanted;
|
|
data->valid = true;
|
|
err = update_devfreq(devfreq);
|
|
if (err == 0)
|
|
err = count;
|
|
mutex_unlock(&devfreq->lock);
|
|
return err;
|
|
}
|
|
|
|
static ssize_t show_freq(struct device *dev, struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
struct devfreq *devfreq = to_devfreq(dev);
|
|
struct userspace_data *data;
|
|
int err = 0;
|
|
|
|
mutex_lock(&devfreq->lock);
|
|
data = devfreq->data;
|
|
|
|
if (data->valid)
|
|
err = sprintf(buf, "%lu\n", data->user_frequency);
|
|
else
|
|
err = sprintf(buf, "undefined\n");
|
|
mutex_unlock(&devfreq->lock);
|
|
return err;
|
|
}
|
|
|
|
static DEVICE_ATTR(set_freq, 0644, show_freq, store_freq);
|
|
static struct attribute *dev_entries[] = {
|
|
&dev_attr_set_freq.attr,
|
|
NULL,
|
|
};
|
|
static struct attribute_group dev_attr_group = {
|
|
.name = "userspace",
|
|
.attrs = dev_entries,
|
|
};
|
|
|
|
static int userspace_init(struct devfreq *devfreq)
|
|
{
|
|
int err = 0;
|
|
struct userspace_data *data = kzalloc(sizeof(struct userspace_data),
|
|
GFP_KERNEL);
|
|
|
|
if (!data) {
|
|
err = -ENOMEM;
|
|
goto out;
|
|
}
|
|
data->valid = false;
|
|
devfreq->data = data;
|
|
|
|
err = sysfs_create_group(&devfreq->dev.kobj, &dev_attr_group);
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
static void userspace_exit(struct devfreq *devfreq)
|
|
{
|
|
sysfs_remove_group(&devfreq->dev.kobj, &dev_attr_group);
|
|
kfree(devfreq->data);
|
|
devfreq->data = NULL;
|
|
}
|
|
|
|
const struct devfreq_governor devfreq_userspace = {
|
|
.name = "userspace",
|
|
.get_target_freq = devfreq_userspace_func,
|
|
.init = userspace_init,
|
|
.exit = userspace_exit,
|
|
.no_central_polling = true,
|
|
};
|