Commit a105e023 authored by Thomas Gleixner's avatar Thomas Gleixner Committed by Ben Hutchings

timekeeping: Maintain ktime_t based offsets for hrtimers

This is a backport of 5b9fe759

We need to update the hrtimer clock offsets from the hrtimer interrupt
context. To avoid conversions from timespec to ktime_t maintain a
ktime_t based representation of those offsets in the timekeeper. This
puts the conversion overhead into the code which updates the
underlying offsets and provides fast accessible values in the hrtimer
interrupt.
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Signed-off-by: default avatarJohn Stultz <johnstul@us.ibm.com>
Reviewed-by: default avatarIngo Molnar <mingo@kernel.org>
Acked-by: default avatarPeter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: default avatarPrarit Bhargava <prarit@redhat.com>
Link: http://lkml.kernel.org/r/1341960205-56738-4-git-send-email-johnstul@us.ibm.comSigned-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
[John Stultz: Backported to 3.2]
Cc: Prarit Bhargava <prarit@redhat.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Linux Kernel <linux-kernel@vger.kernel.org>
Signed-off-by: default avatarJohn Stultz <johnstul@us.ibm.com>
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 8a1ba973
...@@ -161,11 +161,26 @@ static struct timespec xtime __attribute__ ((aligned (16))); ...@@ -161,11 +161,26 @@ static struct timespec xtime __attribute__ ((aligned (16)));
static struct timespec wall_to_monotonic __attribute__ ((aligned (16))); static struct timespec wall_to_monotonic __attribute__ ((aligned (16)));
static struct timespec total_sleep_time; static struct timespec total_sleep_time;
/* Offset clock monotonic -> clock realtime */
static ktime_t offs_real;
/* Offset clock monotonic -> clock boottime */
static ktime_t offs_boot;
/* /*
* The raw monotonic time for the CLOCK_MONOTONIC_RAW posix clock. * The raw monotonic time for the CLOCK_MONOTONIC_RAW posix clock.
*/ */
static struct timespec raw_time; static struct timespec raw_time;
/* must hold write on xtime_lock */
static void update_rt_offset(void)
{
struct timespec tmp, *wtm = &wall_to_monotonic;
set_normalized_timespec(&tmp, -wtm->tv_sec, -wtm->tv_nsec);
offs_real = timespec_to_ktime(tmp);
}
/* must hold write on xtime_lock */ /* must hold write on xtime_lock */
static void timekeeping_update(bool clearntp) static void timekeeping_update(bool clearntp)
{ {
...@@ -173,6 +188,7 @@ static void timekeeping_update(bool clearntp) ...@@ -173,6 +188,7 @@ static void timekeeping_update(bool clearntp)
timekeeper.ntp_error = 0; timekeeper.ntp_error = 0;
ntp_clear(); ntp_clear();
} }
update_rt_offset();
update_vsyscall(&xtime, &wall_to_monotonic, update_vsyscall(&xtime, &wall_to_monotonic,
timekeeper.clock, timekeeper.mult); timekeeper.clock, timekeeper.mult);
} }
...@@ -587,6 +603,7 @@ void __init timekeeping_init(void) ...@@ -587,6 +603,7 @@ void __init timekeeping_init(void)
} }
set_normalized_timespec(&wall_to_monotonic, set_normalized_timespec(&wall_to_monotonic,
-boot.tv_sec, -boot.tv_nsec); -boot.tv_sec, -boot.tv_nsec);
update_rt_offset();
total_sleep_time.tv_sec = 0; total_sleep_time.tv_sec = 0;
total_sleep_time.tv_nsec = 0; total_sleep_time.tv_nsec = 0;
write_sequnlock_irqrestore(&xtime_lock, flags); write_sequnlock_irqrestore(&xtime_lock, flags);
...@@ -595,6 +612,12 @@ void __init timekeeping_init(void) ...@@ -595,6 +612,12 @@ void __init timekeeping_init(void)
/* time in seconds when suspend began */ /* time in seconds when suspend began */
static struct timespec timekeeping_suspend_time; static struct timespec timekeeping_suspend_time;
static void update_sleep_time(struct timespec t)
{
total_sleep_time = t;
offs_boot = timespec_to_ktime(t);
}
/** /**
* __timekeeping_inject_sleeptime - Internal function to add sleep interval * __timekeeping_inject_sleeptime - Internal function to add sleep interval
* @delta: pointer to a timespec delta value * @delta: pointer to a timespec delta value
...@@ -612,7 +635,7 @@ static void __timekeeping_inject_sleeptime(struct timespec *delta) ...@@ -612,7 +635,7 @@ static void __timekeeping_inject_sleeptime(struct timespec *delta)
xtime = timespec_add(xtime, *delta); xtime = timespec_add(xtime, *delta);
wall_to_monotonic = timespec_sub(wall_to_monotonic, *delta); wall_to_monotonic = timespec_sub(wall_to_monotonic, *delta);
total_sleep_time = timespec_add(total_sleep_time, *delta); update_sleep_time(timespec_add(total_sleep_time, *delta));
} }
......
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