Message ID | 1363873693-30902-3-git-send-email-ulf.hansson@stericsson.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Quoting Ulf Hansson (2013-03-21 06:48:12) > @@ -1344,8 +1341,9 @@ out: > int clk_set_parent(struct clk *clk, struct clk *parent) > { > int ret = 0; > + u8 p_index; > > - if (!clk || !clk->ops) > + if (!clk || !clk->ops || !parent) > return -EINVAL; > A NULL clock is valid according to the clk.h api. I would like to allow parent to be NULL, resulting in a migration from the real clock tree to the orphans list. This feature was apparently buggy for some time and Rajagopal sent me a patch off-list to fix it. Rajagopal, can you post your patch publicly? Thanks, Mike
On 22 March 2013 02:30, Mike Turquette <mturquette@linaro.org> wrote: > Quoting Ulf Hansson (2013-03-21 06:48:12) >> @@ -1344,8 +1341,9 @@ out: >> int clk_set_parent(struct clk *clk, struct clk *parent) >> { >> int ret = 0; >> + u8 p_index; >> >> - if (!clk || !clk->ops) >> + if (!clk || !clk->ops || !parent) >> return -EINVAL; >> > > A NULL clock is valid according to the clk.h api. I would like to allow > parent to be NULL, resulting in a migration from the real clock tree to > the orphans list. > > This feature was apparently buggy for some time and Rajagopal sent me a > patch off-list to fix it. Rajagopal, can you post your patch publicly? > Patch is available at https://patchwork.kernel.org/patch/2012221/ Do you want me to rebase it on top of this patchset? > Thanks, > Mike
On 21 March 2013 22:00, Mike Turquette <mturquette@linaro.org> wrote: > Quoting Ulf Hansson (2013-03-21 06:48:12) >> @@ -1344,8 +1341,9 @@ out: >> int clk_set_parent(struct clk *clk, struct clk *parent) >> { >> int ret = 0; >> + u8 p_index; >> >> - if (!clk || !clk->ops) >> + if (!clk || !clk->ops || !parent) >> return -EINVAL; >> > > A NULL clock is valid according to the clk.h api. I would like to allow > parent to be NULL, resulting in a migration from the real clock tree to > the orphans list. Sorry, did not think of this. Will correct. > > This feature was apparently buggy for some time and Rajagopal sent me a > patch off-list to fix it. Rajagopal, can you post your patch publicly? So, that is also why I removed this. Anyway, I guess it is somewhat easier for me to make a fixup so we can accept NULL as parents. I give it a try. > > Thanks, > Mike
Quoting Rajagopal Venkat (2013-03-22 02:59:17) > On 22 March 2013 02:30, Mike Turquette <mturquette@linaro.org> wrote: > > Quoting Ulf Hansson (2013-03-21 06:48:12) > >> @@ -1344,8 +1341,9 @@ out: > >> int clk_set_parent(struct clk *clk, struct clk *parent) > >> { > >> int ret = 0; > >> + u8 p_index; > >> > >> - if (!clk || !clk->ops) > >> + if (!clk || !clk->ops || !parent) > >> return -EINVAL; > >> > > > > A NULL clock is valid according to the clk.h api. I would like to allow > > parent to be NULL, resulting in a migration from the real clock tree to > > the orphans list. > > > > This feature was apparently buggy for some time and Rajagopal sent me a > > patch off-list to fix it. Rajagopal, can you post your patch publicly? > > > > Patch is available at https://patchwork.kernel.org/patch/2012221/ > Do you want me to rebase it on top of this patchset? > Maybe. Let's see what Ulf's next version of the clk_set_parent locking patch looks like. Thanks, Mike > > Thanks, > > Mike > > > > -- > Regards, > Rajagopal
diff --git a/drivers/clk/clk.c b/drivers/clk/clk.c index d73fb0f..c7eb0d7 100644 --- a/drivers/clk/clk.c +++ b/drivers/clk/clk.c @@ -1269,15 +1269,10 @@ void __clk_reparent(struct clk *clk, struct clk *new_parent) __clk_recalc_rates(clk, POST_RATE_CHANGE); } -static int __clk_set_parent(struct clk *clk, struct clk *parent) +static u8 clk_fetch_parent_index(struct clk *clk, struct clk *parent) { - struct clk *old_parent; - unsigned long flags; - int ret = -EINVAL; u8 i; - old_parent = clk->parent; - if (!clk->parents) clk->parents = kzalloc((sizeof(struct clk*) * clk->num_parents), GFP_KERNEL); @@ -1297,11 +1292,14 @@ static int __clk_set_parent(struct clk *clk, struct clk *parent) } } - if (i == clk->num_parents) { - pr_debug("%s: clock %s is not a possible parent of clock %s\n", - __func__, parent->name, clk->name); - goto out; - } + return i; +} + +static int __clk_set_parent(struct clk *clk, struct clk *parent, u8 p_index) +{ + unsigned long flags; + int ret; + struct clk *old_parent = clk->parent; /* migrate prepare and enable */ if (clk->prepare_count) @@ -1314,7 +1312,7 @@ static int __clk_set_parent(struct clk *clk, struct clk *parent) spin_unlock_irqrestore(&enable_lock, flags); /* change clock input source */ - ret = clk->ops->set_parent(clk->hw, i); + ret = clk->ops->set_parent(clk->hw, p_index); /* clean up old prepare and enable */ spin_lock_irqsave(&enable_lock, flags); @@ -1325,7 +1323,6 @@ static int __clk_set_parent(struct clk *clk, struct clk *parent) if (clk->prepare_count) __clk_unprepare(old_parent); -out: return ret; } @@ -1344,8 +1341,9 @@ out: int clk_set_parent(struct clk *clk, struct clk *parent) { int ret = 0; + u8 p_index; - if (!clk || !clk->ops) + if (!clk || !clk->ops || !parent) return -EINVAL; if (!clk->ops->set_parent) @@ -1357,6 +1355,21 @@ int clk_set_parent(struct clk *clk, struct clk *parent) if (clk->parent == parent) goto out; + /* check that we are allowed to re-parent if the clock is in use */ + if ((clk->flags & CLK_SET_PARENT_GATE) && clk->prepare_count) { + ret = -EBUSY; + goto out; + } + + /* try finding the new parent index */ + p_index = clk_fetch_parent_index(clk, parent); + if (p_index == clk->num_parents) { + pr_debug("%s: clock %s is not a possible parent of clock %s\n", + __func__, parent->name, clk->name); + ret = -EINVAL; + goto out; + } + /* propagate PRE_RATE_CHANGE notifications */ if (clk->notifier_count) ret = __clk_speculate_rates(clk, parent->rate); @@ -1365,11 +1378,8 @@ int clk_set_parent(struct clk *clk, struct clk *parent) if (ret == NOTIFY_STOP) goto out; - /* only re-parent if the clock is not in use */ - if ((clk->flags & CLK_SET_PARENT_GATE) && clk->prepare_count) - ret = -EBUSY; - else - ret = __clk_set_parent(clk, parent); + /* do the re-parent */ + ret = __clk_set_parent(clk, parent, p_index); /* propagate ABORT_RATE_CHANGE if .set_parent failed */ if (ret) {