/*
 * 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);
}