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

com.aspectran.utils.timer.CyclicTimeout Maven / Gradle / Ivy

There is a newer version: 8.1.5
Show newest version
/*
 * Copyright (c) 2008-2025 The Aspectran Project
 *
 * 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 com.aspectran.utils.timer;

import com.aspectran.utils.annotation.jsr305.NonNull;
import com.aspectran.utils.logging.Logger;
import com.aspectran.utils.logging.LoggerFactory;
import com.aspectran.utils.thread.Scheduler;

import javax.security.auth.Destroyable;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;

import static java.lang.Long.MAX_VALUE;

/**
 * 

This class is a clone of org.eclipse.jetty.io.CyclicTimeout

* *

An abstract implementation of a timeout.

*

Subclasses should implement {@link #onTimeoutExpired()}.

*

This implementation is optimised assuming that the timeout * will mostly be cancelled and then reused with a similar value.

*

This implementation has a {@link Timeout} holding the time * at which the scheduled task should fire, and a linked list of * {@link Wakeup}, each holding the actual scheduled task.

*

Calling {@link #schedule(long, TimeUnit)} the first time will * create a Timeout with an associated Wakeup and submit a task to * the scheduler. * Calling {@link #schedule(long, TimeUnit)} again with the same or * a larger delay will cancel the previous Timeout, but keep the * previous Wakeup without submitting a new task to the scheduler, * therefore reducing the pressure on the scheduler and avoid it * becomes a bottleneck. * When the Wakeup task fires, it will see that the Timeout is now * in the future and will attach a new Wakeup with the future time * to the Timeout, and submit a scheduler task for the new Wakeup.

*/ public abstract class CyclicTimeout implements Destroyable { private static final Logger logger = LoggerFactory.getLogger(CyclicTimeout.class); private static final Timeout NOT_SET = new Timeout(MAX_VALUE, null); private static final Scheduler.Task DESTROYED = () -> false; /* The underlying scheduler to use */ private final Scheduler scheduler; /* Reference to the current Timeout and chain of Wakeup */ private final AtomicReference timeout = new AtomicReference<>(NOT_SET); /** * @param scheduler A scheduler used to schedule wakeups */ public CyclicTimeout(Scheduler scheduler) { this.scheduler = scheduler; } public Scheduler getScheduler() { return scheduler; } /** * Schedules a timeout, even if already set, cancelled or expired. * If a timeout is already set, it will be cancelled and replaced * by the new one. * @param delay The period of time before the timeout expires. * @param units The unit of time of the period. * @return true if the timeout was already set. */ public boolean schedule(long delay, @NonNull TimeUnit units) { long now = System.nanoTime(); long newTimeoutAt = now + units.toNanos(delay); Wakeup newWakeup = null; boolean result; while (true) { Timeout timeout = this.timeout.get(); result = (timeout.at != MAX_VALUE); // Is the current wakeup good to use? ie before our timeout time? Wakeup wakeup = timeout.wakeup; if (wakeup == null || wakeup.at > newTimeoutAt) { // No, we need an earlier wakeup. wakeup = newWakeup = new Wakeup(newTimeoutAt, wakeup); } if (this.timeout.compareAndSet(timeout, new Timeout(newTimeoutAt, wakeup))) { if (logger.isTraceEnabled()) { logger.trace("Installed timeout in " + units.toMillis(delay) + " ms, " + (newWakeup != null ? "new" : "existing") + " waking up in " + TimeUnit.NANOSECONDS.toMillis(wakeup.at - now) + " ms"); } break; } } // If we created a new wakeup, we need to actually schedule it. // Any wakeup that is created and discarded by the failed CAS will not be // in the wakeup chain, will not have a scheduler task set and will be GC'd. if (newWakeup != null) { newWakeup.schedule(now); } return result; } /** * Cancels this CyclicTimeout so that it won't expire. * After being cancelled, this CyclicTimeout can be scheduled again. * @return true if this CyclicTimeout was scheduled to expire * @see #destroy() */ public boolean cancel() { boolean result; while (true) { Timeout timeout = this.timeout.get(); result = (timeout.at != MAX_VALUE); Wakeup wakeup = timeout.wakeup; Timeout newTimeout = (wakeup == null ? NOT_SET : new Timeout(MAX_VALUE, wakeup)); if (this.timeout.compareAndSet(timeout, newTimeout)) { break; } } return result; } /** * Invoked when the timeout expires. */ public abstract void onTimeoutExpired(); /** * Destroys this CyclicTimeout. * After being destroyed, this CyclicTimeout is not used anymore. */ @Override public void destroy() { Timeout timeout = this.timeout.getAndSet(NOT_SET); Wakeup wakeup = (timeout != null ? timeout.wakeup : null); while (wakeup != null) { wakeup.destroy(); wakeup = wakeup.next; } } /** * A timeout time with a link to a Wakeup chain. */ private static class Timeout { private final long at; private final Wakeup wakeup; private Timeout(long timeoutAt, Wakeup wakeup) { this.at = timeoutAt; this.wakeup = wakeup; } @Override @NonNull public String toString() { return String.format("%s@%x:%dms,%s", getClass().getSimpleName(), hashCode(), TimeUnit.NANOSECONDS.toMillis(at), wakeup); } } /** * A Wakeup chain of real scheduler tasks. */ private class Wakeup implements Runnable { private final AtomicReference task = new AtomicReference<>(); private final long at; private final Wakeup next; private Wakeup(long wakeupAt, Wakeup next) { this.at = wakeupAt; this.next = next; } private void schedule(long now) { task.compareAndSet(null, scheduler.schedule(this, at - now, TimeUnit.NANOSECONDS)); } private void destroy() { Scheduler.Task task = this.task.getAndSet(DESTROYED); if (task != null) { task.cancel(); } } @Override public void run() { long now = System.nanoTime(); Wakeup newWakeup = null; boolean hasExpired = false; while (true) { Timeout timeout = CyclicTimeout.this.timeout.get(); // We must look for ourselves in the current wakeup list. // If we find ourselves, then we act and we use our tail for any new // wakeup list, effectively removing any wakeup before us in the list (and making them no-ops). // If we don't find ourselves, then a wakeup that should have expired after us has already run // and removed us from the list, so we become a noop. Wakeup wakeup = timeout.wakeup; while (wakeup != null) { if (wakeup == this) { break; } // Not us, so look at next wakeup in the list. wakeup = wakeup.next; } if (wakeup == null) { // Not found, we become a noop. return; } // We are in the wakeup list! So we have to act and we know our // tail has not expired (else it would have removed us from the list). // Remove ourselves (and any prior Wakeup) from the wakeup list. wakeup = wakeup.next; Timeout newTimeout; if (timeout.at <= now) { // We have timed out! hasExpired = true; newTimeout = (wakeup == null ? NOT_SET : new Timeout(MAX_VALUE, wakeup)); } else if (timeout.at != MAX_VALUE) { // We have not timed out, but we are set to! // Is the current wakeup good to use? ie before our timeout time? if (wakeup == null || wakeup.at > timeout.at) { // No, we need an earlier wakeup. wakeup = newWakeup = new Wakeup(timeout.at, wakeup); } newTimeout = new Timeout(timeout.at, wakeup); } else { // We don't timeout, preserve scheduled chain. newTimeout = (wakeup == null ? NOT_SET : new Timeout(MAX_VALUE, wakeup)); } // Loop until we succeed in changing state or we are a noop! if (CyclicTimeout.this.timeout.compareAndSet(timeout, newTimeout)) { break; } } // If we created a new wakeup, we need to actually schedule it. if (newWakeup != null) { newWakeup.schedule(now); } // If we expired, then do the callback. if (hasExpired) { onTimeoutExpired(); } } @Override @NonNull public String toString() { return String.format("%s@%x:%dms->%s", getClass().getSimpleName(), hashCode(), (at == MAX_VALUE ? at : TimeUnit.NANOSECONDS.toMillis(at)), next); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy