aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAndy Clement <aclement@pivotal.io>2021-06-30 11:43:27 -0700
committerGitHub <noreply@github.com>2021-06-30 11:43:27 -0700
commita8239df4c8485b36916d1dfc2ec4b73820629e65 (patch)
treeb59dab9e8775da5669b83d9adb9576002385cce7
parent82f13664d0fcc9a392486724c5e1278dc8d9a408 (diff)
parent91024728dc6033889507f6c5d7510931761fb1b8 (diff)
downloadaspectj-a8239df4c8485b36916d1dfc2ec4b73820629e65.tar.gz
aspectj-a8239df4c8485b36916d1dfc2ec4b73820629e65.zip
Merge pull request #77 from SmallGiantGames/bug415838-test
Test for 1.9.6 concurrency bug in LocalVariableTable.unpack()
-rw-r--r--bcel-builder/src/main/java/org/aspectj/apache/bcel/classfile/LocalVariableTable.java14
-rw-r--r--bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/LocalVariableTableConcurrencyTest.java95
2 files changed, 108 insertions, 1 deletions
diff --git a/bcel-builder/src/main/java/org/aspectj/apache/bcel/classfile/LocalVariableTable.java b/bcel-builder/src/main/java/org/aspectj/apache/bcel/classfile/LocalVariableTable.java
index a67caf8c1..96c52d20f 100644
--- a/bcel-builder/src/main/java/org/aspectj/apache/bcel/classfile/LocalVariableTable.java
+++ b/bcel-builder/src/main/java/org/aspectj/apache/bcel/classfile/LocalVariableTable.java
@@ -190,6 +190,18 @@ public class LocalVariableTable extends Attribute {
}
/**
+ * Returns copy of this attribute using same packed state. Used in unit tests.
+ */
+ public synchronized LocalVariableTable copyFromPackedState() {
+ if (!isInPackedState) throw new IllegalStateException("No in packed state");
+ try {
+ return new LocalVariableTable(nameIndex, length, new DataInputStream(new ByteArrayInputStream(data)), getConstantPool());
+ } catch (IOException e) {
+ throw new RuntimeException("Failed to unpack clone", e);
+ }
+ }
+
+ /**
* @return deep copy of this attribute
*/
// public Attribute copy(ConstantPool constant_pool) {
@@ -223,7 +235,7 @@ public class LocalVariableTable extends Attribute {
dis.close();
data = null; // throw it away now
} catch (IOException e) {
- throw new RuntimeException("Unpacking of LocalVariableTable attribute failed");
+ throw new RuntimeException("Unpacking of LocalVariableTable attribute failed", e);
}
isInPackedState = false;
}
diff --git a/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/LocalVariableTableConcurrencyTest.java b/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/LocalVariableTableConcurrencyTest.java
new file mode 100644
index 000000000..b62ffe380
--- /dev/null
+++ b/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/LocalVariableTableConcurrencyTest.java
@@ -0,0 +1,95 @@
+/* *******************************************************************
+ * Copyright (c) 2004 IBM
+ * All rights reserved.
+ * This program and the accompanying materials are made available
+ * under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ * Dmitry Mikhaylov - initial implementation
+ * ******************************************************************/
+
+package org.aspectj.apache.bcel.classfile.tests;
+
+import org.aspectj.apache.bcel.classfile.Code;
+import org.aspectj.apache.bcel.classfile.JavaClass;
+import org.aspectj.apache.bcel.classfile.LocalVariableTable;
+import org.aspectj.apache.bcel.classfile.Method;
+
+import java.util.concurrent.CountDownLatch;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.Semaphore;
+import java.util.concurrent.atomic.AtomicReference;
+
+
+public class LocalVariableTableConcurrencyTest extends BcelTestCase {
+
+ private final int nThreads = Runtime.getRuntime().availableProcessors();
+
+ private final ExecutorService[] workers = new ExecutorService[nThreads];
+
+ private LocalVariableTable reference;
+
+ protected void setUp() throws Exception {
+ super.setUp();
+
+ for (int i = 0; i < nThreads; i++) workers[i] = Executors.newSingleThreadExecutor();
+
+ JavaClass clazz = getClassFromJar("SimpleGenericsProgram");
+ Method mainMethod = getMethod(clazz,"main");
+ Code codeAttr = (Code) findAttribute("Code",mainMethod.getAttributes());
+ reference = (LocalVariableTable) findAttribute("LocalVariableTable",codeAttr.getAttributes());
+ }
+
+ /**
+ * Try to hit concurrency bug in org.aspectj.apache.bcel.classfile.LocalVariableTable.unpack().
+ * We do so by running unpack() on same instance with multiple threads, and artificially
+ * delaying all threads but first so that they enter unpack() the moment first thread is about to leave it.
+ *
+ * Since this test relies on empirically obtained access pattern and number of iterations,
+ * it never has 100% probability of hitting the bug. If it fails - there is certainly a bug.
+ * If it passes, it could mean anything - fully correct code or slightly changed execution order preventing
+ * threads to collide at problematic location.
+ *
+ * As such, it is not really good for unit testing.
+ */
+ public void testLocalVariableTableAttributeConcurrency() throws RuntimeException, InterruptedException {
+ final AtomicReference<RuntimeException> error = new AtomicReference<>();
+ for (int i = 0; i < 10000; i++) {
+ LocalVariableTable sharedInstance = reference.copyFromPackedState();
+ CountDownLatch preStart = new CountDownLatch(nThreads);
+ Semaphore start = new Semaphore(0);
+ CountDownLatch finish = new CountDownLatch(nThreads);
+
+ for (int j = 0; j < nThreads; j++) {
+ final boolean needsDelay = j > 0;
+ workers[j].execute(() -> {
+ preStart.countDown();
+ start.acquireUninterruptibly();
+ // trying to trigger concurrent unpacking bug - one tread should enter unpack() when other is about to leave it
+ if (needsDelay) reference.copyFromPackedState().getTableLength();
+ try {
+ sharedInstance.getTableLength();
+ }
+ catch (RuntimeException ex) {
+ error.compareAndSet(null, ex);
+ }
+ finish.countDown();
+ });
+ }
+
+ preStart.await();
+ start.release(nThreads);
+ finish.await();
+
+ if (error.get() != null) throw error.get();
+ }
+ }
+
+ protected void tearDown() throws Exception {
+ for (int i = 0; i < nThreads; i++) if (workers[i] != null) workers[i].shutdownNow();
+ super.tearDown();
+ }
+}