Message ID | 1430757460-9478-11-git-send-email-rklein@nvidia.com |
---|---|
State | Superseded, archived |
Headers | show |
On Mon, May 4, 2015 at 9:37 AM, Rhyland Klein <rklein@nvidia.com> wrote: > For Tegra210, the logic to calculate out-of-table rates is different > from previous generations. Add callbacks that can be overridden to > allow for different ways of calculating rates. Default to > _cal_rate when not specified. You should mention that you're adding a new flag and new members that may override mdiv as well. > > Based on original work by Aleksandr Frid <afrid@nvidia.com> > > Signed-off-by: Rhyland Klein <rklein@nvidia.com> > --- > diff --git a/drivers/clk/tegra/clk.h b/drivers/clk/tegra/clk.h > index 3f92f1ad3961..b63ef31a2d7a 100644 > --- a/drivers/clk/tegra/clk.h > +++ b/drivers/clk/tegra/clk.h > @@ -199,6 +199,8 @@ struct div_nmp { > * base register. > * TEGRA_PLL_BYPASS - PLL has bypass bit > * TEGRA_PLL_HAS_LOCK_ENABLE - PLL has bit to enable lock monitoring > + * TEGRA_MDIV_NEW - Switch to new method for calculating fixed mdiv > + * it may be more accurate (especially if SDM present) > */ > struct tegra_clk_pll_params { > unsigned long input_min; > @@ -235,7 +237,13 @@ struct tegra_clk_pll_params { > struct div_nmp *div_nmp; > struct tegra_clk_pll_freq_table *freq_table; > unsigned long fixed_rate; > + bool vco_out; vco_out is unused and unrelated to this change? > + u16 mdiv_default; > + u32 (*round_p_to_pdiv)(u32 p, u32 *pdiv); > void (*set_gain)(struct tegra_clk_pll_freq_table *cfg); > + int (*calc_rate)(struct clk_hw *hw, > + struct tegra_clk_pll_freq_table *cfg, > + unsigned long rate, unsigned long parent_rate); Add kerneldoc for new members.
On 5/4/2015 7:34 PM, Benson Leung wrote: > On Mon, May 4, 2015 at 9:37 AM, Rhyland Klein <rklein@nvidia.com> wrote: >> For Tegra210, the logic to calculate out-of-table rates is different >> from previous generations. Add callbacks that can be overridden to >> allow for different ways of calculating rates. Default to >> _cal_rate when not specified. > > You should mention that you're adding a new flag and new members that > may override mdiv as well. Yes, will add that. > >> >> Based on original work by Aleksandr Frid <afrid@nvidia.com> >> >> Signed-off-by: Rhyland Klein <rklein@nvidia.com> >> --- >> diff --git a/drivers/clk/tegra/clk.h b/drivers/clk/tegra/clk.h >> index 3f92f1ad3961..b63ef31a2d7a 100644 >> --- a/drivers/clk/tegra/clk.h >> +++ b/drivers/clk/tegra/clk.h >> @@ -199,6 +199,8 @@ struct div_nmp { >> * base register. >> * TEGRA_PLL_BYPASS - PLL has bypass bit >> * TEGRA_PLL_HAS_LOCK_ENABLE - PLL has bit to enable lock monitoring >> + * TEGRA_MDIV_NEW - Switch to new method for calculating fixed mdiv >> + * it may be more accurate (especially if SDM present) >> */ >> struct tegra_clk_pll_params { >> unsigned long input_min; >> @@ -235,7 +237,13 @@ struct tegra_clk_pll_params { >> struct div_nmp *div_nmp; >> struct tegra_clk_pll_freq_table *freq_table; >> unsigned long fixed_rate; >> + bool vco_out; > > vco_out is unused and unrelated to this change? vco_out is used to mark which PLL's have a vco_out, and is going to be used in the new out-of-table rate calculation in tegra210, so I will move adding this member until that patch. > >> + u16 mdiv_default; >> + u32 (*round_p_to_pdiv)(u32 p, u32 *pdiv); >> void (*set_gain)(struct tegra_clk_pll_freq_table *cfg); >> + int (*calc_rate)(struct clk_hw *hw, >> + struct tegra_clk_pll_freq_table *cfg, >> + unsigned long rate, unsigned long parent_rate); > > Add kerneldoc for new members. Will do, thanks! -rhyland
diff --git a/drivers/clk/tegra/clk-pll.c b/drivers/clk/tegra/clk-pll.c index 2a559408cee1..0b9cbe12a3eb 100644 --- a/drivers/clk/tegra/clk-pll.c +++ b/drivers/clk/tegra/clk-pll.c @@ -687,7 +687,7 @@ static int clk_pll_set_rate(struct clk_hw *hw, unsigned long rate, } if (_get_table_rate(hw, &cfg, rate, parent_rate) && - _calc_rate(hw, &cfg, rate, parent_rate)) { + pll->params->calc_rate(hw, &cfg, rate, parent_rate)) { pr_err("%s: Failed to set %s rate %lu\n", __func__, __clk_get_name(hw->clk), rate); WARN_ON(1); @@ -722,7 +722,7 @@ static long clk_pll_round_rate(struct clk_hw *hw, unsigned long rate, return __clk_get_rate(hw->clk); if (_get_table_rate(hw, &cfg, rate, *prate) && - _calc_rate(hw, &cfg, rate, *prate)) + pll->params->calc_rate(hw, &cfg, rate, *prate)) return -EINVAL; return cfg.output_rate; @@ -912,12 +912,28 @@ const struct clk_ops tegra_clk_plle_ops = { static int _pll_fixed_mdiv(struct tegra_clk_pll_params *pll_params, unsigned long parent_rate) { + u16 mdiv = parent_rate / pll_params->cf_min; + + if (pll_params->flags & TEGRA_MDIV_NEW) + return (!pll_params->mdiv_default ? mdiv : + min(mdiv, pll_params->mdiv_default)); + + if (pll_params->mdiv_default) + return pll_params->mdiv_default; + if (parent_rate > pll_params->cf_max) return 2; else return 1; } +u16 tegra_pll_get_fixed_mdiv(struct clk_hw *hw, unsigned long input_rate) +{ + struct tegra_clk_pll *pll = to_clk_pll(hw); + + return (u16)_pll_fixed_mdiv(pll->params, input_rate); +} + static unsigned long _clip_vco_min(unsigned long vco_min, unsigned long parent_rate) { @@ -1492,6 +1508,10 @@ static struct clk *_tegra_clk_register_pll(struct tegra_clk_pll *pll, init.parent_names = (parent_name ? &parent_name : NULL); init.num_parents = (parent_name ? 1 : 0); + /* Default to _calc_rate if unspecified */ + if (!pll->params->calc_rate) + pll->params->calc_rate = _calc_rate; + /* Data in .init is copied by clk_register(), so stack variable OK */ pll->hw.init = &init; diff --git a/drivers/clk/tegra/clk.h b/drivers/clk/tegra/clk.h index 3f92f1ad3961..b63ef31a2d7a 100644 --- a/drivers/clk/tegra/clk.h +++ b/drivers/clk/tegra/clk.h @@ -199,6 +199,8 @@ struct div_nmp { * base register. * TEGRA_PLL_BYPASS - PLL has bypass bit * TEGRA_PLL_HAS_LOCK_ENABLE - PLL has bit to enable lock monitoring + * TEGRA_MDIV_NEW - Switch to new method for calculating fixed mdiv + * it may be more accurate (especially if SDM present) */ struct tegra_clk_pll_params { unsigned long input_min; @@ -235,7 +237,13 @@ struct tegra_clk_pll_params { struct div_nmp *div_nmp; struct tegra_clk_pll_freq_table *freq_table; unsigned long fixed_rate; + bool vco_out; + u16 mdiv_default; + u32 (*round_p_to_pdiv)(u32 p, u32 *pdiv); void (*set_gain)(struct tegra_clk_pll_freq_table *cfg); + int (*calc_rate)(struct clk_hw *hw, + struct tegra_clk_pll_freq_table *cfg, + unsigned long rate, unsigned long parent_rate); }; #define TEGRA_PLL_USE_LOCK BIT(0) @@ -249,6 +257,7 @@ struct tegra_clk_pll_params { #define TEGRA_PLL_LOCK_MISC BIT(8) #define TEGRA_PLL_BYPASS BIT(9) #define TEGRA_PLL_HAS_LOCK_ENABLE BIT(10) +#define TEGRA_MDIV_NEW BIT(11) /** * struct tegra_clk_pll - Tegra PLL clock @@ -659,5 +668,6 @@ void tegra114_clock_deassert_dfll_dvco_reset(void); typedef void (*tegra_clk_apply_init_table_func)(void); extern tegra_clk_apply_init_table_func tegra_clk_apply_init_table; int tegra_pll_wait_for_lock(struct tegra_clk_pll *pll); +u16 tegra_pll_get_fixed_mdiv(struct clk_hw *hw, unsigned long input_rate); #endif /* TEGRA_CLK_H */
For Tegra210, the logic to calculate out-of-table rates is different from previous generations. Add callbacks that can be overridden to allow for different ways of calculating rates. Default to _cal_rate when not specified. Based on original work by Aleksandr Frid <afrid@nvidia.com> Signed-off-by: Rhyland Klein <rklein@nvidia.com> --- drivers/clk/tegra/clk-pll.c | 24 ++++++++++++++++++++++-- drivers/clk/tegra/clk.h | 10 ++++++++++ 2 files changed, 32 insertions(+), 2 deletions(-)