linux-tegra.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Rhyland Klein <rklein-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
To: Peter De Schrijver
	<pdeschrijver-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
Cc: Mike Turquette
	<mturquette-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org>,
	Stephen Warren <swarren-3lzwWm7+Weoh9ZMKESR00Q@public.gmane.org>,
	Stephen Boyd <sboyd-sgV2jX0FEOL9JmXXK+q4OQ@public.gmane.org>,
	Thierry Reding
	<thierry.reding-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>,
	Alexandre Courbot
	<gnurou-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>,
	linux-clk-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
	linux-tegra-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
	linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
	Bill Huang <bilhuang-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
Subject: Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic
Date: Thu, 30 Apr 2015 11:31:22 -0400	[thread overview]
Message-ID: <55424ACA.9010406@nvidia.com> (raw)
In-Reply-To: <20150430101225.GU3097-Rysk9IDjsxmJz7etNGeUX8VPkgjIgRvpAL8bYrjMMd8@public.gmane.org>

On 4/30/2015 6:12 AM, Peter De Schrijver wrote:
> On Wed, Apr 29, 2015 at 01:21:46PM -0400, Rhyland Klein wrote:
>> From: Bill Huang <bilhuang-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
>>
>> Add logic which (if specified for a pll) can verify that a PLL is set
>> to the proper default value and if not can set it. This can be
>> specified per PLL as each will have different default values.
>>
> 
> Why can't we just set the default values at init time?

Sorry, I did some investigation into this and wrote up a nice response
... and forgot to hit send ...

