diff options
Diffstat (limited to 'arch/arm/mach-omap2/clkt_dpll.c')
-rw-r--r-- | arch/arm/mach-omap2/clkt_dpll.c | 93 |
1 files changed, 24 insertions, 69 deletions
diff --git a/arch/arm/mach-omap2/clkt_dpll.c b/arch/arm/mach-omap2/clkt_dpll.c index 337392c3f549..bcffee001bfa 100644 --- a/arch/arm/mach-omap2/clkt_dpll.c +++ b/arch/arm/mach-omap2/clkt_dpll.c | |||
@@ -77,7 +77,7 @@ static int _dpll_test_fint(struct clk *clk, u8 n) | |||
77 | dd = clk->dpll_data; | 77 | dd = clk->dpll_data; |
78 | 78 | ||
79 | /* DPLL divider must result in a valid jitter correction val */ | 79 | /* DPLL divider must result in a valid jitter correction val */ |
80 | fint = clk->parent->rate / (n + 1); | 80 | fint = clk->parent->rate / n; |
81 | if (fint < DPLL_FINT_BAND1_MIN) { | 81 | if (fint < DPLL_FINT_BAND1_MIN) { |
82 | 82 | ||
83 | pr_debug("rejecting n=%d due to Fint failure, " | 83 | pr_debug("rejecting n=%d due to Fint failure, " |
@@ -178,12 +178,11 @@ void omap2_init_dpll_parent(struct clk *clk) | |||
178 | if (!dd) | 178 | if (!dd) |
179 | return; | 179 | return; |
180 | 180 | ||
181 | /* Return bypass rate if DPLL is bypassed */ | ||
182 | v = __raw_readl(dd->control_reg); | 181 | v = __raw_readl(dd->control_reg); |
183 | v &= dd->enable_mask; | 182 | v &= dd->enable_mask; |
184 | v >>= __ffs(dd->enable_mask); | 183 | v >>= __ffs(dd->enable_mask); |
185 | 184 | ||
186 | /* Reparent in case the dpll is in bypass */ | 185 | /* Reparent the struct clk in case the dpll is in bypass */ |
187 | if (cpu_is_omap24xx()) { | 186 | if (cpu_is_omap24xx()) { |
188 | if (v == OMAP2XXX_EN_DPLL_LPBYPASS || | 187 | if (v == OMAP2XXX_EN_DPLL_LPBYPASS || |
189 | v == OMAP2XXX_EN_DPLL_FRBYPASS) | 188 | v == OMAP2XXX_EN_DPLL_FRBYPASS) |
@@ -260,50 +259,22 @@ u32 omap2_get_dpll_rate(struct clk *clk) | |||
260 | /* DPLL rate rounding code */ | 259 | /* DPLL rate rounding code */ |
261 | 260 | ||
262 | /** | 261 | /** |
263 | * omap2_dpll_set_rate_tolerance: set the error tolerance during rate rounding | ||
264 | * @clk: struct clk * of the DPLL | ||
265 | * @tolerance: maximum rate error tolerance | ||
266 | * | ||
267 | * Set the maximum DPLL rate error tolerance for the rate rounding | ||
268 | * algorithm. The rate tolerance is an attempt to balance DPLL power | ||
269 | * saving (the least divider value "n") vs. rate fidelity (the least | ||
270 | * difference between the desired DPLL target rate and the rounded | ||
271 | * rate out of the algorithm). So, increasing the tolerance is likely | ||
272 | * to decrease DPLL power consumption and increase DPLL rate error. | ||
273 | * Returns -EINVAL if provided a null clock ptr or a clk that is not a | ||
274 | * DPLL; or 0 upon success. | ||
275 | */ | ||
276 | int omap2_dpll_set_rate_tolerance(struct clk *clk, unsigned int tolerance) | ||
277 | { | ||
278 | if (!clk || !clk->dpll_data) | ||
279 | return -EINVAL; | ||
280 | |||
281 | clk->dpll_data->rate_tolerance = tolerance; | ||
282 | |||
283 | return 0; | ||
284 | } | ||
285 | |||
286 | /** | ||
287 | * omap2_dpll_round_rate - round a target rate for an OMAP DPLL | 262 | * omap2_dpll_round_rate - round a target rate for an OMAP DPLL |
288 | * @clk: struct clk * for a DPLL | 263 | * @clk: struct clk * for a DPLL |
289 | * @target_rate: desired DPLL clock rate | 264 | * @target_rate: desired DPLL clock rate |
290 | * | 265 | * |
291 | * Given a DPLL, a desired target rate, and a rate tolerance, round | 266 | * Given a DPLL and a desired target rate, round the target rate to a |
292 | * the target rate to a possible, programmable rate for this DPLL. | 267 | * possible, programmable rate for this DPLL. Attempts to select the |
293 | * Rate tolerance is assumed to be set by the caller before this | 268 | * minimum possible n. Stores the computed (m, n) in the DPLL's |
294 | * function is called. Attempts to select the minimum possible n | 269 | * dpll_data structure so set_rate() will not need to call this |
295 | * within the tolerance to reduce power consumption. Stores the | 270 | * (expensive) function again. Returns ~0 if the target rate cannot |
296 | * computed (m, n) in the DPLL's dpll_data structure so set_rate() | 271 | * be rounded, or the rounded rate upon success. |
297 | * will not need to call this (expensive) function again. Returns ~0 | ||
298 | * if the target rate cannot be rounded, either because the rate is | ||
299 | * too low or because the rate tolerance is set too tightly; or the | ||
300 | * rounded rate upon success. | ||
301 | */ | 272 | */ |
302 | long omap2_dpll_round_rate(struct clk *clk, unsigned long target_rate) | 273 | long omap2_dpll_round_rate(struct clk *clk, unsigned long target_rate) |
303 | { | 274 | { |
304 | int m, n, r, e, scaled_max_m; | 275 | int m, n, r, scaled_max_m; |
305 | unsigned long scaled_rt_rp, new_rate; | 276 | unsigned long scaled_rt_rp; |
306 | int min_e = -1, min_e_m = -1, min_e_n = -1; | 277 | unsigned long new_rate = 0; |
307 | struct dpll_data *dd; | 278 | struct dpll_data *dd; |
308 | 279 | ||
309 | if (!clk || !clk->dpll_data) | 280 | if (!clk || !clk->dpll_data) |
@@ -311,8 +282,8 @@ long omap2_dpll_round_rate(struct clk *clk, unsigned long target_rate) | |||
311 | 282 | ||
312 | dd = clk->dpll_data; | 283 | dd = clk->dpll_data; |
313 | 284 | ||
314 | pr_debug("clock: starting DPLL round_rate for clock %s, target rate " | 285 | pr_debug("clock: %s: starting DPLL round_rate, target rate %ld\n", |
315 | "%ld\n", clk->name, target_rate); | 286 | clk->name, target_rate); |
316 | 287 | ||
317 | scaled_rt_rp = target_rate / (dd->clk_ref->rate / DPLL_SCALE_FACTOR); | 288 | scaled_rt_rp = target_rate / (dd->clk_ref->rate / DPLL_SCALE_FACTOR); |
318 | scaled_max_m = dd->max_multiplier * DPLL_SCALE_FACTOR; | 289 | scaled_max_m = dd->max_multiplier * DPLL_SCALE_FACTOR; |
@@ -347,39 +318,23 @@ long omap2_dpll_round_rate(struct clk *clk, unsigned long target_rate) | |||
347 | if (r == DPLL_MULT_UNDERFLOW) | 318 | if (r == DPLL_MULT_UNDERFLOW) |
348 | continue; | 319 | continue; |
349 | 320 | ||
350 | e = target_rate - new_rate; | 321 | pr_debug("clock: %s: m = %d: n = %d: new_rate = %ld\n", |
351 | pr_debug("clock: n = %d: m = %d: rate error is %d " | 322 | clk->name, m, n, new_rate); |
352 | "(new_rate = %ld)\n", n, m, e, new_rate); | ||
353 | |||
354 | if (min_e == -1 || | ||
355 | min_e >= (int)(abs(e) - dd->rate_tolerance)) { | ||
356 | min_e = e; | ||
357 | min_e_m = m; | ||
358 | min_e_n = n; | ||
359 | |||
360 | pr_debug("clock: found new least error %d\n", min_e); | ||
361 | 323 | ||
362 | /* We found good settings -- bail out now */ | 324 | if (target_rate == new_rate) { |
363 | if (min_e <= dd->rate_tolerance) | 325 | dd->last_rounded_m = m; |
364 | break; | 326 | dd->last_rounded_n = n; |
327 | dd->last_rounded_rate = target_rate; | ||
328 | break; | ||
365 | } | 329 | } |
366 | } | 330 | } |
367 | 331 | ||
368 | if (min_e < 0) { | 332 | if (target_rate != new_rate) { |
369 | pr_debug("clock: error: target rate or tolerance too low\n"); | 333 | pr_debug("clock: %s: cannot round to rate %ld\n", clk->name, |
334 | target_rate); | ||
370 | return ~0; | 335 | return ~0; |
371 | } | 336 | } |
372 | 337 | ||
373 | dd->last_rounded_m = min_e_m; | 338 | return target_rate; |
374 | dd->last_rounded_n = min_e_n; | ||
375 | dd->last_rounded_rate = _dpll_compute_new_rate(dd->clk_ref->rate, | ||
376 | min_e_m, min_e_n); | ||
377 | |||
378 | pr_debug("clock: final least error: e = %d, m = %d, n = %d\n", | ||
379 | min_e, min_e_m, min_e_n); | ||
380 | pr_debug("clock: final rate: %ld (target rate: %ld)\n", | ||
381 | dd->last_rounded_rate, target_rate); | ||
382 | |||
383 | return dd->last_rounded_rate; | ||
384 | } | 339 | } |
385 | 340 | ||