aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/cpufreq/arm_big_little_dt.c
diff options
context:
space:
mode:
authorViresh Kumar <viresh.kumar@linaro.org>2013-05-17 07:25:11 -0400
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>2013-05-22 06:42:34 -0400
commit92a9b5c291c72aa9899021699458f0b6e328b940 (patch)
tree70063db2fa775cf57c182dc00df25929183ece04 /drivers/cpufreq/arm_big_little_dt.c
parentb5f14720a6421aab841d9f03f0129cfbe7db5133 (diff)
cpufreq: arm_big_little_dt: Register driver only if DT has valid data
If arm_big_little_dt driver is enabled, then it will always try to register with big LITTLE cpufreq core driver. In case DT doesn't have relevant data for cpu nodes, i.e. operating points aren't present, then we should exit early and shouldn't register with big LITTLE cpufreq core driver. Otherwise we will fail continuously from the driver->init() routine. This patch fixes this issue. Reported-and-tested-by: Jon Medhurst <tixy@linaro.org> Reviewed-by: Jon Medhurst <tixy@linaro.org> Signed-off-by: Viresh Kumar <viresh.kumar@linaro.org> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Diffstat (limited to 'drivers/cpufreq/arm_big_little_dt.c')
-rw-r--r--drivers/cpufreq/arm_big_little_dt.c73
1 files changed, 42 insertions, 31 deletions
diff --git a/drivers/cpufreq/arm_big_little_dt.c b/drivers/cpufreq/arm_big_little_dt.c
index 173ed059d95f..27e2f45ccdd5 100644
--- a/drivers/cpufreq/arm_big_little_dt.c
+++ b/drivers/cpufreq/arm_big_little_dt.c
@@ -19,6 +19,7 @@
19 19
20#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 20#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
21 21
22#include <linux/cpu.h>
22#include <linux/cpufreq.h> 23#include <linux/cpufreq.h>
23#include <linux/device.h> 24#include <linux/device.h>
24#include <linux/export.h> 25#include <linux/export.h>
@@ -29,60 +30,63 @@
29#include <linux/types.h> 30#include <linux/types.h>
30#include "arm_big_little.h" 31#include "arm_big_little.h"
31 32
32static int dt_init_opp_table(struct device *cpu_dev) 33/* get cpu node with valid operating-points */
34static struct device_node *get_cpu_node_with_valid_op(int cpu)
33{ 35{
34 struct device_node *np, *parent; 36 struct device_node *np = NULL, *parent;
35 int count = 0, ret; 37 int count = 0;
36 38
37 parent = of_find_node_by_path("/cpus"); 39 parent = of_find_node_by_path("/cpus");
38 if (!parent) { 40 if (!parent) {
39 pr_err("failed to find OF /cpus\n"); 41 pr_err("failed to find OF /cpus\n");
40 return -ENOENT; 42 return NULL;
41 } 43 }
42 44
43 for_each_child_of_node(parent, np) { 45 for_each_child_of_node(parent, np) {
44 if (count++ != cpu_dev->id) 46 if (count++ != cpu)
45 continue; 47 continue;
46 if (!of_get_property(np, "operating-points", NULL)) { 48 if (!of_get_property(np, "operating-points", NULL)) {
47 ret = -ENODATA; 49 of_node_put(np);
48 } else { 50 np = NULL;
49 cpu_dev->of_node = np;
50 ret = of_init_opp_table(cpu_dev);
51 } 51 }
52 of_node_put(np);
53 of_node_put(parent);
54 52
55 return ret; 53 break;
56 } 54 }
57 55
58 return -ENODEV; 56 of_node_put(parent);
57 return np;
58}
59
60static int dt_init_opp_table(struct device *cpu_dev)
61{
62 struct device_node *np;
63 int ret;
64
65 np = get_cpu_node_with_valid_op(cpu_dev->id);
66 if (!np)
67 return -ENODATA;
68
69 cpu_dev->of_node = np;
70 ret = of_init_opp_table(cpu_dev);
71 of_node_put(np);
72
73 return ret;
59} 74}
60 75
61static int dt_get_transition_latency(struct device *cpu_dev) 76static int dt_get_transition_latency(struct device *cpu_dev)
62{ 77{
63 struct device_node *np, *parent; 78 struct device_node *np;
64 u32 transition_latency = CPUFREQ_ETERNAL; 79 u32 transition_latency = CPUFREQ_ETERNAL;
65 int count = 0;
66 80
67 parent = of_find_node_by_path("/cpus"); 81 np = get_cpu_node_with_valid_op(cpu_dev->id);
68 if (!parent) { 82 if (!np)
69 pr_info("Failed to find OF /cpus. Use CPUFREQ_ETERNAL transition latency\n");
70 return CPUFREQ_ETERNAL; 83 return CPUFREQ_ETERNAL;
71 }
72
73 for_each_child_of_node(parent, np) {
74 if (count++ != cpu_dev->id)
75 continue;
76
77 of_property_read_u32(np, "clock-latency", &transition_latency);
78 of_node_put(np);
79 of_node_put(parent);
80 84
81 return transition_latency; 85 of_property_read_u32(np, "clock-latency", &transition_latency);
82 } 86 of_node_put(np);
83 87
84 pr_info("clock-latency isn't found, use CPUFREQ_ETERNAL transition latency\n"); 88 pr_debug("%s: clock-latency: %d\n", __func__, transition_latency);
85 return CPUFREQ_ETERNAL; 89 return transition_latency;
86} 90}
87 91
88static struct cpufreq_arm_bL_ops dt_bL_ops = { 92static struct cpufreq_arm_bL_ops dt_bL_ops = {
@@ -93,6 +97,13 @@ static struct cpufreq_arm_bL_ops dt_bL_ops = {
93 97
94static int generic_bL_init(void) 98static int generic_bL_init(void)
95{ 99{
100 struct device_node *np;
101
102 np = get_cpu_node_with_valid_op(0);
103 if (!np)
104 return -ENODEV;
105
106 of_node_put(np);
96 return bL_cpufreq_register(&dt_bL_ops); 107 return bL_cpufreq_register(&dt_bL_ops);
97} 108}
98module_init(generic_bL_init); 109module_init(generic_bL_init);