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

swim.concurrent.Clock Maven / Gradle / Ivy

// Copyright 2015-2020 SWIM.AI inc.
//
// 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 swim.concurrent;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;

/**
 * Hashed wheel timer {@link Schedule}.
 */
public class Clock implements Schedule {

  /**
   * Default number of milliseconds between clock ticks, used by the no-arg
   * {@link #Clock()} constructor.  Defaults to the value of the {@code
   * swim.clock.tick.millis} system property, if defined; otherwise defaults to
   * {@code 100} milliseconds.
   */
  public static final int TICK_MILLIS;
  /**
   * Default number of ticks per clock revolution, used by the no-arg {@link
   * #Clock()} constructor.  Defaults to the value of the {@code
   * swim.clock.tick.count} system property, if defined; otherwise defaults to
   * {@code 512} clock ticks per revolution.
   */
  public static final int TICK_COUNT;
  /**
   * Atomic {@link #status} bit flag indicating that the clock has started, and
   * is currently running.
   */
  static final int STARTED = 1 << 0;
  /**
   * Atomic {@link #status} bit flag indicating that the clock had previously
   * started, but is now permanently stopped.
   */
  static final int STOPPED = 1 << 1;
  /**
   * Atomic {@link #status} field updater, used to linearize clock startup and
   * shutdown.
   */
  static final AtomicIntegerFieldUpdater STATUS =
      AtomicIntegerFieldUpdater.newUpdater(Clock.class, "status");

  static {
    // Initializes the default number of milliseconds between clock ticks.
    int tickMillis;
    try {
      tickMillis = Integer.parseInt(System.getProperty("swim.clock.tick.millis"));
    } catch (NumberFormatException e) {
      tickMillis = 100;
    }
    TICK_MILLIS = tickMillis;

    // Initialize the default number of ticks per clock revolution.
    int tickCount;
    try {
      tickCount = Integer.parseInt(System.getProperty("swim.clock.tick.count"));
    } catch (NumberFormatException e) {
      tickCount = 512;
    }
    TICK_COUNT = tickCount;
  }

  /**
   * Immutable array of {@link #tickCount} timer buckets, each containing a
   * lock-free queue of timer events to execute for a particular modulus of
   * clock ticks.
   */
  final ClockQueue[] dial;
  /**
   * Barrier used to sequence clock startup.
   */
  final CountDownLatch startLatch;
  /**
   * Barrier used to sequence clock shutdown.
   */
  final CountDownLatch stopLatch;
  /**
   * Thread that executes timer events at their scheduled times.
   */
  final ClockThread thread;
  /**
   * Number of nanoseconds between successive clock ticks.
   */
  final long tickNanos;
  /**
   * Number of ticks per clock revolution.
   */
  final int tickCount;
  /**
   * Time at which the clock started, in nanoseconds, with arbitrary origin.
   * Set exactly once when the clock thread starts.
   */
  volatile long startTime;
  /**
   * Atomic bit field with {@link #STARTED} and {@link #STOPPED} flags.
   */
  volatile int status;

  /**
   * Constructs a new {@code Clock} with a timer resolution of {@code
   * tickMillis} milliseconds, and a clock period of {@code tickCount} ticks
   * per revolution.
   */
  public Clock(int tickMillis, int tickCount) {
    // Initialize the number of nanoseconds between clock ticks.
    if (tickMillis <= 0) {
      throw new IllegalArgumentException(Long.toString(tickMillis));
    }
    this.tickNanos = (long) tickMillis * 1000000L;

    // Initialize the number of ticks per clock revolution.
    if (tickCount <= 0) {
      throw new IllegalArgumentException(Integer.toString(tickCount));
    }
    // Round the tick count up to the next power of two.
    tickCount = tickCount - 1;
    tickCount |= tickCount >> 1;
    tickCount |= tickCount >> 2;
    tickCount |= tickCount >> 4;
    tickCount |= tickCount >> 8;
    tickCount |= tickCount >> 16;
    tickCount = tickCount + 1;
    this.tickCount = tickCount;

    // Initialize the clock dial with one revolution worth of clock ticks.
    this.dial = new ClockQueue[tickCount];
    for (int i = 0; i < tickCount; i += 1) {
      this.dial[i] = new ClockQueue((long) i);
    }

    // Initialize the barrier used to sequence clock startup.
    this.startLatch = new CountDownLatch(1);

    // Initialize the barrier used to sequence clock shutdown.
    this.stopLatch = new CountDownLatch(1);

    // Initialize--but don't start--the clock thread.
    this.thread = new ClockThread(this);
  }

