2011-09-29 18:26:50 -06:00
|
|
|
package java.util.concurrent;
|
|
|
|
|
|
|
|
import avian.Atomic;
|
|
|
|
|
|
|
|
public class ConcurrentLinkedQueue<T> {
|
|
|
|
private static final long QueueHead;
|
|
|
|
private static final long QueueTail;
|
|
|
|
private static final long NodeNext;
|
|
|
|
|
|
|
|
static {
|
|
|
|
try {
|
|
|
|
QueueHead = Atomic.getOffset
|
|
|
|
(ConcurrentLinkedQueue.class.getField("head"));
|
|
|
|
|
|
|
|
QueueTail = Atomic.getOffset
|
|
|
|
(ConcurrentLinkedQueue.class.getField("tail"));
|
|
|
|
|
|
|
|
NodeNext = Atomic.getOffset
|
|
|
|
(Node.class.getField("next"));
|
|
|
|
} catch (NoSuchFieldException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-23 14:19:41 -07:00
|
|
|
private volatile Node<T> head = new Node<T>(null, null);
|
2011-09-29 18:26:50 -06:00
|
|
|
private volatile Node<T> tail = head;
|
|
|
|
|
2011-11-07 15:52:42 -07:00
|
|
|
public void clear() {
|
|
|
|
// todo: can we safely make this O(1)?
|
|
|
|
while (poll() != null) { }
|
|
|
|
}
|
|
|
|
|
2011-09-29 18:26:50 -06:00
|
|
|
public boolean add(T value) {
|
2013-12-23 14:19:41 -07:00
|
|
|
Node<T> n = new Node<T>(value, null);
|
2011-09-29 18:26:50 -06:00
|
|
|
while (true) {
|
|
|
|
Node<T> t = tail;
|
|
|
|
Node<T> next = tail.next;
|
|
|
|
if (t == tail) {
|
|
|
|
if (next != null) {
|
|
|
|
Atomic.compareAndSwapObject(this, QueueTail, t, next);
|
|
|
|
} else if (Atomic.compareAndSwapObject(tail, NodeNext, null, n)) {
|
|
|
|
Atomic.compareAndSwapObject(this, QueueTail, t, n);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T peek() {
|
|
|
|
return poll(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public T poll() {
|
|
|
|
return poll(true);
|
|
|
|
}
|
|
|
|
|
2014-02-11 14:54:43 -07:00
|
|
|
private T poll(boolean remove) {
|
2011-09-29 18:26:50 -06:00
|
|
|
while (true) {
|
|
|
|
Node<T> h = head;
|
|
|
|
Node<T> t = tail;
|
|
|
|
Node<T> next = head.next;
|
|
|
|
|
|
|
|
if (h == head) {
|
|
|
|
if (h == t) {
|
|
|
|
if (next != null) {
|
|
|
|
Atomic.compareAndSwapObject(this, QueueTail, t, next);
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
T value = next.value;
|
|
|
|
if ((! remove)
|
|
|
|
|| Atomic.compareAndSwapObject(this, QueueHead, h, next))
|
|
|
|
{
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class Node<T> {
|
|
|
|
public volatile T value;
|
|
|
|
public volatile Node<T> next;
|
|
|
|
|
|
|
|
public Node(T value, Node<T> next) {
|
|
|
|
this.value = value;
|
|
|
|
this.next = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|