Skip to content

Commit 888b8dd

Browse files
committedOct 14, 2024
add extra comments
1 parent 5da98bb commit 888b8dd

8 files changed

+45
-29
lines changed
 

‎src/hotspot/cpu/aarch64/interp_masm_aarch64.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -1539,20 +1539,20 @@ void InterpreterMacroAssembler::call_VM_preemptable(Register oop_result,
15391539

15401540
push_cont_fastpath();
15411541

1542-
// Make VM call. In case of preemption set last_pc to
1543-
// the one we want to resume to.
1542+
// Make VM call. In case of preemption set last_pc to the one we want to resume to.
15441543
adr(rscratch1, resume_pc);
15451544
str(rscratch1, Address(rthread, JavaThread::last_Java_pc_offset()));
15461545
call_VM_base(oop_result, noreg, noreg, entry_point, 1, false /*check_exceptions*/);
15471546

15481547
pop_cont_fastpath();
15491548

1550-
// Check if preempted
1549+
// Check if preempted.
15511550
ldr(rscratch1, Address(rthread, JavaThread::preempt_alternate_return_offset()));
15521551
cbz(rscratch1, not_preempted);
15531552
str(zr, Address(rthread, JavaThread::preempt_alternate_return_offset()));
15541553
br(rscratch1);
15551554

1555+
// In case of preemption, this is where we will resume once we finally acquire the monitor.
15561556
bind(resume_pc);
15571557
restore_after_resume(false /* is_native */);
15581558

‎src/hotspot/cpu/aarch64/stubGenerator_aarch64.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -7150,19 +7150,20 @@ class StubGenerator: public StubCodeGenerator {
71507150
// reset the flag
71517151
__ strb(zr, Address(rthread, JavaThread::preempting_offset()));
71527152

7153-
// Set sp to enterSpecial frame and then remove it from the stack
7153+
// Set sp to enterSpecial frame, i.e. remove all frames copied into the heap.
71547154
__ ldr(rscratch2, Address(rthread, JavaThread::cont_entry_offset()));
71557155
__ mov(sp, rscratch2);
71567156

71577157
Label preemption_cancelled;
71587158
__ ldrb(rscratch1, Address(rthread, JavaThread::preemption_cancelled_offset()));
71597159
__ cbnz(rscratch1, preemption_cancelled);
71607160

7161-
// Remove enterSpecial frame from the stack and return to Continuation.run()
7161+
// Remove enterSpecial frame from the stack and return to Continuation.run() to unmount.
71627162
SharedRuntime::continuation_enter_cleanup(_masm);
71637163
__ leave();
71647164
__ ret(lr);
71657165

7166+
// We acquired the monitor after freezing the frames so call thaw to continue execution.
71667167
__ bind(preemption_cancelled);
71677168
__ strb(zr, Address(rthread, JavaThread::preemption_cancelled_offset()));
71687169
__ lea(rfp, Address(sp, checked_cast<int32_t>(ContinuationEntry::size())));

‎src/hotspot/cpu/riscv/interp_masm_riscv.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -1585,20 +1585,20 @@ void InterpreterMacroAssembler::call_VM_preemptable(Register oop_result,
15851585

15861586
push_cont_fastpath();
15871587

1588-
// Make VM call. In case of preemption set last_pc to
1589-
// the one we want to resume to.
1588+
// Make VM call. In case of preemption set last_pc to the one we want to resume to.
15901589
la(t0, resume_pc);
15911590
sd(t0, Address(xthread, JavaThread::last_Java_pc_offset()));
15921591
call_VM_base(oop_result, noreg, noreg, entry_point, 1, false /*check_exceptions*/);
15931592

15941593
pop_cont_fastpath();
15951594

1596-
// Check if preempted
1595+
// Check if preempted.
15971596
ld(t0, Address(xthread, JavaThread::preempt_alternate_return_offset()));
15981597
beqz(t0, not_preempted);
15991598
sd(zr, Address(xthread, JavaThread::preempt_alternate_return_offset()));
16001599
jr(t0);
16011600

1601+
// In case of preemption, this is where we will resume once we finally acquire the monitor.
16021602
bind(resume_pc);
16031603
restore_after_resume(false /* is_native */);
16041604

‎src/hotspot/cpu/riscv/stubGenerator_riscv.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -3883,18 +3883,19 @@ class StubGenerator: public StubCodeGenerator {
38833883
// reset the flag
38843884
__ sb(zr, Address(xthread, JavaThread::preempting_offset()));
38853885

3886-
// Set sp to enterSpecial frame and then remove it from the stack
3886+
// Set sp to enterSpecial frame, i.e. remove all frames copied into the heap.
38873887
__ ld(sp, Address(xthread, JavaThread::cont_entry_offset()));
38883888

38893889
Label preemption_cancelled;
38903890
__ lbu(t0, Address(xthread, JavaThread::preemption_cancelled_offset()));
38913891
__ bnez(t0, preemption_cancelled);
38923892

3893-
// Remove enterSpecial frame from the stack and return to Continuation.run()
3893+
// Remove enterSpecial frame from the stack and return to Continuation.run() to unmount.
38943894
SharedRuntime::continuation_enter_cleanup(_masm);
38953895
__ leave();
38963896
__ ret();
38973897

3898+
// We acquired the monitor after freezing the frames so call thaw to continue execution.
38983899
__ bind(preemption_cancelled);
38993900
__ sb(zr, Address(xthread, JavaThread::preemption_cancelled_offset()));
39003901
__ la(fp, Address(sp, checked_cast<int32_t>(ContinuationEntry::size() + 2 * wordSize)));

‎src/hotspot/cpu/x86/interp_masm_x86.cpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -345,22 +345,22 @@ void InterpreterMacroAssembler::call_VM_preemptable(Register oop_result,
345345

346346
push_cont_fastpath();
347347

348-
// Make VM call. In case of preemption set last_pc to
349-
// the one we want to resume to.
348+
// Make VM call. In case of preemption set last_pc to the one we want to resume to.
350349
lea(rscratch1, resume_pc);
351350
push(rscratch1);
352351
MacroAssembler::call_VM_helper(oop_result, entry_point, 1, false /*check_exceptions*/);
353352
pop(rscratch1);
354353

355354
pop_cont_fastpath();
356355

357-
// Check if preempted
356+
// Check if preempted.
358357
movptr(rscratch1, Address(r15_thread, JavaThread::preempt_alternate_return_offset()));
359358
cmpptr(rscratch1, NULL_WORD);
360359
jccb(Assembler::zero, not_preempted);
361360
movptr(Address(r15_thread, JavaThread::preempt_alternate_return_offset()), NULL_WORD);
362361
jmp(rscratch1);
363362

363+
// In case of preemption, this is where we will resume once we finally acquire the monitor.
364364
bind(resume_pc);
365365
restore_after_resume(false /* is_native */);
366366

@@ -371,7 +371,7 @@ void InterpreterMacroAssembler::restore_after_resume(bool is_native) {
371371
lea(rscratch1, ExternalAddress(Interpreter::cont_resume_interpreter_adapter()));
372372
call(rscratch1);
373373
if (is_native) {
374-
// On resume we need to set up stack as expected
374+
// On resume we need to set up stack as expected.
375375
push(dtos);
376376
push(ltos);
377377
}

‎src/hotspot/cpu/x86/stubGenerator_x86_64.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -3799,7 +3799,7 @@ address StubGenerator::generate_cont_preempt_stub() {
37993799

38003800
__ reset_last_Java_frame(true);
38013801

3802-
// reset _preempting flag
3802+
// Check and reset _preempting flag.
38033803
#ifdef ASSERT
38043804
{ Label L;
38053805
__ movbool(rscratch1, Address(r15_thread, JavaThread::preempting_offset()));
@@ -3811,19 +3811,20 @@ address StubGenerator::generate_cont_preempt_stub() {
38113811
#endif
38123812
__ movbool(Address(r15_thread, JavaThread::preempting_offset()), false);
38133813

3814-
// Set rsp to enterSpecial frame
3814+
// Set rsp to enterSpecial frame, i.e. remove all frames copied into the heap.
38153815
__ movptr(rsp, Address(r15_thread, JavaThread::cont_entry_offset()));
38163816

38173817
Label preemption_cancelled;
38183818
__ movbool(rscratch1, Address(r15_thread, JavaThread::preemption_cancelled_offset()));
38193819
__ testbool(rscratch1);
38203820
__ jcc(Assembler::notZero, preemption_cancelled);
38213821

3822-
// Remove enterSpecial frame from the stack and return to Continuation.run()
3822+
// Remove enterSpecial frame from the stack and return to Continuation.run() to unmount.
38233823
SharedRuntime::continuation_enter_cleanup(_masm);
38243824
__ pop(rbp);
38253825
__ ret(0);
38263826

3827+
// We acquired the monitor after freezing the frames so call thaw to continue execution.
38273828
__ bind(preemption_cancelled);
38283829
__ movbool(Address(r15_thread, JavaThread::preemption_cancelled_offset()), false);
38293830
__ lea(rbp, Address(rsp, checked_cast<int32_t>(ContinuationEntry::size())));

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

+7-2
Original file line numberDiff line numberDiff line change
@@ -2216,6 +2216,8 @@ NOINLINE intptr_t* Thaw<ConfigT>::thaw_slow(stackChunkOop chunk, Continuation::t
22162216
_preempted_case = chunk->preempted();
22172217
if (_preempted_case) {
22182218
if (chunk->object_waiter() != nullptr) {
2219+
// Mounted again after preemption. Resume the pending monitor operation,
2220+
// which will be either a monitorenter or Object.wait() call.
22192221
assert(chunk->current_pending_monitor() != nullptr || chunk->current_waiting_monitor() != nullptr, "");
22202222
ObjectWaiter* waiter = chunk->object_waiter();
22212223
ObjectMonitor* mon = waiter->monitor();
@@ -2224,15 +2226,16 @@ NOINLINE intptr_t* Thaw<ConfigT>::thaw_slow(stackChunkOop chunk, Continuation::t
22242226
bool mon_acquired = mon->resume_operation(_thread, waiter, _cont);
22252227
assert(!mon_acquired || mon->is_owner(_thread), "invariant");
22262228
if (!mon_acquired) {
2229+
// Failed to aquire monitor. Return to enterSpecial to unmount again.
22272230
return push_cleanup_continuation();
22282231
}
2229-
chunk = _cont.tail(); // reload oop in case of safepoint in resume_operation
2232+
chunk = _cont.tail(); // reload oop in case of safepoint in resume_operation (if posting JVMTI events).
22302233
} else {
22312234
// Preemption cancelled in moniterenter case. We actually acquired
22322235
// the monitor after freezing all frames so nothing to do.
22332236
preempt_kind = Continuation::freeze_on_monitorenter;
22342237
}
2235-
// Call this first to avoid racing with GC threads later when modifying the flags.
2238+
// Call this first to avoid racing with GC threads later when modifying the chunk flags.
22362239
relativize_chunk_concurrently(chunk);
22372240
chunk->set_preempted(false);
22382241
retry_fast_path = true;
@@ -2447,6 +2450,7 @@ intptr_t* ThawBase::handle_preempted_continuation(intptr_t* sp, Continuation::pr
24472450
assert(top.pc() == *(address*)(sp - frame::sender_sp_ret_address_offset()), "");
24482451

24492452
#if INCLUDE_JVMTI
2453+
// Finish the VTMS transition.
24502454
assert(_thread->is_in_VTMS_transition(), "must be");
24512455
bool is_vthread = Continuation::continuation_scope(_cont.continuation()) == java_lang_VirtualThread::vthread_scope();
24522456
if (is_vthread) {
@@ -2469,6 +2473,7 @@ intptr_t* ThawBase::handle_preempted_continuation(intptr_t* sp, Continuation::pr
24692473
}
24702474

24712475
if (preempt_kind == Continuation::freeze_on_wait) {
2476+
// Check now if we need to throw IE exception.
24722477
if (_thread->pending_interrupted_exception()) {
24732478
throw_interrupted_exception(_thread, top);
24742479
_thread->set_pending_interrupted_exception(false);

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

+18-10
Original file line numberDiff line numberDiff line change
@@ -512,6 +512,9 @@ void ObjectMonitor::enter_with_contention_mark(JavaThread *current, ObjectMonito
512512
if (result == freeze_ok) {
513513
bool acquired = VThreadMonitorEnter(current);
514514
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.
515518
current->set_preemption_cancelled(true);
516519
if (JvmtiExport::should_post_monitor_contended_entered()) {
517520
// 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
11511154
oop vthread = current->vthread();
11521155
if (is_succesor(current)) clear_succesor();
11531156

1154-
// Invariant: after clearing _succ a thread *must* retry _owner before parking.
1157+
// Invariant: after clearing _succ a thread *must* retry acquiring the monitor.
11551158
OrderAccess::fence();
11561159

11571160
if (TryLock(current) == TryLockResult::Success) {
11581161
VThreadEpilog(current, node);
11591162
return true;
11601163
}
11611164

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.
11641166
java_lang_VirtualThread::set_state(vthread, java_lang_VirtualThread::BLOCKING);
11651167

11661168
return false;
@@ -1188,6 +1190,7 @@ void ObjectMonitor::VThreadEpilog(JavaThread* current, ObjectWaiter* node) {
11881190
UnlinkAfterAcquire(current, node);
11891191
delete node;
11901192

1193+
// Remove the ObjectWaiter* from the stackChunk.
11911194
oop vthread = current->vthread();
11921195
oop cont = java_lang_VirtualThread::continuation(vthread);
11931196
stackChunkOop chunk = jdk_internal_vm_Continuation::tail(cont);
@@ -1624,6 +1627,10 @@ static void vthread_monitor_waited_event(JavaThread *current, ObjectWaiter* node
16241627
post_monitor_wait_event(event, node->_monitor, node->_notifier_tid, timeout, timed_out);
16251628
}
16261629
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()).
16271634
ThreadOnMonitorWaitedEvent tmwe(current);
16281635
JvmtiExport::vthread_post_monitor_waited(current, node->_monitor, timed_out);
16291636
}
@@ -2004,7 +2011,7 @@ void ObjectMonitor::VThreadWait(JavaThread* current, jlong millis) {
20042011
Thread::SpinRelease(&_WaitSetLock);
20052012

20062013
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
20082015
_waiters++; // increment the number of waiters
20092016
exit(current); // exit the monitor
20102017
guarantee(!is_owner(current), "invariant");
@@ -2013,17 +2020,16 @@ void ObjectMonitor::VThreadWait(JavaThread* current, jlong millis) {
20132020
java_lang_VirtualThread::set_state(vthread, millis == 0 ? java_lang_VirtualThread::WAITING : java_lang_VirtualThread::TIMED_WAITING);
20142021
java_lang_VirtualThread::set_waitTimeout(vthread, millis);
20152022

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.
20182024
oop cont = java_lang_VirtualThread::continuation(vthread);
20192025
stackChunkOop chunk = jdk_internal_vm_Continuation::tail(cont);
20202026
chunk->set_object_waiter(node);
20212027
}
20222028

20232029
bool ObjectMonitor::VThreadWaitReenter(JavaThread* current, ObjectWaiter* node, ContinuationWrapper& cont) {
20242030
// 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.
20272033
if (node->TState == ObjectWaiter::TS_WAIT) {
20282034
Thread::SpinAcquire(&_WaitSetLock, "WaitSet - unlink");
20292035
if (node->TState == ObjectWaiter::TS_WAIT) {
@@ -2034,18 +2040,20 @@ bool ObjectMonitor::VThreadWaitReenter(JavaThread* current, ObjectWaiter* node,
20342040
Thread::SpinRelease(&_WaitSetLock);
20352041
}
20362042

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.
20372045
ObjectWaiter::TStates state = node->TState;
20382046
bool was_notified = state == ObjectWaiter::TS_ENTER || state == ObjectWaiter::TS_CXQ;
20392047
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.
20422048
node->_interrupted = !was_notified && current->is_interrupted(false);
20432049

2050+
// Post JFR and JVMTI events.
20442051
EventJavaMonitorWait event;
20452052
if (event.should_commit() || JvmtiExport::should_post_monitor_waited()) {
20462053
vthread_monitor_waited_event(current, node, cont, &event, !was_notified && !node->_interrupted);
20472054
}
20482055

2056+
// Mark that we are at reenter so that we don't call this method again.
20492057
node->_at_reenter = true;
20502058
assert(!is_owner(current), "invariant");
20512059

0 commit comments

Comments
 (0)
Please sign in to comment.