cregit-Linux how code gets into the kernel

Release 4.18 drivers/cpufreq/tegra20-cpufreq.c

Directory: drivers/cpufreq
/*
 * Copyright (C) 2010 Google, Inc.
 *
 * Author:
 *      Colin Cross <ccross@google.com>
 *      Based on arch/arm/plat-omap/cpu-omap.c, (C) 2005 Nokia Corporation
 *
 * This software is licensed under the terms of the GNU General Public
 * License version 2, as published by the Free Software Foundation, and
 * may be copied, distributed, and modified under those terms.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 */

#include <linux/clk.h>
#include <linux/cpufreq.h>
#include <linux/err.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/types.h>


static struct cpufreq_frequency_table freq_table[] = {
	{ .frequency = 216000 },
	{ .frequency = 312000 },
	{ .frequency = 456000 },
	{ .frequency = 608000 },
	{ .frequency = 760000 },
	{ .frequency = 816000 },
	{ .frequency = 912000 },
	{ .frequency = 1000000 },
	{ .frequency = CPUFREQ_TABLE_END },
};


struct tegra20_cpufreq {
	
struct device *dev;
	
struct cpufreq_driver driver;
	
struct clk *cpu_clk;
	
struct clk *pll_x_clk;
	
struct clk *pll_p_clk;
	
bool pll_x_prepared;
};


static unsigned int tegra_get_intermediate(struct cpufreq_policy *policy, unsigned int index) { struct tegra20_cpufreq *cpufreq = cpufreq_get_driver_data(); unsigned int ifreq = clk_get_rate(cpufreq->pll_p_clk) / 1000; /* * Don't switch to intermediate freq if: * - we are already at it, i.e. policy->cur == ifreq * - index corresponds to ifreq */ if (freq_table[index].frequency == ifreq || policy->cur == ifreq) return 0; return ifreq; }

Contributors

PersonTokensPropCommitsCommitProp
Viresh Kumar3760.66%133.33%
Stephen Warren1422.95%133.33%
Dmitry Osipenko1016.39%133.33%
Total61100.00%3100.00%


