Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-05-11 Thread Rhyland Klein
On 5/11/2015 7:50 AM, Peter De Schrijver wrote:
> On Thu, Apr 30, 2015 at 11:31:22AM -0400, Rhyland Klein wrote:
>> 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 

 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?
> 
> Ok. I see... Should we print a warning (pr_warn()) the bootloader isn't
> initializing the hw correctly if the second pass needs to write the default
> values?

All the set default routines use the inline function
"_pll_misc_chk_default" (used to be a MACRO). Inside, it compares
register values vs expected defaults and warns:

if (boot_val != default_val) {
pr_warn("boot misc%d 0x%x: expected 0x%x\n",
misc_num, boot_val, default_val);
pr_warn(" (comparison mask = 0x%x)\n", mask);
params->defaults_set = false;
}

so this is already done. I suppose the only other place we could add a
warning is if the first past set-defaults can't be fully run since the
clock is on, that might make things a little clearer, so I guess I'll go
ahead and add that.

-rhyland

> 
> Thanks,
> 
> Peter.
> 


-- 
nvpublic
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-05-11 Thread Peter De Schrijver
On Thu, Apr 30, 2015 at 11:31:22AM -0400, Rhyland Klein wrote:
> 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 
> >>
> >> 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?

Ok. I see... Should we print a warning (pr_warn()) the bootloader isn't
initializing the hw correctly if the second pass needs to write the default
values?

Thanks,

Peter.
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-05-11 Thread Peter De Schrijver
On Thu, Apr 30, 2015 at 11:31:22AM -0400, Rhyland Klein wrote:
 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 bilhu...@nvidia.com
 
  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?

Ok. I see... Should we print a warning (pr_warn()) the bootloader isn't
initializing the hw correctly if the second pass needs to write the default
values?

Thanks,

Peter.
--
To unsubscribe from this list: send the line unsubscribe linux-kernel in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-05-11 Thread Rhyland Klein
On 5/11/2015 7:50 AM, Peter De Schrijver wrote:
 On Thu, Apr 30, 2015 at 11:31:22AM -0400, Rhyland Klein wrote:
 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 bilhu...@nvidia.com

 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?
 
 Ok. I see... Should we print a warning (pr_warn()) the bootloader isn't
 initializing the hw correctly if the second pass needs to write the default
 values?

All the set default routines use the inline function
_pll_misc_chk_default (used to be a MACRO). Inside, it compares
register values vs expected defaults and warns:

if (boot_val != default_val) {
pr_warn(boot misc%d 0x%x: expected 0x%x\n,
misc_num, boot_val, default_val);
pr_warn( (comparison mask = 0x%x)\n, mask);
params-defaults_set = false;
}

so this is already done. I suppose the only other place we could add a
warning is if the first past set-defaults can't be fully run since the
clock is on, that might make things a little clearer, so I guess I'll go
ahead and add that.

-rhyland

 
 Thanks,
 
 Peter.
 


-- 
nvpublic
--
To unsubscribe from this list: send the line unsubscribe linux-kernel in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-04-30 Thread Rhyland Klein
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 
>>
>> 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 
>> ---
>> 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, _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 = 
>>  
>> @@ -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)
>> -   

Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-04-30 Thread Peter De Schrijver
On Wed, Apr 29, 2015 at 01:21:46PM -0400, Rhyland Klein wrote:
> From: Bill Huang 
> 
> 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?

> Signed-off-by: Bill Huang 
> ---
> 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, _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 = 
>  
> @@ -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;
> + booldefaults_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 

Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-04-30 Thread Peter De Schrijver
On Wed, Apr 29, 2015 at 01:21:46PM -0400, Rhyland Klein wrote:
 From: Bill Huang bilhu...@nvidia.com
 
 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?

 Signed-off-by: Bill Huang bilhu...@nvidia.com
 ---
 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;
 + booldefaults_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)
 -- 
 

Re: [PATCH v2 16/19] clk: tegra: pll: Add Set_default logic

2015-04-30 Thread Rhyland Klein
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 bilhu...@nvidia.com

 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 bilhu...@nvidia.com
 ---
 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 +