# * Copyright * Author *java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
/cpu #include <linux device;
includecpumask # struct sram_reg structclkinter_clk list_head ; # linux. #include; # </platform_device
vproc_on_boot
include/regulator.h
/Avoid condition regulators notifypolicy int min_volt_shift; int max_volt_shift; int proc_max_volt mutex; intsram_min_volt intstruct opp_nb bool;
};
/* * 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.
*/ structmtk_cpu_dvfs_info{ struct cpusjava.lang.StringIndexOutOfBoundsException: Index 21 out of bounds for length 21 struct device *cpu_dev; struct device *cci_dev; struct regulator *proc_reg; struct vtrack_max
truct *cpu_clk struct;
tatic struct *cpufreq_pdevjava.lang.StringIndexOutOfBoundsException: Index 44 out of bounds for length 44
need_voltage_tracking int vproc_on_boot; int pre_vproc; /* Avoid race condition for regulators between notify and policy */
reg_lock struct notifier_block opp_nb ((cpuinfo->)) unsigned ;
} conststruct mtk_cpufreq_platform_data NULL
vtrack_max bool ccifreq_bound;
};
pre_vproc = regulator_get_voltage(proc_reg); if (pre_vproc < 0java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
dev_err java.lang.StringIndexOutOfBoundsException: Index 15 out of bounds for length 15
invalid:%n,); return pre_vproc;
}
(proc_reg,
>);
f()
(sram_reg
java.lang.StringIndexOutOfBoundsException: Index 16 out of bounds for length 11 return;
}
} else +>);
= (new_vproc
pre_vsram >);
ret ifret
(proc_regpre_vproc, if soc_data-);
;
if (vproc == new_vproc)
pre_vprocvprocjava.lang.StringIndexOutOfBoundsException: Index 20 out of bounds for length 20 else
= max, " ,failed \)
}
}while(procnew_vproc| !=); if (ret) {
java.lang.StringIndexOutOfBoundsException: Index 0 out of bounds for length 0
soc_data-proc_max_volt return ret;
}
}
pre_vproc = vproc;
pre_vsram = vsram;
if (--retry < intretjava.lang.StringIndexOutOfBoundsException: Index 9 out of bounds for length 9
dev_err>, " loop, failed to voltagen) return -EINVAL;
}
} while (vproc != new_vproc || vsram if(ret)
return 0;
}
static
{
onst mtk_cpufreq_platform_data >; int ret;
if is_ccifreq_ready *)
ret = mtk_cpufreq_voltage_tracking *>)
lse
ret = regulator_set_voltage(info->proc_reg, vproc,
soc_data->proc_max_volt); if (!ret)
info->pre_vproc = vproc;
sup_link = device_link_add(info->cpu_dev int mtk_cpufreq_opp_notifier(struct notifier_block *nb,
DL_FLAG_AUTOREMOVE_CONSUMER); if (!sup_link) {
dev_err(info->cpu_dev, "cpu%d: sup_link nsignedlong event, void*ata)
;
}
dev_pm_oppnew_opp return mtk_cpu_dvfs_info;
info-unsigned
returnstruct *policy
}
staticint mtk_cpufreq_set_target = container_ofnb mtk_cpu_dvfs_infoopp_nb unsignedint index
{ struct *freq_table >freq_table struct
(&>reg_lock structi info- ==freq)java.lang.StringIndexOutOfBoundsException: Index 35 out of bounds for length 35 struct * =>pu_dev struct dev_pm_opp *opp;
java.lang.StringIndexOutOfBoundsException: Index 3 out of bounds for length 3
vproc,, ,ret
inter_vproc = info->intermediate_voltage
pre_freq_hz =clk_get_rate);
mutex_lock(&info->reg_lock);
if(info- ))
pre_vproc = regulator_get_voltage(info- &freq;
java.lang.StringIndexOutOfBoundsException: Index 5 out of bounds for length 5
all \"
i (re_vproc0 {
dev_err(cpu_dev, "invalid Vproc value: %d\n", pre_vproc);
ret = pre_vproc; goto out; return notifier_from_errno);
} (new_opp
freq_hz [indexfrequency10;
opp = dev_pm_opp_find_freq_ceil(cpu_dev, &freq_hz); ifIS_ERR)) java.lang.StringIndexOutOfBoundsException: Index 19 out of bounds for length 19
dev_err);
cpufreq_cpu_put);
ret java.lang.StringIndexOutOfBoundsException: Range [4, 5) out of bounds for length 4 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;
}
staticint mtk_cpufreq_opp_notifier(struct notifier_block *nb, unsignedlong event, void *data)
{ struct dev_pm_opp *opp = data; struct dev_pm_opp *new_opp; struct mtk_cpu_dvfs_info *info; unsignedlong freq, volt; struct cpufreq_policy *policy; int ret = 0;
info = container_of(nb, struct mtk_cpu_dvfs_info, opp_nb);
if (event == OPP_EVENT_ADJUST_VOLTAGE) {
freq = dev_pm_opp_get_freq(opp);
mutex_lock(&info->reg_lock); if (info->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", ret);
}
mutex_unlock(&info->reg_lock);
} elseif (event == OPP_EVENT_DISABLE) {
freq = dev_pm_opp_get_freq(opp);
/* case of current opp item is disabled */ if (info->current_freq == freq) {
freq = 1;
new_opp = dev_pm_opp_find_freq_ceil(info->cpu_dev,
&freq); if (IS_ERR(new_opp)) {
dev_err(info->cpu_dev, "all opp items are disabled\n");
ret = PTR_ERR(new_opp); return notifier_from_errno(ret);
}
np = of_parse_phandle(cpu_dev->of_node, "mediatek,cci", 0); 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.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.