diff options
author | Russell King <rmk@dyn-67.arm.linux.org.uk> | 2009-01-31 05:05:51 -0500 |
---|---|---|
committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2009-02-08 12:50:42 -0500 |
commit | 3f0a820c4c0b4670fb5f164baa5582e23c2ef118 (patch) | |
tree | 6af02e1456c0316791ab95e7da9c09496f29c232 /arch/arm/plat-omap | |
parent | b5088c0d90b898802318c62caf2320a53df6ce57 (diff) |
[ARM] omap: create a proper tree of clocks
Traditionally, we've tracked the parent/child relationships between
clk structures by setting the child's parent member to point at the
upstream clock. As a result, when decending the tree, we have had
to scan all clocks to find the children.
Avoid this wasteful scanning by keeping a list of the clock's children.
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'arch/arm/plat-omap')
-rw-r--r-- | arch/arm/plat-omap/clock.c | 51 | ||||
-rw-r--r-- | arch/arm/plat-omap/include/mach/clock.h | 7 |
2 files changed, 37 insertions, 21 deletions
diff --git a/arch/arm/plat-omap/clock.c b/arch/arm/plat-omap/clock.c index 54da27af0bd5..6a1737a74477 100644 --- a/arch/arm/plat-omap/clock.c +++ b/arch/arm/plat-omap/clock.c | |||
@@ -143,8 +143,7 @@ int clk_set_rate(struct clk *clk, unsigned long rate) | |||
143 | if (ret == 0) { | 143 | if (ret == 0) { |
144 | if (clk->recalc) | 144 | if (clk->recalc) |
145 | clk->recalc(clk); | 145 | clk->recalc(clk); |
146 | if (clk->flags & RATE_PROPAGATES) | 146 | propagate_rate(clk); |
147 | propagate_rate(clk); | ||
148 | } | 147 | } |
149 | spin_unlock_irqrestore(&clockfw_lock, flags); | 148 | spin_unlock_irqrestore(&clockfw_lock, flags); |
150 | 149 | ||
@@ -166,8 +165,7 @@ int clk_set_parent(struct clk *clk, struct clk *parent) | |||
166 | if (ret == 0) { | 165 | if (ret == 0) { |
167 | if (clk->recalc) | 166 | if (clk->recalc) |
168 | clk->recalc(clk); | 167 | clk->recalc(clk); |
169 | if (clk->flags & RATE_PROPAGATES) | 168 | propagate_rate(clk); |
170 | propagate_rate(clk); | ||
171 | } | 169 | } |
172 | spin_unlock_irqrestore(&clockfw_lock, flags); | 170 | spin_unlock_irqrestore(&clockfw_lock, flags); |
173 | 171 | ||
@@ -214,24 +212,31 @@ void followparent_recalc(struct clk *clk) | |||
214 | clk->rate = clk->parent->rate; | 212 | clk->rate = clk->parent->rate; |
215 | } | 213 | } |
216 | 214 | ||
215 | void clk_reparent(struct clk *child, struct clk *parent) | ||
216 | { | ||
217 | list_del_init(&child->sibling); | ||
218 | if (parent) | ||
219 | list_add(&child->sibling, &parent->children); | ||
220 | child->parent = parent; | ||
221 | |||
222 | /* now do the debugfs renaming to reattach the child | ||
223 | to the proper parent */ | ||
224 | } | ||
225 | |||
217 | /* Propagate rate to children */ | 226 | /* Propagate rate to children */ |
218 | void propagate_rate(struct clk * tclk) | 227 | void propagate_rate(struct clk * tclk) |
219 | { | 228 | { |
220 | struct clk *clkp; | 229 | struct clk *clkp; |
221 | 230 | ||
222 | if (tclk == NULL || IS_ERR(tclk)) | 231 | list_for_each_entry(clkp, &tclk->children, sibling) { |
223 | return; | ||
224 | |||
225 | list_for_each_entry(clkp, &clocks, node) { | ||
226 | if (likely(clkp->parent != tclk)) | ||
227 | continue; | ||
228 | if (clkp->recalc) | 232 | if (clkp->recalc) |
229 | clkp->recalc(clkp); | 233 | clkp->recalc(clkp); |
230 | if (clkp->flags & RATE_PROPAGATES) | 234 | propagate_rate(clkp); |
231 | propagate_rate(clkp); | ||
232 | } | 235 | } |
233 | } | 236 | } |
234 | 237 | ||
238 | static LIST_HEAD(root_clks); | ||
239 | |||
235 | /** | 240 | /** |
236 | * recalculate_root_clocks - recalculate and propagate all root clocks | 241 | * recalculate_root_clocks - recalculate and propagate all root clocks |
237 | * | 242 | * |
@@ -243,16 +248,18 @@ void recalculate_root_clocks(void) | |||
243 | { | 248 | { |
244 | struct clk *clkp; | 249 | struct clk *clkp; |
245 | 250 | ||
246 | list_for_each_entry(clkp, &clocks, node) { | 251 | list_for_each_entry(clkp, &root_clks, sibling) { |
247 | if (!clkp->parent) { | 252 | if (clkp->recalc) |
248 | if (clkp->recalc) | 253 | clkp->recalc(clkp); |
249 | clkp->recalc(clkp); | 254 | propagate_rate(clkp); |
250 | if (clkp->flags & RATE_PROPAGATES) | ||
251 | propagate_rate(clkp); | ||
252 | } | ||
253 | } | 255 | } |
254 | } | 256 | } |
255 | 257 | ||
258 | void clk_init_one(struct clk *clk) | ||
259 | { | ||
260 | INIT_LIST_HEAD(&clk->children); | ||
261 | } | ||
262 | |||
256 | int clk_register(struct clk *clk) | 263 | int clk_register(struct clk *clk) |
257 | { | 264 | { |
258 | if (clk == NULL || IS_ERR(clk)) | 265 | if (clk == NULL || IS_ERR(clk)) |
@@ -265,6 +272,11 @@ int clk_register(struct clk *clk) | |||
265 | return 0; | 272 | return 0; |
266 | 273 | ||
267 | mutex_lock(&clocks_mutex); | 274 | mutex_lock(&clocks_mutex); |
275 | if (clk->parent) | ||
276 | list_add(&clk->sibling, &clk->parent->children); | ||
277 | else | ||
278 | list_add(&clk->sibling, &root_clks); | ||
279 | |||
268 | list_add(&clk->node, &clocks); | 280 | list_add(&clk->node, &clocks); |
269 | if (clk->init) | 281 | if (clk->init) |
270 | clk->init(clk); | 282 | clk->init(clk); |
@@ -280,6 +292,7 @@ void clk_unregister(struct clk *clk) | |||
280 | return; | 292 | return; |
281 | 293 | ||
282 | mutex_lock(&clocks_mutex); | 294 | mutex_lock(&clocks_mutex); |
295 | list_del(&clk->sibling); | ||
283 | list_del(&clk->node); | 296 | list_del(&clk->node); |
284 | mutex_unlock(&clocks_mutex); | 297 | mutex_unlock(&clocks_mutex); |
285 | } | 298 | } |
diff --git a/arch/arm/plat-omap/include/mach/clock.h b/arch/arm/plat-omap/include/mach/clock.h index 8705902de1d6..af6ae4fa46d6 100644 --- a/arch/arm/plat-omap/include/mach/clock.h +++ b/arch/arm/plat-omap/include/mach/clock.h | |||
@@ -70,6 +70,8 @@ struct clk { | |||
70 | const char *name; | 70 | const char *name; |
71 | int id; | 71 | int id; |
72 | struct clk *parent; | 72 | struct clk *parent; |
73 | struct list_head children; | ||
74 | struct list_head sibling; /* node for children */ | ||
73 | unsigned long rate; | 75 | unsigned long rate; |
74 | __u32 flags; | 76 | __u32 flags; |
75 | void __iomem *enable_reg; | 77 | void __iomem *enable_reg; |
@@ -115,7 +117,9 @@ struct clk_functions { | |||
115 | extern unsigned int mpurate; | 117 | extern unsigned int mpurate; |
116 | 118 | ||
117 | extern int clk_init(struct clk_functions *custom_clocks); | 119 | extern int clk_init(struct clk_functions *custom_clocks); |
120 | extern void clk_init_one(struct clk *clk); | ||
118 | extern int clk_register(struct clk *clk); | 121 | extern int clk_register(struct clk *clk); |
122 | extern void clk_reparent(struct clk *child, struct clk *parent); | ||
119 | extern void clk_unregister(struct clk *clk); | 123 | extern void clk_unregister(struct clk *clk); |
120 | extern void propagate_rate(struct clk *clk); | 124 | extern void propagate_rate(struct clk *clk); |
121 | extern void recalculate_root_clocks(void); | 125 | extern void recalculate_root_clocks(void); |
@@ -131,8 +135,7 @@ extern const struct clkops clkops_null; | |||
131 | /* Clock flags */ | 135 | /* Clock flags */ |
132 | /* bit 0 is free */ | 136 | /* bit 0 is free */ |
133 | #define RATE_FIXED (1 << 1) /* Fixed clock rate */ | 137 | #define RATE_FIXED (1 << 1) /* Fixed clock rate */ |
134 | #define RATE_PROPAGATES (1 << 2) /* Program children too */ | 138 | /* bits 2-4 are free */ |
135 | /* bits 3-4 are free */ | ||
136 | #define ENABLE_REG_32BIT (1 << 5) /* Use 32-bit access */ | 139 | #define ENABLE_REG_32BIT (1 << 5) /* Use 32-bit access */ |
137 | #define CLOCK_IDLE_CONTROL (1 << 7) | 140 | #define CLOCK_IDLE_CONTROL (1 << 7) |
138 | #define CLOCK_NO_IDLE_PARENT (1 << 8) | 141 | #define CLOCK_NO_IDLE_PARENT (1 << 8) |