package java.util.concurrent; import java.util.Collection; import java.util.Iterator; import java.util.NoSuchElementException; import java.util.Queue; import avian.Atomic; public class ConcurrentLinkedQueue implements Queue { 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); } } private volatile Node head = new Node(null, null); private volatile Node tail = head; @Override public void clear() { // TODO - can we safely make this O(1)? while (poll() != null) { } } @Override public boolean offer(T element) { add(element); return true; } @Override public boolean add(T value) { Node n = new Node(value, null); while (true) { Node t = tail; Node 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; } @Override public T peek() { return poll(false); } @Override public T element() { T result = peek(); if (result == null) { throw new NoSuchElementException(); } else { return result; } } @Override public T poll() { return poll(true); } @Override public T remove() { T result = poll(); if (result == null) { throw new NoSuchElementException(); } else { return result; } } private T poll(boolean remove) { while (true) { Node h = head; Node t = tail; Node 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 { public volatile T value; public volatile Node next; public Node(T value, Node next) { this.value = value; this.next = next; } } @Override public int size() { // TODO - implement throw new UnsupportedOperationException(); } @Override public boolean isEmpty() { return size() == 0; } @Override public boolean contains(Object element) { // TODO - implement throw new UnsupportedOperationException(); } @Override public boolean containsAll(Collection c) { // TODO - implement throw new UnsupportedOperationException(); } @Override public boolean addAll(Collection collection) { // TODO - implement throw new UnsupportedOperationException(); } @Override public boolean remove(Object element) { // TODO - implement throw new UnsupportedOperationException(); } @Override public boolean removeAll(Collection c) { // TODO - implement throw new UnsupportedOperationException(); } @Override public Object[] toArray() { // TODO - implement throw new UnsupportedOperationException(); } @Override public S[] toArray(S[] array) { // TODO - implement throw new UnsupportedOperationException(); } @Override public Iterator iterator() { // TODO - implement throw new UnsupportedOperationException(); } }