Hi

On Tue, 23 Jun 2015, Roger Quadros wrote:

> For some hwmods (e.g. DCAN on DRA7) we need the possibility to
> disable HW_AUTO for the clockdomain while the module is active.
> 
> To achieve this there needs to be a refcounting mechanism to
> indicate whether any module in the clockdomain has requested
> to disable HW_AUTO. We keep track of this in 'noidlecount'.
> 
> Hwmod code must use clkdm_hwmod_prevent_hwauto() to prevent
> HW_AUTO of the clockdomain in the future clkdm_hwmod_hwauto() calls.
> 
> It must use clkdm_hwmod_allow_hwauto() to allow HW_AUTO in
> the future clkdm_hwmod_hwauto() calls.
> 
> Hwmod code must use clkdm_hwmod_allow_hwauto() whenever it needs
> to request HW_AUTO of any clockdomain. (Typically after it has
> enabled the module).

How about just modifying clkdm_{allow,deny}_idle*() to do the 
idle-block-counting?  The default state would be to allow idle, assuming 
that the clockdomain flags support that state, and then clkdm_deny_idle*() 
would increment the idle-blocking count, clkdm_allow_idle*() would 
decrement it.  Then on the 0 -> 1 or 1 -> 0 transitions, the hardware 
would be reprogrammed appropiately.

Anyway, seems like that would avoid races with any 
clkdm_{allow,deny}_idle*() users.  

A few other comments below:


> 
> Signed-off-by: Roger Quadros <[email protected]>
> ---
>  arch/arm/mach-omap2/clockdomain.c | 71 
> +++++++++++++++++++++++++++++++++++++++
>  arch/arm/mach-omap2/clockdomain.h |  5 +++
>  2 files changed, 76 insertions(+)
> 
> diff --git a/arch/arm/mach-omap2/clockdomain.c 
> b/arch/arm/mach-omap2/clockdomain.c
> index 2da3b5e..a7190d2 100644
> --- a/arch/arm/mach-omap2/clockdomain.c
> +++ b/arch/arm/mach-omap2/clockdomain.c
> @@ -1212,6 +1212,77 @@ ccd_exit:
>       return 0;
>  }
>  
> +/*
> + * prevent future hwauto for this clkdm. If clkdm->usecount becomes hwauto 
> isn't prevented.
> + * It will only prevnt future hwauto but not bring it out of hwauto.
> + */

If you modify clkdm_{allow,deny}_idle*(), this shouldn't be a major issue 
- but all of the function comments should be fixed so that they are 
understandable and follow kernel-doc-nano specs.

> +int clkdm_hwmod_prevent_hwauto(struct clockdomain *clkdm, struct omap_hwmod 
> *oh)
> +{
> +     /* The clkdm attribute does not exist yet prior OMAP4 */
> +     if (cpu_is_omap24xx() || cpu_is_omap34xx())
> +             return 0;
> +
> +     if (!clkdm || !oh || !arch_clkdm || !arch_clkdm->clkdm_clk_disable)
> +             return -EINVAL;
> +
> +
> +     pwrdm_lock(clkdm->pwrdm.ptr);
> +     clkdm->noidlecount++;
> +     pwrdm_unlock(clkdm->pwrdm.ptr);
> +
> +     return 0;
> +}
> +
> +/*
> + * allow future hwauto for this clkdm
> + * It won't put clkdm into hwauto. use clkdm_hwmod_hwauto() for that.
> + */
> +int clkdm_hwmod_allow_hwauto(struct clockdomain *clkdm, struct omap_hwmod 
> *oh)
> +{
> +     /* The clkdm attribute does not exist yet prior OMAP4 */
> +     if (cpu_is_omap24xx() || cpu_is_omap34xx())
> +             return 0;
> +
> +     if (!clkdm || !oh || !arch_clkdm || !arch_clkdm->clkdm_clk_disable)
> +             return -EINVAL;
> +
> +
> +     pwrdm_lock(clkdm->pwrdm.ptr);
> +
> +     if (clkdm->noidlecount == 0) {
> +             pwrdm_unlock(clkdm->pwrdm.ptr);
> +             WARN_ON(1); /* underflow */
> +             return -ERANGE;
> +     }
> +
> +     clkdm->noidlecount--;
> +     pwrdm_unlock(clkdm->pwrdm.ptr);
> +
> +     return 0;
> +}
> +
> +/*
> + * put clkdm in hwauto if we can. checks noidlecount to see if we can.
> + */
> +int clkdm_hwmod_hwauto(struct clockdomain *clkdm, struct omap_hwmod *oh)
> +{
> +     /* The clkdm attribute does not exist yet prior OMAP4 */
> +     if (cpu_is_omap24xx() || cpu_is_omap34xx())
> +             return 0;
> +
> +     if (!clkdm || !oh || !arch_clkdm || !arch_clkdm->clkdm_clk_disable)
> +             return -EINVAL;
> +
> +
> +     pwrdm_lock(clkdm->pwrdm.ptr);
> +     if (clkdm->noidlecount == 0)
> +             clkdm_allow_idle_nolock(clkdm);
> +
> +     pwrdm_unlock(clkdm->pwrdm.ptr);
> +
> +     return 0;
> +}
> +
>  /**
>   * clkdm_hwmod_enable - add an enabled downstream hwmod to this clkdm
>   * @clkdm: struct clockdomain *
> diff --git a/arch/arm/mach-omap2/clockdomain.h 
> b/arch/arm/mach-omap2/clockdomain.h
> index 77bab5f..8c491be 100644
> --- a/arch/arm/mach-omap2/clockdomain.h
> +++ b/arch/arm/mach-omap2/clockdomain.h
> @@ -114,6 +114,7 @@ struct omap_hwmod;
>   * @wkdep_srcs: Clockdomains that can be told to wake this powerdomain up
>   * @sleepdep_srcs: Clockdomains that can be told to keep this clkdm from 
> inact
>   * @usecount: Usecount tracking
> + * @noidlecount: Noidle count tracking. Domain won't be auto idled this is > 
> 0.
>   * @node: list_head to link all clockdomains together
>   *
>   * @prcm_partition should be a macro from mach-omap2/prcm44xx.h (OMAP4 only)
> @@ -138,6 +139,7 @@ struct clockdomain {
>       struct clkdm_dep *wkdep_srcs;
>       struct clkdm_dep *sleepdep_srcs;
>       int usecount;
> +     int noidlecount;
>       struct list_head node;
>  };
>  
> @@ -211,6 +213,9 @@ int clkdm_clk_enable(struct clockdomain *clkdm, struct 
> clk *clk);
>  int clkdm_clk_disable(struct clockdomain *clkdm, struct clk *clk);
>  int clkdm_hwmod_enable(struct clockdomain *clkdm, struct omap_hwmod *oh);
>  int clkdm_hwmod_disable(struct clockdomain *clkdm, struct omap_hwmod *oh);
> +int clkdm_hwmod_prevent_hwauto(struct clockdomain *clkdm, struct omap_hwmod 
> *oh);
> +int clkdm_hwmod_allow_hwauto(struct clockdomain *clkdm, struct omap_hwmod 
> *oh);
> +int clkdm_hwmod_hwauto(struct clockdomain *clkdm, struct omap_hwmod *oh);
>  
>  extern void __init omap242x_clockdomains_init(void);
>  extern void __init omap243x_clockdomains_init(void);
> -- 
> 2.1.4
> 


- Paul
--
To unsubscribe from this list: send the line "unsubscribe linux-omap" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to