606 lines
14 KiB
C
606 lines
14 KiB
C
/*
|
|
* drivers/cpufreq/sunxi-cpufreq.c
|
|
*
|
|
* Copyright (c) 2014 softwinner.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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.
|
|
*
|
|
*/
|
|
|
|
#include <linux/init.h>
|
|
#include <linux/module.h>
|
|
#include <linux/cpufreq.h>
|
|
#include <linux/cpu.h>
|
|
#include <linux/io.h>
|
|
#include <linux/of.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/err.h>
|
|
#include <linux/regulator/consumer.h>
|
|
#include <linux/pm_opp.h>
|
|
#include <linux/arisc/arisc.h>
|
|
#include <linux/sunxi-sid.h>
|
|
#include <asm/cacheflush.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/sunxi-cpufreq.h>
|
|
#include <linux/delay.h>
|
|
#ifdef CONFIG_SUNXI_ARISC
|
|
#include <linux/arisc/arisc.h>
|
|
#endif
|
|
|
|
#define CPUFREQ_DBG(format, args...) \
|
|
pr_debug("[cpu_freq] DBG: "format, ##args)
|
|
#define CPUFREQ_ERR(format, args...) \
|
|
pr_err("[cpu_freq] ERR: "format, ##args)
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
/* sunxi CPUFreq driver data structure */
|
|
static struct {
|
|
s64 cpufreq_set_us;
|
|
s64 cpufreq_get_us;
|
|
} sunxi_cpufreq;
|
|
#endif
|
|
|
|
/* cpufreq_dvfs_table is global default dvfs table */
|
|
static struct cpufreq_dvfs_table cpufreq_dvfs_table[DVFS_VF_TABLE_MAX] = {
|
|
/*
|
|
* cluster0
|
|
* cpu0 vdd is 1.20v if cpu freq is (600Mhz, 1008Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (420Mhz, 600Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (360Mhz, 420Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (300Mhz, 360Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (240Mhz, 300Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (120Mhz, 240Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (60Mhz, 120Mhz]
|
|
* cpu0 vdd is 1.20v if cpu freq is (0Mhz, 60Mhz]
|
|
*/
|
|
/* freq voltage axi_div */
|
|
{900000000, 1200, 3},
|
|
{600000000, 1200, 3},
|
|
{420000000, 1200, 3},
|
|
{360000000, 1200, 3},
|
|
{300000000, 1200, 3},
|
|
{240000000, 1200, 3},
|
|
{120000000, 1200, 3},
|
|
{60000000, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3},
|
|
{0, 1200, 3}
|
|
};
|
|
|
|
#ifdef CONFIG_ARM_SUNXI_PSENSOR_BIN
|
|
struct psensor_range {
|
|
unsigned int min;
|
|
unsigned int max;
|
|
};
|
|
|
|
static struct psensor_range psensor_range[8];
|
|
static int psensor_range_count;
|
|
#define PSENSOR_REG 0x0300621c
|
|
#endif
|
|
|
|
extern unsigned long dev_pm_opp_axi_bus_divide_ratio(struct dev_pm_opp *opp);
|
|
extern int dev_pm_opp_of_get_sharing_cpus_by_soc_bin(struct device *cpu_dev,
|
|
cpumask_var_t cpumask, int soc_bin);
|
|
extern int dev_pm_opp_of_cpumask_add_table_by_soc_bin(cpumask_var_t cpumask,
|
|
int soc_bin);
|
|
|
|
|
|
static unsigned int sunxi_cpufreq_get(unsigned int cpu)
|
|
{
|
|
unsigned int current_freq = 0;
|
|
#ifdef CONFIG_DEBUG_FS
|
|
ktime_t calltime = ktime_get();
|
|
#endif
|
|
|
|
current_freq = cpufreq_generic_get(cpu);
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
sunxi_cpufreq.cpufreq_get_us =
|
|
ktime_to_us(ktime_sub(ktime_get(), calltime));
|
|
#endif
|
|
|
|
return current_freq;
|
|
}
|
|
|
|
#ifdef CONFIG_SUNXI_ARISC
|
|
static int sunxi_set_cpufreq_and_voltage(struct cpufreq_policy *policy,
|
|
unsigned long freq)
|
|
{
|
|
int ret = 0;
|
|
#ifdef CONFIG_SUNXI_CPUFREQ_ASYN
|
|
unsigned long timeout;
|
|
|
|
arisc_dvfs_set_cpufreq(freq, ARISC_DVFS_PLL1, ARISC_DVFS_ASYN,
|
|
NULL, NULL);
|
|
/* CPUS max latency for cpu freq*/
|
|
timeout = 15;
|
|
while (timeout-- && (clk_get_rate(policy->clk) != freq*1000))
|
|
msleep(1);
|
|
if (clk_get_rate(policy->clk) != freq*1000)
|
|
ret = -1;
|
|
#else
|
|
|
|
ret = arisc_dvfs_set_cpufreq(freq, ARISC_DVFS_PLL1, ARISC_DVFS_SYN,
|
|
NULL, NULL);
|
|
#endif
|
|
return ret;
|
|
}
|
|
#else
|
|
static int sunxi_set_cpufreq_and_voltage(struct cpufreq_policy *policy,
|
|
unsigned long freq)
|
|
{
|
|
struct device *cpu_dev;
|
|
|
|
cpu_dev = get_cpu_device(policy->cpu);
|
|
|
|
|
|
return dev_pm_opp_set_rate(cpu_dev, freq);
|
|
}
|
|
#endif
|
|
|
|
static int sunxi_cpufreq_target_index(struct cpufreq_policy *policy,
|
|
unsigned int index)
|
|
{
|
|
int ret = 0;
|
|
unsigned long freq;
|
|
#ifdef CONFIG_DEBUG_FS
|
|
ktime_t calltime;
|
|
#endif
|
|
|
|
freq = policy->freq_table[index].frequency;
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
calltime = ktime_get();
|
|
#endif
|
|
/* try to set cpu frequency */
|
|
ret = sunxi_set_cpufreq_and_voltage(policy, freq);
|
|
if (ret)
|
|
CPUFREQ_ERR("Set cpu frequency to %luKHz failed!\n", freq);
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
sunxi_cpufreq.cpufreq_set_us = ktime_to_us(ktime_sub(ktime_get(),
|
|
calltime));
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
static int sunxi_cpufreq_set_vf(struct cpufreq_frequency_table *table,
|
|
unsigned int cpu)
|
|
{
|
|
struct cpufreq_frequency_table *pos;
|
|
struct dev_pm_opp *opp;
|
|
struct device *dev;
|
|
unsigned long freq;
|
|
int ret, num = 0;
|
|
int max_opp_num = 0;
|
|
void *kvir = NULL;
|
|
|
|
dev = get_cpu_device(cpu);
|
|
max_opp_num = dev_pm_opp_get_opp_count(dev);
|
|
|
|
#ifndef CONFIG_ARCH_SUN8IW12P1
|
|
for (pos = table; max_opp_num > 0; --max_opp_num) {
|
|
freq = pos[max_opp_num - 1].frequency * 1000;
|
|
#else
|
|
cpufreq_for_each_valid_entry(pos, table) {
|
|
freq = pos->frequency * 1000;
|
|
#endif
|
|
CPUFREQ_DBG("freq: %lu\n", freq);
|
|
rcu_read_lock();
|
|
opp = dev_pm_opp_find_freq_ceil(dev, &freq);
|
|
if (IS_ERR(opp)) {
|
|
ret = PTR_ERR(opp);
|
|
dev_err(dev,
|
|
"%s: failed to find OPP for freq %lu (%d)\n",
|
|
__func__, freq, ret);
|
|
rcu_read_unlock();
|
|
return ret;
|
|
}
|
|
|
|
cpufreq_dvfs_table[num].voltage =
|
|
dev_pm_opp_get_voltage(opp) / 1000;
|
|
cpufreq_dvfs_table[num].freq = dev_pm_opp_get_freq(opp);
|
|
cpufreq_dvfs_table[num].axi_div =
|
|
dev_pm_opp_axi_bus_divide_ratio(opp);
|
|
rcu_read_unlock();
|
|
CPUFREQ_DBG("num:%d, volatge:%d, freq:%d, axi_div:%d ,%s\n",
|
|
num, cpufreq_dvfs_table[num].voltage,
|
|
cpufreq_dvfs_table[num].freq,
|
|
cpufreq_dvfs_table[num].axi_div, __func__);
|
|
num++;
|
|
}
|
|
|
|
kvir =
|
|
kmalloc(num * sizeof(struct cpufreq_frequency_table), GFP_KERNEL);
|
|
if (kvir == NULL) {
|
|
CPUFREQ_ERR("kmalloc error for transmiting vf table\n");
|
|
return -1;
|
|
}
|
|
memcpy((void *)kvir, (void *)cpufreq_dvfs_table,
|
|
num * sizeof(struct cpufreq_frequency_table));
|
|
__dma_flush_area((void *)kvir, num * sizeof(struct cpufreq_frequency_table));
|
|
|
|
arisc_dvfs_cfg_vf_table(0, num, virt_to_phys(kvir));
|
|
|
|
kfree(kvir);
|
|
kvir = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_ARM_SUNXI_PSENSOR_BIN
|
|
static int get_psensor_range(void)
|
|
{
|
|
int i = 0;
|
|
char name[20];
|
|
int psensor_count = 0;
|
|
struct device_node *psensor_node = NULL;
|
|
|
|
psensor_node = of_find_node_by_path("/psensor_table");
|
|
if (psensor_node == NULL) {
|
|
pr_err("%s(%d) has no pensor node\n", __func__, __LINE__);
|
|
return -1;
|
|
}
|
|
|
|
if (of_property_read_u32(psensor_node, "psensor_count",
|
|
&psensor_count)) {
|
|
pr_err("%s(%d) get psensor count error\n", __func__, __LINE__);
|
|
return -1;
|
|
}
|
|
psensor_range_count = psensor_count;
|
|
for (i = 0; i < psensor_count; i++) {
|
|
sprintf(name, "prange_min_%d", i);
|
|
if (of_property_read_u32(psensor_node, name,
|
|
&psensor_range[i].min)) {
|
|
pr_err("get prange_min_%d failed\n", i);
|
|
return -1;
|
|
}
|
|
|
|
sprintf(name, "prange_max_%d", i);
|
|
if (of_property_read_u32(psensor_node, name,
|
|
&psensor_range[i].max)) {
|
|
pr_err("get prange_max_%d failed\n", i);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < psensor_count; ++i) {
|
|
pr_debug("psensor_range_%d.min=%d; psensor_range_%d.max=%d\n",
|
|
i, psensor_range[i].min, i, psensor_range[i].max);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int get_psensor_bin(void)
|
|
{
|
|
unsigned int soc_bin;
|
|
void __iomem *bin_reg = NULL;
|
|
int i;
|
|
|
|
bin_reg = ioremap(PSENSOR_REG, 16);
|
|
|
|
soc_bin = readl(bin_reg);
|
|
|
|
/*use the high 16 bit*/
|
|
soc_bin >>= 16;
|
|
|
|
iounmap(bin_reg);
|
|
|
|
for (i = 0; i < psensor_range_count; ++i) {
|
|
if (soc_bin >= psensor_range[i].min && soc_bin <=
|
|
psensor_range[i].max)
|
|
break;
|
|
}
|
|
|
|
if (i >= psensor_range_count)
|
|
return -1;
|
|
|
|
return i;
|
|
}
|
|
#endif
|
|
|
|
static int sunxi_cpufreq_init(struct cpufreq_policy *policy)
|
|
{
|
|
struct device *cpu_dev;
|
|
struct cpufreq_frequency_table *freq_table;
|
|
struct dev_pm_opp *suspend_opp;
|
|
struct clk *cpu_clk;
|
|
const char *regulator_name;
|
|
struct opp_table *table;
|
|
unsigned int transition_latency;
|
|
int ret, soc_bin;
|
|
unsigned int table_count;
|
|
struct device_node *dvfs_main_np;
|
|
|
|
dvfs_main_np = of_find_node_by_path("/opp_dvfs_table");
|
|
if (!dvfs_main_np) {
|
|
CPUFREQ_ERR("No opp dvfs table node found\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
#ifdef CONFIG_ARM_SUNXI_PSENSOR_BIN
|
|
ret = get_psensor_range();
|
|
if (ret < 0)
|
|
return -EINVAL;
|
|
#endif
|
|
if (of_property_read_u32(dvfs_main_np, "opp_table_count",
|
|
&table_count)) {
|
|
CPUFREQ_ERR("get vf_table_count failed\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (table_count == 1) {
|
|
pr_info("%s: only one opp_table\n", __func__);
|
|
soc_bin = 0;
|
|
} else {
|
|
#ifdef CONFIG_ARM_SUNXI_PSENSOR_BIN
|
|
soc_bin = get_psensor_bin();
|
|
#else
|
|
soc_bin = sunxi_get_soc_bin();
|
|
#endif
|
|
if (soc_bin < 0) {
|
|
pr_err("%s: get the wrong soc bin!\n", __func__);
|
|
return -EINVAL;
|
|
}
|
|
pr_info("%s: support more opp_table and soc bin is %d\n",
|
|
__func__, soc_bin);
|
|
}
|
|
|
|
cpu_dev = get_cpu_device(policy->cpu);
|
|
if (!cpu_dev) {
|
|
CPUFREQ_ERR("Failed to get cpu%d device\n", policy->cpu);
|
|
return -ENODEV;
|
|
}
|
|
|
|
cpu_clk = clk_get(cpu_dev, NULL);
|
|
if (IS_ERR_OR_NULL(cpu_clk)) {
|
|
ret = PTR_ERR(cpu_clk);
|
|
CPUFREQ_ERR("Unable to get PLL CPU clock\n");
|
|
return ret;
|
|
}
|
|
policy->clk = cpu_clk;
|
|
|
|
regulator_name = of_get_property(cpu_dev->of_node, "regulators", NULL);
|
|
if (!regulator_name) {
|
|
CPUFREQ_ERR("Unable to get regulator\n");
|
|
goto lable_1;
|
|
}
|
|
|
|
table = dev_pm_opp_set_regulator(cpu_dev, regulator_name);
|
|
if (!table) {
|
|
CPUFREQ_ERR("Failed to set regulator for cpu\n");
|
|
goto out_err_clk_pll;
|
|
}
|
|
|
|
lable_1:
|
|
/* set policy->cpus according to operating-points-v2 */
|
|
ret = dev_pm_opp_of_get_sharing_cpus_by_soc_bin(cpu_dev,
|
|
policy->cpus, soc_bin);
|
|
if (ret) {
|
|
CPUFREQ_ERR("OPP-v2 opp-shared Error\n");
|
|
goto out_err_clk_pll;
|
|
}
|
|
|
|
ret = dev_pm_opp_of_cpumask_add_table_by_soc_bin(policy->cpus,
|
|
soc_bin);
|
|
if (ret) {
|
|
CPUFREQ_ERR("Failed to add opp table\n");
|
|
goto out_err_clk_pll;
|
|
}
|
|
|
|
ret = dev_pm_opp_init_cpufreq_table(cpu_dev, &freq_table);
|
|
if (ret) {
|
|
CPUFREQ_ERR("Failed to init cpufreq table: %d\n", ret);
|
|
goto out_err_free_opp;
|
|
}
|
|
|
|
ret = cpufreq_table_validate_and_show(policy, freq_table);
|
|
if (ret) {
|
|
CPUFREQ_ERR("Invalid frequency table: %d\n", ret);
|
|
goto out_err_free_opp;
|
|
}
|
|
|
|
transition_latency = dev_pm_opp_get_max_transition_latency(cpu_dev);
|
|
if (!transition_latency)
|
|
transition_latency = CPUFREQ_ETERNAL;
|
|
policy->cpuinfo.transition_latency = transition_latency;
|
|
|
|
rcu_read_lock();
|
|
suspend_opp = dev_pm_opp_get_suspend_opp(cpu_dev);
|
|
if (suspend_opp)
|
|
policy->suspend_freq = dev_pm_opp_get_freq(suspend_opp) / 1000;
|
|
rcu_read_unlock();
|
|
|
|
ret = sunxi_cpufreq_set_vf(freq_table, policy->cpu);
|
|
if (ret) {
|
|
CPUFREQ_ERR("sunxi_cpufreq_set_vf failed: %d\n", ret);
|
|
goto out_err_free_opp;
|
|
}
|
|
|
|
return 0;
|
|
|
|
out_err_free_opp:
|
|
dev_pm_opp_of_cpumask_remove_table(policy->cpus);
|
|
out_err_clk_pll:
|
|
clk_put(cpu_clk);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int sunxi_cpufreq_exit(struct cpufreq_policy *policy)
|
|
{
|
|
struct device *cpu_dev;
|
|
|
|
cpu_dev = get_cpu_device(policy->cpu);
|
|
|
|
dev_pm_opp_free_cpufreq_table(cpu_dev, &policy->freq_table);
|
|
|
|
dev_pm_opp_of_cpumask_remove_table(policy->related_cpus);
|
|
|
|
//dev_pm_opp_put_regulator(cpu_dev);
|
|
|
|
clk_put(policy->clk);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct freq_attr *sunxi_cpufreq_attr[] = {
|
|
&cpufreq_freq_attr_scaling_available_freqs,
|
|
NULL,
|
|
};
|
|
|
|
static struct cpufreq_driver sunxi_cpufreq_driver = {
|
|
.name = "cpufreq-sunxi",
|
|
.flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK,
|
|
.attr = sunxi_cpufreq_attr,
|
|
.init = sunxi_cpufreq_init,
|
|
.get = sunxi_cpufreq_get,
|
|
.target_index = sunxi_cpufreq_target_index,
|
|
.exit = sunxi_cpufreq_exit,
|
|
.verify = cpufreq_generic_frequency_table_verify,
|
|
.suspend = cpufreq_generic_suspend,
|
|
};
|
|
|
|
#ifdef CONFIG_ARCH_SUN8IW7P1
|
|
static int set_pll_cpu_lock_time(void)
|
|
{
|
|
unsigned int value;
|
|
void __iomem *lock_time_vbase = NULL;
|
|
#define PLL_CPU_LOCK_TIME_REG (0x01c20000 + 0x204)
|
|
|
|
lock_time_vbase = ioremap(PLL_CPU_LOCK_TIME_REG, 4);
|
|
if (lock_time_vbase == NULL) {
|
|
pr_err("ioremap pll cpu lock time error\n");
|
|
return -1;
|
|
}
|
|
|
|
value = readl(lock_time_vbase);
|
|
value &= ~(0xffff);
|
|
value |= 0x400;
|
|
writel(value, lock_time_vbase);
|
|
|
|
iounmap(lock_time_vbase);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static int __init sunxi_cpufreq_initcall(void)
|
|
{
|
|
int ret;
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
sunxi_cpufreq.cpufreq_set_us = 0;
|
|
sunxi_cpufreq.cpufreq_get_us = 0;
|
|
#endif
|
|
|
|
#ifdef CONFIG_ARCH_SUN8IW7P1
|
|
if (set_pll_cpu_lock_time())
|
|
return -1;
|
|
#endif
|
|
ret = cpufreq_register_driver(&sunxi_cpufreq_driver);
|
|
if (ret)
|
|
CPUFREQ_ERR("Failed register driver\n");
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void __exit sunxi_cpufreq_exitcall(void)
|
|
{
|
|
cpufreq_unregister_driver(&sunxi_cpufreq_driver);
|
|
}
|
|
|
|
module_init(sunxi_cpufreq_initcall);
|
|
module_exit(sunxi_cpufreq_exitcall);
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
#include <linux/debugfs.h>
|
|
|
|
static struct dentry *debugfs_cpufreq_root;
|
|
|
|
static int cpufreq_debugfs_gettime_show(struct seq_file *s, void *data)
|
|
{
|
|
seq_printf(s, "%lld\n", sunxi_cpufreq.cpufreq_get_us);
|
|
return 0;
|
|
}
|
|
|
|
static int cpufreq_debugfs_gettime_open(struct inode *inode, struct file *file)
|
|
{
|
|
return single_open(file, cpufreq_debugfs_gettime_show,
|
|
inode->i_private);
|
|
}
|
|
|
|
static const struct file_operations cpufreq_debugfs_gettime_fops = {
|
|
.open = cpufreq_debugfs_gettime_open,
|
|
.read = seq_read,
|
|
};
|
|
|
|
static int cpufreq_debugfs_settime_show(struct seq_file *s, void *data)
|
|
{
|
|
seq_printf(s, "%lld\n", sunxi_cpufreq.cpufreq_set_us);
|
|
return 0;
|
|
}
|
|
|
|
static int cpufreq_debugfs_settime_open(struct inode *inode, struct file *file)
|
|
{
|
|
return single_open(file, cpufreq_debugfs_settime_show,
|
|
inode->i_private);
|
|
}
|
|
|
|
static const struct file_operations cpufreq_debugfs_settime_fops = {
|
|
.open = cpufreq_debugfs_settime_open,
|
|
.read = seq_read,
|
|
};
|
|
|
|
static int __init cpufreq_debugfs_init(void)
|
|
{
|
|
int err = 0;
|
|
|
|
debugfs_cpufreq_root = debugfs_create_dir("cpufreq", 0);
|
|
if (!debugfs_cpufreq_root)
|
|
return -ENOMEM;
|
|
|
|
if (!debugfs_create_file("get_time", 0444, debugfs_cpufreq_root, NULL,
|
|
&cpufreq_debugfs_gettime_fops)) {
|
|
err = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
if (!debugfs_create_file("set_time", 0444, debugfs_cpufreq_root, NULL,
|
|
&cpufreq_debugfs_settime_fops)) {
|
|
err = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
return 0;
|
|
|
|
out:
|
|
debugfs_remove_recursive(debugfs_cpufreq_root);
|
|
return err;
|
|
}
|
|
|
|
static void __exit cpufreq_debugfs_exit(void)
|
|
{
|
|
debugfs_remove_recursive(debugfs_cpufreq_root);
|
|
}
|
|
|
|
late_initcall(cpufreq_debugfs_init);
|
|
module_exit(cpufreq_debugfs_exit);
|
|
|
|
#endif /* CONFIG_DEBUG_FS */
|
|
|
|
MODULE_DESCRIPTION("cpufreq driver for sunxi SOCs");
|
|
MODULE_LICENSE("GPL");
|