3789983e89
Reviewed-by: darcy, ihse
409 lines
14 KiB
Java
409 lines
14 KiB
Java
/*
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code 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. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/*
|
|
* This file is available under and governed by the GNU General Public
|
|
* License version 2 only, as published by the Free Software Foundation.
|
|
* However, the following notice accompanied the original version of this
|
|
* file:
|
|
*
|
|
* Written by Martin Buchholz with assistance from members of JCP
|
|
* JSR-166 Expert Group and released to the public domain, as
|
|
* explained at http://creativecommons.org/publicdomain/zero/1.0/
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @modules java.base/java.util.concurrent:open
|
|
* @run testng WhiteBox
|
|
* @summary White box tests of implementation details
|
|
*/
|
|
|
|
import static org.testng.Assert.*;
|
|
import org.testng.annotations.DataProvider;
|
|
import org.testng.annotations.Test;
|
|
|
|
import java.io.ByteArrayInputStream;
|
|
import java.io.ByteArrayOutputStream;
|
|
import java.io.ObjectInputStream;
|
|
import java.io.ObjectOutputStream;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.VarHandle;
|
|
import java.util.ArrayList;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.concurrent.LinkedTransferQueue;
|
|
import java.util.concurrent.ThreadLocalRandom;
|
|
import java.util.concurrent.TimeUnit;
|
|
import static java.util.stream.Collectors.toList;
|
|
import java.util.function.Consumer;
|
|
import java.util.function.Function;
|
|
|
|
@Test
|
|
public class WhiteBox {
|
|
final ThreadLocalRandom rnd = ThreadLocalRandom.current();
|
|
final VarHandle HEAD, TAIL, ITEM, NEXT;
|
|
final int SWEEP_THRESHOLD;
|
|
|
|
public WhiteBox() throws ReflectiveOperationException {
|
|
Class<?> qClass = LinkedTransferQueue.class;
|
|
Class<?> nodeClass = Class.forName(qClass.getName() + "$Node");
|
|
MethodHandles.Lookup lookup
|
|
= MethodHandles.privateLookupIn(qClass, MethodHandles.lookup());
|
|
HEAD = lookup.findVarHandle(qClass, "head", nodeClass);
|
|
TAIL = lookup.findVarHandle(qClass, "tail", nodeClass);
|
|
NEXT = lookup.findVarHandle(nodeClass, "next", nodeClass);
|
|
ITEM = lookup.findVarHandle(nodeClass, "item", Object.class);
|
|
SWEEP_THRESHOLD = (int)
|
|
lookup.findStaticVarHandle(qClass, "SWEEP_THRESHOLD", int.class)
|
|
.get();
|
|
}
|
|
|
|
Object head(LinkedTransferQueue q) { return HEAD.getVolatile(q); }
|
|
Object tail(LinkedTransferQueue q) { return TAIL.getVolatile(q); }
|
|
Object item(Object node) { return ITEM.getVolatile(node); }
|
|
Object next(Object node) { return NEXT.getVolatile(node); }
|
|
|
|
int nodeCount(LinkedTransferQueue q) {
|
|
int i = 0;
|
|
for (Object p = head(q); p != null; ) {
|
|
i++;
|
|
if (p == (p = next(p))) p = head(q);
|
|
}
|
|
return i;
|
|
}
|
|
|
|
int tailCount(LinkedTransferQueue q) {
|
|
int i = 0;
|
|
for (Object p = tail(q); p != null; ) {
|
|
i++;
|
|
if (p == (p = next(p))) p = head(q);
|
|
}
|
|
return i;
|
|
}
|
|
|
|
Object findNode(LinkedTransferQueue q, Object e) {
|
|
for (Object p = head(q); p != null; ) {
|
|
if (item(p) != null && e.equals(item(p)))
|
|
return p;
|
|
if (p == (p = next(p))) p = head(q);
|
|
}
|
|
throw new AssertionError("not found");
|
|
}
|
|
|
|
Iterator iteratorAt(LinkedTransferQueue q, Object e) {
|
|
for (Iterator it = q.iterator(); it.hasNext(); )
|
|
if (it.next().equals(e))
|
|
return it;
|
|
throw new AssertionError("not found");
|
|
}
|
|
|
|
void assertIsSelfLinked(Object node) {
|
|
assertSame(next(node), node);
|
|
assertNull(item(node));
|
|
}
|
|
void assertIsNotSelfLinked(Object node) {
|
|
assertNotSame(node, next(node));
|
|
}
|
|
|
|
@Test
|
|
public void addRemove() {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
assertInvariants(q);
|
|
assertNull(next(head(q)));
|
|
assertNull(item(head(q)));
|
|
q.add(1);
|
|
assertEquals(nodeCount(q), 2);
|
|
assertInvariants(q);
|
|
q.remove(1);
|
|
assertEquals(nodeCount(q), 1);
|
|
assertInvariants(q);
|
|
}
|
|
|
|
/**
|
|
* Traversal actions that visit every node and do nothing, but
|
|
* have side effect of squeezing out dead nodes.
|
|
*/
|
|
@DataProvider
|
|
public Object[][] traversalActions() {
|
|
return List.<Consumer<LinkedTransferQueue>>of(
|
|
q -> q.forEach(e -> {}),
|
|
q -> assertFalse(q.contains(new Object())),
|
|
q -> assertFalse(q.remove(new Object())),
|
|
q -> q.spliterator().forEachRemaining(e -> {}),
|
|
q -> q.stream().collect(toList()),
|
|
q -> assertFalse(q.removeIf(e -> false)),
|
|
q -> assertFalse(q.removeAll(List.of())))
|
|
.stream().map(x -> new Object[]{ x }).toArray(Object[][]::new);
|
|
}
|
|
|
|
@Test(dataProvider = "traversalActions")
|
|
public void traversalOperationsCollapseLeadingNodes(
|
|
Consumer<LinkedTransferQueue> traversalAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
Object oldHead;
|
|
int n = 1 + rnd.nextInt(5);
|
|
for (int i = 0; i < n; i++) q.add(i);
|
|
assertEquals(nodeCount(q), n + 1);
|
|
oldHead = head(q);
|
|
traversalAction.accept(q);
|
|
assertInvariants(q);
|
|
assertEquals(nodeCount(q), n);
|
|
assertIsSelfLinked(oldHead);
|
|
}
|
|
|
|
@Test(dataProvider = "traversalActions")
|
|
public void traversalOperationsCollapseInteriorNodes(
|
|
Consumer<LinkedTransferQueue> traversalAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
int n = 6;
|
|
for (int i = 0; i < n; i++) q.add(i);
|
|
|
|
// We must be quite devious to reliably create an interior dead node
|
|
Object p0 = findNode(q, 0);
|
|
Object p1 = findNode(q, 1);
|
|
Object p2 = findNode(q, 2);
|
|
Object p3 = findNode(q, 3);
|
|
Object p4 = findNode(q, 4);
|
|
Object p5 = findNode(q, 5);
|
|
|
|
Iterator it1 = iteratorAt(q, 1);
|
|
Iterator it2 = iteratorAt(q, 2);
|
|
|
|
it2.remove(); // causes it2's ancestor to advance to 1
|
|
assertSame(next(p1), p3);
|
|
assertSame(next(p2), p3);
|
|
assertNull(item(p2));
|
|
it1.remove(); // removes it2's ancestor
|
|
assertSame(next(p0), p3);
|
|
assertSame(next(p1), p3);
|
|
assertSame(next(p2), p3);
|
|
assertNull(item(p1));
|
|
assertEquals(it2.next(), 3);
|
|
it2.remove(); // it2's ancestor can't unlink
|
|
|
|
assertSame(next(p0), p3); // p3 is now interior dead node
|
|
assertSame(next(p1), p4); // it2 uselessly CASed p1.next
|
|
assertSame(next(p2), p3);
|
|
assertSame(next(p3), p4);
|
|
assertInvariants(q);
|
|
|
|
int c = nodeCount(q);
|
|
traversalAction.accept(q);
|
|
assertEquals(nodeCount(q), c - 1);
|
|
|
|
assertSame(next(p0), p4);
|
|
assertSame(next(p1), p4);
|
|
assertSame(next(p2), p3);
|
|
assertSame(next(p3), p4);
|
|
assertInvariants(q);
|
|
|
|
// trailing nodes are not unlinked
|
|
Iterator it5 = iteratorAt(q, 5); it5.remove();
|
|
traversalAction.accept(q);
|
|
assertSame(next(p4), p5);
|
|
assertNull(next(p5));
|
|
assertEquals(nodeCount(q), c - 1);
|
|
}
|
|
|
|
/**
|
|
* Checks that traversal operations collapse a random pattern of
|
|
* dead nodes as could normally only occur with a race.
|
|
*/
|
|
@Test(dataProvider = "traversalActions")
|
|
public void traversalOperationsCollapseRandomNodes(
|
|
Consumer<LinkedTransferQueue> traversalAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
int n = rnd.nextInt(6);
|
|
for (int i = 0; i < n; i++) q.add(i);
|
|
ArrayList nulledOut = new ArrayList();
|
|
for (Object p = head(q); p != null; p = next(p))
|
|
if (rnd.nextBoolean()) {
|
|
nulledOut.add(item(p));
|
|
ITEM.setVolatile(p, null);
|
|
}
|
|
traversalAction.accept(q);
|
|
int c = nodeCount(q);
|
|
assertEquals(q.size(), c - (q.contains(n - 1) ? 0 : 1));
|
|
for (int i = 0; i < n; i++)
|
|
assertTrue(nulledOut.contains(i) ^ q.contains(i));
|
|
}
|
|
|
|
/**
|
|
* Traversal actions that remove every element, and are also
|
|
* expected to squeeze out dead nodes.
|
|
*/
|
|
@DataProvider
|
|
public Object[][] bulkRemovalActions() {
|
|
return List.<Consumer<LinkedTransferQueue>>of(
|
|
q -> q.clear(),
|
|
q -> assertTrue(q.removeIf(e -> true)),
|
|
q -> assertTrue(q.retainAll(List.of())))
|
|
.stream().map(x -> new Object[]{ x }).toArray(Object[][]::new);
|
|
}
|
|
|
|
@Test(dataProvider = "bulkRemovalActions")
|
|
public void bulkRemovalOperationsCollapseNodes(
|
|
Consumer<LinkedTransferQueue> bulkRemovalAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
int n = 1 + rnd.nextInt(5);
|
|
for (int i = 0; i < n; i++) q.add(i);
|
|
bulkRemovalAction.accept(q);
|
|
assertEquals(nodeCount(q), 1);
|
|
assertInvariants(q);
|
|
}
|
|
|
|
/**
|
|
* Actions that remove the first element, and are expected to
|
|
* leave at most one slack dead node at head.
|
|
*/
|
|
@DataProvider
|
|
public Object[][] pollActions() {
|
|
return List.<Consumer<LinkedTransferQueue>>of(
|
|
q -> assertNotNull(q.poll()),
|
|
q -> { try { assertNotNull(q.poll(1L, TimeUnit.DAYS)); }
|
|
catch (Throwable x) { throw new AssertionError(x); }},
|
|
q -> { try { assertNotNull(q.take()); }
|
|
catch (Throwable x) { throw new AssertionError(x); }},
|
|
q -> assertNotNull(q.remove()))
|
|
.stream().map(x -> new Object[]{ x }).toArray(Object[][]::new);
|
|
}
|
|
|
|
@Test(dataProvider = "pollActions")
|
|
public void pollActionsOneNodeSlack(
|
|
Consumer<LinkedTransferQueue> pollAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
int n = 1 + rnd.nextInt(5);
|
|
for (int i = 0; i < n; i++) q.add(i);
|
|
assertEquals(nodeCount(q), n + 1);
|
|
for (int i = 0; i < n; i++) {
|
|
int c = nodeCount(q);
|
|
boolean slack = item(head(q)) == null;
|
|
if (slack) assertNotNull(item(next(head(q))));
|
|
pollAction.accept(q);
|
|
assertEquals(nodeCount(q), q.isEmpty() ? 1 : c - (slack ? 2 : 0));
|
|
}
|
|
assertInvariants(q);
|
|
}
|
|
|
|
/**
|
|
* Actions that append an element, and are expected to
|
|
* leave at most one slack node at tail.
|
|
*/
|
|
@DataProvider
|
|
public Object[][] addActions() {
|
|
return List.<Consumer<LinkedTransferQueue>>of(
|
|
q -> q.add(1),
|
|
q -> q.offer(1))
|
|
.stream().map(x -> new Object[]{ x }).toArray(Object[][]::new);
|
|
}
|
|
|
|
@Test(dataProvider = "addActions")
|
|
public void addActionsOneNodeSlack(
|
|
Consumer<LinkedTransferQueue> addAction) {
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
int n = 1 + rnd.nextInt(9);
|
|
for (int i = 0; i < n; i++) {
|
|
boolean slack = next(tail(q)) != null;
|
|
addAction.accept(q);
|
|
if (slack)
|
|
assertNull(next(tail(q)));
|
|
else {
|
|
assertNotNull(next(tail(q)));
|
|
assertNull(next(next(tail(q))));
|
|
}
|
|
assertInvariants(q);
|
|
}
|
|
}
|
|
|
|
byte[] serialBytes(Object o) {
|
|
try {
|
|
ByteArrayOutputStream bos = new ByteArrayOutputStream();
|
|
ObjectOutputStream oos = new ObjectOutputStream(bos);
|
|
oos.writeObject(o);
|
|
oos.flush();
|
|
oos.close();
|
|
return bos.toByteArray();
|
|
} catch (Exception fail) {
|
|
throw new AssertionError(fail);
|
|
}
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
<T> T serialClone(T o) {
|
|
try {
|
|
ObjectInputStream ois = new ObjectInputStream
|
|
(new ByteArrayInputStream(serialBytes(o)));
|
|
T clone = (T) ois.readObject();
|
|
assertNotSame(o, clone);
|
|
assertSame(o.getClass(), clone.getClass());
|
|
return clone;
|
|
} catch (Exception fail) {
|
|
throw new AssertionError(fail);
|
|
}
|
|
}
|
|
|
|
public void testSerialization() {
|
|
LinkedTransferQueue q = serialClone(new LinkedTransferQueue());
|
|
assertInvariants(q);
|
|
}
|
|
|
|
public void cancelledNodeSweeping() throws Throwable {
|
|
assertEquals(SWEEP_THRESHOLD & (SWEEP_THRESHOLD - 1), 0);
|
|
LinkedTransferQueue q = new LinkedTransferQueue();
|
|
Thread blockHead = null;
|
|
if (rnd.nextBoolean()) {
|
|
blockHead = new Thread(
|
|
() -> { try { q.take(); } catch (InterruptedException ok) {}});
|
|
blockHead.start();
|
|
while (nodeCount(q) != 2) { Thread.yield(); }
|
|
assertTrue(q.hasWaitingConsumer());
|
|
assertEquals(q.getWaitingConsumerCount(), 1);
|
|
}
|
|
int initialNodeCount = nodeCount(q);
|
|
|
|
// Some dead nodes do in fact accumulate ...
|
|
if (blockHead != null)
|
|
while (nodeCount(q) < initialNodeCount + SWEEP_THRESHOLD / 2)
|
|
q.poll(1L, TimeUnit.MICROSECONDS);
|
|
|
|
// ... but no more than SWEEP_THRESHOLD nodes accumulate
|
|
for (int i = rnd.nextInt(SWEEP_THRESHOLD * 10); i-->0; )
|
|
q.poll(1L, TimeUnit.MICROSECONDS);
|
|
assertTrue(nodeCount(q) <= initialNodeCount + SWEEP_THRESHOLD);
|
|
|
|
if (blockHead != null) {
|
|
blockHead.interrupt();
|
|
blockHead.join();
|
|
}
|
|
}
|
|
|
|
/** Checks conditions which should always be true. */
|
|
void assertInvariants(LinkedTransferQueue q) {
|
|
assertNotNull(head(q));
|
|
assertNotNull(tail(q));
|
|
// head is never self-linked (but tail may!)
|
|
for (Object h; next(h = head(q)) == h; )
|
|
assertNotSame(h, head(q)); // must be update race
|
|
}
|
|
}
|