Commit aa87aa34 authored by Paul Mundt's avatar Paul Mundt

sh: clkfwk: Improve the generic clk_set_parent() implementation.

This causes the generic clk_set_parent() implementation to be a bit more
intelligent. A clk_reparent() is added to move the clock over to the new
parent's sibling list, which then allows the generic rate propagation
code to succeed. This also becomes a nop if the new and old parents are
unchanged.
Signed-off-by: default avatarPaul Mundt <lethal@linux-sh.org>
parent ae891a42
...@@ -48,6 +48,7 @@ int clk_init(void); ...@@ -48,6 +48,7 @@ int clk_init(void);
unsigned long followparent_recalc(struct clk *); unsigned long followparent_recalc(struct clk *);
void recalculate_root_clocks(void); void recalculate_root_clocks(void);
void propagate_rate(struct clk *); void propagate_rate(struct clk *);
int clk_reparent(struct clk *child, struct clk *parent);
void clk_recalc_rate(struct clk *); void clk_recalc_rate(struct clk *);
int clk_register(struct clk *); int clk_register(struct clk *);
void clk_unregister(struct clk *); void clk_unregister(struct clk *);
......
...@@ -81,6 +81,19 @@ unsigned long followparent_recalc(struct clk *clk) ...@@ -81,6 +81,19 @@ unsigned long followparent_recalc(struct clk *clk)
return clk->parent->rate; return clk->parent->rate;
} }
int clk_reparent(struct clk *child, struct clk *parent)
{
list_del_init(&child->sibling);
if (parent)
list_add(&child->sibling, &parent->children);
child->parent = parent;
/* now do the debugfs renaming to reattach the child
to the proper parent */
return 0;
}
/* Propagate rate to children */ /* Propagate rate to children */
void propagate_rate(struct clk *tclk) void propagate_rate(struct clk *tclk)
{ {
...@@ -288,12 +301,19 @@ int clk_set_parent(struct clk *clk, struct clk *parent) ...@@ -288,12 +301,19 @@ int clk_set_parent(struct clk *clk, struct clk *parent)
if (!parent || !clk) if (!parent || !clk)
return ret; return ret;
if (clk->parent == parent)
return 0;
spin_lock_irqsave(&clock_lock, flags); spin_lock_irqsave(&clock_lock, flags);
if (clk->usecount == 0) { if (clk->usecount == 0) {
if (clk->ops->set_parent) if (clk->ops->set_parent)
ret = clk->ops->set_parent(clk, parent); ret = clk->ops->set_parent(clk, parent);
else
ret = clk_reparent(clk, parent);
if (ret == 0) { if (ret == 0) {
pr_debug("clock: set parent of %s to %s (new rate %ld)\n",
clk->name, clk->parent->name, clk->rate);
if (clk->ops->recalc) if (clk->ops->recalc)
clk->rate = clk->ops->recalc(clk); clk->rate = clk->ops->recalc(clk);
propagate_rate(clk); propagate_rate(clk);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment