config ARM_BIG_LITTLE_CPUFREQ
tristate "Generic ARM big LITTLE CPUfreq driver"
depends on ((ARM && ARM_CPU_TOPOLOGY) || (ARM64 && SMP)) && BIG_LITTLE && HAVE_CLK
+ # if CPU_THERMAL is on and THERMAL=m, ARM_BIG_LITTLE_CPUFREQ cannot be =y:
+ depends on !CPU_THERMAL || THERMAL
select PM_OPP
help
This enables the Generic CPUfreq driver for ARM big.LITTLE platforms.
#include <linux/slab.h>
#include <linux/topology.h>
#include <linux/types.h>
+#include <linux/cpu_cooling.h>
#include <asm/bL_switcher.h>
#include "arm_big_little.h"
{
u32 cur_cluster = cpu_to_cluster(policy->cpu);
struct device *cpu_dev;
+ struct private_data *priv;
int ret;
cpu_dev = get_cpu_device(policy->cpu);
return -ENODEV;
}
+ priv = devm_kzalloc(cpu_dev, sizeof(*priv), GFP_KERNEL);
+ if (!priv)
+ return -ENOMEM;
+ priv->cpu_dev = cpu_dev;
+ policy->driver_data = priv;
+
ret = get_cluster_clk_and_freq_table(cpu_dev);
if (ret)
return ret;
return 0;
}
+static void bL_cpufreq_ready(struct cpufreq_policy *policy)
+{
+ struct private_data *priv = policy->driver_data;
+ struct device_node *np = of_node_get(priv->cpu_dev->of_node);
+
+ if (of_find_property(np, "#cooling-cells", NULL)) {
+ priv->cdev = of_cpufreq_cooling_register(np,
+ policy->related_cpus);
+ if (IS_ERR(priv->cdev)) {
+ dev_err(priv->cpu_dev,
+ "running cpufreq without cooling device: %ld\n",
+ PTR_ERR(priv->cdev));
+
+ priv->cdev = NULL;
+ }
+ }
+ of_node_put(np);
+}
+
static struct cpufreq_driver bL_cpufreq_driver = {
.name = "arm-big-little",
.flags = CPUFREQ_STICKY |
.get = bL_cpufreq_get_rate,
.init = bL_cpufreq_init,
.exit = bL_cpufreq_exit,
+ .ready = bL_cpufreq_ready,
.attr = cpufreq_generic_attr,
};
#include <linux/cpufreq.h>
#include <linux/device.h>
#include <linux/types.h>
+#include <linux/thermal.h>
+
+struct private_data {
+ struct device *cpu_dev;
+ struct thermal_cooling_device *cdev;
+};
struct cpufreq_arm_bL_ops {
char name[CPUFREQ_NAME_LEN];