Make sure cancelled Timeouts are able to be GC'ed fast.
Motivation: At the moment the HashedWheelTimer will only remove the cancelled Timeouts once the HashedWheelBucket is processed again. Until this the instance will not be able to be GC'ed as there are still strong referenced to it even if the user not reference it by himself/herself. This can cause to waste a lot of memory even if the Timeout was cancelled before. Modification: Add a new queue which holds CancelTasks that will be processed on each tick to remove cancelled Timeouts. Because all of this is done only by the WorkerThread there is no need for synchronization and only one extra object creation is needed when cancel() is executed. For addTimeout(...) no new overhead is introduced. Result: Less memory usage for cancelled Timeouts.
This commit is contained in:
parent
2fa9400a59
commit
2b401d4587
@ -393,6 +393,13 @@ public class HashedWheelTimer implements Timer {
|
||||
// all processed
|
||||
break;
|
||||
}
|
||||
if (timeout.state() == HashedWheelTimeout.ST_CANCELLED
|
||||
|| !timeout.compareAndSetState(HashedWheelTimeout.ST_INIT, HashedWheelTimeout.ST_IN_BUCKET)) {
|
||||
// Was cancelled in the meantime. So just remove it and continue with next HashedWheelTimeout
|
||||
// in the queue
|
||||
timeout.remove();
|
||||
continue;
|
||||
}
|
||||
long calculated = timeout.deadline / tickDuration;
|
||||
long remainingRounds = (calculated - tick) / wheel.length;
|
||||
timeout.remainingRounds = remainingRounds;
|
||||
@ -452,8 +459,9 @@ public class HashedWheelTimer implements Timer {
|
||||
private static final class HashedWheelTimeout implements Timeout {
|
||||
|
||||
private static final int ST_INIT = 0;
|
||||
private static final int ST_CANCELLED = 1;
|
||||
private static final int ST_EXPIRED = 2;
|
||||
private static final int ST_IN_BUCKET = 1;
|
||||
private static final int ST_CANCELLED = 2;
|
||||
private static final int ST_EXPIRED = 3;
|
||||
private static final AtomicIntegerFieldUpdater<HashedWheelTimeout> STATE_UPDATER =
|
||||
AtomicIntegerFieldUpdater.newUpdater(HashedWheelTimeout.class, "state");
|
||||
|
||||
@ -473,6 +481,9 @@ public class HashedWheelTimer implements Timer {
|
||||
HashedWheelTimeout next;
|
||||
HashedWheelTimeout prev;
|
||||
|
||||
// The bucket to which the timeout was added
|
||||
HashedWheelBucket bucket;
|
||||
|
||||
HashedWheelTimeout(HashedWheelTimer timer, TimerTask task, long deadline) {
|
||||
this.timer = timer;
|
||||
this.task = task;
|
||||
@ -488,16 +499,47 @@ public class HashedWheelTimer implements Timer {
|
||||
}
|
||||
|
||||
public void cancel() {
|
||||
int state = state();
|
||||
if (state >= ST_CANCELLED) {
|
||||
// fail fast if the task was cancelled or expired before.
|
||||
return;
|
||||
}
|
||||
if (state != ST_IN_BUCKET && compareAndSetState(ST_INIT, ST_CANCELLED)) {
|
||||
// Was cancelled before the HashedWheelTimeout was added to its HashedWheelBucket.
|
||||
// In this case we can just return here as it will be discarded by the WorkerThread when handling
|
||||
// the adding of HashedWheelTimeout to the HashedWheelBuckets.
|
||||
return;
|
||||
}
|
||||
// only update the state it will be removed from HashedWheelBucket on next tick.
|
||||
STATE_UPDATER.compareAndSet(this, ST_INIT, ST_CANCELLED);
|
||||
if (!compareAndSetState(ST_IN_BUCKET, ST_CANCELLED)) {
|
||||
return;
|
||||
}
|
||||
// Add the HashedWheelTimeout back to the timeouts queue so it will be picked up on the next tick
|
||||
// and remove this HashedTimeTask from the HashedWheelBucket. After this is done it is ready to get
|
||||
// GC'ed once the user has no reference to it anymore.
|
||||
timer.timeouts.add(this);
|
||||
}
|
||||
|
||||
public void remove() {
|
||||
if (bucket != null) {
|
||||
bucket.remove(this);
|
||||
}
|
||||
}
|
||||
|
||||
public boolean compareAndSetState(int expected, int state) {
|
||||
return STATE_UPDATER.compareAndSet(this, expected, state);
|
||||
}
|
||||
|
||||
public int state() {
|
||||
return state;
|
||||
}
|
||||
|
||||
public boolean isCancelled() {
|
||||
return STATE_UPDATER.get(this) == ST_CANCELLED;
|
||||
return state == ST_CANCELLED;
|
||||
}
|
||||
|
||||
public boolean isExpired() {
|
||||
return STATE_UPDATER.get(this) != ST_INIT;
|
||||
return state > ST_IN_BUCKET;
|
||||
}
|
||||
|
||||
public HashedWheelTimeout value() {
|
||||
@ -505,7 +547,8 @@ public class HashedWheelTimer implements Timer {
|
||||
}
|
||||
|
||||
public void expire() {
|
||||
if (!STATE_UPDATER.compareAndSet(this, ST_INIT, ST_EXPIRED)) {
|
||||
if (!compareAndSetState(ST_IN_BUCKET, ST_EXPIRED)) {
|
||||
assert state() != ST_INIT;
|
||||
return;
|
||||
}
|
||||
|
||||
@ -564,6 +607,8 @@ public class HashedWheelTimer implements Timer {
|
||||
* Add {@link HashedWheelTimeout} to this bucket.
|
||||
*/
|
||||
public void addTimeout(HashedWheelTimeout timeout) {
|
||||
assert timeout.bucket == null;
|
||||
timeout.bucket = this;
|
||||
if (head == null) {
|
||||
head = tail = timeout;
|
||||
} else {
|
||||
@ -599,31 +644,38 @@ public class HashedWheelTimer implements Timer {
|
||||
// store reference to next as we may null out timeout.next in the remove block.
|
||||
HashedWheelTimeout next = timeout.next;
|
||||
if (remove) {
|
||||
remove(timeout);
|
||||
}
|
||||
timeout = next;
|
||||
}
|
||||
}
|
||||
|
||||
public void remove(HashedWheelTimeout timeout) {
|
||||
HashedWheelTimeout next = timeout.next;
|
||||
// remove timeout that was either processed or cancelled by updating the linked-list
|
||||
if (timeout.prev != null) {
|
||||
timeout.prev.next = timeout.next;
|
||||
timeout.prev.next = next;
|
||||
}
|
||||
if (timeout.next != null) {
|
||||
timeout.next.prev = timeout.prev;
|
||||
}
|
||||
|
||||
if (timeout == head) {
|
||||
// if timeout is head we need to replace the head with the next entry
|
||||
head = next;
|
||||
if (timeout == tail) {
|
||||
// if timeout is also the tail we need to adjust the entry too
|
||||
tail = timeout.next;
|
||||
if (timeout == tail) {
|
||||
tail = null;
|
||||
head = null;
|
||||
} else {
|
||||
head = next;
|
||||
}
|
||||
} else if (timeout == tail) {
|
||||
// if the timeout is the tail modify the tail to be the prev node.
|
||||
tail = timeout.prev;
|
||||
}
|
||||
// null out prev and next to allow for GC.
|
||||
// null out prev, next and bucket to allow for GC.
|
||||
timeout.prev = null;
|
||||
timeout.next = null;
|
||||
}
|
||||
timeout = next;
|
||||
}
|
||||
timeout.bucket = null;
|
||||
}
|
||||
|
||||
/**
|
||||
|
Loading…
x
Reference in New Issue
Block a user