All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.internal.time.MonotonicClock Maven / Gradle / Ivy

/*
 * Copyright 2017 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.gradle.internal.time;

import com.google.common.annotations.VisibleForTesting;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;

/**
 * A clock that is guaranteed to not go backwards.
 * 

* It aims to strike a balance between never going backwards (allowing timestamps to represent causality) * and keeping in sync with the system wall clock so that time values make sense in comparison with the system wall clock, * including timestamps generated from other processes. *

* This clock effectively measures time by duration (according to System.nanoTime()), * in between syncs with the system wall clock. * When issuing the first timestamp after the sync interval has expired, * The system wall clock will be read, and the current time set to the max of wall clock time or the most recently issued timestamp. * All other timestamps are calculated as the wall clock time at last sync + elapsed time since. *

* This clock deals relatively well when the system wall clock shift is adjusted by small amounts. * It also deals relatively well when the system wall clock jumps forward by large amounts (this clock will jump with it). * It does not deal as well with large jumps back in time. *

* When the system wall clock jumps back in time, this clock will effectively slow down until it is back in sync. * All syncing timestamps will be the same as the previously issued timestamp. * The rate by which this clock slows, and therefore the time it takes to resync, * is determined by how frequently the clock is read. * If timestamps are only requested at a rate greater than the sync interval, * all timestamps will have the same value until the clocks synchronize (i.e. this clock will pause). * If timestamps are requested more frequently than the sync interval, * timestamps before and after the sync point will under represent the actual elapsed time, * gradually bringing the clocks back into sync. */ class MonotonicClock implements Clock { private static final long SYNC_INTERVAL_MILLIS = TimeUnit.SECONDS.toMillis(3); private final long syncIntervalMillis; private final TimeSource timeSource; private final AtomicLong syncMillisRef; private final AtomicLong syncNanosRef; private final AtomicLong currentTime = new AtomicLong(0); MonotonicClock() { this(TimeSource.SYSTEM, SYNC_INTERVAL_MILLIS); } @VisibleForTesting MonotonicClock(TimeSource timeSource, long syncIntervalMillis) { long nanoTime = timeSource.nanoTime(); long currentTimeMillis = timeSource.currentTimeMillis(); this.timeSource = timeSource; this.syncIntervalMillis = syncIntervalMillis; this.syncNanosRef = new AtomicLong(nanoTime); this.syncMillisRef = new AtomicLong(currentTimeMillis); this.currentTime.set(currentTimeMillis); } @Override public long getCurrentTime() { long nowNanos = timeSource.nanoTime(); long syncNanos = syncNanosRef.get(); long syncMillis = syncMillisRef.get(); long sinceSyncNanos = nowNanos - syncNanos; long sinceSyncMillis = TimeUnit.NANOSECONDS.toMillis(sinceSyncNanos); if (syncIsDue(nowNanos, syncNanos, sinceSyncMillis)) { return sync(syncMillis); } else { return advance(syncMillis + sinceSyncMillis); } } private boolean syncIsDue(long nowNanos, long syncNanos, long sinceSyncMillis) { return sinceSyncMillis >= syncIntervalMillis && syncNanosRef.compareAndSet(syncNanos, nowNanos); } /** * Syncs our internal clock with the system clock and returns the new time. * Marks the current time as the last synchronization point, unless another thread already did a synchronization in the meantime. */ private long sync(long syncMillis) { long newSyncMillis = advance(timeSource.currentTimeMillis()); syncMillisRef.compareAndSet(syncMillis, newSyncMillis); return newSyncMillis; } /** * Advance the clock to the given timestamp and return the new time. * The returned time may not be the one passed in, in case another thread already advanced the clock further. * This ensures that all threads share a consistent time. */ private long advance(long newTime) { while (true) { long current = currentTime.get(); if (newTime <= current) { return current; } else if (currentTime.compareAndSet(current, newTime)) { return newTime; } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy