cpufreq: cpu0: rename driver and internals to 'cpufreq_dt'
authorViresh Kumar <viresh.kumar@linaro.org>
Tue, 9 Sep 2014 14:28:03 +0000 (19:58 +0530)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Fri, 3 Oct 2014 13:37:54 +0000 (15:37 +0200)
The naming convention of this driver was always under the scanner, people
complained that it should have a more generic name than cpu0, as it manages all
CPUs that are sharing clock lines.

Also, in future it will be modified to support any number of clusters with
separate clock/voltage lines.

Lets rename it to 'cpufreq_dt' from 'cpufreq_cpu0'.

Tested-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Viresh Kumar <viresh.kumar@linaro.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
21 files changed:
Documentation/devicetree/bindings/cpufreq/cpufreq-cpu0.txt [deleted file]
Documentation/devicetree/bindings/cpufreq/cpufreq-dt.txt [new file with mode: 0644]
arch/arm/configs/mvebu_v7_defconfig
arch/arm/mach-imx/imx27-dt.c
arch/arm/mach-imx/mach-imx51.c
arch/arm/mach-mvebu/pmsu.c
arch/arm/mach-omap2/pm.c
arch/arm/mach-shmobile/board-ape6evm-reference.c
arch/arm/mach-shmobile/cpufreq.c
arch/arm/mach-shmobile/setup-sh73a0.c
arch/arm/mach-zynq/common.c
drivers/cpufreq/Kconfig
drivers/cpufreq/Kconfig.arm
drivers/cpufreq/Makefile
drivers/cpufreq/cpufreq-cpu0.c [deleted file]
drivers/cpufreq/cpufreq-dt.c [new file with mode: 0644]
drivers/cpufreq/exynos4210-cpufreq.c
drivers/cpufreq/exynos4x12-cpufreq.c
drivers/cpufreq/exynos5250-cpufreq.c
drivers/cpufreq/highbank-cpufreq.c
drivers/cpufreq/s5pv210-cpufreq.c

diff --git a/Documentation/devicetree/bindings/cpufreq/cpufreq-cpu0.txt b/Documentation/devicetree/bindings/cpufreq/cpufreq-cpu0.txt
deleted file mode 100644 (file)
index 366690c..0000000
+++ /dev/null
@@ -1,64 +0,0 @@
-Generic CPU0 cpufreq driver
-
-It is a generic cpufreq driver for CPU0 frequency management.  It
-supports both uniprocessor (UP) and symmetric multiprocessor (SMP)
-systems which share clock and voltage across all CPUs.
-
-Both required and optional properties listed below must be defined
-under node /cpus/cpu@0.
-
-Required properties:
-- None
-
-Optional properties:
-- operating-points: Refer to Documentation/devicetree/bindings/power/opp.txt for
-  details. OPPs *must* be supplied either via DT, i.e. this property, or
-  populated at runtime.
-- clock-latency: Specify the possible maximum transition latency for clock,
-  in unit of nanoseconds.
-- voltage-tolerance: Specify the CPU voltage tolerance in percentage.
-- #cooling-cells:
-- cooling-min-level:
-- cooling-max-level:
-     Please refer to Documentation/devicetree/bindings/thermal/thermal.txt.
-
-Examples:
-
-cpus {
-       #address-cells = <1>;
-       #size-cells = <0>;
-
-       cpu@0 {
-               compatible = "arm,cortex-a9";
-               reg = <0>;
-               next-level-cache = <&L2>;
-               operating-points = <
-                       /* kHz    uV */
-                       792000  1100000
-                       396000  950000
-                       198000  850000
-               >;
-               clock-latency = <61036>; /* two CLK32 periods */
-               #cooling-cells = <2>;
-               cooling-min-level = <0>;
-               cooling-max-level = <2>;
-       };
-
-       cpu@1 {
-               compatible = "arm,cortex-a9";
-               reg = <1>;
-               next-level-cache = <&L2>;
-       };
-
-       cpu@2 {
-               compatible = "arm,cortex-a9";
-               reg = <2>;
-               next-level-cache = <&L2>;
-       };
-
-       cpu@3 {
-               compatible = "arm,cortex-a9";
-               reg = <3>;
-               next-level-cache = <&L2>;
-       };
-};
diff --git a/Documentation/devicetree/bindings/cpufreq/cpufreq-dt.txt b/Documentation/devicetree/bindings/cpufreq/cpufreq-dt.txt
new file mode 100644 (file)
index 0000000..e41c98f
--- /dev/null
@@ -0,0 +1,64 @@
+Generic cpufreq driver
+
+It is a generic DT based cpufreq driver for frequency management.  It supports
+both uniprocessor (UP) and symmetric multiprocessor (SMP) systems which share
+clock and voltage across all CPUs.
+
+Both required and optional properties listed below must be defined
+under node /cpus/cpu@0.
+
+Required properties:
+- None
+
+Optional properties:
+- operating-points: Refer to Documentation/devicetree/bindings/power/opp.txt for
+  details. OPPs *must* be supplied either via DT, i.e. this property, or
+  populated at runtime.
+- clock-latency: Specify the possible maximum transition latency for clock,
+  in unit of nanoseconds.
+- voltage-tolerance: Specify the CPU voltage tolerance in percentage.
+- #cooling-cells:
+- cooling-min-level:
+- cooling-max-level:
+     Please refer to Documentation/devicetree/bindings/thermal/thermal.txt.
+
+Examples:
+
+cpus {
+       #address-cells = <1>;
+       #size-cells = <0>;
+
+       cpu@0 {
+               compatible = "arm,cortex-a9";
+               reg = <0>;
+               next-level-cache = <&L2>;
+               operating-points = <
+                       /* kHz    uV */
+                       792000  1100000
+                       396000  950000
+                       198000  850000
+               >;
+               clock-latency = <61036>; /* two CLK32 periods */
+               #cooling-cells = <2>;
+               cooling-min-level = <0>;
+               cooling-max-level = <2>;
+       };
+
+       cpu@1 {
+               compatible = "arm,cortex-a9";
+               reg = <1>;
+               next-level-cache = <&L2>;
+       };
+
+       cpu@2 {
+               compatible = "arm,cortex-a9";
+               reg = <2>;
+               next-level-cache = <&L2>;
+       };
+
+       cpu@3 {
+               compatible = "arm,cortex-a9";
+               reg = <3>;
+               next-level-cache = <&L2>;
+       };
+};
index fdfda1fa95212cfb9e594f06ceb49381502db78e..7309988b0f1fe1aa808ba75bdef74531b847b561 100644 (file)
@@ -32,7 +32,7 @@ CONFIG_ARM_ATAG_DTB_COMPAT=y
 CONFIG_CPU_IDLE=y
 CONFIG_ARM_MVEBU_V7_CPUIDLE=y
 CONFIG_CPU_FREQ=y
