diff options
Diffstat (limited to 'kernel/time/timekeeping.c')
-rw-r--r-- | kernel/time/timekeeping.c | 41 |
1 files changed, 28 insertions, 13 deletions
diff --git a/kernel/time/timekeeping.c b/kernel/time/timekeeping.c index b8b70fb545fc..016a2591d719 100644 --- a/kernel/time/timekeeping.c +++ b/kernel/time/timekeeping.c | |||
@@ -79,7 +79,7 @@ static void clocksource_forward_now(void) | |||
79 | cycle_t cycle_now, cycle_delta; | 79 | cycle_t cycle_now, cycle_delta; |
80 | s64 nsec; | 80 | s64 nsec; |
81 | 81 | ||
82 | cycle_now = clocksource_read(clock); | 82 | cycle_now = clock->read(clock); |
83 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; | 83 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; |
84 | clock->cycle_last = cycle_now; | 84 | clock->cycle_last = cycle_now; |
85 | 85 | ||
@@ -114,7 +114,7 @@ void getnstimeofday(struct timespec *ts) | |||
114 | *ts = xtime; | 114 | *ts = xtime; |
115 | 115 | ||
116 | /* read clocksource: */ | 116 | /* read clocksource: */ |
117 | cycle_now = clocksource_read(clock); | 117 | cycle_now = clock->read(clock); |
118 | 118 | ||
119 | /* calculate the delta since the last update_wall_time: */ | 119 | /* calculate the delta since the last update_wall_time: */ |
120 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; | 120 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; |
@@ -146,7 +146,7 @@ ktime_t ktime_get(void) | |||
146 | nsecs = xtime.tv_nsec + wall_to_monotonic.tv_nsec; | 146 | nsecs = xtime.tv_nsec + wall_to_monotonic.tv_nsec; |
147 | 147 | ||
148 | /* read clocksource: */ | 148 | /* read clocksource: */ |
149 | cycle_now = clocksource_read(clock); | 149 | cycle_now = clock->read(clock); |
150 | 150 | ||
151 | /* calculate the delta since the last update_wall_time: */ | 151 | /* calculate the delta since the last update_wall_time: */ |
152 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; | 152 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; |
@@ -186,7 +186,7 @@ void ktime_get_ts(struct timespec *ts) | |||
186 | tomono = wall_to_monotonic; | 186 | tomono = wall_to_monotonic; |
187 | 187 | ||
188 | /* read clocksource: */ | 188 | /* read clocksource: */ |
189 | cycle_now = clocksource_read(clock); | 189 | cycle_now = clock->read(clock); |
190 | 190 | ||
191 | /* calculate the delta since the last update_wall_time: */ | 191 | /* calculate the delta since the last update_wall_time: */ |
192 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; | 192 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; |
@@ -274,16 +274,29 @@ static void change_clocksource(void) | |||
274 | 274 | ||
275 | clocksource_forward_now(); | 275 | clocksource_forward_now(); |
276 | 276 | ||
277 | if (clocksource_enable(new)) | 277 | if (new->enable && !new->enable(new)) |
278 | return; | 278 | return; |
279 | /* | ||
280 | * The frequency may have changed while the clocksource | ||
281 | * was disabled. If so the code in ->enable() must update | ||
282 | * the mult value to reflect the new frequency. Make sure | ||
283 | * mult_orig follows this change. | ||
284 | */ | ||
285 | new->mult_orig = new->mult; | ||
279 | 286 | ||
280 | new->raw_time = clock->raw_time; | 287 | new->raw_time = clock->raw_time; |
281 | old = clock; | 288 | old = clock; |
282 | clock = new; | 289 | clock = new; |
283 | clocksource_disable(old); | 290 | /* |
291 | * Save mult_orig in mult so that the value can be restored | ||
292 | * regardless if ->enable() updates the value of mult or not. | ||
293 | */ | ||
294 | old->mult = old->mult_orig; | ||
295 | if (old->disable) | ||
296 | old->disable(old); | ||
284 | 297 | ||
285 | clock->cycle_last = 0; | 298 | clock->cycle_last = 0; |
286 | clock->cycle_last = clocksource_read(clock); | 299 | clock->cycle_last = clock->read(clock); |
287 | clock->error = 0; | 300 | clock->error = 0; |
288 | clock->xtime_nsec = 0; | 301 | clock->xtime_nsec = 0; |
289 | clocksource_calculate_interval(clock, NTP_INTERVAL_LENGTH); | 302 | clocksource_calculate_interval(clock, NTP_INTERVAL_LENGTH); |
@@ -373,7 +386,7 @@ void getrawmonotonic(struct timespec *ts) | |||
373 | seq = read_seqbegin(&xtime_lock); | 386 | seq = read_seqbegin(&xtime_lock); |
374 | 387 | ||
375 | /* read clocksource: */ | 388 | /* read clocksource: */ |
376 | cycle_now = clocksource_read(clock); | 389 | cycle_now = clock->read(clock); |
377 | 390 | ||
378 | /* calculate the delta since the last update_wall_time: */ | 391 | /* calculate the delta since the last update_wall_time: */ |
379 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; | 392 | cycle_delta = (cycle_now - clock->cycle_last) & clock->mask; |
@@ -435,9 +448,12 @@ void __init timekeeping_init(void) | |||
435 | ntp_init(); | 448 | ntp_init(); |
436 | 449 | ||
437 | clock = clocksource_get_next(); | 450 | clock = clocksource_get_next(); |
438 | clocksource_enable(clock); | 451 | if (clock->enable) |
452 | clock->enable(clock); | ||
453 | /* set mult_orig on enable */ | ||
454 | clock->mult_orig = clock->mult; | ||
439 | clocksource_calculate_interval(clock, NTP_INTERVAL_LENGTH); | 455 | clocksource_calculate_interval(clock, NTP_INTERVAL_LENGTH); |
440 | clock->cycle_last = clocksource_read(clock); | 456 | clock->cycle_last = clock->read(clock); |
441 | 457 | ||
442 | xtime.tv_sec = sec; | 458 | xtime.tv_sec = sec; |
443 | xtime.tv_nsec = 0; | 459 | xtime.tv_nsec = 0; |
@@ -477,8 +493,7 @@ static int timekeeping_resume(struct sys_device *dev) | |||
477 | } | 493 | } |
478 | update_xtime_cache(0); | 494 | update_xtime_cache(0); |
479 | /* re-base the last cycle value */ | 495 | /* re-base the last cycle value */ |
480 | clock->cycle_last = 0; | 496 | clock->cycle_last = clock->read(clock); |
481 | clock->cycle_last = clocksource_read(clock); | ||
482 | clock->error = 0; | 497 | clock->error = 0; |
483 | timekeeping_suspended = 0; | 498 | timekeeping_suspended = 0; |
484 | write_sequnlock_irqrestore(&xtime_lock, flags); | 499 | write_sequnlock_irqrestore(&xtime_lock, flags); |
@@ -630,7 +645,7 @@ void update_wall_time(void) | |||
630 | return; | 645 | return; |
631 | 646 | ||
632 | #ifdef CONFIG_GENERIC_TIME | 647 | #ifdef CONFIG_GENERIC_TIME |
633 | offset = (clocksource_read(clock) - clock->cycle_last) & clock->mask; | 648 | offset = (clock->read(clock) - clock->cycle_last) & clock->mask; |
634 | #else | 649 | #else |
635 | offset = clock->cycle_interval; | 650 | offset = clock->cycle_interval; |
636 | #endif | 651 | #endif |