  /**
   * Constructs a new {@code Clock} with the timer resolution and clock period
   * specified by the given {@code clockDef}.
   */
  public Clock(ClockDef clockDef) {
    this(clockDef.tickMillis, clockDef.tickCount);
  }

  /**
   * Constructs a new {@code Clock} with a timer resolution of {@link
   * #TICK_MILLIS} milliseconds, and a clock period of {@link #TICK_COUNT}
   * ticks per revolution.
   */
  public Clock() {
    this(TICK_MILLIS, TICK_COUNT);
  }

  /**
   * Returns the tick sequence number of the lowest clock tick that has yet to
   * finish executing.
   */
  public final long tick() {
    return this.thread.tick;
  }

  /**
   * Ensures that this {@code Clock} is up and running, starting up the clock
   * thread if it has not yet been started.
   *
   * @throws ScheduleException if this {@code Clock} has been stopped.
   */
  public final void start() {
    do {
      final int oldStatus = STATUS.get(this);
      if ((oldStatus & STOPPED) == 0) {
        // Clock hasn't yet stopped; make sure it has started.
        if ((oldStatus & STARTED) == 0) {
          final int newStatus = oldStatus | STARTED;
          // Try to set the STARTED flag; linearization point for clock startup.
          if (STATUS.compareAndSet(this, oldStatus, newStatus)) {
            // Initiate clock thread startup.
            willStart();
            this.thread.start();
            break;
          }
        } else {
          // Clock thread already started.
          break;
        }
      } else {
        throw new ScheduleException("Can't restart stopped clock");
      }
    } while (true);

    // Loop while the clock thread is not yet up and running.
    boolean interrupted = false;
    while (this.startLatch.getCount() != 0) {
      try {
        // Wait for clock thread startup to complete.
        this.startLatch.await();
      } catch (InterruptedException error) {
        interrupted = true;
      }
    }
    if (interrupted) {
      Thread.currentThread().interrupt();
    }
  }

  /**
   * Ensures that this {@code Clock} has been permanently stopped, shutting
   * down the clock thread, if it's currently running.  Upon return, this
   * {@code Clock} is guaranteed to be in the stopped state.
   */
  public final void stop() {
    // Clock hasn't been started
    if ((STATUS.get(this) & STARTED) == 0) {
      return;
    }

    boolean interrupted = false;
    do {
      final int oldStatus = STATUS.get(this);
      if ((oldStatus & STOPPED) == 0) {
        // Clock hasn't yet stopped; try to stop it.
        final int newStatus = oldStatus | STOPPED;
        // Try to set the STOPPED flag; linearization point for clock shutdown.
        if (STATUS.compareAndSet(this, oldStatus, newStatus)) {
          // Loop while the clock thread is still running.
          while (this.thread.isAlive()) {
            // Interrupt the clock thread so it will wakeup and die.
            this.thread.interrupt();
            try {
              // Wait for the clock thread to exit.
              this.thread.join(100);
            } catch (InterruptedException error) {
              interrupted = true;
            }
          }
        }
      } else {
        // Clock thread already stopped.
        break;
      }
    } while (true);

    // Loop while the clock thread is still running.
    while (this.stopLatch.getCount() != 0) {
      try {
        // Wait for clock thread shutdown to complete.
        this.stopLatch.await();
      } catch (InterruptedException e) {
        interrupted = true;
      }
    }
    if (interrupted) {
      Thread.currentThread().interrupt();
    }
  }

  @Override
  public TimerRef timer(TimerFunction timer) {
    // Ensure that the clock has started.
    start();

    // Create the context that binds the timer to this clock.
    final ClockTimer context = new ClockTimer(this, timer);
    if (timer instanceof Timer) {
      ((Timer) timer).setTimerContext(context);
    }

    // Return the timer context.
    return context;
  }