-CONFIG_CPUFREQ_GENERIC=y
+CONFIG_CPUFREQ_DT=y
 CONFIG_VFP=y
 CONFIG_NET=y
 CONFIG_INET=y
index 080e66c6a1d02722022f12a7991cf0b0b566f72e..dc8f1a6f45f20a9551045105400a30587d00b1a1 100644 (file)
@@ -20,7 +20,7 @@
 
 static void __init imx27_dt_init(void)
 {
-       struct platform_device_info devinfo = { .name = "cpufreq-cpu0", };
+       struct platform_device_info devinfo = { .name = "cpufreq-dt", };
 
        mxc_arch_reset_init_dt();
 
index c77deb3f08939f50304797ea5ac1c1b17983fef7..2c5fcaf8675b96bfdd4ee391871c1602eba5ae08 100644 (file)
@@ -51,7 +51,7 @@ static void __init imx51_ipu_mipi_setup(void)
 
 static void __init imx51_dt_init(void)
 {
-       struct platform_device_info devinfo = { .name = "cpufreq-cpu0", };
+       struct platform_device_info devinfo = { .name = "cpufreq-dt", };
 
        mxc_arch_reset_init_dt();
        imx51_ipu_mipi_setup();
index 8a70a51533fd4507e02f10cb576667500cfbf575..bbd8664d1bacb2732ec58072d630e70d963904de 100644 (file)
@@ -644,7 +644,7 @@ static int __init armada_xp_pmsu_cpufreq_init(void)
                }
        }
 
-       platform_device_register_simple("cpufreq-generic", -1, NULL, 0);
+       platform_device_register_simple("cpufreq-dt", -1, NULL, 0);
        return 0;
 }
 
index 828aee9ea6a8b4146cff3b90ae1c1e909e0bc217..58920bc8807bce963536296bc7914db421c2b6aa 100644 (file)
@@ -282,7 +282,7 @@ static inline void omap_init_cpufreq(void)
        if (!of_have_populated_dt())
                devinfo.name = "omap-cpufreq";
        else
-               devinfo.name = "cpufreq-cpu0";
+               devinfo.name = "cpufreq-dt";
        platform_device_register_full(&devinfo);
 }
 
index 2f7723e5fe91ba1f7adb5e10d38e342277a9fd10..0110751da511c2e33373481af8fbb37105d5f46d 100644 (file)
@@ -50,7 +50,7 @@ static void __init ape6evm_add_standard_devices(void)
 
        r8a73a4_add_dt_devices();
        of_platform_populate(NULL, of_default_bus_match_table, NULL, NULL);
-       platform_device_register_simple("cpufreq-cpu0", -1, NULL, 0);
+       platform_device_register_simple("cpufreq-dt", -1, NULL, 0);
 }
 
 static const char *ape6evm_boards_compat_dt[] __initdata = {
index 8a24b2be46ae34165a7c613b1d483097ce67a861..57fbff024dcd5dd6ccf23afb94e09d6b0ae47796 100644 (file)
@@ -12,6 +12,6 @@
 
 int __init shmobile_cpufreq_init(void)
 {
-       platform_device_register_simple("cpufreq-cpu0", -1, NULL, 0);
+       platform_device_register_simple("cpufreq-dt", -1, NULL, 0);
        return 0;
 }
index 2c802ae9b241332f11d120b2dd188e6c21fc1e92..15b990cd8c70bc571175ed98483923abda858d09 100644 (file)
@@ -775,7 +775,7 @@ void __init sh73a0_add_early_devices(void)
 
 void __init sh73a0_add_standard_devices_dt(void)
 {
-       struct platform_device_info devinfo = { .name = "cpufreq-cpu0", .id = -1, };
+       struct platform_device_info devinfo = { .name = "cpufreq-dt", .id = -1, };
 
        /* clocks are setup late during boot in the case of DT */
        sh73a0_clock_init();
@@ -784,7 +784,7 @@ void __init sh73a0_add_standard_devices_dt(void)
                             ARRAY_SIZE(sh73a0_devices_dt));
        of_platform_populate(NULL, of_default_bus_match_table, NULL, NULL);
 
-       /* Instantiate cpufreq-cpu0 */
+       /* Instantiate cpufreq-dt */
        platform_device_register_full(&devinfo);
 }
 
