On 04/19/13 09:28, James Hogan wrote: > Add core support to allow clock implementations to select the best > parent clock when rounding a rate, e.g. the one which can provide the > closest clock rate to that requested. This is by way of adding a new > clock op, determine_rate(), which is like round_rate() but has an extra > parameter to allow the clock implementation to optionally select a > different parent clock. The core then takes care of reparenting the > clock when setting the rate. > > The parent change takes place with the help of two new private data > members. struct clk::new_parent specifies a clock's new parent (NULL > indicates no change), and struct clk::new_child specifies a clock's new > child (whose new_parent member points back to it). The purpose of these > are to allow correct walking of the future tree for notifications prior > to actually reparenting any clocks, specifically to skip child clocks > who are being reparented to another clock (they will be notified via the > new parent), and to include any new child clock. These pointers are set > by clk_calc_subtree(), and the new_child pointer gets cleared when a > child is actually reparented to avoid duplicate POST_RATE_CHANGE > notifications. > > Each place where round_rate() is called, determine_rate is checked first > and called in preference, passing NULL to the new parent argument if not > needed (e.g. in __clk_round_rate). This restructures a few of the call > sites to simplify the logic into if/else blocks. > > A new __clk_set_parent_no_recalc() is created similar to > clk_set_parent() but without calling __clk_recalc_rates(). This is for > clk_change_rate() to use, where rate recalculation and notifications are > already handled. > > Signed-off-by: James Hogan <james.ho...@imgtec.com>
I believe we'll need to update the check in __clk_init() to allow you to have either a .round_rate or a .determine_rate op if you have a .recalc_rate op. Right now it fails to register the clock and then oopses later on while setting up clock debugfs. Here's a (probably whitespace damaged) patch for the first one. ----8<----- diff --git a/drivers/clk/clk.c b/drivers/clk/clk.c index e6e759e..d56291c 100644 --- a/drivers/clk/clk.c +++ b/drivers/clk/clk.c @@ -1668,8 +1668,8 @@ int __clk_init(struct device *dev, struct clk *clk) /* check that clk_ops are sane. See Documentation/clk.txt */ if (clk->ops->set_rate && - !(clk->ops->round_rate && clk->ops->recalc_rate)) { - pr_warning("%s: %s must implement .round_rate & .recalc_rate\n", + !((clk->ops->round_rate || clk->ops->determine_rate) && clk->ops->recalc_rate)) { + pr_warning("%s: %s must implement .round_rate or .determine_rate in addition to .recalc_rate\n", __func__, clk->name); ret = -EINVAL; goto out; -- Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum, hosted by The Linux Foundation -- 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/