@@ -512,6 +512,9 @@ void ObjectMonitor::enter_with_contention_mark(JavaThread *current, ObjectMonito
512
512
if (result == freeze_ok) {
513
513
bool acquired = VThreadMonitorEnter(current);
514
514
if (acquired) {
515
+ // We actually acquired the monitor while trying to add the vthread to the
516
+ // _cxq so cancel preemption. We will still go through the preempt stub
517
+ // but instead of unmounting we will call thaw to continue execution.
515
518
current->set_preemption_cancelled(true);
516
519
if (JvmtiExport::should_post_monitor_contended_entered()) {
517
520
// We are going to call thaw again after this and finish the VMTS
@@ -1151,16 +1154,15 @@ bool ObjectMonitor::resume_operation(JavaThread* current, ObjectWaiter* node, Co
1151
1154
oop vthread = current->vthread();
1152
1155
if (is_succesor(current)) clear_succesor();
1153
1156
1154
- // Invariant: after clearing _succ a thread *must* retry _owner before parking .
1157
+ // Invariant: after clearing _succ a thread *must* retry acquiring the monitor .
1155
1158
OrderAccess::fence();
1156
1159
1157
1160
if (TryLock(current) == TryLockResult::Success) {
1158
1161
VThreadEpilog(current, node);
1159
1162
return true;
1160
1163
}
1161
1164
1162
- // The JT will read this variable on return to the resume_monitor_operation stub
1163
- // and will unmount (enterSpecial frame removed and return to Continuation.run()).
1165
+ // We will return to Continuation.run() and unmount so set the right state.
1164
1166
java_lang_VirtualThread::set_state(vthread, java_lang_VirtualThread::BLOCKING);
1165
1167
1166
1168
return false;
@@ -1188,6 +1190,7 @@ void ObjectMonitor::VThreadEpilog(JavaThread* current, ObjectWaiter* node) {
1188
1190
UnlinkAfterAcquire(current, node);
1189
1191
delete node;
1190
1192
1193
+ // Remove the ObjectWaiter* from the stackChunk.
1191
1194
oop vthread = current->vthread();
1192
1195
oop cont = java_lang_VirtualThread::continuation(vthread);
1193
1196
stackChunkOop chunk = jdk_internal_vm_Continuation::tail(cont);
@@ -1624,6 +1627,10 @@ static void vthread_monitor_waited_event(JavaThread *current, ObjectWaiter* node
1624
1627
post_monitor_wait_event(event, node->_monitor, node->_notifier_tid, timeout, timed_out);
1625
1628
}
1626
1629
if (JvmtiExport::should_post_monitor_waited()) {
1630
+ // We mark this call in case of an upcall to Java while posting the event.
1631
+ // If somebody walks the stack in that case, processing the enterSpecial
1632
+ // frame should not include processing callee arguments since there is no
1633
+ // actual callee (see nmethod::preserve_callee_argument_oops()).
1627
1634
ThreadOnMonitorWaitedEvent tmwe(current);
1628
1635
JvmtiExport::vthread_post_monitor_waited(current, node->_monitor, timed_out);
1629
1636
}
@@ -2004,7 +2011,7 @@ void ObjectMonitor::VThreadWait(JavaThread* current, jlong millis) {
2004
2011
Thread::SpinRelease(&_WaitSetLock);
2005
2012
2006
2013
node->_recursions = _recursions; // record the old recursion count
2007
- _recursions = 0; // set the recursion level to be 1
2014
+ _recursions = 0; // set the recursion level to be 0
2008
2015
_waiters++; // increment the number of waiters
2009
2016
exit(current); // exit the monitor
2010
2017
guarantee(!is_owner(current), "invariant");
@@ -2013,17 +2020,16 @@ void ObjectMonitor::VThreadWait(JavaThread* current, jlong millis) {
2013
2020
java_lang_VirtualThread::set_state(vthread, millis == 0 ? java_lang_VirtualThread::WAITING : java_lang_VirtualThread::TIMED_WAITING);
2014
2021
java_lang_VirtualThread::set_waitTimeout(vthread, millis);
2015
2022
2016
- // Save the ObjectWaiter* in the chunk since we will need it
2017
- // when resuming execution.
2023
+ // Save the ObjectWaiter* in the chunk since we will need it when resuming execution.
2018
2024
oop cont = java_lang_VirtualThread::continuation(vthread);
2019
2025
stackChunkOop chunk = jdk_internal_vm_Continuation::tail(cont);
2020
2026
chunk->set_object_waiter(node);
2021
2027
}
2022
2028
2023
2029
bool ObjectMonitor::VThreadWaitReenter(JavaThread* current, ObjectWaiter* node, ContinuationWrapper& cont) {
2024
2030
// First time we run after being preempted on Object.wait().
2025
- // We need to check if we were interrupted or wait() timed-out
2026
- // and in that case remove ourselves from the _WaitSet queue.
2031
+ // Check if we were interrupted or the wait timed-out, and in
2032
+ // that case remove ourselves from the _WaitSet queue.
2027
2033
if (node->TState == ObjectWaiter::TS_WAIT) {
2028
2034
Thread::SpinAcquire(&_WaitSetLock, "WaitSet - unlink");
2029
2035
if (node->TState == ObjectWaiter::TS_WAIT) {
@@ -2034,18 +2040,20 @@ bool ObjectMonitor::VThreadWaitReenter(JavaThread* current, ObjectWaiter* node,
2034
2040
Thread::SpinRelease(&_WaitSetLock);
2035
2041
}
2036
2042
2043
+ // If this was an interrupted case, set the _interrupted boolean so that
2044
+ // once we re-acquire the monitor we know if we need to throw IE or not.
2037
2045
ObjectWaiter::TStates state = node->TState;
2038
2046
bool was_notified = state == ObjectWaiter::TS_ENTER || state == ObjectWaiter::TS_CXQ;
2039
2047
assert(was_notified || state == ObjectWaiter::TS_RUN, "");
2040
-
2041
- // save it so that once we re-acquire the monitor we know if we need to throw IE.
2042
2048
node->_interrupted = !was_notified && current->is_interrupted(false);
2043
2049
2050
+ // Post JFR and JVMTI events.
2044
2051
EventJavaMonitorWait event;
2045
2052
if (event.should_commit() || JvmtiExport::should_post_monitor_waited()) {
2046
2053
vthread_monitor_waited_event(current, node, cont, &event, !was_notified && !node->_interrupted);
2047
2054
}
2048
2055
2056
+ // Mark that we are at reenter so that we don't call this method again.
2049
2057
node->_at_reenter = true;
2050
2058
assert(!is_owner(current), "invariant");
2051
2059
0 commit comments