This closes #2910
This commit is contained in:
commit
faed83432c
|
@ -161,9 +161,13 @@ public class LinkedListImpl<E> implements LinkedList<E> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void clear() {
|
public void clear() {
|
||||||
tail = head.next = null;
|
// Clearing all of the links between nodes is "unnecessary", but:
|
||||||
|
// - helps a generational GC if the discarded nodes inhabit
|
||||||
|
// more than one generation
|
||||||
|
// - is sure to free memory even if there is a reachable Iterator
|
||||||
|
while (poll() != null) {
|
||||||
|
|
||||||
size = 0;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -308,6 +312,14 @@ public class LinkedListImpl<E> implements LinkedList<E> {
|
||||||
return (T) this;
|
return (T) this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected final LinkedListImpl.Node<T> next() {
|
||||||
|
return next;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final LinkedListImpl.Node<T> prev() {
|
||||||
|
return prev;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return val() == this ? "Intrusive Node" : "Node, value = " + val();
|
return val() == this ? "Intrusive Node" : "Node, value = " + val();
|
||||||
|
|
|
@ -16,13 +16,10 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.artemis.tests.unit.util;
|
package org.apache.activemq.artemis.tests.unit.util;
|
||||||
|
|
||||||
import java.lang.ref.WeakReference;
|
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.NoSuchElementException;
|
import java.util.NoSuchElementException;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
|
||||||
|
|
||||||
import org.apache.activemq.artemis.tests.util.ActiveMQTestBase;
|
import org.apache.activemq.artemis.tests.util.ActiveMQTestBase;
|
||||||
import org.apache.activemq.artemis.tests.util.RandomUtil;
|
import org.apache.activemq.artemis.tests.util.RandomUtil;
|
||||||
|
@ -40,7 +37,6 @@ public class LinkedListTest extends ActiveMQTestBase {
|
||||||
@Before
|
@Before
|
||||||
public void setUp() throws Exception {
|
public void setUp() throws Exception {
|
||||||
super.setUp();
|
super.setUp();
|
||||||
|
|
||||||
list = new LinkedListImpl<>(integerComparator);
|
list = new LinkedListImpl<>(integerComparator);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -111,151 +107,111 @@ public class LinkedListTest extends ActiveMQTestBase {
|
||||||
integerIterator.close();
|
integerIterator.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final class ObservableNode extends LinkedListImpl.Node<ObservableNode> {
|
||||||
|
|
||||||
|
ObservableNode() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public LinkedListImpl.Node<ObservableNode> publicNext() {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
|
||||||
|
public LinkedListImpl.Node<ObservableNode> publicPrev() {
|
||||||
|
return prev();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testAddAndRemove() {
|
public void testAddAndRemove() {
|
||||||
final AtomicInteger count = new AtomicInteger(0);
|
LinkedListImpl<ObservableNode> objs = new LinkedListImpl<>();
|
||||||
class MyObject {
|
|
||||||
|
|
||||||
private final byte[] payload;
|
|
||||||
|
|
||||||
MyObject() {
|
|
||||||
count.incrementAndGet();
|
|
||||||
payload = new byte[10 * 1024];
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void finalize() throws Exception {
|
|
||||||
count.decrementAndGet();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LinkedListImpl<MyObject> objs = new LinkedListImpl<>();
|
|
||||||
|
|
||||||
// Initial add
|
// Initial add
|
||||||
for (int i = 0; i < 100; i++) {
|
for (int i = 0; i < 100; i++) {
|
||||||
objs.addTail(new MyObject());
|
final ObservableNode o = new ObservableNode();
|
||||||
|
objs.addTail(o);
|
||||||
}
|
}
|
||||||
|
|
||||||
LinkedListIterator<MyObject> iter = objs.iterator();
|
try (LinkedListIterator<ObservableNode> iter = objs.iterator()) {
|
||||||
|
|
||||||
for (int i = 0; i < 500; i++) {
|
for (int i = 0; i < 500; i++) {
|
||||||
|
|
||||||
for (int add = 0; add < 1000; add++) {
|
for (int add = 0; add < 1000; add++) {
|
||||||
objs.addTail(new MyObject());
|
final ObservableNode o = new ObservableNode();
|
||||||
|
objs.addTail(o);
|
||||||
|
assertNotNull("prev", o.publicPrev());
|
||||||
|
assertNull("next", o.publicNext());
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int remove = 0; remove < 1000; remove++) {
|
for (int remove = 0; remove < 1000; remove++) {
|
||||||
assertNotNull(iter.next());
|
final ObservableNode next = iter.next();
|
||||||
|
assertNotNull(next);
|
||||||
|
assertNotNull("prev", next.publicPrev());
|
||||||
|
//it's ok to check this, because we've *at least* 100 elements left!
|
||||||
|
assertNotNull("next", next.publicNext());
|
||||||
iter.remove();
|
iter.remove();
|
||||||
|
assertNull("prev", next.publicPrev());
|
||||||
|
assertNull("next", next.publicNext());
|
||||||
}
|
}
|
||||||
|
assertEquals(100, objs.size());
|
||||||
if (i % 100 == 0) {
|
|
||||||
assertCount(100, count);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
assertCount(100, count);
|
|
||||||
|
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
iter.next();
|
final ObservableNode next = iter.next();
|
||||||
|
assertNotNull(next);
|
||||||
iter.remove();
|
iter.remove();
|
||||||
|
assertNull("prev", next.publicPrev());
|
||||||
|
assertNull("next", next.publicNext());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
assertCount(0, count);
|
assertEquals(0, objs.size());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testAddHeadAndRemove() {
|
public void testAddHeadAndRemove() {
|
||||||
final AtomicInteger count = new AtomicInteger(0);
|
LinkedListImpl<ObservableNode> objs = new LinkedListImpl<>();
|
||||||
class MyObject {
|
|
||||||
|
|
||||||
public int payload;
|
|
||||||
|
|
||||||
MyObject(int payloadcount) {
|
|
||||||
count.incrementAndGet();
|
|
||||||
this.payload = payloadcount;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void finalize() throws Exception {
|
|
||||||
count.decrementAndGet();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return "" + payload;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LinkedListImpl<MyObject> objs = new LinkedListImpl<>();
|
|
||||||
|
|
||||||
// Initial add
|
// Initial add
|
||||||
for (int i = 1000; i >= 0; i--) {
|
for (int i = 0; i < 1001; i++) {
|
||||||
objs.addHead(new MyObject(i));
|
final ObservableNode o = new ObservableNode();
|
||||||
|
objs.addHead(o);
|
||||||
}
|
}
|
||||||
assertCount(1001, count);
|
assertEquals(1001, objs.size());
|
||||||
|
|
||||||
LinkedListIterator<MyObject> iter = objs.iterator();
|
|
||||||
|
|
||||||
int countLoop = 0;
|
int countLoop = 0;
|
||||||
|
|
||||||
|
try (LinkedListIterator<ObservableNode> iter = objs.iterator()) {
|
||||||
|
int removed = 0;
|
||||||
for (countLoop = 0; countLoop <= 1000; countLoop++) {
|
for (countLoop = 0; countLoop <= 1000; countLoop++) {
|
||||||
MyObject obj = iter.next();
|
final ObservableNode obj = iter.next();
|
||||||
assertEquals(countLoop, obj.payload);
|
Assert.assertNotNull(obj);
|
||||||
if (countLoop == 500 || countLoop == 1000) {
|
if (countLoop == 500 || countLoop == 1000) {
|
||||||
|
assertNotNull("prev", obj.publicPrev());
|
||||||
iter.remove();
|
iter.remove();
|
||||||
|
assertNull("prev", obj.publicPrev());
|
||||||
|
assertNull("next", obj.publicNext());
|
||||||
|
removed++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
assertEquals(1001 - removed, objs.size());
|
||||||
|
}
|
||||||
|
|
||||||
iter.close();
|
final int expectedSize = objs.size();
|
||||||
|
try (LinkedListIterator<ObservableNode> iter = objs.iterator()) {
|
||||||
iter = objs.iterator();
|
|
||||||
|
|
||||||
countLoop = 0;
|
countLoop = 0;
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
if (countLoop == 500 || countLoop == 1000) {
|
final ObservableNode obj = iter.next();
|
||||||
System.out.println("Jumping " + countLoop);
|
assertNotNull(obj);
|
||||||
countLoop++;
|
countLoop++;
|
||||||
}
|
}
|
||||||
MyObject obj = iter.next();
|
Assert.assertEquals(expectedSize, countLoop);
|
||||||
assertEquals(countLoop, obj.payload);
|
|
||||||
countLoop++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
assertCount(999, count);
|
|
||||||
|
|
||||||
// it's needed to add this line here because IBM JDK calls finalize on all objects in list
|
// it's needed to add this line here because IBM JDK calls finalize on all objects in list
|
||||||
// before previous assert is called and fails the test, this will prevent it
|
// before previous assert is called and fails the test, this will prevent it
|
||||||
objs.clear();
|
objs.clear();
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param count
|
|
||||||
*/
|
|
||||||
private void assertCount(final int expected, final AtomicInteger count) {
|
|
||||||
long timeout = System.currentTimeMillis() + 15000;
|
|
||||||
|
|
||||||
int seqCount = 0;
|
|
||||||
while (timeout > System.currentTimeMillis() && count.get() != expected) {
|
|
||||||
seqCount++;
|
|
||||||
if (seqCount > 5) {
|
|
||||||
LinkedList<String> toOME = new LinkedList<>();
|
|
||||||
int someCount = 0;
|
|
||||||
try {
|
|
||||||
WeakReference<Object> ref = new WeakReference<>(new Object());
|
|
||||||
while (ref.get() != null) {
|
|
||||||
toOME.add("sdlfkjshadlfkjhas dlfkjhas dlfkjhads lkjfhads lfkjhads flkjashdf " + someCount++);
|
|
||||||
}
|
|
||||||
} catch (Throwable expectedThrowable) {
|
|
||||||
}
|
|
||||||
|
|
||||||
toOME.clear();
|
|
||||||
}
|
|
||||||
forceGC();
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(expected, count.get());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -850,6 +806,49 @@ public class LinkedListTest extends ActiveMQTestBase {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGCNepotismPoll() {
|
||||||
|
final int count = 100;
|
||||||
|
final LinkedListImpl<ObservableNode> list = new LinkedListImpl<>();
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
final ObservableNode node = new ObservableNode();
|
||||||
|
assertNull(node.publicPrev());
|
||||||
|
assertNull(node.publicNext());
|
||||||
|
list.addTail(node);
|
||||||
|
assertNotNull(node.publicPrev());
|
||||||
|
}
|
||||||
|
ObservableNode node;
|
||||||
|
int removed = 0;
|
||||||
|
while ((node = list.poll()) != null) {
|
||||||
|
assertNull(node.publicPrev());
|
||||||
|
assertNull(node.publicNext());
|
||||||
|
removed++;
|
||||||
|
}
|
||||||
|
assertEquals(count, removed);
|
||||||
|
assertEquals(0, list.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGCNepotismClear() {
|
||||||
|
final int count = 100;
|
||||||
|
final ObservableNode[] nodes = new ObservableNode[count];
|
||||||
|
final LinkedListImpl<ObservableNode> list = new LinkedListImpl<>();
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
final ObservableNode node = new ObservableNode();
|
||||||
|
assertNull(node.publicPrev());
|
||||||
|
assertNull(node.publicNext());
|
||||||
|
nodes[i] = node;
|
||||||
|
list.addTail(node);
|
||||||
|
assertNotNull(node.publicPrev());
|
||||||
|
}
|
||||||
|
list.clear();
|
||||||
|
for (ObservableNode node : nodes) {
|
||||||
|
assertNull(node.publicPrev());
|
||||||
|
assertNull(node.publicNext());
|
||||||
|
}
|
||||||
|
assertEquals(0, list.size());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testClear() {
|
public void testClear() {
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue