diff options
author | Viresh Kumar <viresh.kumar@linaro.org> | 2014-01-09 10:08:43 -0500 |
---|---|---|
committer | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2014-01-16 20:00:44 -0500 |
commit | 652ed95d5fa6074b3c4ea245deb0691f1acb6656 (patch) | |
tree | 7b01079f3a582b0d6b08f541a8a344440bec8eae /drivers/cpufreq/exynos5440-cpufreq.c | |
parent | 0ad04fb30db0341d0b1134e2f592d9146c9abb64 (diff) |
cpufreq: introduce cpufreq_generic_get() routine
CPUFreq drivers that use clock frameworks interface,i.e. clk_get_rate(),
to get CPUs clk rate, have similar sort of code used in most of them.
This patch adds a generic ->get() which will do the same thing for them.
All those drivers are required to now is to set .get to cpufreq_generic_get()
and set their clk pointer in policy->clk during ->init().
Acked-by: Hans-Christian Egtvedt <egtvedt@samfundet.no>
Acked-by: Shawn Guo <shawn.guo@linaro.org>
Acked-by: Linus Walleij <linus.walleij@linaro.org>
Acked-by: Shawn Guo <shawn.guo@linaro.org>
Acked-by: Stephen Warren <swarren@nvidia.com>
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/exynos5440-cpufreq.c')
-rw-r--r-- | drivers/cpufreq/exynos5440-cpufreq.c | 33 |
1 files changed, 14 insertions, 19 deletions
diff --git a/drivers/cpufreq/exynos5440-cpufreq.c b/drivers/cpufreq/exynos5440-cpufreq.c index ffe6faea3a5f..49b756015316 100644 --- a/drivers/cpufreq/exynos5440-cpufreq.c +++ b/drivers/cpufreq/exynos5440-cpufreq.c | |||
@@ -100,7 +100,6 @@ struct exynos_dvfs_data { | |||
100 | struct resource *mem; | 100 | struct resource *mem; |
101 | int irq; | 101 | int irq; |
102 | struct clk *cpu_clk; | 102 | struct clk *cpu_clk; |
103 | unsigned int cur_frequency; | ||
104 | unsigned int latency; | 103 | unsigned int latency; |
105 | struct cpufreq_frequency_table *freq_table; | 104 | struct cpufreq_frequency_table *freq_table; |
106 | unsigned int freq_count; | 105 | unsigned int freq_count; |
@@ -165,7 +164,7 @@ static int init_div_table(void) | |||
165 | return 0; | 164 | return 0; |
166 | } | 165 | } |
167 | 166 | ||
168 | static void exynos_enable_dvfs(void) | 167 | static void exynos_enable_dvfs(unsigned int cur_frequency) |
169 | { | 168 | { |
170 | unsigned int tmp, i, cpu; | 169 | unsigned int tmp, i, cpu; |
171 | struct cpufreq_frequency_table *freq_table = dvfs_info->freq_table; | 170 | struct cpufreq_frequency_table *freq_table = dvfs_info->freq_table; |
@@ -184,18 +183,18 @@ static void exynos_enable_dvfs(void) | |||
184 | 183 | ||
185 | /* Set initial performance index */ | 184 | /* Set initial performance index */ |
186 | for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) | 185 | for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) |
187 | if (freq_table[i].frequency == dvfs_info->cur_frequency) | 186 | if (freq_table[i].frequency == cur_frequency) |
188 | break; | 187 | break; |
189 | 188 | ||
190 | if (freq_table[i].frequency == CPUFREQ_TABLE_END) { | 189 | if (freq_table[i].frequency == CPUFREQ_TABLE_END) { |
191 | dev_crit(dvfs_info->dev, "Boot up frequency not supported\n"); | 190 | dev_crit(dvfs_info->dev, "Boot up frequency not supported\n"); |
192 | /* Assign the highest frequency */ | 191 | /* Assign the highest frequency */ |
193 | i = 0; | 192 | i = 0; |
194 | dvfs_info->cur_frequency = freq_table[i].frequency; | 193 | cur_frequency = freq_table[i].frequency; |
195 | } | 194 | } |
196 | 195 | ||
197 | dev_info(dvfs_info->dev, "Setting dvfs initial frequency = %uKHZ", | 196 | dev_info(dvfs_info->dev, "Setting dvfs initial frequency = %uKHZ", |
198 | dvfs_info->cur_frequency); | 197 | cur_frequency); |
199 | 198 | ||
200 | for (cpu = 0; cpu < CONFIG_NR_CPUS; cpu++) { | 199 | for (cpu = 0; cpu < CONFIG_NR_CPUS; cpu++) { |
201 | tmp = __raw_readl(dvfs_info->base + XMU_C0_3_PSTATE + cpu * 4); | 200 | tmp = __raw_readl(dvfs_info->base + XMU_C0_3_PSTATE + cpu * 4); |
@@ -209,11 +208,6 @@ static void exynos_enable_dvfs(void) | |||
209 | dvfs_info->base + XMU_DVFS_CTRL); | 208 | dvfs_info->base + XMU_DVFS_CTRL); |
210 | } | 209 | } |
211 | 210 | ||
212 | static unsigned int exynos_getspeed(unsigned int cpu) | ||
213 | { | ||
214 | return dvfs_info->cur_frequency; | ||
215 | } | ||
216 | |||
217 | static int exynos_target(struct cpufreq_policy *policy, unsigned int index) | 211 | static int exynos_target(struct cpufreq_policy *policy, unsigned int index) |
218 | { | 212 | { |
219 | unsigned int tmp; | 213 | unsigned int tmp; |
@@ -222,7 +216,7 @@ static int exynos_target(struct cpufreq_policy *policy, unsigned int index) | |||
222 | 216 | ||
223 | mutex_lock(&cpufreq_lock); | 217 | mutex_lock(&cpufreq_lock); |
224 | 218 | ||
225 | freqs.old = dvfs_info->cur_frequency; | 219 | freqs.old = policy->cur; |
226 | freqs.new = freq_table[index].frequency; | 220 | freqs.new = freq_table[index].frequency; |
227 | 221 | ||
228 | cpufreq_notify_transition(policy, &freqs, CPUFREQ_PRECHANGE); | 222 | cpufreq_notify_transition(policy, &freqs, CPUFREQ_PRECHANGE); |
@@ -250,7 +244,7 @@ static void exynos_cpufreq_work(struct work_struct *work) | |||
250 | goto skip_work; | 244 | goto skip_work; |
251 | 245 | ||
252 | mutex_lock(&cpufreq_lock); | 246 | mutex_lock(&cpufreq_lock); |
253 | freqs.old = dvfs_info->cur_frequency; | 247 | freqs.old = policy->cur; |
254 | 248 | ||
255 | cur_pstate = __raw_readl(dvfs_info->base + XMU_P_STATUS); | 249 | cur_pstate = __raw_readl(dvfs_info->base + XMU_P_STATUS); |
256 | if (cur_pstate >> C0_3_PSTATE_VALID_SHIFT & 0x1) | 250 | if (cur_pstate >> C0_3_PSTATE_VALID_SHIFT & 0x1) |
@@ -260,10 +254,9 @@ static void exynos_cpufreq_work(struct work_struct *work) | |||
260 | 254 | ||
261 | if (likely(index < dvfs_info->freq_count)) { | 255 | if (likely(index < dvfs_info->freq_count)) { |
262 | freqs.new = freq_table[index].frequency; | 256 | freqs.new = freq_table[index].frequency; |
263 | dvfs_info->cur_frequency = freqs.new; | ||
264 | } else { | 257 | } else { |
265 | dev_crit(dvfs_info->dev, "New frequency out of range\n"); | 258 | dev_crit(dvfs_info->dev, "New frequency out of range\n"); |
266 | freqs.new = dvfs_info->cur_frequency; | 259 | freqs.new = freqs.old; |
267 | } | 260 | } |
268 | cpufreq_notify_transition(policy, &freqs, CPUFREQ_POSTCHANGE); | 261 | cpufreq_notify_transition(policy, &freqs, CPUFREQ_POSTCHANGE); |
269 | 262 | ||
@@ -307,6 +300,7 @@ static void exynos_sort_descend_freq_table(void) | |||
307 | 300 | ||
308 | static int exynos_cpufreq_cpu_init(struct cpufreq_policy *policy) | 301 | static int exynos_cpufreq_cpu_init(struct cpufreq_policy *policy) |
309 | { | 302 | { |
303 | policy->clk = dvfs_info->cpu_clk; | ||
310 | return cpufreq_generic_init(policy, dvfs_info->freq_table, | 304 | return cpufreq_generic_init(policy, dvfs_info->freq_table, |
311 | dvfs_info->latency); | 305 | dvfs_info->latency); |
312 | } | 306 | } |
@@ -316,7 +310,7 @@ static struct cpufreq_driver exynos_driver = { | |||
316 | CPUFREQ_NEED_INITIAL_FREQ_CHECK, | 310 | CPUFREQ_NEED_INITIAL_FREQ_CHECK, |
317 | .verify = cpufreq_generic_frequency_table_verify, | 311 | .verify = cpufreq_generic_frequency_table_verify, |
318 | .target_index = exynos_target, | 312 | .target_index = exynos_target, |
319 | .get = exynos_getspeed, | 313 | .get = cpufreq_generic_get, |
320 | .init = exynos_cpufreq_cpu_init, | 314 | .init = exynos_cpufreq_cpu_init, |
321 | .exit = cpufreq_generic_exit, | 315 | .exit = cpufreq_generic_exit, |
322 | .name = CPUFREQ_NAME, | 316 | .name = CPUFREQ_NAME, |
@@ -336,6 +330,7 @@ static int exynos_cpufreq_probe(struct platform_device *pdev) | |||
336 | int ret = -EINVAL; | 330 | int ret = -EINVAL; |
337 | struct device_node *np; | 331 | struct device_node *np; |
338 | struct resource res; | 332 | struct resource res; |
333 | unsigned int cur_frequency; | ||
339 | 334 | ||
340 | np = pdev->dev.of_node; | 335 | np = pdev->dev.of_node; |
341 | if (!np) | 336 | if (!np) |
@@ -392,13 +387,13 @@ static int exynos_cpufreq_probe(struct platform_device *pdev) | |||
392 | goto err_free_table; | 387 | goto err_free_table; |
393 | } | 388 | } |
394 | 389 | ||
395 | dvfs_info->cur_frequency = clk_get_rate(dvfs_info->cpu_clk); | 390 | cur_frequency = clk_get_rate(dvfs_info->cpu_clk); |
396 | if (!dvfs_info->cur_frequency) { | 391 | if (!cur_frequency) { |
397 | dev_err(dvfs_info->dev, "Failed to get clock rate\n"); | 392 | dev_err(dvfs_info->dev, "Failed to get clock rate\n"); |
398 | ret = -EINVAL; | 393 | ret = -EINVAL; |
399 | goto err_free_table; | 394 | goto err_free_table; |
400 | } | 395 | } |
401 | dvfs_info->cur_frequency /= 1000; | 396 | cur_frequency /= 1000; |
402 | 397 | ||
403 | INIT_WORK(&dvfs_info->irq_work, exynos_cpufreq_work); | 398 | INIT_WORK(&dvfs_info->irq_work, exynos_cpufreq_work); |
404 | ret = devm_request_irq(dvfs_info->dev, dvfs_info->irq, | 399 | ret = devm_request_irq(dvfs_info->dev, dvfs_info->irq, |
@@ -415,7 +410,7 @@ static int exynos_cpufreq_probe(struct platform_device *pdev) | |||
415 | goto err_free_table; | 410 | goto err_free_table; |
416 | } | 411 | } |
417 | 412 | ||
418 | exynos_enable_dvfs(); | 413 | exynos_enable_dvfs(cur_frequency); |
419 | ret = cpufreq_register_driver(&exynos_driver); | 414 | ret = cpufreq_register_driver(&exynos_driver); |
420 | if (ret) { | 415 | if (ret) { |
421 | dev_err(dvfs_info->dev, | 416 | dev_err(dvfs_info->dev, |