com.caucho.util.AlarmClock Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.util;
import java.util.ArrayList;
import java.util.concurrent.atomic.AtomicLong;
import java.util.logging.Logger;
import com.caucho.env.thread.ThreadPool;
/**
* The alarm class provides a lightweight event scheduler. This allows
* an objects to schedule a timeout without creating a new thread.
*
* A separate thread periodically tests the queue for alarms ready.
*/
public class AlarmClock {
private static final Logger log
= Logger.getLogger(AlarmClock.class.getName());
private static final int CLOCK_PERIOD = 60 * 1000;
private static final int CLOCK_NEXT = 5 * 1000;
private static final int CLOCK_INTERVAL = 1;
private Alarm []_clockArray = new Alarm[CLOCK_PERIOD];
private AtomicLong _lastExtractTime = new AtomicLong();
private AtomicLong _nextAlarmTime = new AtomicLong();
private final ArrayList _currentAlarms = new ArrayList();
private long _lastTime;
private Object _lock = new Object();
private ThreadPool _threadPool = ThreadPool.getThreadPool();
/**
* Queue the alarm for wakeup.
*
* @param delta time in milliseconds to wake
*/
public boolean queueAt(Alarm alarm, long wakeTime)
{
boolean isEarliest = false;
long prevNextAlarmTime;
do {
prevNextAlarmTime = _nextAlarmTime.get();
} while (wakeTime > 0 && wakeTime < prevNextAlarmTime
&& ! _nextAlarmTime.compareAndSet(prevNextAlarmTime, wakeTime));
if (wakeTime < prevNextAlarmTime)
isEarliest = true;
long oldWakeTime = alarm.getAndSetWakeTime(wakeTime);
if (oldWakeTime == wakeTime)
return false;
if (oldWakeTime > 0) {
if (! dequeueImpl(alarm)) {
/*
System.out.println("FAIL: " + oldWakeTime + " " + wakeTime
+ " " + alarm);
*/
}
}
if (wakeTime <= 0)
return false;
long lastExtractTime = _lastExtractTime.get();
if (wakeTime <= lastExtractTime) {
queueCurrent(alarm);
return true;
}
synchronized (_lock) {
if (alarm.getBucket() >= 0)
return false;
int bucket = getBucket(wakeTime);
alarm.setBucket(bucket);
Alarm top = _clockArray[bucket];
alarm.setNext(top);
_clockArray[bucket] = alarm;
}
lastExtractTime = _lastExtractTime.get();
long nextWakeTime = alarm.getWakeTime();
if (nextWakeTime <= lastExtractTime && nextWakeTime > 0) {
if (dequeueImpl(alarm)) {
queueCurrent(alarm);
}
}
return isEarliest;
}
private void queueCurrent(Alarm alarm)
{
synchronized (_currentAlarms) {
_currentAlarms.add(alarm);
}
}
void dequeue(Alarm alarm)
{
// long oldWakeTime = alarm.getAndSetWakeTime(0);
// long oldWakeTime = alarm.getAndSetWakeTime(0);
alarm.setWakeTime(0);
if (alarm.getBucket() >= 0) {
dequeueImpl(alarm);
}
}
private boolean dequeueImpl(Alarm alarm)
{
synchronized (_lock) {
int bucket = alarm.getBucket();
Alarm next = alarm.getNext();
alarm.setBucket(-1);
alarm.setNext(null);
if (bucket < 0)
return false;
Alarm head = _clockArray[bucket];
if (head == null)
return false;
if (head == alarm) {
_clockArray[bucket] = next;
return true;
}
Alarm prev = head;
Alarm ptr = prev.getNext();
while (ptr != null) {
if (ptr == alarm) {
prev.setNext(next);
return true;
}
prev = ptr;
ptr = ptr.getNext();
}
}
return false;
}
private Alarm extractNextAlarm(int bucket, long time, boolean isTest)
{
if (_clockArray[bucket] == null)
return null;
synchronized (_lock) {
Alarm ptr = _clockArray[bucket];
Alarm prev = null;
while (ptr != null) {
Alarm next = ptr.getNext();
long wakeTime = ptr.getWakeTime();
if (wakeTime <= time) {
ptr.setNext(null);
ptr.setBucket(-1);
if (prev != null) {
prev.setNext(next);
}
else {
_clockArray[bucket] = next;
}
return ptr;
}
prev = ptr;
ptr = next;
}
}
return null;
}
/**
* Returns the next alarm ready to run
*/
public long extractAlarm(long now, boolean isTest)
{
long lastTime = _lastExtractTime.getAndSet(now);
long nextTime = _nextAlarmTime.get();
if (now < nextTime) {
return nextTime;
}
_nextAlarmTime.set(now + CLOCK_NEXT);
int delta;
delta = (int) (now - lastTime) / CLOCK_INTERVAL;
delta = Math.min(delta, CLOCK_PERIOD);
Alarm alarm;
int bucket = getBucket(lastTime);
for (int i = 0; i <= delta; i++) {
// long time = lastTime + i;
while ((alarm = extractNextAlarm(bucket, now, isTest)) != null) {
dispatch(alarm, now, isTest);
}
bucket = (bucket + 1) % CLOCK_PERIOD;
}
while ((alarm = extractNextCurrentAlarm()) != null) {
dispatch(alarm, now, isTest);
}
long next = updateNextAlarmTime(now);
_lastTime = now;
return next;
}
private Alarm extractNextCurrentAlarm()
{
if (_currentAlarms.size() == 0)
return null;
synchronized (_currentAlarms) {
if (_currentAlarms.size() > 0) {
Alarm alarm = _currentAlarms.remove(_currentAlarms.size() - 1);
return alarm;
}
return null;
}
}
private long updateNextAlarmTime(long now)
{
long nextTime = _nextAlarmTime.get();
long delta = Math.min(nextTime - now, CLOCK_PERIOD);
synchronized (_lock) {
for (int i = 0; i < delta; i++) {
long time = now + i;
if (nextTime < time)
return nextTime;
int bucket = getBucket(time);
for (Alarm ptr = _clockArray[bucket]; ptr != null; ptr = ptr.getNext()) {
// Alarm alarm = _clockArray[bucket];
long alarmTime = ptr.getWakeTime();
while (alarmTime < nextTime && alarmTime > 0) {
_nextAlarmTime.compareAndSet(nextTime, alarmTime);
nextTime = _nextAlarmTime.get();
}
}
}
}
return nextTime;
}
private void dispatch(Alarm alarm, long now, boolean isTest)
{
boolean isStressTest = false;
if (isStressTest)
now = CurrentTime.getExactTime();
else
now = CurrentTime.getCurrentTime();
long wakeTime = alarm.getAndSetWakeTime(0);
long delta = now - wakeTime;
if (wakeTime == 0) {
// #5854
}
else if (delta > 10000) {
log.warning(this + " slow alarm " + alarm + " " + delta + "ms"
+ " coordinator-delta " + (now - _lastTime) + "ms");
}
else if (isStressTest && delta > 100) {
System.out.println(this + " slow alarm " + alarm + " " + delta
+ " coordinator-delta " + (now - _lastTime) + "ms");
}
if (isTest) {
try {
alarm.run();
} catch (Throwable e) {
e.printStackTrace();
}
}
else if (alarm.isPriority()) {
_threadPool.schedulePriority(alarm);
}
else {
_threadPool.schedule(alarm);
}
}
private int getBucket(long time)
{
return (int) ((time + CLOCK_INTERVAL - 1) / CLOCK_INTERVAL % CLOCK_PERIOD);
}
/**
* Returns the next alarm ready to run
*/
long getNextAlarmTime()
{
return _nextAlarmTime.get();
}
// test
void testClear()
{
_lastExtractTime.set(0);
_nextAlarmTime.set(0);
synchronized (_lock) {
_currentAlarms.clear();
for (int i = CLOCK_PERIOD - 1; i >= 0; i--) {
Alarm alarm = _clockArray[i];
_clockArray[i] = null;
while (alarm != null) {
Alarm next = alarm.getNext();
alarm.setNext(null);
alarm.setBucket(-1);
alarm.setWakeTime(alarm.getWakeTime(), 0);
alarm = next;
}
}
}
}
@Override
public String toString()
{
return getClass().getSimpleName();
}
}