/*
* Copyright (C) 2012 Texas Instruments, Inc
*
* 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.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along with
* this program. If not, see .
*/
#include
#include "sgxfreq.h"
static int activeidle_start(struct sgxfreq_sgx_data *data);
static void activeidle_stop(void);
static void activeidle_sgx_active(void);
static void activeidle_sgx_idle(void);
static struct activeidle_data {
unsigned long freq_active;
unsigned long freq_idle;
struct mutex mutex;
bool sgx_active;
} aid;
static struct sgxfreq_governor activeidle_gov = {
.name = "activeidle",
.gov_start = activeidle_start,
.gov_stop = activeidle_stop,
.sgx_active = activeidle_sgx_active,
.sgx_idle = activeidle_sgx_idle,
};
/*********************** begin sysfs interface ***********************/
extern struct kobject *sgxfreq_kobj;
static ssize_t show_freq_active(struct device *dev,
struct device_attribute *attr,
char *buf)
{
return sprintf(buf, "%lu\n", aid.freq_active);
}
static ssize_t store_freq_active(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
int ret;
unsigned long freq;
ret = sscanf(buf, "%lu", &freq);
if (ret != 1)
return -EINVAL;
freq = sgxfreq_get_freq_ceil(freq);
mutex_lock(&aid.mutex);
aid.freq_active = freq;
if (aid.sgx_active)
sgxfreq_set_freq_request(aid.freq_active);
mutex_unlock(&aid.mutex);
return count;
}
static ssize_t show_freq_idle(struct device *dev, struct device_attribute *attr,
char *buf)
{
return sprintf(buf, "%lu\n", aid.freq_idle);
}
static ssize_t store_freq_idle(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
int ret;
unsigned long freq;
ret = sscanf(buf, "%lu", &freq);
if (ret != 1)
return -EINVAL;
freq = sgxfreq_get_freq_floor(freq);
mutex_lock(&aid.mutex);
aid.freq_idle = freq;
if (!aid.sgx_active)
sgxfreq_set_freq_request(aid.freq_idle);
mutex_unlock(&aid.mutex);
return count;
}
static DEVICE_ATTR(freq_active, 0644, show_freq_active, store_freq_active);
static DEVICE_ATTR(freq_idle, 0644, show_freq_idle, store_freq_idle);
static struct attribute *activeidle_attributes[] = {
&dev_attr_freq_active.attr,
&dev_attr_freq_idle.attr,
NULL
};
static struct attribute_group activeidle_attr_group = {
.attrs = activeidle_attributes,
.name = "activeidle",
};
/************************ end sysfs interface ************************/
int activeidle_init(void)
{
int ret;
mutex_init(&aid.mutex);
ret = sgxfreq_register_governor(&activeidle_gov);
if (ret)
return ret;
aid.freq_idle = sgxfreq_get_freq_min();
aid.freq_active = sgxfreq_get_freq_max();
return 0;
}
int activeidle_deinit(void)
{
return 0;
}
static int activeidle_start(struct sgxfreq_sgx_data *data)
{
int ret;
aid.sgx_active = data->active;
ret = sysfs_create_group(sgxfreq_kobj, &activeidle_attr_group);
if (ret)
return ret;
if (aid.sgx_active)
sgxfreq_set_freq_request(aid.freq_active);
else
sgxfreq_set_freq_request(aid.freq_idle);
return 0;
}
static void activeidle_stop(void)
{
sysfs_remove_group(sgxfreq_kobj, &activeidle_attr_group);
}
static void activeidle_sgx_active(void)
{
mutex_lock(&aid.mutex);
aid.sgx_active = true;
sgxfreq_set_freq_request(aid.freq_active);
mutex_unlock(&aid.mutex);
}
static void activeidle_sgx_idle(void)
{
mutex_lock(&aid.mutex);
aid.sgx_active = false;
sgxfreq_set_freq_request(aid.freq_idle);
mutex_unlock(&aid.mutex);
}