  @Override
  public TimerRef setTimer(long millis, TimerFunction timer) {
    if (millis < 0L) {
      throw new TimerException("negative timeout: " + Long.toString(millis));
    }

    // Ensure that the clock has started.
    start();

    // Create the context that binds the timer to this clock.
    final ClockTimer context = new ClockTimer(this, timer);
    if (timer instanceof Timer) {
      ((Timer) timer).setTimerContext(context);
    }

    // Schedule the timer for execution.
    schedule(millis, context);

    // Return the timer context.
    return context;
  }

  /**
   * Schedules a bound timer {@code context} for execution after {@code millis}
   * milliseconds has elapsed.
   */
  final void schedule(long millis, ClockTimer context) {
    // Invoke timer scheduling introspection callbacks.
    timerWillSchedule(context.timer, millis);
    if (context.timer instanceof Timer) {
      ((Timer) context.timer).timerWillSchedule(millis);
    }

    // Convert the timeout to nanoseconds.
    final long nanos = millis * 1000000L;
    // Compute the deadline for the timer in nanoseconds since clock start.
    final long deadline = Math.max(0L, nanoTime() + nanos - this.startTime);
    // Divide the deadline by the tick interval to get the tick sequence number
    // at which to fire the timer, rounding up to the next tick.
    long targetTick = (deadline + (this.tickNanos - 1L)) / this.tickNanos;
    // Take the modulus of the target tick with respect to to the number of
    // ticks per clock revolution, yielding the index in the dial at which to
    // insert the event.
    int targetHand = (int) (targetTick % (long) this.tickCount);

    // Create a timer event to insert into the clock.
    final ClockEvent newEvent = new ClockEvent(0L, targetTick, context, context.timer);
    // Atomically get the current timer event, and replace it with the
    // to-be-scheduled event; linearization point for timer un-cancellation.
    final ClockEvent oldEvent = ClockTimer.EVENT.getAndSet(context, newEvent);
    // Check if the timer had a previously scheduled event.
    if (oldEvent != null) {
      // Remove the timer from the previously scheduled event;
      // linearization point for timer cancellation.
      oldEvent.cancel();
    }

    // Get the event queue for the target hand of the clock.
    ClockQueue queue = this.dial[targetHand];
    // Capture the current foot of the queue.
    ClockEvent foot = queue.foot;
    // Search for the last event of in the queue, starting with foot.
    ClockEvent prev = foot;
    // Loop until the event is inserted into the first queue that will execute
    // after the timer deadline.
    do {
      // Load the next event after the currently referenced last event.
      final ClockEvent next = prev.next;
      if (next == null) {
        // prev is the last event in the queue.
        if (targetTick >= prev.insertTick) {
          // prev was inserted before the target tick, indicating that the
          // timer thread hasn't finished executing the target tick yet.
          // prev.insertTick is the next tick sequence number that the clock
          // thread will execute for the queue; set event.insertTick to match.
          newEvent.insertTick = prev.insertTick;
          // Try to insert the new event to the end of the queue;
          // linearization point for timer scheduling.
          if (ClockEvent.NEXT.compareAndSet(prev, null, newEvent)) {
            // Only update the foot reference if it lags at least two events
            // behind the last event in the queue.
            if (prev != foot) {
              // Try to update the foot reference; ok if this fails.
              ClockQueue.FOOT.compareAndSet(queue, foot, newEvent);
            }
            break;
          }
          // Lost insertion race to another thread; try again.
        } else {
          // The clock thread is currently executing, or has already executed,
          // the target tick; try the next hand of the clock.
          targetTick += 1L;
          targetHand = (int) (targetTick % (long) this.tickCount);
          queue = this.dial[targetHand];
          foot = queue.foot;
          prev = foot;
        }
      } else {
        // Jump to the new foot, if the previously loaded foot lags at least two
        // events behind the prev event; otherwise advance to the next event.
        final ClockEvent newFoot = queue.foot;
        if (foot != newFoot) {
          foot = newFoot;
          prev = foot;
        } else {
          prev = next;
        }
      }
    } while (true);
  }

