diff options
author | john stultz <johnstul@us.ibm.com> | 2007-07-24 21:38:34 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-07-25 13:17:44 -0400 |
commit | 17c38b7490b3f0300c7812aefdae2ddda7ab4112 (patch) | |
tree | 0f7a9ee0c691aef3497030e38c3ba8e3c13a985c | |
parent | 2c6b47de17c75d553de3e2fb426d8298d2074585 (diff) |
Cache xtime every call to update_wall_time
This avoids xtime lag seen with dynticks, because while 'xtime' itself
is still not updated often, we keep a 'xtime_cache' variable around that
contains the approximate real-time that _is_ updated each time we do a
'update_wall_time()', and is thus never off by more than one tick.
IOW, this restores the original semantics for 'xtime' users, as long as
you use the proper abstraction functions (ie 'current_kernel_time()' or
'get_seconds()' depending on whether you want a timespec or just the
seconds field).
[ Updated Patch. As penance for my sins I've also yanked another #ifdef
that was added to avoid the xtime lag w/ hrtimers. ]
Signed-off-by: John Stultz <johnstul@us.ibm.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r-- | include/linux/time.h | 6 | ||||
-rw-r--r-- | kernel/hrtimer.c | 4 | ||||
-rw-r--r-- | kernel/time/timekeeping.c | 26 |
3 files changed, 24 insertions, 12 deletions
diff --git a/include/linux/time.h b/include/linux/time.h index 71181df8b744..6a5f503b4f1d 100644 --- a/include/linux/time.h +++ b/include/linux/time.h | |||
@@ -99,11 +99,7 @@ extern int update_persistent_clock(struct timespec now); | |||
99 | extern int no_sync_cmos_clock __read_mostly; | 99 | extern int no_sync_cmos_clock __read_mostly; |
100 | void timekeeping_init(void); | 100 | void timekeeping_init(void); |
101 | 101 | ||
102 | static inline unsigned long get_seconds(void) | 102 | unsigned long get_seconds(void); |
103 | { | ||
104 | return xtime.tv_sec; | ||
105 | } | ||
106 | |||
107 | struct timespec current_kernel_time(void); | 103 | struct timespec current_kernel_time(void); |
108 | 104 | ||
109 | #define CURRENT_TIME (current_kernel_time()) | 105 | #define CURRENT_TIME (current_kernel_time()) |
diff --git a/kernel/hrtimer.c b/kernel/hrtimer.c index a7bb05e6cb63..c21ca6bfaa66 100644 --- a/kernel/hrtimer.c +++ b/kernel/hrtimer.c | |||
@@ -141,11 +141,7 @@ static void hrtimer_get_softirq_time(struct hrtimer_cpu_base *base) | |||
141 | 141 | ||
142 | do { | 142 | do { |
143 | seq = read_seqbegin(&xtime_lock); | 143 | seq = read_seqbegin(&xtime_lock); |
144 | #ifdef CONFIG_NO_HZ | ||
145 | getnstimeofday(&xts); | ||
146 | #else | ||
147 | xts = current_kernel_time(); | 144 | xts = current_kernel_time(); |
148 | #endif | ||
149 | tom = wall_to_monotonic; | 145 | tom = wall_to_monotonic; |
150 | } while (read_seqretry(&xtime_lock, seq)); | 146 | } while (read_seqretry(&xtime_lock, seq)); |
151 | 147 | ||
diff --git a/kernel/time/timekeeping.c b/kernel/time/timekeeping.c index 07a3f1420c27..acc417b5a9b7 100644 --- a/kernel/time/timekeeping.c +++ b/kernel/time/timekeeping.c | |||
@@ -47,10 +47,22 @@ EXPORT_SYMBOL(xtime_lock); | |||
47 | struct timespec xtime __attribute__ ((aligned (16))); | 47 | struct timespec xtime __attribute__ ((aligned (16))); |
48 | struct timespec wall_to_monotonic __attribute__ ((aligned (16))); | 48 | struct timespec wall_to_monotonic __attribute__ ((aligned (16))); |
49 | static unsigned long total_sleep_time; /* seconds */ | 49 | static unsigned long total_sleep_time; /* seconds */ |
50 | |||
51 | EXPORT_SYMBOL(xtime); | 50 | EXPORT_SYMBOL(xtime); |
52 | 51 | ||
53 | 52 | ||
53 | #ifdef CONFIG_NO_HZ | ||
54 | static struct timespec xtime_cache __attribute__ ((aligned (16))); | ||
55 | static inline void update_xtime_cache(u64 nsec) | ||
56 | { | ||
57 | xtime_cache = xtime; | ||
58 | timespec_add_ns(&xtime_cache, nsec); | ||
59 | } | ||
60 | #else | ||
61 | #define xtime_cache xtime | ||
62 | /* We do *not* want to evaluate the argument for this case */ | ||
63 | #define update_xtime_cache(n) do { } while (0) | ||
64 | #endif | ||
65 | |||
54 | static struct clocksource *clock; /* pointer to current clocksource */ | 66 | static struct clocksource *clock; /* pointer to current clocksource */ |
55 | 67 | ||
56 | 68 | ||
@@ -478,6 +490,8 @@ void update_wall_time(void) | |||
478 | xtime.tv_nsec = (s64)clock->xtime_nsec >> clock->shift; | 490 | xtime.tv_nsec = (s64)clock->xtime_nsec >> clock->shift; |
479 | clock->xtime_nsec -= (s64)xtime.tv_nsec << clock->shift; | 491 | clock->xtime_nsec -= (s64)xtime.tv_nsec << clock->shift; |
480 | 492 | ||
493 | update_xtime_cache(cyc2ns(clock, offset)); | ||
494 | |||
481 | /* check to see if there is a new clocksource to use */ | 495 | /* check to see if there is a new clocksource to use */ |
482 | change_clocksource(); | 496 | change_clocksource(); |
483 | update_vsyscall(&xtime, clock); | 497 | update_vsyscall(&xtime, clock); |
@@ -510,6 +524,13 @@ void monotonic_to_bootbased(struct timespec *ts) | |||
510 | ts->tv_sec += total_sleep_time; | 524 | ts->tv_sec += total_sleep_time; |
511 | } | 525 | } |
512 | 526 | ||
527 | unsigned long get_seconds(void) | ||
528 | { | ||
529 | return xtime_cache.tv_sec; | ||
530 | } | ||
531 | EXPORT_SYMBOL(get_seconds); | ||
532 | |||
533 | |||
513 | struct timespec current_kernel_time(void) | 534 | struct timespec current_kernel_time(void) |
514 | { | 535 | { |
515 | struct timespec now; | 536 | struct timespec now; |
@@ -518,10 +539,9 @@ struct timespec current_kernel_time(void) | |||
518 | do { | 539 | do { |
519 | seq = read_seqbegin(&xtime_lock); | 540 | seq = read_seqbegin(&xtime_lock); |
520 | 541 | ||
521 | now = xtime; | 542 | now = xtime_cache; |
522 | } while (read_seqretry(&xtime_lock, seq)); | 543 | } while (read_seqretry(&xtime_lock, seq)); |
523 | 544 | ||
524 | return now; | 545 | return now; |
525 | } | 546 | } |
526 | |||
527 | EXPORT_SYMBOL(current_kernel_time); | 547 | EXPORT_SYMBOL(current_kernel_time); |