|
42 | 42 | * -XX:+WhiteBoxAPI
|
43 | 43 | * -Xbatch
|
44 | 44 | * -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks
|
| 45 | + * -XX:LockingMode=1 |
45 | 46 | * @run driver EATests
|
46 | 47 | * -XX:+UnlockDiagnosticVMOptions
|
47 | 48 | * -Xms256m -Xmx256m
|
|
50 | 51 | * -XX:+WhiteBoxAPI
|
51 | 52 | * -Xbatch
|
52 | 53 | * -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:-EliminateLocks -XX:+EliminateNestedLocks
|
| 54 | + * -XX:LockingMode=1 |
53 | 55 | * @run driver EATests
|
54 | 56 | * -XX:+UnlockDiagnosticVMOptions
|
55 | 57 | * -Xms256m -Xmx256m
|
|
58 | 60 | * -XX:+WhiteBoxAPI
|
59 | 61 | * -Xbatch
|
60 | 62 | * -XX:+DoEscapeAnalysis -XX:-EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks
|
| 63 | + * -XX:LockingMode=1 |
61 | 64 | * @run driver EATests
|
62 | 65 | * -XX:+UnlockDiagnosticVMOptions
|
63 | 66 | * -Xms256m -Xmx256m
|
|
66 | 69 | * -XX:+WhiteBoxAPI
|
67 | 70 | * -Xbatch
|
68 | 71 | * -XX:-DoEscapeAnalysis -XX:-EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks
|
| 72 | + * -XX:LockingMode=1 |
| 73 | + * |
| 74 | + * @run driver EATests |
| 75 | + * -XX:+UnlockDiagnosticVMOptions |
| 76 | + * -Xms256m -Xmx256m |
| 77 | + * -Xbootclasspath/a:. |
| 78 | + * -XX:CompileCommand=dontinline,*::dontinline_* |
| 79 | + * -XX:+WhiteBoxAPI |
| 80 | + * -Xbatch |
| 81 | + * -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks |
| 82 | + * -XX:LockingMode=2 |
| 83 | + * @run driver EATests |
| 84 | + * -XX:+UnlockDiagnosticVMOptions |
| 85 | + * -Xms256m -Xmx256m |
| 86 | + * -Xbootclasspath/a:. |
| 87 | + * -XX:CompileCommand=dontinline,*::dontinline_* |
| 88 | + * -XX:+WhiteBoxAPI |
| 89 | + * -Xbatch |
| 90 | + * -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:-EliminateLocks -XX:+EliminateNestedLocks |
| 91 | + * -XX:LockingMode=2 |
| 92 | + * @run driver EATests |
| 93 | + * -XX:+UnlockDiagnosticVMOptions |
| 94 | + * -Xms256m -Xmx256m |
| 95 | + * -Xbootclasspath/a:. |
| 96 | + * -XX:CompileCommand=dontinline,*::dontinline_* |
| 97 | + * -XX:+WhiteBoxAPI |
| 98 | + * -Xbatch |
| 99 | + * -XX:+DoEscapeAnalysis -XX:-EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks |
| 100 | + * -XX:LockingMode=2 |
| 101 | + * @run driver EATests |
| 102 | + * -XX:+UnlockDiagnosticVMOptions |
| 103 | + * -Xms256m -Xmx256m |
| 104 | + * -Xbootclasspath/a:. |
| 105 | + * -XX:CompileCommand=dontinline,*::dontinline_* |
| 106 | + * -XX:+WhiteBoxAPI |
| 107 | + * -Xbatch |
| 108 | + * -XX:-DoEscapeAnalysis -XX:-EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks |
| 109 | + * -XX:LockingMode=2 |
69 | 110 | *
|
70 | 111 | * @comment Excercise -XX:+DeoptimizeObjectsALot. Mostly to prevent bit-rot because the option is meant to stress object deoptimization
|
71 | 112 | * with non-synthetic workloads.
|
@@ -208,11 +249,13 @@ public static void main(String[] args) {
|
208 | 249 |
|
209 | 250 | // Relocking test cases
|
210 | 251 | new EARelockingSimpleTarget() .run();
|
| 252 | + new EARelockingSimpleWithAccessInOtherThreadTarget() .run(); |
211 | 253 | new EARelockingRecursiveTarget() .run();
|
212 | 254 | new EARelockingNestedInflatedTarget() .run();
|
213 | 255 | new EARelockingNestedInflated_02Target() .run();
|
214 | 256 | new EARelockingArgEscapeLWLockedInCalleeFrameTarget() .run();
|
215 | 257 | new EARelockingArgEscapeLWLockedInCalleeFrame_2Target() .run();
|
| 258 | + new EARelockingArgEscapeLWLockedInCalleeFrameNoRecursiveTarget() .run(); |
216 | 259 | new EAGetOwnedMonitorsTarget() .run();
|
217 | 260 | new EAEntryCountTarget() .run();
|
218 | 261 | new EARelockingObjectCurrentlyWaitingOnTarget() .run();
|
@@ -328,11 +371,13 @@ protected void runTests() throws Exception {
|
328 | 371 |
|
329 | 372 | // Relocking test cases
|
330 | 373 | new EARelockingSimple() .run(this);
|
| 374 | + new EARelockingSimpleWithAccessInOtherThread() .run(this); |
331 | 375 | new EARelockingRecursive() .run(this);
|
332 | 376 | new EARelockingNestedInflated() .run(this);
|
333 | 377 | new EARelockingNestedInflated_02() .run(this);
|
334 | 378 | new EARelockingArgEscapeLWLockedInCalleeFrame() .run(this);
|
335 | 379 | new EARelockingArgEscapeLWLockedInCalleeFrame_2() .run(this);
|
| 380 | + new EARelockingArgEscapeLWLockedInCalleeFrameNoRecursive() .run(this); |
336 | 381 | new EAGetOwnedMonitors() .run(this);
|
337 | 382 | new EAEntryCount() .run(this);
|
338 | 383 | new EARelockingObjectCurrentlyWaitingOn() .run(this);
|
@@ -1707,6 +1752,62 @@ public void dontinline_testMethod() {
|
1707 | 1752 |
|
1708 | 1753 | /////////////////////////////////////////////////////////////////////////////
|
1709 | 1754 |
|
| 1755 | +// The debugger reads and publishes an object with eliminated locking to an instance field. |
| 1756 | +// A 2nd thread in the debuggee finds it there and changes its state using a synchronized method. |
| 1757 | +// Without eager relocking the accesses are unsynchronized which can be observed. |
| 1758 | +class EARelockingSimpleWithAccessInOtherThread extends EATestCaseBaseDebugger { |
| 1759 | + |
| 1760 | + public void runTestCase() throws Exception { |
| 1761 | + BreakpointEvent bpe = resumeTo(TARGET_TESTCASE_BASE_NAME, "dontinline_brkpt", "()V"); |
| 1762 | + printStack(bpe.thread()); |
| 1763 | + String l1ClassName = EARelockingSimpleWithAccessInOtherThreadTarget.SyncCounter.class.getName(); |
| 1764 | + ObjectReference ctr = getLocalRef(bpe.thread().frame(1), l1ClassName, "l1"); |
| 1765 | + setField(testCase, "sharedCounter", ctr); |
| 1766 | + terminateEndlessLoop(); |
| 1767 | + } |
| 1768 | +} |
| 1769 | + |
| 1770 | +class EARelockingSimpleWithAccessInOtherThreadTarget extends EATestCaseBaseTarget { |
| 1771 | + |
| 1772 | + public static class SyncCounter { |
| 1773 | + private int val; |
| 1774 | + public synchronized int inc() { return val++; } |
| 1775 | + } |
| 1776 | + |
| 1777 | + public volatile SyncCounter sharedCounter; |
| 1778 | + |
| 1779 | + @Override |
| 1780 | + public void setUp() { |
| 1781 | + super.setUp(); |
| 1782 | + doLoop = true; |
| 1783 | + Thread.ofPlatform().daemon().start(() -> { |
| 1784 | + while (doLoop) { |
| 1785 | + SyncCounter ctr = sharedCounter; |
| 1786 | + if (ctr != null) { |
| 1787 | + ctr.inc(); |
| 1788 | + } |
| 1789 | + } |
| 1790 | + }); |
| 1791 | + } |
| 1792 | + |
| 1793 | + public void dontinline_testMethod() { |
| 1794 | + SyncCounter l1 = new SyncCounter(); |
| 1795 | + synchronized (l1) { // Eliminated locking |
| 1796 | + l1.inc(); |
| 1797 | + dontinline_brkpt(); // Debugger publishes l1 to sharedCounter. |
| 1798 | + iResult = l1.inc(); // Changes by the 2nd thread will be observed if l1 |
| 1799 | + // was not relocked before passing it to the debugger. |
| 1800 | + } |
| 1801 | + } |
| 1802 | + |
| 1803 | + @Override |
| 1804 | + public int getExpectedIResult() { |
| 1805 | + return 1; |
| 1806 | + } |
| 1807 | +} |
| 1808 | + |
| 1809 | +///////////////////////////////////////////////////////////////////////////// |
| 1810 | + |
1710 | 1811 | // Test recursive locking
|
1711 | 1812 | class EARelockingRecursiveTarget extends EATestCaseBaseTarget {
|
1712 | 1813 |
|
@@ -1905,6 +2006,48 @@ public int getExpectedIResult() {
|
1905 | 2006 |
|
1906 | 2007 | /////////////////////////////////////////////////////////////////////////////
|
1907 | 2008 |
|
| 2009 | +/** |
| 2010 | + * Similar to {@link EARelockingArgEscapeLWLockedInCalleeFrame_2Target}. It does |
| 2011 | + * not use recursive locking and exposed a bug in the lightweight-locking implementation. |
| 2012 | + */ |
| 2013 | +class EARelockingArgEscapeLWLockedInCalleeFrameNoRecursive extends EATestCaseBaseDebugger { |
| 2014 | + |
| 2015 | + public void runTestCase() throws Exception { |
| 2016 | + BreakpointEvent bpe = resumeTo(TARGET_TESTCASE_BASE_NAME, "dontinline_brkpt", "()V"); |
| 2017 | + printStack(bpe.thread()); |
| 2018 | + @SuppressWarnings("unused") |
| 2019 | + ObjectReference o = getLocalRef(bpe.thread().frame(2), XYVAL_NAME, "l1"); |
| 2020 | + } |
| 2021 | +} |
| 2022 | + |
| 2023 | +class EARelockingArgEscapeLWLockedInCalleeFrameNoRecursiveTarget extends EATestCaseBaseTarget { |
| 2024 | + |
| 2025 | + @Override |
| 2026 | + public void setUp() { |
| 2027 | + super.setUp(); |
| 2028 | + testMethodDepth = 2; |
| 2029 | + } |
| 2030 | + |
| 2031 | + public void dontinline_testMethod() { |
| 2032 | + XYVal l1 = new XYVal(1, 1); // NoEscape, scalar replaced |
| 2033 | + XYVal l2 = new XYVal(4, 2); // NoEscape, scalar replaced |
| 2034 | + XYVal l3 = new XYVal(5, 3); // ArgEscape |
| 2035 | + synchronized (l1) { // eliminated |
| 2036 | + synchronized (l2) { // eliminated |
| 2037 | + l3.dontinline_sync_method(this); // l3 escapes |
| 2038 | + } |
| 2039 | + } |
| 2040 | + iResult = l2.x + l2.y; |
| 2041 | + } |
| 2042 | + |
| 2043 | + @Override |
| 2044 | + public int getExpectedIResult() { |
| 2045 | + return 6; |
| 2046 | + } |
| 2047 | +} |
| 2048 | + |
| 2049 | +///////////////////////////////////////////////////////////////////////////// |
| 2050 | + |
1908 | 2051 | /**
|
1909 | 2052 | * Test relocking eliminated (nested) locks of an object on which the
|
1910 | 2053 | * target thread currently waits.
|
|
0 commit comments