  /**
   * Lifecycle callback invoked before the clock thread starts.
   */
  protected void willStart() {
    // stub
  }

  /**
   * Lifecycle callback invoked after the clock thread starts.
   */
  protected void didStart() {
    // stub
  }

  /**
   * Introspection callback invoked after each tick of the clock.  {@code tick}
   * is the sequence number of the tick that was executed; {@code waitedMillis}
   * is the number of milliseconds the clock thread slept before executing the
   * tick.  If {@code waitedMillis} is negative, then the clock thread didn't
   * start executing the tick until {@code -waitedMillis} milliseconds after
   * the scheduled tick deadline.
   */
  protected void didTick(long tick, long waitedMillis) {
    // stub
  }

  /**
   * Lifecycle callback invoked before the clock thread stops.
   */
  protected void willStop() {
    // stub
  }

  /**
   * Lifecycle callback invoked after the clock thread stops.
   */
  protected void didStop() {
    // stub
  }

  /**
   * Lifecycle callback invoked if the timer thread throws a fatal {@code
   * error}.  The clock thread will stop after invoking {@code didFail}.
   */
  protected void didFail(Throwable error) {
    error.printStackTrace();
  }

  /**
   * Introspection callback invoked before a {@code timer} is scheduled for
   * execution with a delay of {@code millis} milliseconds.
   */
  protected void timerWillSchedule(TimerFunction timer, long millis) {
    // stub
  }

  /**
   * Introspection callback invoked after a {@code timer} has been explicitly
   * cancelled; not invoked when a timer is implicitly cancelled, such as when
   * rescheduling an already scheduled timer.
   */
  protected void timerDidCancel(TimerFunction timer) {
    // stub
  }

  /**
   * Introspection callback invoked before a {@code timer} is executed.
   */
  protected void timerWillRun(TimerFunction timer) {
    // stub
  }

  /**
   * Invokes {@code timer.runTimer()}, or arranges for the asynchronous
   * execution of the provided {@code runnable}, which will itself invoke
   * {@code timer.runTimer()}.
   */
  protected void runTimer(TimerFunction timer, Runnable runnable) {
    timer.runTimer();
  }

  /**
   * Introspection callback invoked after a {@code timer} executes nominally.
   */
  protected void timerDidRun(TimerFunction timer) {
    // stub
  }

  /**
   * Introspection callback invoked after a {@code timer} execution fails by
   * throwing an {@code error}.
   */
  protected void timerDidFail(TimerFunction timer, Throwable error) {
    // stub
  }

  /**
   * Returns the current time, in nanoseconds, with arbitrary origin.
   * Used by the clock thread to determine the current time.  Defaults
   * to {@link System#nanoTime()}.  Can be overridden to substitute an
   * alternative time source.
   */
  protected long nanoTime() {
    return System.nanoTime();
  }

  /**
   * Parks the current thread for the specified number of {@code millis}.
   * Used by the clock thread to wait for the next clock tick.  Defaults
   * to {@link Thread#sleep(long)}.  Can be overridden to substitute an
   * alternative wait mechanism.
   */
  protected void sleep(long millis) throws InterruptedException {
    Thread.sleep(millis);
  }

}

/**
 * Context that binds a {@code TimerFunction} to a {@code Clock}, and manages
 * the scheduling of at most one live {@code ClockEvent} at a time.
 */
final class ClockTimer implements TimerContext {

  /**
   * Atomic {@link #event} field updater, used to linearize cancellation and
   * rescheduling of the {@code timer}.
   */
  static final AtomicReferenceFieldUpdater EVENT =
      AtomicReferenceFieldUpdater.newUpdater(ClockTimer.class, ClockEvent.class, "event");
  /**
   * {@code Clock} to which the {@code timer} is bound.
   */
  final Clock clock;
  /**
   * {@code TimerFunction} to invoke when a scheduled event fires.
   */
  final TimerFunction timer;
  /**
   * Atomic reference to the currently scheduled event that will execute the
   * {@code timer} when fired; {@code null} when the {@code timer} is not
   * currently scheduled.
   */
  volatile ClockEvent event;

  /**
   * Constructs a new {@code ClockTimer} that binds the {@code timer} to the
   * {@code clock}.
   */
  ClockTimer(Clock clock, TimerFunction timer) {
    this.clock = clock;
    this.timer = timer;
  }

  @Override
  public Schedule schedule() {
    return this.clock;
  }

  @Override
  public boolean isScheduled() {
    final ClockEvent event = EVENT.get(this);
    return event != null && event.isScheduled();
  }

  @Override
  public void reschedule(long millis) {
    if (millis < 0L) {
      throw new TimerException("negative timeout: " + Long.toString(millis));
    }
    this.clock.schedule(millis, this);
  }

  @Override
  public boolean cancel() {
    // Atomically get the current timer event, and replace it with null.
    final ClockEvent event = EVENT.getAndSet(this, null);
    // Check if the timer has a previously scheduled event.
    if (event != null) {
      // Remove the timer from the previously scheduled timer event;
      // linearization point for timer cancellation.
      final TimerFunction timer = event.cancel();
      // Check if the timer event hadn't yet been fired or cancelled.
      if (timer != null) {
        // Invoke timer cancellation introspection callbacks.
        if (timer instanceof Timer) {
          ((Timer) timer).timerDidCancel();
        }
        clock.timerDidCancel(timer);
        return true;
      }
    }
    return false;
  }

}

/**
 * Lock-free, multiple publisher, single consumer queue of linked {@code
 * ClockEvent} items.
 */
final class ClockQueue {

  /**
   * Atomic {@link #foot} field updater, used to optimize event insertion.
   */
  static final AtomicReferenceFieldUpdater FOOT =
      AtomicReferenceFieldUpdater.newUpdater(ClockQueue.class, ClockEvent.class, "foot");
  /**
   * First {@code ClockEvent} in the queue, from which all live events in the
   * queue are reachable; always non-{@code null}.  Only the clock thread is
   * permitted to advance the head of the queue.
   */
  volatile ClockEvent head;
  /**
   * Atomic reference from which the last {@code ClockEvent} in the queue can
   * be reached in constant time; always non-{@code null}.
   */
  volatile ClockEvent foot;

  /**
   * Constructs a new {@code ClockQueue} that will next execute the {@code
   * insertTick} sequence number.
   */
  ClockQueue(long insertTick) {
    this.head = new ClockEvent(insertTick, insertTick, null, null);
    this.foot = head;
  }

}

/**
 * Linked {@code ClockQueue} item holding a {@code TimerFunction} to execute.
 */
final class ClockEvent implements Runnable {

  /**
   * Atomic {@link #timer} field updater, used to linearize event cancellation.
   */
  static final AtomicReferenceFieldUpdater TIMER =
      AtomicReferenceFieldUpdater.newUpdater(ClockEvent.class, TimerFunction.class, "timer");
  /**
   * Atomic {@link #next} field updater, used to linearize event scheduling.
   */
  static final AtomicReferenceFieldUpdater NEXT =
      AtomicReferenceFieldUpdater.newUpdater(ClockEvent.class, ClockEvent.class, "next");
  /**
   * {@code ClockTimer} on behalf of whom this {@code ClockEvent} is scheduled,
   * or {@code null} if this is a placeholder event.
   */
  final ClockTimer context;
  /**
   * Tick sequence number of the next tick that the clock thread will execute
   * at the time this {@code ClockEvent} was inserted into the queue.  {@code
   * insertTick} is non-decreasing over successive queued events, and the
   * {@code insertTick} of the last event in the queue always represents the
   * very next tick that the clock thread will execute.
   */
  long insertTick;
  /**
   * Tick sequence number during which to fire this event.
   */
  long targetTick;
  /**
   * Atomic reference to the {@code TimerFunction} to invoke when firing this
   * event; {@code null} when this event has been cancelled.
   */
  volatile TimerFunction timer;
  /**
   * Next {@code ClockEvent} in the linked queue; {@code null} if this is the
   * last event in the queue.
   */
  volatile ClockEvent next;

  /**
   * Constructs a new {@code ClockEvent} that will fire the {@code timer}
   * at the {@code targetTick} sequence number, and will be inserted into the
   * queue during the {@code insertTick} sequence number.
   */
  ClockEvent(long insertTick, long targetTick, ClockTimer context, TimerFunction timer) {
    this.insertTick = insertTick;
    this.targetTick = targetTick;
    this.context = context;
    this.timer = timer;
  }

  /**
   * Returns {@code true} if the {@link #timer} is non-{@code null}, indicating
   * that this {@code ClockEvent} is scheduled for execution.
   */
  boolean isScheduled() {
    return TIMER.get(this) != null;
  }

  /**
   * Atomically gets the scheduled {@link #timer}, and replaces it with {@code
   * null}, thereby preventing future execution, or concurrent cancellation.
   */
  TimerFunction cancel() {
    // Atomically get the scheduled timer, and replace it with null;
    // linearization point for timer execution and cancellation.
    return TIMER.getAndSet(this, null);
  }

  /**
   * Invokes the timer function of the associated timer context.
   */
  @Override
  public void run() {
    this.context.timer.runTimer();
  }

}

/**
 * Thread of execution that fires clock events at the appropriate times.
 */
final class ClockThread extends Thread {

  /**
   * Total number of clock threads that have ever been instantiated.  Used to
   * uniquely name clock threads.
   */
  static final AtomicInteger THREAD_COUNT = new AtomicInteger(0);
  /**
   * {@code Clock} whose events this {@code ClockThread} fires.
   */
  final Clock clock;
  /**
   * Next tick sequence number that this {@code ClockThread} will execute.
   */
  long tick;

  /**
   * Constructs a new {@code ClockThread} that fires events for {@code clock}.
   */
  ClockThread(Clock clock) {
    setName("SwimClock" + THREAD_COUNT.getAndIncrement());
    setDaemon(true);
    this.clock = clock;
  }

  /**
   * Parks the clock thread until the {@code clock} time has reached the
   * elapsed time of the taregt {@code tick}.  Returns the total number of
   * milliseconds waited; the returned wait time can be negative if it's
   * taking longer than the clock's tick interval to execute timers.
   * Returns {@code Long.MIN_VALUE} if the clock was stopped while waiting
   * for the target {@code tick}.
   */
  static long waitForTick(final Clock clock, final long tick) {
    // The clock elapsed time of the target tick.
    final long deadline = clock.tickNanos * tick;
    // The clock elapsed time of the first wait for the target tick.
    final long initialTime = clock.nanoTime() - clock.startTime;
    // The current clock elapsed time.
    long currentTime = initialTime;
    do {
      // Check for elapsed time overflow.
      if (currentTime < 0L) {
        // Can't run for longer than about 292 years.
        throw new InternalError("Clock elapsed time overflow");
      }
      // Calculate the number of milliseconds until the deadline for the target
      // tick, rounding up to the next millisecond.
      final long sleepMillis = ((deadline - currentTime) + 999999L) / 1000000L;
      // Check if the deadline for the target tick is in the future.
      if (sleepMillis > 0L) {
        // Park the timer thread for the number of milliseconds until the
        // deadline for the target tick.
        try {
          clock.sleep(sleepMillis);
        } catch (InterruptedException e) {
          // Interrupted while waiting for the target tick; check if the clock
          // has been stopped.
          if ((Clock.STATUS.get(clock) & Clock.STOPPED) != 0) {
            // Return a sentinel value to signal that the clock stopped.
            return Long.MIN_VALUE;
          }
        }
        // Recompute the current clock elapsed time.
        currentTime = clock.nanoTime() - clock.startTime;
      } else {
        // We've reached the deadline for the target tick; recompute the
        // current clock elapsed time.
        currentTime = clock.nanoTime() - clock.startTime;
        // Compute the total number of milliseconds we waited.
        return (currentTime - initialTime) / 1000000L;
      }
    } while (true);
  }

  /**
   * Executes all {@code clock} timers set to fire at the target {@code tick}.
   */
  static void executeTick(final Clock clock, final long tick) {
    // Compute the tick sequence number for the next revolution of the clock.
    final long nextTick = tick + (long) clock.tickCount;
    // Compute the index in the clock dial of the target clock tick.
    final int hand = (int) (tick % (long) clock.tickCount);

    // Get the event queue for the target hand of the clock dial.
    final ClockQueue queue = clock.dial[hand];
    // The first known still scheduled event to keep in the queue.
    ClockEvent head = null;
    // The last known still scheduled event to keep in the queue.
    ClockEvent prev = null;
    // The next queued event to process.
    ClockEvent next = queue.head;
    // The sentinel event that will be inserted at the end of the queue to
    // complete the execution of this tick.
    final ClockEvent nextFoot = new ClockEvent(nextTick, nextTick, null, null);
    // Loop until no events scheduled for this tick remain in the queue.
    do {
      if (next.targetTick <= tick) {
        // The next event is scheduled for this tick; remove its timer.
        final TimerFunction timer = next.cancel();
        // Clear the event from the associated timer context.
        if (next.context != null) {
          ClockTimer.EVENT.compareAndSet(next.context, next, null);
        }
        if (timer != null) {
          // The timer wasn't cancelled; fire the event.
          try {
            clock.timerWillRun(timer);
            clock.runTimer(timer, next);
            clock.timerDidRun(timer);
          } catch (Throwable error) {
            if (Conts.isNonFatal(error)) {
              // The timer failed with a non-fatal error.
              clock.timerDidFail(timer, error);
            } else {
              // The timer failed with a fatal error.
              throw error;
            }
          }
        }
      } else if (next.isScheduled()) {
        // The next event is scheduled for a future revolution of the clock.
        if (prev != null) {
          // Insert the next event after the last kept event in the queue,
          // bypassing any fired or cancelled events.
          prev.next = next;
        } else {
          // The next event is the first event to keep in the queue.
          head = next;
        }
        // The next event is now the last known event to keep in the queue.
        prev = next;
      }
      // Check if the next event is the last in the queue.
      if (next.next == null) {
        // Try to finish tick execution by appending a cancelled event to the
        // end of the queue, whose insertTick is the next tick sequence number
        // that the clock thread will execute for this queue, preventing
        // further scheduling of events for the current tick.
        if (ClockEvent.NEXT.compareAndSet(next, null, nextFoot)) {
          // All events that will ever be scheduled for this tick have now been
          // cancelled or fired; update the foot of the queue to reference the
          // new foot event.
          ClockQueue.FOOT.set(queue, nextFoot);
          // Check if no events were kept in the queue.
          if (head == null) {
            // In which case the new foot is also the new head.
            head = nextFoot;
          }
          // Update the head of the queue.
          queue.head = head;
          break;
        }
      }
      // Advance to the next event in the queue.
      next = next.next;
    } while (true);
  }

  @Override
  public void run() {
    final Clock clock = this.clock;

    try {
      // Initialize the relative clock start time.
      long startTime = clock.nanoTime();
      if (startTime == 0L) {
        // Avoid clash with sentinel value that signifies an unstarted clock.
        startTime = 1L;
      }
      clock.startTime = startTime;

      // Linearization point for clock start.
      clock.startLatch.countDown();
      clock.didStart();

      // Loop while the clock has not been stopped.
      do {
        final long tick = this.tick;
        // Wait for the clock to reach the elapsed time of the next tick.
        final long waitedMillis = waitForTick(clock, tick);
        // Check if we had a nominal wakeup.
        if (waitedMillis != Long.MIN_VALUE) {
          // Execute the clock tick.
          executeTick(clock, tick);
          // Invoke the clock tick introspection callback, with a measure of the
          // clock latency.
          clock.didTick(tick, waitedMillis);
          // Increment the tick sequence number.
          this.tick = tick + 1L;
        }
      } while ((Clock.STATUS.get(clock) & Clock.STOPPED) == 0);

      clock.willStop();
    } catch (Throwable error) {
      if (Conts.isNonFatal(error)) {
        // Report internal clock error.
        clock.didFail(error);
      } else {
        // Rethrow fatal error.
        throw error;
      }
    } finally {
      // Force the clock into the stopped state.
      Clock.STATUS.set(clock, Clock.STOPPED);
      // Linearization point for clock stop.
      clock.stopLatch.countDown();
      clock.didStop();
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy