diff options
author | Andy Clement <aclement@pivotal.io> | 2021-06-30 11:43:27 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-06-30 11:43:27 -0700 |
commit | a8239df4c8485b36916d1dfc2ec4b73820629e65 (patch) | |
tree | b59dab9e8775da5669b83d9adb9576002385cce7 | |
parent | 82f13664d0fcc9a392486724c5e1278dc8d9a408 (diff) | |
parent | 91024728dc6033889507f6c5d7510931761fb1b8 (diff) | |
download | aspectj-a8239df4c8485b36916d1dfc2ec4b73820629e65.tar.gz aspectj-a8239df4c8485b36916d1dfc2ec4b73820629e65.zip |
Merge pull request #77 from SmallGiantGames/bug415838-test
Test for 1.9.6 concurrency bug in LocalVariableTable.unpack()
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(); + } +} |