index 31a6fa40ba37ef0b37da4d9439258ba3e2b8726b..ec03ec40e9c6fea3c107704bb85c5498ca0d8b77 100644 (file)
@@ -104,7 +104,7 @@ static int __init zynq_get_revision(void)
  */
 static void __init zynq_init_machine(void)
 {
-       struct platform_device_info devinfo = { .name = "cpufreq-cpu0", };
+       struct platform_device_info devinfo = { .name = "cpufreq-dt", };
        struct soc_device_attribute *soc_dev_attr;
        struct soc_device *soc_dev;
        struct device *parent = NULL;
index ffe350f86bca570e879177c63395efff5de6587d..3489f8f5fadabee1b8db494c7b5dbd996bd70e33 100644 (file)
@@ -183,14 +183,14 @@ config CPU_FREQ_GOV_CONSERVATIVE
 
          If in doubt, say N.
 
-config GENERIC_CPUFREQ_CPU0
-       tristate "Generic CPU0 cpufreq driver"
+config CPUFREQ_DT
+       tristate "Generic DT based cpufreq driver"
        depends on HAVE_CLK && OF
-       # if CPU_THERMAL is on and THERMAL=m, CPU0 cannot be =y:
+       # if CPU_THERMAL is on and THERMAL=m, CPUFREQ_DT cannot be =y:
        depends on !CPU_THERMAL || THERMAL
        select PM_OPP
        help
-         This adds a generic cpufreq driver for CPU0 frequency management.
+         This adds a generic DT based cpufreq driver for frequency management.
          It supports both uniprocessor (UP) and symmetric multiprocessor (SMP)
          systems which share clock and voltage across all CPUs.
 
index 7364a538e0562a82b855c95305f983d8a63eb2be..48ed28b789f7605e0d9c241fa70fce4f40332a83 100644 (file)
@@ -92,7 +92,7 @@ config ARM_EXYNOS_CPU_FREQ_BOOST_SW
 
 config ARM_HIGHBANK_CPUFREQ
        tristate "Calxeda Highbank-based"
-       depends on ARCH_HIGHBANK && GENERIC_CPUFREQ_CPU0 && REGULATOR
+       depends on ARCH_HIGHBANK && CPUFREQ_DT && REGULATOR
        default m
        help
          This adds the CPUFreq driver for Calxeda Highbank SoC
index db6d9a2fea4d534f135af08880f229d511633c91..40c53dc1937ec6a4fc22e61f497939ba2ada54bd 100644 (file)
@@ -13,7 +13,7 @@ obj-$(CONFIG_CPU_FREQ_GOV_ONDEMAND)   += cpufreq_ondemand.o
 obj-$(CONFIG_CPU_FREQ_GOV_CONSERVATIVE)        += cpufreq_conservative.o
 obj-$(CONFIG_CPU_FREQ_GOV_COMMON)              += cpufreq_governor.o
 
-obj-$(CONFIG_GENERIC_CPUFREQ_CPU0)     += cpufreq-cpu0.o
+obj-$(CONFIG_CPUFREQ_DT)               += cpufreq-dt.o
 
 ##################################################################################
 # x86 drivers.
diff --git a/drivers/cpufreq/cpufreq-cpu0.c b/drivers/cpufreq/cpufreq-cpu0.c
deleted file mode 100644 (file)
index a5f8c5f..0000000
+++ /dev/null
@@ -1,363 +0,0 @@
-/*
- * Copyright (C) 2012 Freescale Semiconductor, Inc.
- *
- * Copyright (C) 2014 Linaro.
- * Viresh Kumar <viresh.kumar@linaro.org>
- *
- * The OPP code in function cpu0_set_target() is reused from
- * drivers/cpufreq/omap-cpufreq.c
- *
- * 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.
- */
-
-#define pr_fmt(fmt)    KBUILD_MODNAME ": " fmt
-
-#include <linux/clk.h>
-#include <linux/cpu.h>
-#include <linux/cpu_cooling.h>
-#include <linux/cpufreq.h>
-#include <linux/cpumask.h>
-#include <linux/err.h>
-#include <linux/module.h>
-#include <linux/of.h>
-#include <linux/pm_opp.h>
-#include <linux/platform_device.h>
-#include <linux/regulator/consumer.h>
-#include <linux/slab.h>
-#include <linux/thermal.h>
-
-struct private_data {
-       struct device *cpu_dev;
-       struct regulator *cpu_reg;
-       struct thermal_cooling_device *cdev;
-       unsigned int voltage_tolerance; /* in percentage */
-};
-
-static int cpu0_set_target(struct cpufreq_policy *policy, unsigned int index)
-{
-       struct dev_pm_opp *opp;
-       struct cpufreq_frequency_table *freq_table = policy->freq_table;
-       struct clk *cpu_clk = policy->clk;
-       struct private_data *priv = policy->driver_data;
-       struct device *cpu_dev = priv->cpu_dev;
-       struct regulator *cpu_reg = priv->cpu_reg;
-       unsigned long volt = 0, volt_old = 0, tol = 0;
-       unsigned int old_freq, new_freq;
-       long freq_Hz, freq_exact;
-       int ret;
-
-       freq_Hz = clk_round_rate(cpu_clk, freq_table[index].frequency * 1000);
-       if (freq_Hz <= 0)
-               freq_Hz = freq_table[index].frequency * 1000;
-
-       freq_exact = freq_Hz;
-       new_freq = freq_Hz / 1000;
-       old_freq = clk_get_rate(cpu_clk) / 1000;
-
-       if (!IS_ERR(cpu_reg)) {
-               rcu_read_lock();
-               opp = dev_pm_opp_find_freq_ceil(cpu_dev, &freq_Hz);
-               if (IS_ERR(opp)) {
-                       rcu_read_unlock();
-                       dev_err(cpu_dev, "failed to find OPP for %ld\n",
-                               freq_Hz);
-                       return PTR_ERR(opp);
-               }
-               volt = dev_pm_opp_get_voltage(opp);
-               rcu_read_unlock();
-               tol = volt * priv->voltage_tolerance / 100;
-               volt_old = regulator_get_voltage(cpu_reg);
-       }
-
-       dev_dbg(cpu_dev, "%u MHz, %ld mV --> %u MHz, %ld mV\n",
-               old_freq / 1000, volt_old ? volt_old / 1000 : -1,
-               new_freq / 1000, volt ? volt / 1000 : -1);
-
-       /* scaling up?  scale voltage before frequency */
-       if (!IS_ERR(cpu_reg) && new_freq > old_freq) {
-               ret = regulator_set_voltage_tol(cpu_reg, volt, tol);
-               if (ret) {
-                       dev_err(cpu_dev, "failed to scale voltage up: %d\n",
-                               ret);
-                       return ret;
-               }
-       }
-
-       ret = clk_set_rate(cpu_clk, freq_exact);
-       if (ret) {
-               dev_err(cpu_dev, "failed to set clock rate: %d\n", ret);
-               if (!IS_ERR(cpu_reg))
-                       regulator_set_voltage_tol(cpu_reg, volt_old, tol);
-               return ret;
-       }
-
-       /* scaling down?  scale voltage after frequency */
-       if (!IS_ERR(cpu_reg) && new_freq < old_freq) {
-               ret = regulator_set_voltage_tol(cpu_reg, volt, tol);
-               if (ret) {
-                       dev_err(cpu_dev, "failed to scale voltage down: %d\n",
-                               ret);
-                       clk_set_rate(cpu_clk, old_freq * 1000);
-               }
-       }
-
-       return ret;
-}
-
-static int allocate_resources(int cpu, struct device **cdev,
-                             struct regulator **creg, struct clk **cclk)
-{
-       struct device *cpu_dev;
-       struct regulator *cpu_reg;
-       struct clk *cpu_clk;
-       int ret = 0;
-       char *reg_cpu0 = "cpu0", *reg_cpu = "cpu", *reg;
-
-       cpu_dev = get_cpu_device(cpu);
-       if (!cpu_dev) {
-               pr_err("failed to get cpu%d device\n", cpu);
-               return -ENODEV;
-       }
-
-       /* Try "cpu0" for older DTs */
-       if (!cpu)
-               reg = reg_cpu0;
-       else
-               reg = reg_cpu;
-
-try_again:
-       cpu_reg = regulator_get_optional(cpu_dev, reg);
-       if (IS_ERR(cpu_reg)) {
-               /*
-                * If cpu's regulator supply node is present, but regulator is
-                * not yet registered, we should try defering probe.
-                */
-               if (PTR_ERR(cpu_reg) == -EPROBE_DEFER) {
-                       dev_dbg(cpu_dev, "cpu%d regulator not ready, retry\n",
-                               cpu);
-                       return -EPROBE_DEFER;
-               }
-
-               /* Try with "cpu-supply" */
-               if (reg == reg_cpu0) {
-                       reg = reg_cpu;
-                       goto try_again;
-               }
-
-               dev_warn(cpu_dev, "failed to get cpu%d regulator: %ld\n",
-                        cpu, PTR_ERR(cpu_reg));
-       }
-
-       cpu_clk = clk_get(cpu_dev, NULL);
-       if (IS_ERR(cpu_clk)) {
-               /* put regulator */
-               if (!IS_ERR(cpu_reg))
-                       regulator_put(cpu_reg);
-
-               ret = PTR_ERR(cpu_clk);
-
-               /*
-                * If cpu's clk node is present, but clock is not yet
-                * registered, we should try defering probe.
-                */
-               if (ret == -EPROBE_DEFER)
-                       dev_dbg(cpu_dev, "cpu%d clock not ready, retry\n", cpu);
-               else
-                       dev_err(cpu_dev, "failed to get cpu%d clock: %d\n", ret,
-                               cpu);
-       } else {
-               *cdev = cpu_dev;
-               *creg = cpu_reg;
-               *cclk = cpu_clk;
-       }
-
-       return ret;
-}
-
-static int cpu0_cpufreq_init(struct cpufreq_policy *policy)
-{
-       struct cpufreq_frequency_table *freq_table;
-       struct thermal_cooling_device *cdev;
-       struct device_node *np;
-       struct private_data *priv;
-       struct device *cpu_dev;
-       struct regulator *cpu_reg;
-       struct clk *cpu_clk;
-       unsigned int transition_latency;
-       int ret;
-
-       ret = allocate_resources(policy->cpu, &cpu_dev, &cpu_reg, &cpu_clk);
-       if (ret) {
-               pr_err("%s: Failed to allocate resources\n: %d", __func__, ret);
-               return ret;
-       }
-
-       np = of_node_get(cpu_dev->of_node);
-       if (!np) {
-               dev_err(cpu_dev, "failed to find cpu%d node\n", policy->cpu);
-               ret = -ENOENT;
-               goto out_put_reg_clk;
-       }
-
-       /* OPPs might be populated at runtime, don't check for error here */
-       of_init_opp_table(cpu_dev);
-
-       ret = dev_pm_opp_init_cpufreq_table(cpu_dev, &freq_table);
-       if (ret) {
-               dev_err(cpu_dev, "failed to init cpufreq table: %d\n", ret);
-               goto out_put_node;
-       }
-
-       priv = kzalloc(sizeof(*priv), GFP_KERNEL);
-       if (!priv) {
-               ret = -ENOMEM;
-               goto out_free_table;
-       }
-
-       of_property_read_u32(np, "voltage-tolerance", &priv->voltage_tolerance);
-
-       if (of_property_read_u32(np, "clock-latency", &transition_latency))
-               transition_latency = CPUFREQ_ETERNAL;
-
-       if (!IS_ERR(cpu_reg)) {
-               struct dev_pm_opp *opp;
-               unsigned long min_uV, max_uV;
-               int i;
-
-               /*
-                * OPP is maintained in order of increasing frequency, and
-                * freq_table initialised from OPP is therefore sorted in the
-                * same order.
-                */
-               for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++)
-                       ;
-               rcu_read_lock();
-               opp = dev_pm_opp_find_freq_exact(cpu_dev,
-                               freq_table[0].frequency * 1000, true);
-               min_uV = dev_pm_opp_get_voltage(opp);
-               opp = dev_pm_opp_find_freq_exact(cpu_dev,
-                               freq_table[i-1].frequency * 1000, true);
-               max_uV = dev_pm_opp_get_voltage(opp);
-               rcu_read_unlock();
-               ret = regulator_set_voltage_time(cpu_reg, min_uV, max_uV);
-               if (ret > 0)
-                       transition_latency += ret * 1000;
-       }
-
-       /*
-        * For now, just loading the cooling device;
-        * thermal DT code takes care of matching them.
-        */
-       if (of_find_property(np, "#cooling-cells", NULL)) {
-               cdev = of_cpufreq_cooling_register(np, cpu_present_mask);
-               if (IS_ERR(cdev))
-                       dev_err(cpu_dev,
-                               "running cpufreq without cooling device: %ld\n",
-                               PTR_ERR(cdev));
-               else
-                       priv->cdev = cdev;
-       }
-       of_node_put(np);
-
-       priv->cpu_dev = cpu_dev;
-       priv->cpu_reg = cpu_reg;
-       policy->driver_data = priv;
-
-       policy->clk = cpu_clk;
-       ret = cpufreq_generic_init(policy, freq_table, transition_latency);
-       if (ret)
-               goto out_cooling_unregister;
-
-       return 0;
-
-out_cooling_unregister:
-       cpufreq_cooling_unregister(priv->cdev);
-       kfree(priv);
-out_free_table:
-       dev_pm_opp_free_cpufreq_table(cpu_dev, &freq_table);
-out_put_node:
-       of_node_put(np);
-out_put_reg_clk:
-       clk_put(cpu_clk);
-       if (!IS_ERR(cpu_reg))
-               regulator_put(cpu_reg);
-
-       return ret;
-}
-
-static int cpu0_cpufreq_exit(struct cpufreq_policy *policy)
-{
-       struct private_data *priv = policy->driver_data;
-
-       cpufreq_cooling_unregister(priv->cdev);
-       dev_pm_opp_free_cpufreq_table(priv->cpu_dev, &policy->freq_table);
-       clk_put(policy->clk);
-       if (!IS_ERR(priv->cpu_reg))
-               regulator_put(priv->cpu_reg);
-       kfree(priv);
-
-       return 0;
-}
-
-static struct cpufreq_driver cpu0_cpufreq_driver = {
-       .flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK,
-       .verify = cpufreq_generic_frequency_table_verify,
-       .target_index = cpu0_set_target,
-       .get = cpufreq_generic_get,
-       .init = cpu0_cpufreq_init,
-       .exit = cpu0_cpufreq_exit,
-       .name = "generic_cpu0",
-       .attr = cpufreq_generic_attr,
-};
-
-static int cpu0_cpufreq_probe(struct platform_device *pdev)
-{
-       struct device *cpu_dev;
-       struct regulator *cpu_reg;
-       struct clk *cpu_clk;
-       int ret;
-
-       /*
-        * All per-cluster (CPUs sharing clock/voltages) initialization is done
-        * from ->init(). In probe(), we just need to make sure that clk and
-        * regulators are available. Else defer probe and retry.
-        *
-        * FIXME: Is checking this only for CPU0 sufficient ?
-        */
-       ret = allocate_resources(0, &cpu_dev, &cpu_reg, &cpu_clk);
-       if (ret)
-               return ret;
-
-       clk_put(cpu_clk);
-       if (!IS_ERR(cpu_reg))
-               regulator_put(cpu_reg);
-
-       ret = cpufreq_register_driver(&cpu0_cpufreq_driver);
-       if (ret)
-               dev_err(cpu_dev, "failed register driver: %d\n", ret);
-
-       return ret;
-}
-
-static int cpu0_cpufreq_remove(struct platform_device *pdev)
-{
-       cpufreq_unregister_driver(&cpu0_cpufreq_driver);
-       return 0;
-}
-
-static struct platform_driver cpu0_cpufreq_platdrv = {
-       .driver = {
-               .name   = "cpufreq-cpu0",
-               .owner  = THIS_MODULE,
-       },
-       .probe          = cpu0_cpufreq_probe,
-       .remove         = cpu0_cpufreq_remove,
-};
-module_platform_driver(cpu0_cpufreq_platdrv);
-
-MODULE_AUTHOR("Viresh Kumar <viresh.kumar@linaro.org>");
-MODULE_AUTHOR("Shawn Guo <shawn.guo@linaro.org>");
-MODULE_DESCRIPTION("Generic CPU0 cpufreq driver");
-MODULE_LICENSE("GPL");
diff --git a/drivers/cpufreq/cpufreq-dt.c b/drivers/cpufreq/cpufreq-dt.c
new file mode 100644 (file)
index 0000000..e002650
--- /dev/null
@@ -0,0 +1,363 @@
+/*
+ * Copyright (C) 2012 Freescale Semiconductor, Inc.
+ *
+ * Copyright (C) 2014 Linaro.
+ * Viresh Kumar <viresh.kumar@linaro.org>
+ *
+ * The OPP code in function set_target() is reused from
+ * drivers/cpufreq/omap-cpufreq.c
+ *
+ * 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.
+ */
+
+#define pr_fmt(fmt)    KBUILD_MODNAME ": " fmt
+
+#include <linux/clk.h>
+#include <linux/cpu.h>
+#include <linux/cpu_cooling.h>
+#include <linux/cpufreq.h>
+#include <linux/cpumask.h>
+#include <linux/err.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/pm_opp.h>
+#include <linux/platform_device.h>
+#include <linux/regulator/consumer.h>
+#include <linux/slab.h>
+#include <linux/thermal.h>
+
+struct private_data {
+       struct device *cpu_dev;
+       struct regulator *cpu_reg;
+       struct thermal_cooling_device *cdev;
+       unsigned int voltage_tolerance; /* in percentage */
+};
+
+static int set_target(struct cpufreq_policy *policy, unsigned int index)
+{
+       struct dev_pm_opp *opp;
+       struct cpufreq_frequency_table *freq_table = policy->freq_table;
+       struct clk *cpu_clk = policy->clk;
+       struct private_data *priv = policy->driver_data;
+       struct device *cpu_dev = priv->cpu_dev;
+       struct regulator *cpu_reg = priv->cpu_reg;
+       unsigned long volt = 0, volt_old = 0, tol = 0;
+       unsigned int old_freq, new_freq;
+       long freq_Hz, freq_exact;
+       int ret;
+
+       freq_Hz = clk_round_rate(cpu_clk, freq_table[index].frequency * 1000);
+       if (freq_Hz <= 0)
+               freq_Hz = freq_table[index].frequency * 1000;
+
+       freq_exact = freq_Hz;
+       new_freq = freq_Hz / 1000;
+       old_freq = clk_get_rate(cpu_clk) / 1000;
+
+       if (!IS_ERR(cpu_reg)) {
+               rcu_read_lock();
+               opp = dev_pm_opp_find_freq_ceil(cpu_dev, &freq_Hz);
+               if (IS_ERR(opp)) {
+                       rcu_read_unlock();
+                       dev_err(cpu_dev, "failed to find OPP for %ld\n",
+                               freq_Hz);
+                       return PTR_ERR(opp);
+               }
+               volt = dev_pm_opp_get_voltage(opp);
+               rcu_read_unlock();
+               tol = volt * priv->voltage_tolerance / 100;
+               volt_old = regulator_get_voltage(cpu_reg);
+       }
+
+       dev_dbg(cpu_dev, "%u MHz, %ld mV --> %u MHz, %ld mV\n",
+               old_freq / 1000, volt_old ? volt_old / 1000 : -1,
+               new_freq / 1000, volt ? volt / 1000 : -1);
+
+       /* scaling up?  scale voltage before frequency */
+       if (!IS_ERR(cpu_reg) && new_freq > old_freq) {
+               ret = regulator_set_voltage_tol(cpu_reg, volt, tol);
+               if (ret) {
+                       dev_err(cpu_dev, "failed to scale voltage up: %d\n",
+                               ret);
+                       return ret;
+               }
+       }
+
+       ret = clk_set_rate(cpu_clk, freq_exact);
+       if (ret) {
+               dev_err(cpu_dev, "failed to set clock rate: %d\n", ret);
+               if (!IS_ERR(cpu_reg))
+                       regulator_set_voltage_tol(cpu_reg, volt_old, tol);
+               return ret;
+       }
+
+       /* scaling down?  scale voltage after frequency */
+       if (!IS_ERR(cpu_reg) && new_freq < old_freq) {
+               ret = regulator_set_voltage_tol(cpu_reg, volt, tol);
+               if (ret) {
+                       dev_err(cpu_dev, "failed to scale voltage down: %d\n",
+                               ret);
+                       clk_set_rate(cpu_clk, old_freq * 1000);
+               }
+       }
+
+       return ret;
+}
+
+static int allocate_resources(int cpu, struct device **cdev,
+                             struct regulator **creg, struct clk **cclk)
+{
+       struct device *cpu_dev;
+       struct regulator *cpu_reg;
+       struct clk *cpu_clk;
+       int ret = 0;
+       char *reg_cpu0 = "cpu0", *reg_cpu = "cpu", *reg;
+
+       cpu_dev = get_cpu_device(cpu);
+       if (!cpu_dev) {
+               pr_err("failed to get cpu%d device\n", cpu);
+               return -ENODEV;
+       }
+
+       /* Try "cpu0" for older DTs */
+       if (!cpu)
+               reg = reg_cpu0;
+       else
+               reg = reg_cpu;
+
+try_again:
+       cpu_reg = regulator_get_optional(cpu_dev, reg);
+       if (IS_ERR(cpu_reg)) {
+               /*
+                * If cpu's regulator supply node is present, but regulator is
+                * not yet registered, we should try defering probe.
+                */
+               if (PTR_ERR(cpu_reg) == -EPROBE_DEFER) {
+                       dev_dbg(cpu_dev, "cpu%d regulator not ready, retry\n",
+                               cpu);
+                       return -EPROBE_DEFER;
+               }
+
+               /* Try with "cpu-supply" */
+               if (reg == reg_cpu0) {
+                       reg = reg_cpu;
+                       goto try_again;
+               }
+
+               dev_warn(cpu_dev, "failed to get cpu%d regulator: %ld\n",
+                        cpu, PTR_ERR(cpu_reg));
+       }
+
+       cpu_clk = clk_get(cpu_dev, NULL);
+       if (IS_ERR(cpu_clk)) {
+               /* put regulator */
+               if (!IS_ERR(cpu_reg))
+                       regulator_put(cpu_reg);
+
+               ret = PTR_ERR(cpu_clk);
+
+               /*
+                * If cpu's clk node is present, but clock is not yet
+                * registered, we should try defering probe.
+                */
+               if (ret == -EPROBE_DEFER)
+                       dev_dbg(cpu_dev, "cpu%d clock not ready, retry\n", cpu);
+               else
+                       dev_err(cpu_dev, "failed to get cpu%d clock: %d\n", ret,
+                               cpu);
+       } else {
+               *cdev = cpu_dev;
+               *creg = cpu_reg;
+               *cclk = cpu_clk;
+       }
+
+       return ret;
+}
+
+static int cpufreq_init(struct cpufreq_policy *policy)
+{
+       struct cpufreq_frequency_table *freq_table;
+       struct thermal_cooling_device *cdev;
+       struct device_node *np;
+       struct private_data *priv;
+       struct device *cpu_dev;
+       struct regulator *cpu_reg;
+       struct clk *cpu_clk;
+       unsigned int transition_latency;
+       int ret;
+
+       ret = allocate_resources(policy->cpu, &cpu_dev, &cpu_reg, &cpu_clk);
+       if (ret) {
+               pr_err("%s: Failed to allocate resources\n: %d", __func__, ret);
+               return ret;
+       }
+
+       np = of_node_get(cpu_dev->of_node);
+       if (!np) {
+               dev_err(cpu_dev, "failed to find cpu%d node\n", policy->cpu);
+               ret = -ENOENT;
+               goto out_put_reg_clk;
+       }
+
+       /* OPPs might be populated at runtime, don't check for error here */
+       of_init_opp_table(cpu_dev);
+
+       ret = dev_pm_opp_init_cpufreq_table(cpu_dev, &freq_table);
+       if (ret) {
+               dev_err(cpu_dev, "failed to init cpufreq table: %d\n", ret);
+               goto out_put_node;
+       }
+
+       priv = kzalloc(sizeof(*priv), GFP_KERNEL);
+       if (!priv) {
+               ret = -ENOMEM;
+               goto out_free_table;
+       }
+
+       of_property_read_u32(np, "voltage-tolerance", &priv->voltage_tolerance);
+
+       if (of_property_read_u32(np, "clock-latency", &transition_latency))
+               transition_latency = CPUFREQ_ETERNAL;
+
+       if (!IS_ERR(cpu_reg)) {
+               struct dev_pm_opp *opp;
+               unsigned long min_uV, max_uV;
+               int i;
+
+               /*
+                * OPP is maintained in order of increasing frequency, and
+                * freq_table initialised from OPP is therefore sorted in the
+                * same order.
+                */
+               for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++)
+                       ;
+               rcu_read_lock();
+               opp = dev_pm_opp_find_freq_exact(cpu_dev,
+                               freq_table[0].frequency * 1000, true);
+               min_uV = dev_pm_opp_get_voltage(opp);
+               opp = dev_pm_opp_find_freq_exact(cpu_dev,
+                               freq_table[i-1].frequency * 1000, true);
+               max_uV = dev_pm_opp_get_voltage(opp);
+               rcu_read_unlock();
+               ret = regulator_set_voltage_time(cpu_reg, min_uV, max_uV);
+               if (ret > 0)
+                       transition_latency += ret * 1000;
+       }
+
+       /*
+        * For now, just loading the cooling device;
+        * thermal DT code takes care of matching them.
+        */
+       if (of_find_property(np, "#cooling-cells", NULL)) {
+               cdev = of_cpufreq_cooling_register(np, cpu_present_mask);
+               if (IS_ERR(cdev))
+                       dev_err(cpu_dev,
+                               "running cpufreq without cooling device: %ld\n",
+                               PTR_ERR(cdev));
+               else
+                       priv->cdev = cdev;
+       }
+       of_node_put(np);
+
+       priv->cpu_dev = cpu_dev;
+       priv->cpu_reg = cpu_reg;
+       policy->driver_data = priv;
+
+       policy->clk = cpu_clk;
+       ret = cpufreq_generic_init(policy, freq_table, transition_latency);
+       if (ret)
+               goto out_cooling_unregister;
+
+       return 0;
+
+out_cooling_unregister:
+       cpufreq_cooling_unregister(priv->cdev);
+       kfree(priv);
+out_free_table:
+       dev_pm_opp_free_cpufreq_table(cpu_dev, &freq_table);
+out_put_node:
+       of_node_put(np);
+out_put_reg_clk:
+       clk_put(cpu_clk);
+       if (!IS_ERR(cpu_reg))
+               regulator_put(cpu_reg);
+
+       return ret;
+}
+
+static int cpufreq_exit(struct cpufreq_policy *policy)
+{
+       struct private_data *priv = policy->driver_data;
+
+       cpufreq_cooling_unregister(priv->cdev);
+       dev_pm_opp_free_cpufreq_table(priv->cpu_dev, &policy->freq_table);
+       clk_put(policy->clk);
+       if (!IS_ERR(priv->cpu_reg))
+               regulator_put(priv->cpu_reg);
+       kfree(priv);
+
+       return 0;
+}
+
+static struct cpufreq_driver dt_cpufreq_driver = {
+       .flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK,
+       .verify = cpufreq_generic_frequency_table_verify,
+       .target_index = set_target,
+       .get = cpufreq_generic_get,
+       .init = cpufreq_init,
+       .exit = cpufreq_exit,
+       .name = "cpufreq-dt",
+       .attr = cpufreq_generic_attr,
+};
+
+static int dt_cpufreq_probe(struct platform_device *pdev)
+{
+       struct device *cpu_dev;
+       struct regulator *cpu_reg;
+       struct clk *cpu_clk;
+       int ret;
+
+       /*
+        * All per-cluster (CPUs sharing clock/voltages) initialization is done
+        * from ->init(). In probe(), we just need to make sure that clk and
+        * regulators are available. Else defer probe and retry.
+        *
+        * FIXME: Is checking this only for CPU0 sufficient ?
+        */
+       ret = allocate_resources(0, &cpu_dev, &cpu_reg, &cpu_clk);
+       if (ret)
+               return ret;
+
+       clk_put(cpu_clk);
+       if (!IS_ERR(cpu_reg))
+               regulator_put(cpu_reg);
+
+       ret = cpufreq_register_driver(&dt_cpufreq_driver);
+       if (ret)
+               dev_err(cpu_dev, "failed register driver: %d\n", ret);
+
+       return ret;
+}
+
+static int dt_cpufreq_remove(struct platform_device *pdev)
+{
+       cpufreq_unregister_driver(&dt_cpufreq_driver);
+       return 0;
+}
+
+static struct platform_driver dt_cpufreq_platdrv = {
+       .driver = {
+               .name   = "cpufreq-dt",
+               .owner  = THIS_MODULE,
+       },
+       .probe          = dt_cpufreq_probe,
+       .remove         = dt_cpufreq_remove,
+};
+module_platform_driver(dt_cpufreq_platdrv);
+
+MODULE_AUTHOR("Viresh Kumar <viresh.kumar@linaro.org>");
+MODULE_AUTHOR("Shawn Guo <shawn.guo@linaro.org>");
+MODULE_DESCRIPTION("Generic cpufreq driver");
+MODULE_LICENSE("GPL");
index 61a54310a1b9df6923a1fbfbb3ab6e1872c914eb..843ec824fd91051db1af8751d155018261d9043c 100644 (file)
@@ -127,7 +127,7 @@ int exynos4210_cpufreq_init(struct exynos_dvfs_info *info)
         * dependencies on platform headers. It is necessary to enable
         * Exynos multi-platform support and will be removed together with
         * this whole driver as soon as Exynos gets migrated to use
