/* * linux/drivers/devfreq/governor_adaptive.c * * Copyright(c) 2013-2015 Allwinnertech Co., Ltd. * * Author: Pan Nan * * 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 #include #include #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");