199 lines
4.5 KiB
C
Executable File
199 lines
4.5 KiB
C
Executable File
/*
|
|
* linux/drivers/devfreq/governor_adaptive.c
|
|
*
|
|
* Copyright(c) 2013-2015 Allwinnertech Co., Ltd.
|
|
*
|
|
* Author: Pan Nan <pannan@allwinnertech.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/module.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/sunxi_dramfreq.h>
|
|
#include "governor.h"
|
|
|
|
extern struct sunxi_dramfreq *dramfreq;
|
|
|
|
static ssize_t show_pause(struct device *dev, struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
if (!dramfreq)
|
|
return sprintf(buf, "paras error\n");
|
|
return sprintf(buf, "%u\n", dramfreq->pause);
|
|
}
|
|
|
|
static ssize_t store_pause(struct device *dev, struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
unsigned int value;
|
|
int ret;
|
|
|
|
if (!dramfreq) {
|
|
pr_err("%s: paras error\n", __func__);
|
|
ret = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
ret = kstrtouint(buf, 0, &value);
|
|
if (ret != 0)
|
|
goto out;
|
|
|
|
if (value && (!dramfreq->pause))
|
|
dramfreq->governor_state_update(
|
|
dramfreq->devfreq->governor_name, STATE_PAUSE);
|
|
else if ((!value) && dramfreq->pause)
|
|
dramfreq->governor_state_update(
|
|
dramfreq->devfreq->governor_name, STATE_RUNNING);
|
|
|
|
ret = count;
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static DEVICE_ATTR(pause, 0644, show_pause, store_pause);
|
|
static struct attribute *dev_entries[] = {
|
|
&dev_attr_pause.attr,
|
|
NULL,
|
|
};
|
|
static struct attribute_group dev_attr_group = {
|
|
.name = "adaptive",
|
|
.attrs = dev_entries,
|
|
};
|
|
|
|
static int devfreq_adaptive_func(struct devfreq *df, unsigned long *freq)
|
|
{
|
|
if (!dramfreq) {
|
|
pr_err("%s: paras error\n", __func__);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!dramfreq->pause) {
|
|
#if defined(CONFIG_ARCH_SUN50IW1) || defined(CONFIG_ARCH_SUN50IW2) || \
|
|
defined(CONFIG_ARCH_SUN8IW11)
|
|
if (dramfreq->key_masters[MASTER_DE] == 0 &&
|
|
dramfreq->key_masters[MASTER_GPU] == 0 &&
|
|
dramfreq->key_masters[MASTER_CSI] == 0) {
|
|
*freq = SUNXI_DRAMFREQ_IDLE;
|
|
} else if (dramfreq->key_masters[MASTER_DE] == 1 &&
|
|
dramfreq->key_masters[MASTER_GPU] == 0 &&
|
|
dramfreq->key_masters[MASTER_CSI] == 0) {
|
|
*freq = SUNXI_DRAMFREQ_NORMAL;
|
|
} else {
|
|
*freq = df->max_freq;
|
|
}
|
|
#elif defined(CONFIG_ARCH_SUN50IW3)
|
|
if (dramfreq->key_masters[MASTER_DE] == 0)
|
|
*freq = SUNXI_DRAMFREQ_IDLE;
|
|
else if (dramfreq->key_masters[MASTER_DE] == 1 &&
|
|
dramfreq->key_masters[MASTER_GPU] == 0 &&
|
|
dramfreq->key_masters[MASTER_VE] == 0)
|
|
*freq = SUNXI_DRAMFREQ_NORMAL;
|
|
else
|
|
*freq = df->max_freq;
|
|
#elif defined(CONFIG_ARCH_SUN8IW10)
|
|
#ifdef CONFIG_EINK_PANEL_USED
|
|
if (dramfreq->key_masters[MASTER_EINK0] == 0
|
|
&& dramfreq->key_masters[MASTER_EDMA] == 0
|
|
&& dramfreq->key_masters[MASTER_EINK1] == 0) {
|
|
*freq = SUNXI_DRAMFREQ_IDLE;
|
|
} else {
|
|
*freq = df->max_freq;
|
|
}
|
|
#else
|
|
if (dramfreq->key_masters[MASTER_DE] == 0 &&
|
|
dramfreq->key_masters[MASTER_CSI] == 0) {
|
|
*freq = SUNXI_DRAMFREQ_IDLE;
|
|
} else if (dramfreq->key_masters[MASTER_DE] == 1 &&
|
|
dramfreq->key_masters[MASTER_CSI] == 0) {
|
|
*freq = SUNXI_DRAMFREQ_NORMAL;
|
|
} else {
|
|
*freq = df->max_freq;
|
|
}
|
|
#endif
|
|
#endif
|
|
} else {
|
|
*freq = df->max_freq;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int adaptive_init(struct devfreq *devfreq)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (!devfreq)
|
|
return -EINVAL;
|
|
|
|
ret = sysfs_create_group(&devfreq->dev.kobj, &dev_attr_group);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (!dramfreq) {
|
|
pr_err("%s: paras error\n", __func__);
|
|
return -EINVAL;
|
|
}
|
|
|
|
dramfreq->governor_state_update(devfreq->governor_name, STATE_INIT);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void adaptive_exit(struct devfreq *devfreq)
|
|
{
|
|
if (!devfreq)
|
|
return;
|
|
|
|
sysfs_remove_group(&devfreq->dev.kobj, &dev_attr_group);
|
|
|
|
if (!dramfreq) {
|
|
pr_err("%s: paras error\n", __func__);
|
|
return;
|
|
}
|
|
|
|
dramfreq->governor_state_update(devfreq->governor_name, STATE_EXIT);
|
|
}
|
|
|
|
static int devfreq_adaptive_handler(struct devfreq *devfreq,
|
|
unsigned int event, void *data)
|
|
{
|
|
int ret = 0;
|
|
|
|
switch (event) {
|
|
case DEVFREQ_GOV_START:
|
|
ret = adaptive_init(devfreq);
|
|
break;
|
|
case DEVFREQ_GOV_STOP:
|
|
adaptive_exit(devfreq);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static struct devfreq_governor devfreq_adaptive = {
|
|
.name = "adaptive",
|
|
.get_target_freq = devfreq_adaptive_func,
|
|
.event_handler = devfreq_adaptive_handler,
|
|
};
|
|
|
|
static int __init devfreq_adaptive_init(void)
|
|
{
|
|
return devfreq_add_governor(&devfreq_adaptive);
|
|
}
|
|
late_initcall(devfreq_adaptive_init);
|
|
|
|
static void __exit devfreq_adaptive_exit(void)
|
|
{
|
|
if (devfreq_remove_governor(&devfreq_adaptive))
|
|
pr_err("%s: failed remove governor\n", __func__);
|
|
}
|
|
module_exit(devfreq_adaptive_exit);
|
|
MODULE_LICENSE("GPL");
|