mirror of
https://github.com/rd-stuffs/msm-4.14.git
synced 2025-02-20 11:45:48 +08:00
During cpufreq driver's registration, if the ->init() callback for all the CPUs fail then there is not much point in keeping the driver around as it will only account for more of unnecessary noise, for example cpufreq core will try to suspend/resume the driver which never got registered properly. The removal of such a driver is avoided if the driver carries the CPUFREQ_STICKY flag. This was added way back [1] in 2004 and perhaps no one should ever need it now. A lot of drivers do set this flag, probably because they just copied it from other drivers. This was added earlier for some platforms [2] because their cpufreq drivers were getting registered before the CPUs were registered with subsys framework. And hence they used to fail. The same isn't true anymore though. The current code flow in the kernel is: start_kernel() -> kernel_init() -> kernel_init_freeable() -> do_basic_setup() -> driver_init() -> cpu_dev_init() -> subsys_system_register() //For CPUs -> do_initcalls() -> cpufreq_register_driver() Clearly, the CPUs will always get registered with subsys framework before any cpufreq driver can get probed. Remove the flag and update the relevant drivers. Link: https://git.kernel.org/pub/scm/linux/kernel/git/tglx/history.git/commit/include/linux/cpufreq.h?id=7cc9f0d9a1ab04cedc60d64fd8dcf7df224a3b4d # [1] Link: https://git.kernel.org/pub/scm/linux/kernel/git/tglx/history.git/commit/arch/arm/mach-sa1100/cpu-sa1100.c?id=f59d3bbe35f6268d729f51be82af8325d62f20f5 # [2] Change-Id: If2509516ec7fa7518750e23e0198614654dbf9b7 Signed-off-by: Viresh Kumar <viresh.kumar@linaro.org> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Signed-off-by: Samuel Pascua <pascua.samuel.14@gmail.com> Signed-off-by: Richard Raya <rdxzv.dev@gmail.com>
127 lines
3.0 KiB
C
127 lines
3.0 KiB
C
/*
|
|
* Copyright (C) 2004-2007 Atmel Corporation
|
|
*
|
|
* Based on MIPS implementation arch/mips/kernel/time.c
|
|
* Copyright 2001 MontaVista Software Inc.
|
|
*
|
|
* 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 DEBUG*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/types.h>
|
|
#include <linux/init.h>
|
|
#include <linux/cpufreq.h>
|
|
#include <linux/io.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/err.h>
|
|
#include <linux/export.h>
|
|
#include <linux/slab.h>
|
|
|
|
static struct cpufreq_frequency_table *freq_table;
|
|
|
|
static unsigned int ref_freq;
|
|
static unsigned long loops_per_jiffy_ref;
|
|
|
|
static int at32_set_target(struct cpufreq_policy *policy, unsigned int index)
|
|
{
|
|
unsigned int old_freq, new_freq;
|
|
|
|
old_freq = policy->cur;
|
|
new_freq = freq_table[index].frequency;
|
|
|
|
if (!ref_freq) {
|
|
ref_freq = old_freq;
|
|
loops_per_jiffy_ref = boot_cpu_data.loops_per_jiffy;
|
|
}
|
|
|
|
if (old_freq < new_freq)
|
|
boot_cpu_data.loops_per_jiffy = cpufreq_scale(
|
|
loops_per_jiffy_ref, ref_freq, new_freq);
|
|
clk_set_rate(policy->clk, new_freq * 1000);
|
|
if (new_freq < old_freq)
|
|
boot_cpu_data.loops_per_jiffy = cpufreq_scale(
|
|
loops_per_jiffy_ref, ref_freq, new_freq);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int at32_cpufreq_driver_init(struct cpufreq_policy *policy)
|
|
{
|
|
unsigned int frequency, rate, min_freq;
|
|
struct clk *cpuclk;
|
|
int retval, steps, i;
|
|
|
|
if (policy->cpu != 0)
|
|
return -EINVAL;
|
|
|
|
cpuclk = clk_get(NULL, "cpu");
|
|
if (IS_ERR(cpuclk)) {
|
|
pr_debug("cpufreq: could not get CPU clk\n");
|
|
retval = PTR_ERR(cpuclk);
|
|
goto out_err;
|
|
}
|
|
|
|
min_freq = (clk_round_rate(cpuclk, 1) + 500) / 1000;
|
|
frequency = (clk_round_rate(cpuclk, ~0UL) + 500) / 1000;
|
|
policy->cpuinfo.transition_latency = 0;
|
|
|
|
/*
|
|
* AVR32 CPU frequency rate scales in power of two between maximum and
|
|
* minimum, also add space for the table end marker.
|
|
*
|
|
* Further validate that the frequency is usable, and append it to the
|
|
* frequency table.
|
|
*/
|
|
steps = fls(frequency / min_freq) + 1;
|
|
freq_table = kzalloc(steps * sizeof(struct cpufreq_frequency_table),
|
|
GFP_KERNEL);
|
|
if (!freq_table) {
|
|
retval = -ENOMEM;
|
|
goto out_err_put_clk;
|
|
}
|
|
|
|
for (i = 0; i < (steps - 1); i++) {
|
|
rate = clk_round_rate(cpuclk, frequency * 1000) / 1000;
|
|
|
|
if (rate != frequency)
|
|
freq_table[i].frequency = CPUFREQ_ENTRY_INVALID;
|
|
else
|
|
freq_table[i].frequency = frequency;
|
|
|
|
frequency /= 2;
|
|
}
|
|
|
|
policy->clk = cpuclk;
|
|
freq_table[steps - 1].frequency = CPUFREQ_TABLE_END;
|
|
|
|
retval = cpufreq_table_validate_and_show(policy, freq_table);
|
|
if (!retval) {
|
|
printk("cpufreq: AT32AP CPU frequency driver\n");
|
|
return 0;
|
|
}
|
|
|
|
kfree(freq_table);
|
|
out_err_put_clk:
|
|
clk_put(cpuclk);
|
|
out_err:
|
|
return retval;
|
|
}
|
|
|
|
static struct cpufreq_driver at32_driver = {
|
|
.name = "at32ap",
|
|
.init = at32_cpufreq_driver_init,
|
|
.verify = cpufreq_generic_frequency_table_verify,
|
|
.target_index = at32_set_target,
|
|
.get = cpufreq_generic_get,
|
|
};
|
|
|
|
static int __init at32_cpufreq_init(void)
|
|
{
|
|
return cpufreq_register_driver(&at32_driver);
|
|
}
|
|
late_initcall(at32_cpufreq_init);
|