Hi,
On 03/09/15 10:39, Roger Quadros wrote:
> On 28/07/15 14:34, Roger Quadros wrote:
>> Paul,
>>
>> On 16/07/15 16:56, Roger Quadros wrote:
>>> On 16/07/15 04:25, Paul Walmsley wrote:
>>>> 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.
>>>
>>> That is not possible with current hwmod code as clkdm_allow_idle() and
>>> clkdm_deny_idle()
>>> are not symmetrically placed.
>>>
>>> The usual flow is
>>> clkdm_enable_hwmod();
>>> if (hwsup)
>>> clkdm_allow_idle();
>>>
>>> This is mainly because it is redundant to disable auto idle before enableing
>>> (SW_WKUP) the clockdomain.
>>>
>>> If we take your proposal above then we have to modify all users like so.
>>>
>>> if (hwsup)
>>> clkdm_deny_idle();
>>> clkdm_enable_hwmod();
>>> if (hwsup)
>>> clkdm_allow_idle();
>>>
>>> Is this really what we want?
>>
>> Any comments on this?
>
> Paul. I'm waiting on your input to rework this series if needed.
> Early input would be much appreciated. Thanks.
Not sure if Paul is receiving my e-mails but I'd like
others to give their opinion on how we can proceed with this. Thanks.
cheers,
-roger
>
>>
>>>
>>>>
>>>> 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.
>>>
>>> OK for updating the comments.
>>>
>>>
>>> cheers,
>>> -roger
>>>
>>>>
>>>>> +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
>>>
>> --
>> 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
>>
> --
> 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
>
--
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