2019-06-04 16:11:33 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2011-10-02 06:19:34 +08:00
|
|
|
/*
|
2017-01-16 20:26:04 +08:00
|
|
|
* linux/drivers/devfreq/governor_userspace.c
|
2011-10-02 06:19:34 +08:00
|
|
|
*
|
|
|
|
* Copyright (C) 2011 Samsung Electronics
|
|
|
|
* MyungJoo Ham <myungjoo.ham@samsung.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/devfreq.h>
|
|
|
|
#include <linux/pm.h>
|
|
|
|
#include <linux/mutex.h>
|
2012-10-30 04:01:46 +08:00
|
|
|
#include <linux/module.h>
|
2011-10-02 06:19:34 +08:00
|
|
|
#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;
|
|
|
|
|
2018-08-04 04:05:10 +08:00
|
|
|
if (data->valid)
|
|
|
|
*freq = data->user_frequency;
|
|
|
|
else
|
2011-10-02 06:19:34 +08:00
|
|
|
*freq = df->previous_freq; /* No user freq specified yet */
|
2018-08-04 04:05:10 +08:00
|
|
|
|
2011-10-02 06:19:34 +08:00
|
|
|
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,
|
|
|
|
};
|
2017-07-03 18:10:04 +08:00
|
|
|
static const struct attribute_group dev_attr_group = {
|
2017-10-23 09:32:12 +08:00
|
|
|
.name = DEVFREQ_GOV_USERSPACE,
|
2011-10-02 06:19:34 +08:00
|
|
|
.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)
|
|
|
|
{
|
2017-01-12 22:57:41 +08:00
|
|
|
/*
|
|
|
|
* Remove the sysfs entry, unless this is being called after
|
|
|
|
* device_del(), which should have done this already via kobject_del().
|
|
|
|
*/
|
|
|
|
if (devfreq->dev.kobj.sd)
|
|
|
|
sysfs_remove_group(&devfreq->dev.kobj, &dev_attr_group);
|
|
|
|
|
2011-10-02 06:19:34 +08:00
|
|
|
kfree(devfreq->data);
|
|
|
|
devfreq->data = NULL;
|
|
|
|
}
|
|
|
|
|
2012-10-26 07:50:09 +08:00
|
|
|
static int devfreq_userspace_handler(struct devfreq *devfreq,
|
|
|
|
unsigned int event, void *data)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case DEVFREQ_GOV_START:
|
|
|
|
ret = userspace_init(devfreq);
|
|
|
|
break;
|
|
|
|
case DEVFREQ_GOV_STOP:
|
|
|
|
userspace_exit(devfreq);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-10-30 04:01:45 +08:00
|
|
|
static struct devfreq_governor devfreq_userspace = {
|
2011-10-02 06:19:34 +08:00
|
|
|
.name = "userspace",
|
|
|
|
.get_target_freq = devfreq_userspace_func,
|
2012-10-26 07:50:09 +08:00
|
|
|
.event_handler = devfreq_userspace_handler,
|
2011-10-02 06:19:34 +08:00
|
|
|
};
|
2012-10-30 04:01:44 +08:00
|
|
|
|
|
|
|
static int __init devfreq_userspace_init(void)
|
|
|
|
{
|
|
|
|
return devfreq_add_governor(&devfreq_userspace);
|
|
|
|
}
|
|
|
|
subsys_initcall(devfreq_userspace_init);
|
|
|
|
|
|
|
|
static void __exit devfreq_userspace_exit(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = devfreq_remove_governor(&devfreq_userspace);
|
|
|
|
if (ret)
|
|
|
|
pr_err("%s: failed remove governor %d\n", __func__, ret);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
module_exit(devfreq_userspace_exit);
|
2012-10-30 04:01:46 +08:00
|
|
|
MODULE_LICENSE("GPL");
|