The reason this can't be run only once at init time is the following. In
reality, we want to have the defined default values written as early as
possible. Idealy, the bootloader could write these, so the kernel need
only check, see they are right, and not touch them. However, since we
can't rely on the bootloader to do so, the kernel needs the support to
be able to write these default values. At init time, some pll's will be
enabled (from bootloader) and because they are enabled (and the rest of
the clk framework isn't done being setup yet) we can't disable them to
write the full register values. Therefore, the set_defaults logic uses a
2-pass system.

first pass: Try to set defaults at init/registration time. If pll is
disabled, this works fine. If it is enabled, then we update a subset of
the register as a "best effort" setting of the defaults.

second pass: Should only run the first time we go through set_rate for a
pll. If the first pass already wrote default value, then it skips this
step. Otherwise it will go in, once the pll is disabled in the set_rate
path, and write the full register default.

This is required because some registers need to be reset to the default
values we have to ensure locking works correctly. Does that make sense?

-rhyland

> 
>> Signed-off-by: Bill Huang <bilhuang-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
>> ---
>> v2:
>>   - Remove MACRO for PLL_MISC_CHECK_DEFAULT as suggested, and instead
>>     the tegra210 driver will include an inline version of this function.
>>
>>  drivers/clk/tegra/clk-pll.c |   46 ++++++++++++++++++++++++++++++++-----------
>>  drivers/clk/tegra/clk.h     |    2 ++
>>  2 files changed, 37 insertions(+), 11 deletions(-)
>>
>> diff --git a/drivers/clk/tegra/clk-pll.c b/drivers/clk/tegra/clk-pll.c
>> index 00f0e621533a..28a2376cd8a7 100644
>> --- a/drivers/clk/tegra/clk-pll.c
>> +++ b/drivers/clk/tegra/clk-pll.c
>> @@ -658,15 +658,28 @@ static int _program_pll(struct clk_hw *hw, struct tegra_clk_pll_freq_table *cfg,
>>  			unsigned long rate)
>>  {
>>  	struct tegra_clk_pll *pll = to_clk_pll(hw);
>> +	struct tegra_clk_pll_freq_table old_cfg;
>>  	int state, ret = 0;
>>  
>>  	state = clk_pll_is_enabled(hw);
>>  
>> +	_get_pll_mnp(pll, &old_cfg);
>> +
>> +	if (state && pll->params->defaults_set && pll->params->dyn_ramp &&
>> +			(cfg->m == old_cfg.m) && (cfg->p == old_cfg.p)) {
>> +		ret = pll->params->dyn_ramp(pll, cfg);
>> +		if (!ret)
>> +			return 0;
>> +	}
>> +
>>  	if (state) {
>>  		pll_clk_stop_ss(pll);
>>  		_clk_pll_disable(hw);
>>  	}
>>  
>> +	if (!pll->params->defaults_set && pll->params->set_defaults)
>> +		pll->params->set_defaults(pll);
>> +
>>  	_update_pll_mnp(pll, cfg);
>>  
>>  	if (pll->params->flags & TEGRA_PLL_HAS_CPCON)
>> @@ -1526,6 +1539,9 @@ static struct clk *_tegra_clk_register_pll(struct tegra_clk_pll *pll,
>>  	if (!pll->params->calc_rate)
>>  		pll->params->calc_rate = _calc_rate;
>>  
>> +	if (pll->params->set_defaults)
>> +		pll->params->set_defaults(pll);
>> +
>>  	/* Data in .init is copied by clk_register(), so stack variable OK */
>>  	pll->hw.init = &init;
>>  
>> @@ -1644,7 +1660,6 @@ struct clk *tegra_clk_register_pllxc(const char *name, const char *parent_name,
>>  	struct tegra_clk_pll *pll;
>>  	struct clk *clk, *parent;
>>  	unsigned long parent_rate;
>> -	int err;
>>  	u32 val, val_iddq;
>>  
>>  	parent = __clk_lookup(parent_name);
>> @@ -1665,18 +1680,27 @@ struct clk *tegra_clk_register_pllxc(const char *name, const char *parent_name,
>>  		pll_params->vco_min = pll_params->adjust_vco(pll_params,
>>  							     parent_rate);
>>  
>> -	err = _setup_dynamic_ramp(pll_params, clk_base, parent_rate);
>> -	if (err)
>> -		return ERR_PTR(err);
>> +	/*
>> +	 * If the pll has a set_defaults callback, it will take care of
>> +	 * configuring dynamic ramping and setting IDDQ in that path.
>> +	 */
>> +	if (!pll_params->set_defaults) {
>> +		int err;
>>  
>> -	val = readl_relaxed(clk_base + pll_params->base_reg);
>> -	val_iddq = readl_relaxed(clk_base + pll_params->iddq_reg);
>> +		err = _setup_dynamic_ramp(pll_params, clk_base, parent_rate);
>> +		if (err)
>> +			return ERR_PTR(err);
>>  
>> -	if (val & PLL_BASE_ENABLE)
>> -		WARN_ON(val_iddq & BIT(pll_params->iddq_bit_idx));
>> -	else {
>> -		val_iddq |= BIT(pll_params->iddq_bit_idx);
>> -		writel_relaxed(val_iddq, clk_base + pll_params->iddq_reg);
>> +		val = readl_relaxed(clk_base + pll_params->base_reg);
>> +		val_iddq = readl_relaxed(clk_base + pll_params->iddq_reg);
>> +
>> +		if (val & PLL_BASE_ENABLE)
>> +			WARN_ON(val_iddq & BIT(pll_params->iddq_bit_idx));
>> +		else {
>> +			val_iddq |= BIT(pll_params->iddq_bit_idx);
>> +			writel_relaxed(val_iddq,
>> +				       clk_base + pll_params->iddq_reg);
>> +		}
>>  	}
>>  
>>  	pll = _tegra_init_pll(clk_base, pmc, pll_params, lock);
>> diff --git a/drivers/clk/tegra/clk.h b/drivers/clk/tegra/clk.h
>> index 850521d42be6..84c417576c18 100644
>> --- a/drivers/clk/tegra/clk.h
>> +++ b/drivers/clk/tegra/clk.h
>> @@ -239,6 +239,7 @@ struct tegra_clk_pll_params {
>>  	int		stepb_shift;
>>  	int		lock_delay;
>>  	int		max_p;
>> +	bool		defaults_set;
>>  	struct pdiv_map *pdiv_tohw;
>>  	struct div_nmp	*div_nmp;
>>  	struct tegra_clk_pll_freq_table	*freq_table;
>> @@ -254,6 +255,7 @@ struct tegra_clk_pll_params {
>>  				unsigned long parent_rate);
>>  	int	(*dyn_ramp)(struct tegra_clk_pll *pll,
>>  			struct tegra_clk_pll_freq_table *cfg);
>> +	void	(*set_defaults)(struct tegra_clk_pll *pll);
>>  };
>>  
>>  #define TEGRA_PLL_USE_LOCK BIT(0)
>> -- 
>> 1.7.9.5
>>


-- 
nvpublic

  parent reply	other threads:[~2015-04-30 15:31 UTC|newest]

Thread overview: 29+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2015-04-29 17:21 [PATCH v2 00/19] Tegra210 Clock Support Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 02/19] clk: tegra: periph: add new periph clks and muxes for Tegra210 Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 04/19] clk: tegra: pll: simplify clk_enable_path Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 05/19] clk: tegra: pll: update warning msg Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 06/19] clk: tegra: pll-params: change misc_reg count from 3 -> 6 Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 07/19] clk: tegra: pll: Don't unconditionally set LOCK flags Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 08/19] clk: tegra: pll: Add logic for handling SDM data Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 09/19] clk: tegra: pll: Add logic for SS Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 11/19] clk: tegra: pll: Add code to handle if resets are supported by PLL Rhyland Klein
2015-04-29 17:21 ` [PATCH v2 12/19] clk: tegra: pll: Add specialized logic for T210 Rhyland Klein
     [not found]   ` <1430328109-537-13-git-send-email-rklein-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
2015-04-29 18:27     ` Andrew Bresticker
2015-04-29 21:42       ` Rhyland Klein
     [not found] ` <1430328109-537-1-git-send-email-rklein-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
2015-04-29 17:21   ` [PATCH v2 01/19] clk: tegra: Modify tegra_audio_clk_init to accept more plls Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 03/19] clk: tegra: pll: add tegra_pll_wait_for_lock to clk header Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 10/19] clk: tegra: pll: Add logic for out-of-table rates for T210 Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 13/19] clk: tegra: pll: Add support for PLLMB " Rhyland Klein
2015-04-30 10:11     ` Peter De Schrijver
2015-04-29 17:21   ` [PATCH v2 14/19] clk: tegra: pll: Adjust vco_min if SDM present Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 15/19] clk: tegra: pll: Add dyn_ramp callback Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic Rhyland Klein
     [not found]     ` <1430328109-537-17-git-send-email-rklein-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
2015-04-30 10:12       ` Peter De Schrijver
     [not found]         ` <20150430101225.GU3097-Rysk9IDjsxmJz7etNGeUX8VPkgjIgRvpAL8bYrjMMd8@public.gmane.org>
2015-04-30 15:31           ` Rhyland Klein [this message]
     [not found]             ` <55424ACA.9010406-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
2015-05-11 11:50               ` Peter De Schrijver
     [not found]                 ` <20150511115036.GG22637-Rysk9IDjsxmJz7etNGeUX8VPkgjIgRvpAL8bYrjMMd8@public.gmane.org>
2015-05-11 15:07                   ` Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 17/19] clk: tegra: pll: Fix _pll_ramp_calc_pll logic and _calc_dynamic_ramp_rate Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 18/19] clk: tegra: Add Super Gen5 Logic Rhyland Klein
2015-04-29 17:21   ` [PATCH v2 19/19] clk: tegra210: add support for Tegra210 clocks Rhyland Klein
2015-04-30 20:43     ` Andrew Bresticker
     [not found]       ` <CAL1qeaHdjb57D8U-NBeyicV=JP2pNkyx3Xfn2RDgWivdw5jWNg-JsoAwUIsXosN+BqQ9rBEUg@public.gmane.org>
2015-04-30 20:57         ` Rhyland Klein

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=55424ACA.9010406@nvidia.com \
    --to=rklein-ddmlm1+adcrqt0dzr+alfa@public.gmane.org \
    --cc=bilhuang-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org \
    --cc=gnurou-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
    --cc=linux-clk-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
    --cc=linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
    --cc=linux-tegra-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
    --cc=mturquette-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org \
    --cc=pdeschrijver-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org \
    --cc=sboyd-sgV2jX0FEOL9JmXXK+q4OQ@public.gmane.org \
    --cc=swarren-3lzwWm7+Weoh9ZMKESR00Q@public.gmane.org \
    --cc=thierry.reding-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).