Skip to content

Commit 1e52c8e

Browse files
committedOct 13, 2020
fixed deadlocks in JVMTI S/R support for cthreads and VTMT disabling mechanism
1 parent 7071a8a commit 1e52c8e

File tree

6 files changed

+14
-37
lines changed

6 files changed

+14
-37
lines changed
 

‎src/hotspot/share/prims/jvmtiEnvBase.cpp

+7-18
Original file line numberDiff line numberDiff line change
@@ -655,7 +655,7 @@ JvmtiEnvBase::get_thread_state(oop thread_oop, JavaThread* jt) {
655655
// We have a JavaThread* so add more state bits.
656656
JavaThreadState jts = jt->thread_state();
657657

658-
if (cthread_with_mounted_vthread(jt) && jt->is_cthread_pending_suspend()) {
658+
if (jt->is_cthread_pending_suspend()) {
659659
// Suspended carrier thread with a mounted virtual thread.
660660
state |= JVMTI_THREAD_STATE_SUSPENDED;
661661
}
@@ -1455,11 +1455,10 @@ JvmtiEnvBase::suspend_thread(oop thread_oop, JavaThread* java_thread, bool singl
14551455
oop mounted_vt = java_thread->mounted_vthread();
14561456

14571457
if (single_suspend && JvmtiExport::can_support_virtual_threads() &&
1458+
!java_lang_VirtualThread::is_instance(thread_oop) &&
14581459
mounted_vt != NULL && thread_oop != mounted_vt) {
14591460
// A case of a carrier thread executing a mounted virtual thread.
1460-
assert(!java_lang_VirtualThread::is_instance(thread_oop) &&
1461-
java_lang_VirtualThread::is_instance(mounted_vt),
1462-
"sanity check");
1461+
assert(java_lang_VirtualThread::is_instance(mounted_vt), "sanity check");
14631462
if (java_thread->is_cthread_pending_suspend()) {
14641463
return JVMTI_ERROR_THREAD_SUSPENDED;
14651464
}
@@ -1528,20 +1527,10 @@ JvmtiEnvBase::resume_thread(oop thread_oop, JavaThread* java_thread, bool single
15281527
if (java_thread->is_hidden_from_external_view()) {
15291528
return JVMTI_ERROR_NONE;
15301529
}
1531-
1532-
oop mounted_vt = java_thread->mounted_vthread();
1533-
if (single_suspend && JvmtiExport::can_support_virtual_threads() &&
1534-
mounted_vt != NULL && thread_oop != java_thread->mounted_vthread()) {
1535-
// A case of a carrier thread executing a mounted virtual thread.
1536-
assert(!java_lang_VirtualThread::is_instance(thread_oop) &&
1537-
java_lang_VirtualThread::is_instance(mounted_vt),
1538-
"sanity check");
1539-
if (java_thread->is_cthread_pending_suspend()) {
1540-
java_thread->clear_cthread_pending_suspend();
1541-
return JVMTI_ERROR_NONE;
1542-
} else {
1543-
return JVMTI_ERROR_THREAD_NOT_SUSPENDED;
1544-
}
1530+
// A case of a carrier thread executing a mounted virtual thread.
1531+
if (java_thread->is_cthread_pending_suspend()) {
1532+
java_thread->clear_cthread_pending_suspend();
1533+
return JVMTI_ERROR_NONE;
15451534
}
15461535
if (!java_thread->is_being_ext_suspended()) {
15471536
return JVMTI_ERROR_THREAD_NOT_SUSPENDED;

‎src/hotspot/share/prims/jvmtiThreadState.cpp

+3-11
Original file line numberDiff line numberDiff line change
@@ -205,11 +205,7 @@ JvmtiVTMTDisabler::disable_VTMT() {
205205
_VTMT_disable_count++;
206206

207207
// Block while some mount/unmount transitions are in progress.
208-
// Thread in VTMT can call JVMTI functions in mount/unmount event handlers.
209-
// The JVMTI functions supporting virtual threads normally disable VTMT.
210-
// To avoid deadlocks, such VTMT disablers are not blocked on wait below.
211-
// Other VTMT disablers are waiting until all VTMT's go through.
212-
while (_VTMT_count > 0 && !thread->in_vtmt()) {
208+
while (_VTMT_count > 0) {
213209
ml.wait();
214210
}
215211
}
@@ -220,8 +216,7 @@ JvmtiVTMTDisabler::enable_VTMT() {
220216
return;
221217
}
222218
MonitorLocker ml(JvmtiVTMT_lock, Mutex::_no_safepoint_check_flag);
223-
assert((JavaThread::current()->in_vtmt() || _VTMT_count == 0) && _VTMT_disable_count > 0,
224-
"VTMT sanity check");
219+
assert(_VTMT_count == 0 && _VTMT_disable_count > 0, "VTMT sanity check");
225220

226221
if (--_VTMT_disable_count == 0) {
227222
ml.notify_all();
@@ -247,7 +242,6 @@ JvmtiVTMTDisabler::start_VTMT(jthread vthread, int callsite_tag) {
247242
ml.wait();
248243
}
249244
_VTMT_count++;
250-
thread->set_in_vtmt(true);
251245
}
252246

253247
void
@@ -259,18 +253,16 @@ JvmtiVTMTDisabler::finish_VTMT(jthread vthread, int callsite_tag) {
259253
MonitorLocker ml(JvmtiVTMT_lock, Mutex::_no_safepoint_check_flag);
260254

261255
_VTMT_count--;
262-
thread->set_in_vtmt(false);
263256

264257
// unblock waiting VTMT disablers
265258
if (_VTMT_disable_count > 0) {
266259
ml.notify_all();
267260
}
268261
if (callsite_tag == 1) { // finish_VTMT for a vthread unmount
262+
MonitorLocker ml(thread->SR_lock(), Mutex::_no_safepoint_check_flag);
269263
if (thread->is_cthread_pending_suspend()) {
270264
thread->clear_cthread_pending_suspend();
271-
// Safe to do without grabbing SR_lock as the thread is current.
272265
// The JavaThread* will be suspended upon return to Java.
273-
MonitorLocker ml(thread->SR_lock(), Mutex::_no_safepoint_check_flag);
274266
thread->set_external_suspend();
275267
}
276268
}

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

-1
Original file line numberDiff line numberDiff line change
@@ -1686,7 +1686,6 @@ void JavaThread::initialize() {
16861686
set_deopt_compiled_method(NULL);
16871687
set_monitor_chunks(NULL);
16881688
_on_thread_list = false;
1689-
_in_vtmt = false;
16901689
_thread_state = _thread_new;
16911690
_terminated = _not_terminated;
16921691
_suspend_equivalent = false;

‎src/hotspot/share/runtime/thread.hpp

-3
Original file line numberDiff line numberDiff line change
@@ -1036,7 +1036,6 @@ class JavaThread: public Thread {
10361036
friend class Continuation;
10371037
private:
10381038
bool _on_thread_list; // Is set when this JavaThread is added to the Threads list
1039-
bool _in_vtmt;
10401039
OopHandle _threadObj; // The Java level thread object
10411040
OopHandle _vthread;
10421041
OopHandle _scopedCache;
@@ -1363,8 +1362,6 @@ class JavaThread: public Thread {
13631362
void smr_delete();
13641363
bool on_thread_list() const { return _on_thread_list; }
13651364
void set_on_thread_list() { _on_thread_list = true; }
1366-
bool in_vtmt() const { return _in_vtmt; }
1367-
void set_in_vtmt(bool val) { _in_vtmt = val; }
13681365

13691366
// thread has called JavaThread::exit() or is terminated
13701367
bool is_exiting() const;

‎src/java.base/share/classes/java/lang/VirtualThread.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -907,25 +907,25 @@ private void notifyJvmtiMountBegin(boolean firstRun) {
907907

908908
private void notifyJvmtiMountEnd(boolean firstRun) {
909909
Thread carrier = Thread.currentCarrierThread();
910+
notifyVTMTFinish(this, 0);
910911
if (firstRun) {
911912
notifyStarted(carrier, this);
912913
}
913914
notifyMount(carrier, this);
914-
notifyVTMTFinish(this, 0);
915915
}
916916

917917
private void notifyJvmtiUnmountBegin() {
918-
notifyVTMTStart(this, 1);
919918
notifyUnmount(Thread.currentCarrierThread(), this);
919+
notifyVTMTStart(this, 1);
920920
}
921921

922922
private void notifyJvmtiUnmountEnd() {
923923
notifyVTMTFinish(this, 1);
924924
}
925925

926926
private void notifyJvmtiTerminate() {
927-
notifyVTMTStart(this, 0);
928927
notifyTerminated(Thread.currentCarrierThread(), this);
928+
notifyVTMTStart(this, 0);
929929
notifyVTMTFinish(this, 0);
930930
}
931931

‎test/hotspot/jtreg/vmTestbase/nsk/jvmti/SuspendThread/suspendvthr001.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,7 @@ private int test_vthreads() {
9494
if (status != Consts.TEST_PASSED) {
9595
return status;
9696
}
97-
sleep(100); // let tested vthreads work while they are tested in native agent
97+
sleep(3000); // let tested vthreads work while they are tested in native agent
9898

9999
System.out.println("\n## Java: runIt: Finishing vthreads");
100100
try {

0 commit comments

Comments
 (0)
Please sign in to comment.