Skip to content

Commit ea1ffa3

Browse files
rkennkexmas92reinrich
committedNov 10, 2023
8318895: Deoptimization results in incorrect lightweight locking stack
Co-authored-by: Axel Boldt-Christmas <aboldtch@openjdk.org> Co-authored-by: Richard Reingruber <rrich@openjdk.org> Reviewed-by: dlong, rrich
1 parent c9657ca commit ea1ffa3

File tree

2 files changed

+157
-3
lines changed

2 files changed

+157
-3
lines changed
 

‎src/hotspot/share/runtime/deoptimization.cpp

+14-3
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,7 @@
8383
#include "runtime/stackValue.hpp"
8484
#include "runtime/stackWatermarkSet.hpp"
8585
#include "runtime/stubRoutines.hpp"
86+
#include "runtime/synchronizer.hpp"
8687
#include "runtime/threadSMR.hpp"
8788
#include "runtime/threadWXSetters.inline.hpp"
8889
#include "runtime/vframe.hpp"
@@ -1636,9 +1637,19 @@ bool Deoptimization::relock_objects(JavaThread* thread, GrowableArray<MonitorInf
16361637
}
16371638
}
16381639
}
1639-
BasicLock* lock = mon_info->lock();
1640-
ObjectSynchronizer::enter(obj, lock, deoptee_thread);
1641-
assert(mon_info->owner()->is_locked(), "object must be locked now");
1640+
if (LockingMode == LM_LIGHTWEIGHT && exec_mode == Unpack_none) {
1641+
// We have lost information about the correct state of the lock stack.
1642+
// Inflate the locks instead. Enter then inflate to avoid races with
1643+
// deflation.
1644+
ObjectSynchronizer::enter(obj, nullptr, deoptee_thread);
1645+
assert(mon_info->owner()->is_locked(), "object must be locked now");
1646+
ObjectMonitor* mon = ObjectSynchronizer::inflate(deoptee_thread, obj(), ObjectSynchronizer::inflate_cause_vm_internal);
1647+
assert(mon->owner() == deoptee_thread, "must be");
1648+
} else {
1649+
BasicLock* lock = mon_info->lock();
1650+
ObjectSynchronizer::enter(obj, lock, deoptee_thread);
1651+
assert(mon_info->owner()->is_locked(), "object must be locked now");
1652+
}
16421653
}
16431654
}
16441655
}

‎test/jdk/com/sun/jdi/EATests.java

+143
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@
4242
* -XX:+WhiteBoxAPI
4343
* -Xbatch
4444
* -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks
45+
* -XX:LockingMode=1
4546
* @run driver EATests
4647
* -XX:+UnlockDiagnosticVMOptions
4748
* -Xms256m -Xmx256m
@@ -50,6 +51,7 @@
5051
* -XX:+WhiteBoxAPI
5152
* -Xbatch
5253
* -XX:+DoEscapeAnalysis -XX:+EliminateAllocations -XX:-EliminateLocks -XX:+EliminateNestedLocks
54+
* -XX:LockingMode=1
5355
* @run driver EATests
5456
* -XX:+UnlockDiagnosticVMOptions
5557
* -Xms256m -Xmx256m
@@ -58,6 +60,7 @@
5860
* -XX:+WhiteBoxAPI
5961
* -Xbatch
6062
* -XX:+DoEscapeAnalysis -XX:-EliminateAllocations -XX:+EliminateLocks -XX:+EliminateNestedLocks
63+
* -XX:LockingMode=1
6164
* @run driver EATests
6265
* -XX:+UnlockDiagnosticVMOptions
6366
* -Xms256m -Xmx256m
@@ -66,6 +69,44 @@
6669
* -XX:+WhiteBoxAPI
6770
* -Xbatch
6871
* -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
69110
*
70111
* @comment Excercise -XX:+DeoptimizeObjectsALot. Mostly to prevent bit-rot because the option is meant to stress object deoptimization
71112
* with non-synthetic workloads.
@@ -208,11 +249,13 @@ public static void main(String[] args) {
208249

209250
// Relocking test cases
210251
new EARelockingSimpleTarget() .run();
252+
new EARelockingSimpleWithAccessInOtherThreadTarget() .run();
211253
new EARelockingRecursiveTarget() .run();
212254
new EARelockingNestedInflatedTarget() .run();
213255
new EARelockingNestedInflated_02Target() .run();
214256
new EARelockingArgEscapeLWLockedInCalleeFrameTarget() .run();
215257
new EARelockingArgEscapeLWLockedInCalleeFrame_2Target() .run();
258+
new EARelockingArgEscapeLWLockedInCalleeFrameNoRecursiveTarget() .run();
216259
new EAGetOwnedMonitorsTarget() .run();
217260
new EAEntryCountTarget() .run();
218261
new EARelockingObjectCurrentlyWaitingOnTarget() .run();
@@ -328,11 +371,13 @@ protected void runTests() throws Exception {
328371

329372
// Relocking test cases
330373
new EARelockingSimple() .run(this);
374+
new EARelockingSimpleWithAccessInOtherThread() .run(this);
331375
new EARelockingRecursive() .run(this);
332376
new EARelockingNestedInflated() .run(this);
333377
new EARelockingNestedInflated_02() .run(this);
334378
new EARelockingArgEscapeLWLockedInCalleeFrame() .run(this);
335379
new EARelockingArgEscapeLWLockedInCalleeFrame_2() .run(this);
380+
new EARelockingArgEscapeLWLockedInCalleeFrameNoRecursive() .run(this);
336381
new EAGetOwnedMonitors() .run(this);
337382
new EAEntryCount() .run(this);
338383
new EARelockingObjectCurrentlyWaitingOn() .run(this);
@@ -1707,6 +1752,62 @@ public void dontinline_testMethod() {
17071752

17081753
/////////////////////////////////////////////////////////////////////////////
17091754

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+
17101811
// Test recursive locking
17111812
class EARelockingRecursiveTarget extends EATestCaseBaseTarget {
17121813

@@ -1905,6 +2006,48 @@ public int getExpectedIResult() {
19052006

19062007
/////////////////////////////////////////////////////////////////////////////
19072008

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+
19082051
/**
19092052
* Test relocking eliminated (nested) locks of an object on which the
19102053
* target thread currently waits.

0 commit comments

Comments
 (0)
Please sign in to comment.