|
@@ -291,6 +291,26 @@ int pvclock_gtod_unregister_notifier(struct notifier_block *nb)
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(pvclock_gtod_unregister_notifier);
|
|
EXPORT_SYMBOL_GPL(pvclock_gtod_unregister_notifier);
|
|
|
|
|
|
|
|
+/*
|
|
|
|
+ * Update the ktime_t based scalar nsec members of the timekeeper
|
|
|
|
+ */
|
|
|
|
+static inline void tk_update_ktime_data(struct timekeeper *tk)
|
|
|
|
+{
|
|
|
|
+ s64 nsec;
|
|
|
|
+
|
|
|
|
+ /*
|
|
|
|
+ * The xtime based monotonic readout is:
|
|
|
|
+ * nsec = (xtime_sec + wtm_sec) * 1e9 + wtm_nsec + now();
|
|
|
|
+ * The ktime based monotonic readout is:
|
|
|
|
+ * nsec = base_mono + now();
|
|
|
|
+ * ==> base_mono = (xtime_sec + wtm_sec) * 1e9 + wtm_nsec
|
|
|
|
+ */
|
|
|
|
+ nsec = (s64)(tk->xtime_sec + tk->wall_to_monotonic.tv_sec);
|
|
|
|
+ nsec *= NSEC_PER_SEC;
|
|
|
|
+ nsec += tk->wall_to_monotonic.tv_nsec;
|
|
|
|
+ tk->base_mono = ns_to_ktime(nsec);
|
|
|
|
+}
|
|
|
|
+
|
|
/* must hold timekeeper_lock */
|
|
/* must hold timekeeper_lock */
|
|
static void timekeeping_update(struct timekeeper *tk, unsigned int action)
|
|
static void timekeeping_update(struct timekeeper *tk, unsigned int action)
|
|
{
|
|
{
|
|
@@ -301,6 +321,8 @@ static void timekeeping_update(struct timekeeper *tk, unsigned int action)
|
|
update_vsyscall(tk);
|
|
update_vsyscall(tk);
|
|
update_pvclock_gtod(tk, action & TK_CLOCK_WAS_SET);
|
|
update_pvclock_gtod(tk, action & TK_CLOCK_WAS_SET);
|
|
|
|
|
|
|
|
+ tk_update_ktime_data(tk);
|
|
|
|
+
|
|
if (action & TK_MIRROR)
|
|
if (action & TK_MIRROR)
|
|
memcpy(&shadow_timekeeper, &tk_core.timekeeper,
|
|
memcpy(&shadow_timekeeper, &tk_core.timekeeper,
|
|
sizeof(tk_core.timekeeper));
|
|
sizeof(tk_core.timekeeper));
|