-        * cpufreq-cpu0 driver.
+        * cpufreq-dt driver.
         */
        np = of_find_compatible_node(NULL, NULL, "samsung,exynos4210-clock");
        if (!np) {
index 351a2074cfea784c8a522b3fa6080a67c59e7180..9e78a850e29f4dc967fceb422f22cdc6a571355c 100644 (file)
@@ -174,7 +174,7 @@ int exynos4x12_cpufreq_init(struct exynos_dvfs_info *info)
         * dependencies on platform headers. It is necessary to enable
         * Exynos multi-platform support and will be removed together with
         * this whole driver as soon as Exynos gets migrated to use
-        * cpufreq-cpu0 driver.
+        * cpufreq-dt driver.
         */
        np = of_find_compatible_node(NULL, NULL, "samsung,exynos4412-clock");
        if (!np) {
index c91ce69dc63101d3a1f866b906acaf780c20070a..3eafdc7ba7877f4eedc2484fa7208c32f5cf49eb 100644 (file)
@@ -153,7 +153,7 @@ int exynos5250_cpufreq_init(struct exynos_dvfs_info *info)
         * dependencies on platform headers. It is necessary to enable
         * Exynos multi-platform support and will be removed together with
         * this whole driver as soon as Exynos gets migrated to use
-        * cpufreq-cpu0 driver.
+        * cpufreq-dt driver.
         */
        np = of_find_compatible_node(NULL, NULL, "samsung,exynos5250-clock");
        if (!np) {
index bf8902a0866dd4d767cdab4ba45d84ca397033fe..ec399ad2f059379891a4d384e5b24dc3b9c3ab59 100644 (file)
@@ -6,7 +6,7 @@
  * published by the Free Software Foundation.
  *
  * This driver provides the clk notifier callbacks that are used when
- * the cpufreq-cpu0 driver changes to frequency to alert the highbank
+ * the cpufreq-dt driver changes to frequency to alert the highbank
  * EnergyCore Management Engine (ECME) about the need to change
  * voltage. The ECME interfaces with the actual voltage regulators.
  */
@@ -60,7 +60,7 @@ static struct notifier_block hb_cpufreq_clk_nb = {
 
 static int hb_cpufreq_driver_init(void)
 {
-       struct platform_device_info devinfo = { .name = "cpufreq-cpu0", };
+       struct platform_device_info devinfo = { .name = "cpufreq-dt", };
        struct device *cpu_dev;
        struct clk *cpu_clk;
        struct device_node *np;
@@ -95,7 +95,7 @@ static int hb_cpufreq_driver_init(void)
                goto out_put_node;
        }
 
-       /* Instantiate cpufreq-cpu0 */
+       /* Instantiate cpufreq-dt */
        platform_device_register_full(&devinfo);
 
 out_put_node:
index 3f9791f07b8ea05f745a2bcbce6b738379443087..567caa6313fffa447f19992d7af5df940c4d0068 100644 (file)
@@ -597,7 +597,7 @@ static int s5pv210_cpufreq_probe(struct platform_device *pdev)
         * and dependencies on platform headers. It is necessary to enable
         * S5PV210 multi-platform support and will be removed together with
         * this whole driver as soon as S5PV210 gets migrated to use
-        * cpufreq-cpu0 driver.
+        * cpufreq-dt driver.
         */
        np = of_find_compatible_node(NULL, NULL, "samsung,s5pv210-clock");
        if (!np) {