Re: moving Tegra30 to the common clock framework

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 05/14/2012 02:36 PM, Turquette, Mike wrote:
On Fri, May 11, 2012 at 7:58 PM, Saravana Kannan<skannan@xxxxxxxxxxxxxx>  wrote:
Mike,

I was looking at the code to make the changes and I noticed this snippet
(reformatted for email) in clk_change_rate():

       if (clk->ops->set_rate)
                clk->ops->set_rate(clk->hw, clk->new_rate,
                                        clk->parent->rate);


        if (clk->ops->recalc_rate)
                clk->rate = clk->ops->recalc_rate(clk->hw,
                                clk->parent->rate);
        else
                clk->rate = clk->parent->rate;

I'm a bit confused. I thought recalc_rates was optional. But if I don't
implement it, the clocks rate will get set to parent's rate? Or is that a
bug in the code?


It is not a bug.  The handsome ascii art in Documentation/clk.txt covers
this requirement.  I have copy/pasted the relevant bits below for
convenience:

                            clock hardware characteristics
	     -----------------------------------------------------------
              | gate | change rate | single parent | multiplexer | root |
              |------|-------------|---------------|-------------|------|
	     				...
              |      |             |               |             |      |
.recalc_rate |      | y           |               |             |      |
.round_rate  |      | y           |               |             |      |
.set_rate    |      | y           |               |             |      |
					...
              |      |             |               |             |      |
	     -----------------------------------------------------------

The take-away is that a clock that can adjust its rate (eg: implements a
.set_rate callback) must also implement .recalc_rate and .round_rate
callbacks.

I get the round_rate ops part. But I don't see a need to force a recalc_rate ops. Can we just check for the existence of .set_rate() to figure out if the clock will take up the rate of the parent or will output a different rate?


Also, if the clock's rate was just set with set_rate, why do we need to
recalc the rate by reading hardware? I'm a bit confused. Can you please
clarify what's going on here?


This is simply being very cautious.  For platforms adjusting dividers
with direct register writes this might feel unnecessary.  However this
strict checking is in anticipation of clock hardware that might not
actually output the precise rate passed into .set_rate.  In principal
this isn't different from how CPUfreq and devfreq drivers inspect rates
after requesting them.

Sorry, this still doesn't make much sense to me. This essentially means we can't trust the HW to do what we are asking it to do?

CPUfreq and devfreq are clients of external clock APIs. That's different from whether the HW will do what the platform driver will ask it to.

Even if this unusual HW exists, I certainly don't want to deal with recalc_rate(). It's also quite a bit of register reads that I would like to avoid. I want to keep the clock APIs as fast as I can since it affects power.

Can we do something like this in clk_change_rate()?

if (ops->set_rate) {
	ops->set_rate(clk->new_rate,...);
	clk->rate = clk->new_rate;
} else {
	clk->rate = clk->parent->rate;
}

if (ops->recalc_rate()) {
	clk->rate = ops->recalc_rate(...);
}

This code also makes the assumptions more intuitive and easy to understand.

Would you mind adding more comments inside clk_calc_new_rates() and
clk_change_rate() trying to explain what cases you are trying to account
for?


Someday I'll have a comment/kerneldoc cleanup session.  Things in the
clock core are likely to change in the next couple of release cycles
which will deprecate some of the kerneldoc making a big cleanup
unavoidable.

In the mean time are there any other specific question you have for
clk_change_rate?

Also, in clk_calc_new_rates(),

        if (!clk->ops->round_rate) {
                top = clk_calc_new_rates(clk->parent, rate);
                new_rate = clk->parent->new_rate;

                goto out;
        }

Is the code assuming that if there is no round rate ops that that clock node
is only a gating clock (as in, can't change frequency the input freq)? Just
trying to understand the assumptions made in the code.


This is also covered by the ascii art above.  There is no assumption
about gating, per se.  However if a clock can adjust it's rate (more
specifically, if the input rate for a clock differs from the output
rate) then a .round_rate callback must exist.

The code above reads like it does because in the absence of a
.round_rate callback it is implied that the clock cannot set it's own
rate and should thus rely on its parent's rate.

I agree that anyone implementing .set_rate should also provide .round_rate. May be we should enforce that in clk_init/clk_register?

But can we please the above "if" check more explicit? "Your rate has to be same as the parent rate since you don't implement set rate" is clearer than "Your rate has to be the same as the parent rate since you don't implement round rate".

	if (!clk->ops->set_rate) {
		top = clk_calc_new_rates(clk->parent, rate);
		new_rate = clk->parent->new_rate;

		goto out;
	}

Thanks,
Saravana

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--
To unsubscribe from this list: send the line "unsubscribe linux-tegra" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [ARM Kernel]     [Linux ARM]     [Linux ARM MSM]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux