ARTEMIS-3802 Avoiding races on deleteWithID and Iterator.remove

This commit is contained in:
Clebert Suconic 2022-04-25 18:43:11 -04:00 committed by clebertsuconic
parent 502461fc9b
commit 1946351fce
2 changed files with 133 additions and 40 deletions

View File

@ -103,7 +103,7 @@ public class LinkedListImpl<E> implements LinkedList<E> {
}
@Override
public E removeWithID(String listID, long id) {
public synchronized E removeWithID(String listID, long id) {
assert nodeStore != null; // it is assumed the code will call setNodeStore before callin removeWithID
Node<E> node = nodeStore.getNode(listID, id);
@ -422,6 +422,7 @@ public class LinkedListImpl<E> implements LinkedList<E> {
@Override
public boolean hasNext() {
synchronized (LinkedListImpl.this) {
Node<E> e = getNode();
if (e != null && (e != last || repeat)) {
@ -430,9 +431,11 @@ public class LinkedListImpl<E> implements LinkedList<E> {
return canAdvance();
}
}
@Override
public E next() {
synchronized (LinkedListImpl.this) {
Node<E> e = getNode();
if (repeat) {
@ -469,21 +472,28 @@ public class LinkedListImpl<E> implements LinkedList<E> {
return e.val();
}
}
@Override
public void remove() {
synchronized (LinkedListImpl.this) {
if (last == null) {
throw new NoSuchElementException();
}
if (current == null) {
throw new NoSuchElementException();
return;
}
LinkedListImpl.this.removeAfter(current.prev);
Node<E> prev = current.prev;
if (prev != null) {
LinkedListImpl.this.removeAfter(prev);
last = null;
}
}
}
@Override
public void close() {

View File

@ -22,6 +22,12 @@ import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import io.netty.util.collection.LongObjectHashMap;
import org.apache.activemq.artemis.tests.util.ActiveMQTestBase;
@ -313,6 +319,83 @@ public class LinkedListTest extends ActiveMQTestBase {
}
}
@Test
public void testRaceRemovingID() throws Exception {
ExecutorService executor = Executors.newFixedThreadPool(2);
int elements = 1000;
try {
LinkedListImpl<ObservableNode> objs = new LinkedListImpl<>();
ListNodeStore nodeStore = new ListNodeStore();
objs.setNodeStore(nodeStore);
final String serverID = RandomUtil.randomString();
for (int i = 0; i < elements; i++) {
objs.addHead(new ObservableNode(serverID, i));
}
Assert.assertEquals(elements, objs.size());
CyclicBarrier barrier = new CyclicBarrier(2);
CountDownLatch latch = new CountDownLatch(2);
AtomicInteger errors = new AtomicInteger(0);
executor.execute(() -> {
try {
barrier.await(10, TimeUnit.SECONDS);
} catch (Exception e) {
e.printStackTrace();
}
try {
for (int i = 0; i < elements; i++) {
objs.removeWithID(serverID, i);
}
} catch (Exception e) {
e.printStackTrace();
errors.incrementAndGet();
} finally {
latch.countDown();
}
});
executor.execute(() -> {
LinkedListIterator iterator = objs.iterator();
try {
barrier.await(10, TimeUnit.SECONDS);
} catch (Exception e) {
e.printStackTrace();
}
try {
while (iterator.hasNext()) {
Object value = iterator.next();
iterator.remove();
}
} catch (Exception e) {
if (e instanceof NoSuchElementException) {
// this is ok
} else {
e.printStackTrace();
errors.incrementAndGet();
}
} finally {
latch.countDown();
}
});
Assert.assertTrue(latch.await(10, TimeUnit.SECONDS));
Assert.assertEquals(0, objs.size());
Assert.assertEquals(0, errors.get());
} finally {
executor.shutdownNow();
}
}
@Test
public void testAddHeadAndRemove() {
LinkedListImpl<ObservableNode> objs = new LinkedListImpl<>();