aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/cpufreq/powernv-cpufreq.c
diff options
context:
space:
mode:
authorPreeti U Murthy <preeti@linux.vnet.ibm.com>2014-09-29 09:47:53 -0400
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>2014-09-29 09:47:53 -0400
commitb120339c787b243cdbe3f67401b4aa5625d88ff3 (patch)
tree4569887e9616b3a639a4171e9837d7f92cf27b25 /drivers/cpufreq/powernv-cpufreq.c
parent789ca243740de236a39146fc3d3bbaeb4d3ae0ba (diff)
cpufreq: powernv: Set the pstate of the last hotplugged out cpu in policy->cpus to minimum
Its possible today that the pstate of a core is held at a high even after the entire core is hotplugged out if a load had just run on the hotplugged cpu. This is fair, since it is assumed that the pstate does not matter to a cpu in a deep idle state, which is the expected state of a hotplugged core on powerpc. However on powerpc, the pstate at a socket level is held at the maximum of the pstates of each core. Even if the pstates of the active cores on that socket is low, the socket pstate is held high due to the pstate of the hotplugged core in the above mentioned scenario. This can cost significant amount of power loss for no good. Besides, since it is a non active core, nothing can be done from the kernel's end to set the frequency of the core right. Hence make use of the stop_cpu callback to explicitly set the pstate of the core to a minimum when the last cpu of the core gets hotplugged out. Signed-off-by: Preeti U Murthy <preeti@linux.vnet.ibm.com> Acked-by: Viresh Kumar <viresh.kumar@linaro.org> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Diffstat (limited to 'drivers/cpufreq/powernv-cpufreq.c')
-rw-r--r--drivers/cpufreq/powernv-cpufreq.c9
1 files changed, 9 insertions, 0 deletions
diff --git a/drivers/cpufreq/powernv-cpufreq.c b/drivers/cpufreq/powernv-cpufreq.c
index 379c0837f5a9..5a628f1e2cdf 100644
--- a/drivers/cpufreq/powernv-cpufreq.c
+++ b/drivers/cpufreq/powernv-cpufreq.c
@@ -317,6 +317,14 @@ static int powernv_cpufreq_cpu_init(struct cpufreq_policy *policy)
317 return cpufreq_table_validate_and_show(policy, powernv_freqs); 317 return cpufreq_table_validate_and_show(policy, powernv_freqs);
318} 318}
319 319
320static void powernv_cpufreq_stop_cpu(struct cpufreq_policy *policy)
321{
322 struct powernv_smp_call_data freq_data;
323
324 freq_data.pstate_id = powernv_pstate_info.min;
325 smp_call_function_single(policy->cpu, set_pstate, &freq_data, 1);
326}
327
320static struct cpufreq_driver powernv_cpufreq_driver = { 328static struct cpufreq_driver powernv_cpufreq_driver = {
321 .name = "powernv-cpufreq", 329 .name = "powernv-cpufreq",
322 .flags = CPUFREQ_CONST_LOOPS, 330 .flags = CPUFREQ_CONST_LOOPS,
@@ -324,6 +332,7 @@ static struct cpufreq_driver powernv_cpufreq_driver = {
324 .verify = cpufreq_generic_frequency_table_verify, 332 .verify = cpufreq_generic_frequency_table_verify,
325 .target_index = powernv_cpufreq_target_index, 333 .target_index = powernv_cpufreq_target_index,
326 .get = powernv_cpufreq_get, 334 .get = powernv_cpufreq_get,
335 .stop_cpu = powernv_cpufreq_stop_cpu,
327 .attr = powernv_cpu_freq_attr, 336 .attr = powernv_cpu_freq_attr,
328}; 337};
329 338