/*
* 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 onoff_start(struct sgxfreq_sgx_data *data);
static void onoff_stop(void);
static void onoff_sgx_clk_on(void);
static void onoff_sgx_clk_off(void);
static struct onoff_data {
unsigned long freq_off;
unsigned long freq_on;
struct mutex mutex;
bool sgx_clk_on;
} ood;
static struct sgxfreq_governor onoff_gov = {
.name = "onoff",
.gov_start = onoff_start,
.gov_stop = onoff_stop,
.sgx_clk_on = onoff_sgx_clk_on,
.sgx_clk_off = onoff_sgx_clk_off,
};
/*********************** begin sysfs interface ***********************/
extern struct kobject *sgxfreq_kobj;
static ssize_t show_freq_on(struct device *dev, struct device_attribute *attr,
char *buf)
{
return sprintf(buf, "%lu\n", ood.freq_on);
}
static ssize_t store_freq_on(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(&ood.mutex);
ood.freq_on = freq;
if (ood.sgx_clk_on)
sgxfreq_set_freq_request(ood.freq_on);
mutex_unlock(&ood.mutex);
return count;
}
static ssize_t show_freq_off(struct device *dev, struct device_attribute *attr,
char *buf)
{
return sprintf(buf, "%lu\n", ood.freq_off);
}
static ssize_t store_freq_off(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(&ood.mutex);
ood.freq_off = freq;
if (!ood.sgx_clk_on)
sgxfreq_set_freq_request(ood.freq_off);
mutex_unlock(&ood.mutex);
return count;
}
static DEVICE_ATTR(freq_on, 0644, show_freq_on, store_freq_on);
static DEVICE_ATTR(freq_off, 0644, show_freq_off, store_freq_off);
static struct attribute *onoff_attributes[] = {
&dev_attr_freq_on.attr,
&dev_attr_freq_off.attr,
NULL
};
static struct attribute_group onoff_attr_group = {
.attrs = onoff_attributes,
.name = "onoff",
};
/************************ end sysfs interface ************************/
int onoff_init(void)
{
int ret;
mutex_init(&ood.mutex);
ret = sgxfreq_register_governor(&onoff_gov);
if (ret)
return ret;
ood.freq_off = sgxfreq_get_freq_min();
ood.freq_on = sgxfreq_get_freq_max();
return 0;
}
int onoff_deinit(void)
{
return 0;
}
static int onoff_start(struct sgxfreq_sgx_data *data)
{
int ret;
ood.sgx_clk_on = data->clk_on;
ret = sysfs_create_group(sgxfreq_kobj, &onoff_attr_group);
if (ret)
return ret;
if (ood.sgx_clk_on)
sgxfreq_set_freq_request(ood.freq_on);
else
sgxfreq_set_freq_request(ood.freq_off);
return 0;
}
static void onoff_stop(void)
{
sysfs_remove_group(sgxfreq_kobj, &onoff_attr_group);
}
static void onoff_sgx_clk_on(void)
{
mutex_lock(&ood.mutex);
ood.sgx_clk_on = true;
sgxfreq_set_freq_request(ood.freq_on);
mutex_unlock(&ood.mutex);
}
static void onoff_sgx_clk_off(void)
{
mutex_lock(&ood.mutex);
ood.sgx_clk_on = false;
sgxfreq_set_freq_request(ood.freq_off);
mutex_unlock(&ood.mutex);
}