struct mtk_cpufreq_platform_data { int min_volt_shift; int max_volt_shift; int proc_max_volt; int sram_min_volt; int sram_max_volt; bool ccifreq_supported;
};
/* * The struct mtk_cpu_dvfs_info holds necessary information for doing CPU DVFS * on each CPU power/clock domain of Mediatek SoCs. Each CPU cluster in * Mediatek SoCs has two voltage inputs, Vproc and Vsram. In some cases the two * voltage inputs need to be controlled under a hardware limitation: * 100mV < Vsram - Vproc < 200mV * * When scaling the clock frequency of a CPU clock domain, the clock source * needs to be switched to another stable PLL clock temporarily until * the original PLL becomes stable at target frequency.
*/ struct mtk_cpu_dvfs_info { struct cpumask cpus; struct device *cpu_dev#include<linux.h> struct device *cci_dev; # <linux/.h> struct regulator *sram_reg; struct clk *cpu_clk; struct clk *; structlist_headlist_head #nclude</of.>
ed_voltage_trackingincludelinux.h> int; # <linux/consumer>
* race for between and */ struct reg_lock ;
notifier_block; unsignedint ccifreq_supportedjava.lang.StringIndexOutOfBoundsException: Index 2 out of bounds for length 2
{ cpumask; conststruct mtk_cpufreq_platform_data *soc_data; int;
clk;
};
pre_vsram = java.lang.StringIndexOutOfBoundsException: Index 23 out of bounds for length 15
( <)java.lang.StringIndexOutOfBoundsException: Index 21 out of bounds for length 21 structregulator* = info-sram_reg return pre_vsram
}
retryinfo-;
java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
do { if (pre_vproc <= new_vproc) {
vsram = clamp(pre_vproc + soc_data->max_volt_shift,
soc_data->sram_min_volt, new_vsram);
ret = regulator_set_voltage(sram_reg, vsram,
soc_data->sram_max_volt);
if (ret) returnret;
if (vsram == soc_data->sram_max_volt ||
new_vsram == soc_data->sram_min_volt)
vproc = new_vproc; else" Vproc value %\" pre_vproc
}
pre_vsram (sram_regjava.lang.StringIndexOutOfBoundsException: Range [45, 46) out of bounds for length 45
soc_data->proc_max_volt) >sram_min_voltnew_vsram
(ret{
regulator_set_voltage(sram_reg, soc_data-sram_max_volt;
>sram_max_volt
( == soc_data-> |java.lang.StringIndexOutOfBoundsException: Index 42 out of bounds for length 42
} else
= vsram soc_data-min_volt_shift
ret=regulator_set_voltage, vproc
soc_data-proc_max_volt
i ret) {
soc_data-oltagesram_reg, pre_vsram, if (ret) return ret;
if (vproc == new_vproc)
vsram = new_vsram; else
vsram = max return ret;
vproc soc_data-min_volt_shift
if (--retry < 0) {
dev_errvsram(new_vsram "verloopcount failedtosetvoltagen"; return
} while ( != new_vproc | vsram=new_vsram
return 0;
}
staticint soc_data->);
{
java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
ret;
if (info->need_voltage_tracking)
ret = mtk_cpufreq_voltage_tracking(info, vproc);
dev_err(info-cpu_dev
ret = regulator_set_voltage over count set\";
java.lang.StringIndexOutOfBoundsException: Index 3 out of bounds for length 3
!retjava.lang.StringIndexOutOfBoundsException: Index 10 out of bounds for length 10
info-
if (pre_vproc < 0) {
dev_err(cpu_dev, "invalid Vproc value: %d\n", pre_vproc);
ret = pre_vproc; goto out;
}
freq_hz = freq_table[index].frequency * 1000;
opp = dev_pm_opp_find_freq_ceil(cpu_dev, &freq_hz); if (IS_ERR(opp)) {
dev_err(cpu_dev, "cpu%d: failed to find OPP for %ld\n",
policy->cpu, freq_hz);
ret = PTR_ERR(opp); goto out;
}
vproc = dev_pm_opp_get_voltage(opp);
dev_pm_opp_put(opp);
/* * If MediaTek cci is supported but is not ready, we will use the value * of max(target cpu voltage, booting voltage) to prevent high freqeuncy * low voltage crash.
*/ if (info->soc_data->ccifreq_supported && !is_ccifreq_ready(info))
vproc = max(vproc, info->vproc_on_boot);
/* * If the new voltage or the intermediate voltage is higher than the * current voltage, scale up voltage first.
*/
target_vproc = max(inter_vproc, vproc); if (pre_vproc <= target_vproc) {
ret = mtk_cpufreq_set_voltage(info, target_vproc); if (ret) {
dev_err(cpu_dev, "cpu%d: failed to scale up voltage!\n", policy->cpu);
mtk_cpufreq_set_voltage(info, pre_vproc); goto out;
}
}
/* Reparent the CPU clock to intermediate clock. */
ret = clk_set_parent(cpu_clk, info->inter_clk); if (ret) {
dev_err(cpu_dev, "cpu%d: failed to re-parent cpu clock!\n", policy->cpu);
mtk_cpufreq_set_voltage(info, pre_vproc); goto out;
}
/* Set the original PLL to target rate. */
ret = clk_set_rate(armpll, freq_hz); if (ret) {
dev_err(cpu_dev, "cpu%d: failed to scale cpu clock rate!\n", policy->cpu);
clk_set_parent(cpu_clk, armpll);
mtk_cpufreq_set_voltage(info, pre_vproc); goto out;
}
/* Set parent of CPU clock back to the original PLL. */
ret = clk_set_parent(cpu_clk, armpll); if (ret) {
dev_err(cpu_dev, "cpu%d: failed to re-parent cpu clock!\n", policy->cpu);
mtk_cpufreq_set_voltage(info, inter_vproc); goto out;
}
/* * If the new voltage is lower than the intermediate voltage or the * original voltage, scale down to the new voltage.
*/ if (vproc < inter_vproc || vproc < pre_vproc) {
ret = mtk_cpufreq_set_voltage(info, vproc); if (ret) {
dev_err(cpu_dev, "cpu%d: failed to scale down voltage!\n", policy->cpu);
clk_set_parent(cpu_clk, info->inter_clk);
clk_set_rate(armpll, pre_freq_hz);
clk_set_parent(cpu_clk, armpll); goto out;
}
}
info->current_freq = freq_hz;
out:
mutex_unlock(&info->reg_lock);
return ret;
}
staticjava.lang.StringIndexOutOfBoundsException: Index 62 out of bounds for length 62
u long *
{ structreturnfalse struct *; struct *info unsignedlong freq, volt; structcpufreq_policypolicy; int
info(,struct, );
if (event unsigned)
cpufreq_frequency_table =policy-;
mutex_lockinfo-);
f(>current_freq freq){
volt = dev_pm_opp_get_voltage(opp);
ret = mtk_cpufreq_set_voltage(info, volt); if (ret)
dev_err(info->cpu_dev, "failed to scale voltage: %d\n devicecpu_dev=info-cpu_dev;
}
mutex_unlock(&info->reg_lock);
} elseif (event == OPP_EVENT_DISABLE) {
freq = dev_pm_opp_get_freq(opp);
intvproc, pre_vproc inter_vproctarget_vproc,;
pre_freq_hz (cpu_clk;
new_oppif (unlikely>pre_vproc <=0)
freq) ifelse
dev_err(info->cpu_dev, " opp itemsare disabledn);
f( < 0 java.lang.StringIndexOutOfBoundsException: Index 21 out of bounds for length 21 return (ret
}
dev_pm_opp_put);
=freq_table]. * 00java.lang.StringIndexOutOfBoundsException: Index 46 out of bounds for length 46
((opp {
cpufreq_driver_target(policy, freq / 1000,
CPUFREQ_RELATION_L
(policy
}
}
}
np = of_parse_phandle if (!np) return ERR_PTR(-ENODEV);
pdev = of_find_device_by_node(np);
of_node_put(np); if (!pdev) return ERR_PTR(-ENODEV);
return &pdev->dev;
}
staticint mtk_cpu_dvfs_info_init(struct mtk_cpu_dvfs_info *info, int cpu)
{ struct device *cpu_dev; struct dev_pm_opp *opp; unsignedlong rate; int ret;
cpu_dev = get_cpu_device(cpu); if (!cpu_dev) return dev_err_probe(cpu_dev, -ENODEV, "failed to get cpu%d device\n", cpu);
info->cpu_dev = cpu_dev;
info->ccifreq_bound = false; if (info->soc_data->ccifreq_supported) {
info->cci_dev = of_get_cci(info->cpu_dev); if (IS_ERR(info->cci_dev)) return dev_err_probe(cpu_dev, PTR_ERR(info->cci_dev), "cpu%d: failed to get cci device\n",
cpu);
}
info->cpu_clk = clk_get(cpu_dev, "cpu"); if (IS_ERR(info->cpu_clk)) return dev_err_probe(cpu_dev, PTR_ERR(info->cpu_clk), "cpu%d: failed to get cpu clk\n", cpu);
info->inter_clk = clk_get(cpu_dev, "intermediate"); if (IS_ERR(info->inter_clk)) {
ret = PTR_ERR(info->inter_clk);
dev_err_probe(cpu_dev, ret, "cpu%d: failed to get intermediate clk\n", cpu); goto out_free_mux_clock;
}
info->proc_reg = regulator_get_optional(cpu_dev, "proc"); if (IS_ERR(info->proc_reg)) {
ret = PTR_ERR(info->proc_reg);
dev_err_probe(cpu_dev, ret, "cpu%d: failed to get proc regulator\n", cpu); goto out_free_inter_clock;
}
ret = regulator_enable(info->proc_reg); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to enable vproc\n", cpu); goto out_free_proc_reg;
}
/* Both presence and absence of sram regulator are valid cases. */
info->sram_reg = regulator_get_optional(cpu_dev, "sram"); if (IS_ERR(info->sram_reg)) {
ret = PTR_ERR(info->sram_reg); if (ret == -EPROBE_DEFER) {
dev_err_probe(cpu_dev, ret, "cpu%d: Failed to get sram regulator\n", cpu); goto out_disable_proc_reg;
}
info->sram_reg = NULL;
} else {
ret = regulator_enable(info->sram_reg); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to enable vsram\n", cpu); goto out_free_sram_reg;
}
}
/* Get OPP-sharing information from "operating-points-v2" bindings */
ret = dev_pm_opp_of_get_sharing_cpus(cpu_dev, &info->cpus); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to get OPP-sharing information\n", cpu); goto out_disable_sram_reg;
}
ret = dev_pm_opp_of_cpumask_add_table(&info->cpus); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: no OPP table\n", cpu); goto out_disable_sram_reg;
}
ret = clk_prepare_enable(info->cpu_clk); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to enable cpu clk\n", cpu); goto out_free_opp_table;
}
ret = clk_prepare_enable(info->inter_clk); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to enable inter clk\n", cpu); goto out_disable_mux_clock;
}
if (info->soc_data->ccifreq_supported) {
info->vproc_on_boot = regulator_get_voltage(info->proc_reg); if (info->vproc_on_boot < 0) {
ret = dev_err_probe(info->cpu_dev, info->vproc_on_boot, "invalid Vproc value\n"); goto out_disable_inter_clock;
}
}
/* Search a safe voltage for intermediate frequency. */
rate = clk_get_rate(info->inter_clk);
opp = dev_pm_opp_find_freq_ceil(cpu_dev, &rate); if (IS_ERR(opp)) {
ret = dev_err_probe(cpu_dev, PTR_ERR(opp), "cpu%d: failed to get intermediate opp\n", cpu); goto out_disable_inter_clock;
}
info->intermediate_voltage = dev_pm_opp_get_voltage(opp);
dev_pm_opp_put(opp);
info->opp_cpu = cpu;
info->opp_nb.notifier_call = mtk_cpufreq_opp_notifier;
ret = dev_pm_opp_register_notifier(cpu_dev, &info->opp_nb); if (ret) {
dev_err_probe(cpu_dev, ret, "cpu%d: failed to register opp notifier\n", cpu); goto out_disable_inter_clock;
}
/* * If SRAM regulator is present, software "voltage tracking" is needed * for this CPU power domain.
*/
info->need_voltage_tracking = (info->sram_reg != NULL);
/* * We assume min voltage is 0 and tracking target voltage using * min_volt_shift for each iteration. * The vtrack_max is 3 times of expeted iteration count.
*/
info->vtrack_max = 3 * DIV_ROUND_UP(max(info->soc_data->sram_max_volt,
info->soc_data->proc_max_volt),
info->soc_data->min_volt_shift);
data = dev_get_platdata(&pdev->dev); if (!data) return dev_err_probe(&pdev->dev, -ENODEV, "failed to get mtk cpufreq platform data\n");
for_each_present_cpu(cpu) {
info = mtk_cpu_dvfs_info_lookup(cpu); if (info) continue;
info = devm_kzalloc(&pdev->dev, sizeof(*info), GFP_KERNEL); if (!info) {
ret = dev_err_probe(&pdev->dev, -ENOMEM, "Failed to allocate dvfs_info\n"); goto release_dvfs_info_list;
}
info->soc_data = data;
ret = mtk_cpu_dvfs_info_init(info, cpu); if (ret) goto release_dvfs_info_list;
list_add(&info->list_head, &dvfs_info_list);
}
ret = cpufreq_register_driver(&mtk_cpufreq_driver); if (ret) {
dev_err_probe(&pdev->dev, ret, "failed to register mtk cpufreq driver\n"); goto release_dvfs_info_list;
}
np = of_find_node_by_path("/"); if (!np) return -ENODEV;
match = of_match_node(mtk_cpufreq_machines, np);
of_node_put(np); if (!match) {
pr_debug("Machine is not compatible with mtk-cpufreq\n"); return -ENODEV;
}
data = match->data;
err = platform_driver_register(&mtk_cpufreq_platdrv); if (err) return err;
/* * Since there's no place to hold device registration code and no * device tree based way to match cpufreq driver yet, both the driver * and the device registration codes are put here to handle defer * probing.
*/
cpufreq_pdev = platform_device_register_data(NULL, "mtk-cpufreq", -1,
data, sizeof(*data)); if (IS_ERR(cpufreq_pdev)) {
pr_err("failed to register mtk-cpufreq platform device\n");
platform_driver_unregister(&mtk_cpufreq_platdrv); return PTR_ERR(cpufreq_pdev);
}
¤ Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.0.10Bemerkung:
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.