static int tegra_target_intermediate(struct cpufreq_policy *policy, unsigned int index) { struct tegra20_cpufreq *cpufreq = cpufreq_get_driver_data(); int ret; /* * Take an extra reference to the main pll so it doesn't turn * off when we move the cpu off of it as enabling it again while we * switch to it from tegra_target() would take additional time. * * When target-freq is equal to intermediate freq we don't need to * switch to an intermediate freq and so this routine isn't called. * Also, we wouldn't be using pll_x anymore and must not take extra * reference to it, as it can be disabled now to save some power. */ clk_prepare_enable(cpufreq->pll_x_clk); ret = clk_set_parent(cpufreq->cpu_clk, cpufreq->pll_p_clk); if (ret) clk_disable_unprepare(cpufreq->pll_x_clk); else cpufreq->pll_x_prepared = true; return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Stephen Warren2638.24%125.00%
Viresh Kumar2435.29%250.00%
Dmitry Osipenko1826.47%125.00%
Total68100.00%4100.00%


static int tegra_target(struct cpufreq_policy *policy, unsigned int index) { struct tegra20_cpufreq *cpufreq = cpufreq_get_driver_data(); unsigned long rate = freq_table[index].frequency; unsigned int ifreq = clk_get_rate(cpufreq->pll_p_clk) / 1000; int ret; /* * target freq == pll_p, don't need to take extra reference to pll_x_clk * as it isn't used anymore. */ if (rate == ifreq) return clk_set_parent(cpufreq->cpu_clk, cpufreq->pll_p_clk); ret = clk_set_rate(cpufreq->pll_x_clk, rate * 1000); /* Restore to earlier frequency on error, i.e. pll_x */ if (ret) dev_err(cpufreq->dev, "Failed to change pll_x to %lu\n", rate); ret = clk_set_parent(cpufreq->cpu_clk, cpufreq->pll_x_clk); /* This shouldn't fail while changing or restoring */ WARN_ON(ret); /* * Drop count to pll_x clock only if we switched to intermediate freq * earlier while transitioning to a target frequency. */ if (cpufreq->pll_x_prepared) { clk_disable_unprepare(cpufreq->pll_x_clk); cpufreq->pll_x_prepared = false; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Viresh Kumar8559.86%562.50%
Dmitry Osipenko3121.83%112.50%
Colin Cross2618.31%225.00%
Total142100.00%8100.00%


static int tegra_cpu_init(struct cpufreq_policy *policy) { struct tegra20_cpufreq *cpufreq = cpufreq_get_driver_data(); int ret; clk_prepare_enable(cpufreq->cpu_clk); /* FIXME: what's the actual transition time? */ ret = cpufreq_generic_init(policy, freq_table, 300 * 1000); if (ret) { clk_disable_unprepare(cpufreq->cpu_clk); return ret; } policy->clk = cpufreq->cpu_clk; policy->suspend_freq = freq_table[0].frequency; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Viresh Kumar3644.44%333.33%
Colin Cross3037.04%444.44%
Dmitry Osipenko1417.28%111.11%
Prashant Gaikwad11.23%111.11%
Total81100.00%9100.00%


static int tegra_cpu_exit(struct cpufreq_policy *policy) { struct tegra20_cpufreq *cpufreq = cpufreq_get_driver_data(); clk_disable_unprepare(cpufreq->cpu_clk); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Colin Cross1551.72%240.00%
Dmitry Osipenko1034.48%120.00%
Viresh Kumar310.34%120.00%
Prashant Gaikwad13.45%120.00%
Total29100.00%5100.00%


static int tegra20_cpufreq_probe(struct platform_device *pdev) { struct tegra20_cpufreq *cpufreq; int err; cpufreq = devm_kzalloc(&pdev->dev, sizeof(*cpufreq), GFP_KERNEL); if (!cpufreq) return -ENOMEM; cpufreq->cpu_clk = clk_get_sys(NULL, "cclk"); if (IS_ERR(cpufreq->cpu_clk)) return PTR_ERR(cpufreq->cpu_clk); cpufreq->pll_x_clk = clk_get_sys(NULL, "pll_x"); if (IS_ERR(cpufreq->pll_x_clk)) { err = PTR_ERR(cpufreq->pll_x_clk); goto put_cpu; } cpufreq->pll_p_clk = clk_get_sys(NULL, "pll_p"); if (IS_ERR(cpufreq->pll_p_clk)) { err = PTR_ERR(cpufreq->pll_p_clk); goto put_pll_x; } cpufreq->dev = &pdev->dev; cpufreq->driver.get = cpufreq_generic_get; cpufreq->driver.attr = cpufreq_generic_attr; cpufreq->driver.init = tegra_cpu_init; cpufreq->driver.exit = tegra_cpu_exit; cpufreq->driver.flags = CPUFREQ_NEED_INITIAL_FREQ_CHECK; cpufreq->driver.verify = cpufreq_generic_frequency_table_verify; cpufreq->driver.suspend = cpufreq_generic_suspend; cpufreq->driver.driver_data = cpufreq; cpufreq->driver.target_index = tegra_target; cpufreq->driver.get_intermediate = tegra_get_intermediate; cpufreq->driver.target_intermediate = tegra_target_intermediate; snprintf(cpufreq->driver.name, CPUFREQ_NAME_LEN, "tegra"); err = cpufreq_register_driver(&cpufreq->driver); if (err) goto put_pll_p; platform_set_drvdata(pdev, cpufreq); return 0; put_pll_p: clk_put(cpufreq->pll_p_clk); put_pll_x: clk_put(cpufreq->pll_x_clk); put_cpu: clk_put(cpufreq->cpu_clk); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Dmitry Osipenko23676.38%350.00%
Richard Zhao6019.42%116.67%
Colin Cross113.56%116.67%
Joseph Lo20.65%116.67%
Total309100.00%6100.00%


static int tegra20_cpufreq_remove(struct platform_device *pdev) { struct tegra20_cpufreq *cpufreq = platform_get_drvdata(pdev); cpufreq_unregister_driver(&cpufreq->driver); clk_put(cpufreq->pll_p_clk); clk_put(cpufreq->pll_x_clk); clk_put(cpufreq->cpu_clk); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Dmitry Osipenko3871.70%250.00%
Colin Cross1018.87%125.00%
Richard Zhao59.43%125.00%
Total53100.00%4100.00%

static struct platform_driver tegra20_cpufreq_driver = { .probe = tegra20_cpufreq_probe, .remove = tegra20_cpufreq_remove, .driver = { .name = "tegra20-cpufreq", }, }; module_platform_driver(tegra20_cpufreq_driver); MODULE_ALIAS("platform:tegra20-cpufreq"); MODULE_AUTHOR("Colin Cross <ccross@android.com>"); MODULE_DESCRIPTION("NVIDIA Tegra20 cpufreq driver"); MODULE_LICENSE("GPL");

Overall Contributors

PersonTokensPropCommitsCommitProp
Dmitry Osipenko41545.11%417.39%
Viresh Kumar21623.48%1147.83%
Colin Cross17018.48%417.39%
Richard Zhao657.07%14.35%
Stephen Warren505.43%14.35%
Prashant Gaikwad20.22%14.35%
Joseph Lo20.22%14.35%
Total920100.00%23100.00%
Directory: drivers/cpufreq
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.