clk = devm_clk_get(dev, NULL); if (IS_ERR(clk)) {
dev_err(dev, "failed to get clk: %pe\n", clk); return PTR_ERR(clk);
}
rate = clk_get_rate(clk); if (!rate) {
dev_err(dev, "failed to get clk rate\n"); return -EINVAL;
}
/* * Runtime PM of the device must be enabled in order to set up * GENPD's performance properly because GENPD core checks whether * device is suspended and this check doesn't work while RPM is * disabled. This makes sure the OPP vote below gets cached in * GENPD for the device. Instead, the vote is done the next time * the device gets runtime resumed.
*/
rpm_enabled = pm_runtime_enabled(dev); if (!rpm_enabled)
pm_runtime_enable(dev);
/* should never happen in practice */ if (!pm_runtime_enabled(dev)) {
dev_WARN(dev, "failed to enable runtime PM\n");
pm_runtime_disable(dev); return -EINVAL;
}
/* first dummy rate-setting initializes voltage vote */
err = dev_pm_opp_set_rate(dev, rate);
if (!rpm_enabled)
pm_runtime_disable(dev);
if (err) {
dev_err(dev, "failed to initialize OPP clock: %d\n", err); return err;
}
return 0;
}
/** * devm_tegra_core_dev_init_opp_table() - initialize OPP table * @dev: device for which OPP table is initialized * @params: pointer to the OPP table configuration * * This function will initialize OPP table and sync OPP state of a Tegra SoC * core device. * * Return: 0 on success or errorno.
*/ int devm_tegra_core_dev_init_opp_table(struct device *dev, struct tegra_core_opp_params *params)
{
u32 hw_version; int err; /* * The clk's connection id to set is NULL and this is a NULL terminated * array, hence two NULL entries.
*/ constchar *clk_names[] = { NULL, NULL }; struct dev_pm_opp_config config = { /* * For some devices we don't have any OPP table in the DT, and * in order to use the same code path for all the devices, we * create a dummy OPP table for them via this. The dummy OPP * table is only capable of doing clk_set_rate() on invocation * of dev_pm_opp_set_rate() and doesn't provide any other * functionality.
*/
.clk_names = clk_names,
};
err = devm_pm_opp_set_config(dev, &config); if (err) {
dev_err(dev, "failed to set OPP config: %d\n", err); return err;
}
/* * Tegra114+ doesn't support OPP yet, return early for non tegra20/30 * case.
*/ if (!config.supported_hw) return -ENODEV;
/* * Older device-trees have an empty OPP table, we will get * -ENODEV from devm_pm_opp_of_add_table() in this case.
*/
err = devm_pm_opp_of_add_table(dev); if (err) { if (err != -ENODEV)
dev_err(dev, "failed to add OPP table: %d\n", err);
return err;
}
if (params->init_state) {
err = tegra_core_dev_init_opp_state(dev); if (err) return err;
}
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.