8293613: need to properly handle and hide tmp VTMS transitions
Reviewed-by: cjplummer, lmesnik
This commit is contained in:
parent
5e1e449c11
commit
79ccc791f2
@ -224,4 +224,5 @@ JVM_VirtualThreadMountBegin
|
|||||||
JVM_VirtualThreadMountEnd
|
JVM_VirtualThreadMountEnd
|
||||||
JVM_VirtualThreadUnmountBegin
|
JVM_VirtualThreadUnmountBegin
|
||||||
JVM_VirtualThreadUnmountEnd
|
JVM_VirtualThreadUnmountEnd
|
||||||
|
JVM_VirtualThreadHideFrames
|
||||||
#
|
#
|
||||||
|
@ -1150,6 +1150,9 @@ JVM_VirtualThreadUnmountBegin(JNIEnv* env, jobject vthread, jboolean last_unmoun
|
|||||||
JNIEXPORT void JNICALL
|
JNIEXPORT void JNICALL
|
||||||
JVM_VirtualThreadUnmountEnd(JNIEnv* env, jobject vthread, jboolean last_unmount);
|
JVM_VirtualThreadUnmountEnd(JNIEnv* env, jobject vthread, jboolean last_unmount);
|
||||||
|
|
||||||
|
JNIEXPORT void JNICALL
|
||||||
|
JVM_VirtualThreadHideFrames(JNIEnv* env, jobject vthread, jboolean hide);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Core reflection support.
|
* Core reflection support.
|
||||||
*/
|
*/
|
||||||
|
@ -3882,6 +3882,8 @@ JVM_ENTRY(void, JVM_VirtualThreadMountBegin(JNIEnv* env, jobject vthread, jboole
|
|||||||
assert(!JvmtiExport::can_support_virtual_threads(), "sanity check");
|
assert(!JvmtiExport::can_support_virtual_threads(), "sanity check");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
assert(!thread->is_in_tmp_VTMS_transition(), "sanity check");
|
||||||
|
assert(!thread->is_in_VTMS_transition(), "sanity check");
|
||||||
JvmtiVTMSTransitionDisabler::start_VTMS_transition(vthread, /* is_mount */ true);
|
JvmtiVTMSTransitionDisabler::start_VTMS_transition(vthread, /* is_mount */ true);
|
||||||
#else
|
#else
|
||||||
fatal("Should only be called with JVMTI enabled");
|
fatal("Should only be called with JVMTI enabled");
|
||||||
@ -3906,6 +3908,7 @@ JVM_ENTRY(void, JVM_VirtualThreadMountEnd(JNIEnv* env, jobject vthread, jboolean
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert(thread->is_in_VTMS_transition(), "sanity check");
|
assert(thread->is_in_VTMS_transition(), "sanity check");
|
||||||
|
assert(!thread->is_in_tmp_VTMS_transition(), "sanity check");
|
||||||
JvmtiVTMSTransitionDisabler::finish_VTMS_transition(vthread, /* is_mount */ true);
|
JvmtiVTMSTransitionDisabler::finish_VTMS_transition(vthread, /* is_mount */ true);
|
||||||
if (first_mount) {
|
if (first_mount) {
|
||||||
// thread start
|
// thread start
|
||||||
@ -3954,7 +3957,7 @@ JVM_ENTRY(void, JVM_VirtualThreadUnmountBegin(JNIEnv* env, jobject vthread, jboo
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
assert(!thread->is_in_tmp_VTMS_transition(), "sanity check");
|
||||||
assert(!thread->is_in_VTMS_transition(), "sanity check");
|
assert(!thread->is_in_VTMS_transition(), "sanity check");
|
||||||
JvmtiVTMSTransitionDisabler::start_VTMS_transition(vthread, /* is_mount */ false);
|
JvmtiVTMSTransitionDisabler::start_VTMS_transition(vthread, /* is_mount */ false);
|
||||||
|
|
||||||
@ -3977,12 +3980,27 @@ JVM_ENTRY(void, JVM_VirtualThreadUnmountEnd(JNIEnv* env, jobject vthread, jboole
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
assert(thread->is_in_VTMS_transition(), "sanity check");
|
assert(thread->is_in_VTMS_transition(), "sanity check");
|
||||||
|
assert(!thread->is_in_tmp_VTMS_transition(), "sanity check");
|
||||||
JvmtiVTMSTransitionDisabler::finish_VTMS_transition(vthread, /* is_mount */ false);
|
JvmtiVTMSTransitionDisabler::finish_VTMS_transition(vthread, /* is_mount */ false);
|
||||||
#else
|
#else
|
||||||
fatal("Should only be called with JVMTI enabled");
|
fatal("Should only be called with JVMTI enabled");
|
||||||
#endif
|
#endif
|
||||||
JVM_END
|
JVM_END
|
||||||
|
|
||||||
|
JVM_ENTRY(void, JVM_VirtualThreadHideFrames(JNIEnv* env, jobject vthread, jboolean hide))
|
||||||
|
#if INCLUDE_JVMTI
|
||||||
|
if (!DoJVMTIVirtualThreadTransitions) {
|
||||||
|
assert(!JvmtiExport::can_support_virtual_threads(), "sanity check");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
assert(!thread->is_in_VTMS_transition(), "sanity check");
|
||||||
|
assert(thread->is_in_tmp_VTMS_transition() != (bool)hide, "sanity check");
|
||||||
|
thread->toggle_is_in_tmp_VTMS_transition();
|
||||||
|
#else
|
||||||
|
fatal("Should only be called with JVMTI enabled");
|
||||||
|
#endif
|
||||||
|
JVM_END
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return the current class's class file version. The low order 16 bits of the
|
* Return the current class's class file version. The low order 16 bits of the
|
||||||
* returned jint contain the class's major version. The high order 16 bits
|
* returned jint contain the class's major version. The high order 16 bits
|
||||||
|
@ -703,10 +703,14 @@ JvmtiEnvBase::get_vthread_jvf(oop vthread) {
|
|||||||
javaVFrame*
|
javaVFrame*
|
||||||
JvmtiEnvBase::get_cthread_last_java_vframe(JavaThread* jt, RegisterMap* reg_map_p) {
|
JvmtiEnvBase::get_cthread_last_java_vframe(JavaThread* jt, RegisterMap* reg_map_p) {
|
||||||
// Strip vthread frames in case of carrier thread with mounted continuation.
|
// Strip vthread frames in case of carrier thread with mounted continuation.
|
||||||
javaVFrame *jvf = JvmtiEnvBase::is_cthread_with_continuation(jt) ?
|
bool cthread_with_cont = JvmtiEnvBase::is_cthread_with_continuation(jt);
|
||||||
jt->carrier_last_java_vframe(reg_map_p) :
|
javaVFrame *jvf = cthread_with_cont ? jt->carrier_last_java_vframe(reg_map_p)
|
||||||
jt->last_java_vframe(reg_map_p);
|
: jt->last_java_vframe(reg_map_p);
|
||||||
|
// Skip hidden frames only for carrier threads
|
||||||
|
// which are in non-temporary VTMS transition.
|
||||||
|
if (jt->is_in_VTMS_transition()) {
|
||||||
jvf = check_and_skip_hidden_frames(jt, jvf);
|
jvf = check_and_skip_hidden_frames(jt, jvf);
|
||||||
|
}
|
||||||
return jvf;
|
return jvf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -893,7 +893,7 @@ class JvmtiClassFileLoadHookPoster : public StackObj {
|
|||||||
_cached_class_file_ptr = cache_ptr;
|
_cached_class_file_ptr = cache_ptr;
|
||||||
_has_been_modified = false;
|
_has_been_modified = false;
|
||||||
|
|
||||||
assert(!_thread->is_in_VTMS_transition(), "CFLH events are not allowed in VTMS transition");
|
assert(!_thread->is_in_any_VTMS_transition(), "CFLH events are not allowed in any VTMS transition");
|
||||||
_state = _thread->jvmti_thread_state();
|
_state = _thread->jvmti_thread_state();
|
||||||
if (_state != NULL) {
|
if (_state != NULL) {
|
||||||
_class_being_redefined = _state->get_class_being_redefined();
|
_class_being_redefined = _state->get_class_being_redefined();
|
||||||
@ -1050,6 +1050,9 @@ bool JvmtiExport::post_class_file_load_hook(Symbol* h_name,
|
|||||||
if (JvmtiEnv::get_phase() < JVMTI_PHASE_PRIMORDIAL) {
|
if (JvmtiEnv::get_phase() < JVMTI_PHASE_PRIMORDIAL) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
if (JavaThread::current()->is_in_tmp_VTMS_transition()) {
|
||||||
|
return false; // skip CFLH events in tmp VTMS transition
|
||||||
|
}
|
||||||
|
|
||||||
JvmtiClassFileLoadHookPoster poster(h_name, class_loader,
|
JvmtiClassFileLoadHookPoster poster(h_name, class_loader,
|
||||||
h_protection_domain,
|
h_protection_domain,
|
||||||
@ -1184,8 +1187,8 @@ void JvmtiExport::post_raw_breakpoint(JavaThread *thread, Method* method, addres
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_BREAKPOINT, ("[%s] Trg Breakpoint triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_BREAKPOINT, ("[%s] Trg Breakpoint triggered",
|
||||||
@ -1340,7 +1343,10 @@ void JvmtiExport::post_class_load(JavaThread *thread, Klass* klass) {
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
assert(!thread->is_in_VTMS_transition(), "class load events are not allowed in VTMS transition");
|
if (thread->is_in_tmp_VTMS_transition()) {
|
||||||
|
return; // skip ClassLoad events in tmp VTMS transition
|
||||||
|
}
|
||||||
|
assert(!thread->is_in_any_VTMS_transition(), "class load events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_CLASS_LOAD, ("[%s] Trg Class Load triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_CLASS_LOAD, ("[%s] Trg Class Load triggered",
|
||||||
JvmtiTrace::safe_get_thread_name(thread)));
|
JvmtiTrace::safe_get_thread_name(thread)));
|
||||||
@ -1375,7 +1381,10 @@ void JvmtiExport::post_class_prepare(JavaThread *thread, Klass* klass) {
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
assert(!thread->is_in_VTMS_transition(), "class prepare events are not allowed in VTMS transition");
|
if (thread->is_in_tmp_VTMS_transition()) {
|
||||||
|
return; // skip ClassPrepare events in tmp VTMS transition
|
||||||
|
}
|
||||||
|
assert(!thread->is_in_any_VTMS_transition(), "class prepare events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_CLASS_PREPARE, ("[%s] Trg Class Prepare triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_CLASS_PREPARE, ("[%s] Trg Class Prepare triggered",
|
||||||
JvmtiTrace::safe_get_thread_name(thread)));
|
JvmtiTrace::safe_get_thread_name(thread)));
|
||||||
@ -1694,8 +1703,8 @@ void JvmtiExport::post_object_free(JvmtiEnv* env, GrowableArray<jlong>* objects)
|
|||||||
assert(objects != NULL, "Nothing to post");
|
assert(objects != NULL, "Nothing to post");
|
||||||
|
|
||||||
JavaThread *javaThread = JavaThread::current();
|
JavaThread *javaThread = JavaThread::current();
|
||||||
if (javaThread->is_in_VTMS_transition()) {
|
if (javaThread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
if (!env->is_enabled(JVMTI_EVENT_OBJECT_FREE)) {
|
if (!env->is_enabled(JVMTI_EVENT_OBJECT_FREE)) {
|
||||||
return; // the event type has been already disabled
|
return; // the event type has been already disabled
|
||||||
@ -1718,8 +1727,8 @@ void JvmtiExport::post_resource_exhausted(jint resource_exhausted_flags, const c
|
|||||||
|
|
||||||
JavaThread *thread = JavaThread::current();
|
JavaThread *thread = JavaThread::current();
|
||||||
|
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
log_error(jvmti)("Posting Resource Exhausted event: %s",
|
log_error(jvmti)("Posting Resource Exhausted event: %s",
|
||||||
@ -1761,8 +1770,8 @@ void JvmtiExport::post_method_entry(JavaThread *thread, Method* method, frame cu
|
|||||||
// for any thread that actually wants method entry, interp_only_mode is set
|
// for any thread that actually wants method entry, interp_only_mode is set
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (mh->jvmti_mount_transition() || thread->is_in_VTMS_transition()) {
|
if (mh->jvmti_mount_transition() || thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_METHOD_ENTRY, ("[%s] Trg Method Entry triggered %s.%s",
|
EVT_TRIG_TRACE(JVMTI_EVENT_METHOD_ENTRY, ("[%s] Trg Method Entry triggered %s.%s",
|
||||||
JvmtiTrace::safe_get_thread_name(thread),
|
JvmtiTrace::safe_get_thread_name(thread),
|
||||||
@ -1845,8 +1854,8 @@ void JvmtiExport::post_method_exit_inner(JavaThread* thread,
|
|||||||
bool exception_exit,
|
bool exception_exit,
|
||||||
frame current_frame,
|
frame current_frame,
|
||||||
jvalue& value) {
|
jvalue& value) {
|
||||||
if (mh->jvmti_mount_transition() || thread->is_in_VTMS_transition()) {
|
if (mh->jvmti_mount_transition() || thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_METHOD_EXIT, ("[%s] Trg Method Exit triggered %s.%s",
|
EVT_TRIG_TRACE(JVMTI_EVENT_METHOD_EXIT, ("[%s] Trg Method Exit triggered %s.%s",
|
||||||
@ -1921,8 +1930,8 @@ void JvmtiExport::post_single_step(JavaThread *thread, Method* method, address l
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (mh->jvmti_mount_transition() || thread->is_in_VTMS_transition()) {
|
if (mh->jvmti_mount_transition() || thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
JvmtiEnvThreadStateIterator it(state);
|
JvmtiEnvThreadStateIterator it(state);
|
||||||
@ -1958,8 +1967,8 @@ void JvmtiExport::post_exception_throw(JavaThread *thread, Method* method, addre
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_EXCEPTION, ("[%s] Trg Exception thrown triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_EXCEPTION, ("[%s] Trg Exception thrown triggered",
|
||||||
@ -2080,8 +2089,8 @@ void JvmtiExport::notice_unwind_due_to_exception(JavaThread *thread, Method* met
|
|||||||
assert(!state->is_exception_caught(), "exception must not be caught yet.");
|
assert(!state->is_exception_caught(), "exception must not be caught yet.");
|
||||||
state->set_exception_caught();
|
state->set_exception_caught();
|
||||||
|
|
||||||
if (mh->jvmti_mount_transition() || thread->is_in_VTMS_transition()) {
|
if (mh->jvmti_mount_transition() || thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
JvmtiEnvThreadStateIterator it(state);
|
JvmtiEnvThreadStateIterator it(state);
|
||||||
for (JvmtiEnvThreadState* ets = it.first(); ets != NULL; ets = it.next(ets)) {
|
for (JvmtiEnvThreadState* ets = it.first(); ets != NULL; ets = it.next(ets)) {
|
||||||
@ -2126,8 +2135,8 @@ void JvmtiExport::post_field_access_by_jni(JavaThread *thread, oop obj,
|
|||||||
// function don't make the call unless there is a Java context.
|
// function don't make the call unless there is a Java context.
|
||||||
assert(thread->has_last_Java_frame(), "must be called with a Java context");
|
assert(thread->has_last_Java_frame(), "must be called with a Java context");
|
||||||
|
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
@ -2162,8 +2171,8 @@ void JvmtiExport::post_field_access(JavaThread *thread, Method* method,
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_FIELD_ACCESS, ("[%s] Trg Field Access event triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_FIELD_ACCESS, ("[%s] Trg Field Access event triggered",
|
||||||
@ -2212,8 +2221,8 @@ void JvmtiExport::post_field_modification_by_jni(JavaThread *thread, oop obj,
|
|||||||
// function don't make the call unless there is a Java context.
|
// function don't make the call unless there is a Java context.
|
||||||
assert(thread->has_last_Java_frame(), "must be called with Java context");
|
assert(thread->has_last_Java_frame(), "must be called with Java context");
|
||||||
|
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
@ -2243,8 +2252,8 @@ void JvmtiExport::post_raw_field_modification(JavaThread *thread, Method* method
|
|||||||
address location, Klass* field_klass, Handle object, jfieldID field,
|
address location, Klass* field_klass, Handle object, jfieldID field,
|
||||||
char sig_type, jvalue *value) {
|
char sig_type, jvalue *value) {
|
||||||
|
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sig_type == JVM_SIGNATURE_INT || sig_type == JVM_SIGNATURE_BOOLEAN ||
|
if (sig_type == JVM_SIGNATURE_INT || sig_type == JVM_SIGNATURE_BOOLEAN ||
|
||||||
@ -2318,8 +2327,8 @@ void JvmtiExport::post_field_modification(JavaThread *thread, Method* method,
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_FIELD_MODIFICATION,
|
EVT_TRIG_TRACE(JVMTI_EVENT_FIELD_MODIFICATION,
|
||||||
@ -2357,8 +2366,8 @@ void JvmtiExport::post_native_method_bind(Method* method, address* function_ptr)
|
|||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
methodHandle mh(thread, method);
|
methodHandle mh(thread, method);
|
||||||
|
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_NATIVE_METHOD_BIND, ("[%s] Trg Native Method Bind event triggered",
|
EVT_TRIG_TRACE(JVMTI_EVENT_NATIVE_METHOD_BIND, ("[%s] Trg Native Method Bind event triggered",
|
||||||
JvmtiTrace::safe_get_thread_name(thread)));
|
JvmtiTrace::safe_get_thread_name(thread)));
|
||||||
@ -2431,7 +2440,7 @@ void JvmtiExport::post_compiled_method_load(nmethod *nm) {
|
|||||||
}
|
}
|
||||||
JavaThread* thread = JavaThread::current();
|
JavaThread* thread = JavaThread::current();
|
||||||
|
|
||||||
assert(!thread->is_in_VTMS_transition(), "compiled method load events are not allowed in VTMS transition");
|
assert(!thread->is_in_any_VTMS_transition(), "compiled method load events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_COMPILED_METHOD_LOAD,
|
EVT_TRIG_TRACE(JVMTI_EVENT_COMPILED_METHOD_LOAD,
|
||||||
("[%s] method compile load event triggered",
|
("[%s] method compile load event triggered",
|
||||||
@ -2454,7 +2463,7 @@ void JvmtiExport::post_compiled_method_load(JvmtiEnv* env, nmethod *nm) {
|
|||||||
}
|
}
|
||||||
JavaThread* thread = JavaThread::current();
|
JavaThread* thread = JavaThread::current();
|
||||||
|
|
||||||
assert(!thread->is_in_VTMS_transition(), "compiled method load events are not allowed in VTMS transition");
|
assert(!thread->is_in_any_VTMS_transition(), "compiled method load events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
EVT_TRACE(JVMTI_EVENT_COMPILED_METHOD_LOAD,
|
EVT_TRACE(JVMTI_EVENT_COMPILED_METHOD_LOAD,
|
||||||
("[%s] method compile load event sent %s.%s ",
|
("[%s] method compile load event sent %s.%s ",
|
||||||
@ -2479,7 +2488,7 @@ void JvmtiExport::post_dynamic_code_generated_internal(const char *name, const v
|
|||||||
|
|
||||||
JavaThread* thread = JavaThread::current();
|
JavaThread* thread = JavaThread::current();
|
||||||
|
|
||||||
assert(!thread->is_in_VTMS_transition(), "dynamic code generated events are not allowed in VTMS transition");
|
assert(!thread->is_in_any_VTMS_transition(), "dynamic code generated events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
// In theory everyone coming thru here is in_vm but we need to be certain
|
// In theory everyone coming thru here is in_vm but we need to be certain
|
||||||
// because a callee will do a vm->native transition
|
// because a callee will do a vm->native transition
|
||||||
@ -2527,7 +2536,7 @@ void JvmtiExport::post_dynamic_code_generated(JvmtiEnv* env, const char *name,
|
|||||||
{
|
{
|
||||||
JavaThread* thread = JavaThread::current();
|
JavaThread* thread = JavaThread::current();
|
||||||
|
|
||||||
assert(!thread->is_in_VTMS_transition(), "dynamic code generated events are not allowed in VTMS transition");
|
assert(!thread->is_in_any_VTMS_transition(), "dynamic code generated events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
EVT_TRIG_TRACE(JVMTI_EVENT_DYNAMIC_CODE_GENERATED,
|
EVT_TRIG_TRACE(JVMTI_EVENT_DYNAMIC_CODE_GENERATED,
|
||||||
("[%s] dynamic code generated event triggered (by GenerateEvents)",
|
("[%s] dynamic code generated event triggered (by GenerateEvents)",
|
||||||
@ -2551,7 +2560,7 @@ void JvmtiExport::post_dynamic_code_generated_while_holding_locks(const char* na
|
|||||||
address code_begin, address code_end)
|
address code_begin, address code_end)
|
||||||
{
|
{
|
||||||
JavaThread* thread = JavaThread::current();
|
JavaThread* thread = JavaThread::current();
|
||||||
assert(!thread->is_in_VTMS_transition(), "dynamic code generated events are not allowed in VTMS transition");
|
assert(!thread->is_in_any_VTMS_transition(), "dynamic code generated events are not allowed in any VTMS transition");
|
||||||
|
|
||||||
// register the stub with the current dynamic code event collector
|
// register the stub with the current dynamic code event collector
|
||||||
// Cannot take safepoint here so do not use state_for to get
|
// Cannot take safepoint here so do not use state_for to get
|
||||||
@ -2681,8 +2690,8 @@ void JvmtiExport::post_monitor_contended_enter(JavaThread *thread, ObjectMonitor
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
@ -2714,8 +2723,8 @@ void JvmtiExport::post_monitor_contended_entered(JavaThread *thread, ObjectMonit
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
@ -2748,8 +2757,8 @@ void JvmtiExport::post_monitor_wait(JavaThread *thread, oop object,
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
@ -2782,8 +2791,8 @@ void JvmtiExport::post_monitor_waited(JavaThread *thread, ObjectMonitor *obj_mnt
|
|||||||
if (state == NULL) {
|
if (state == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
|
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
@ -2814,8 +2823,8 @@ void JvmtiExport::post_vm_object_alloc(JavaThread *thread, oop object) {
|
|||||||
if (object == NULL) {
|
if (object == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
Handle h(thread, object);
|
Handle h(thread, object);
|
||||||
@ -2848,8 +2857,8 @@ void JvmtiExport::post_sampled_object_alloc(JavaThread *thread, oop object) {
|
|||||||
if (object == NULL) {
|
if (object == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (thread->is_in_VTMS_transition()) {
|
if (thread->is_in_any_VTMS_transition()) {
|
||||||
return; // no events should be posted if thread is in a VTMS transition
|
return; // no events should be posted if thread is in any VTMS transition
|
||||||
}
|
}
|
||||||
HandleMark hm(thread);
|
HandleMark hm(thread);
|
||||||
Handle h(thread, object);
|
Handle h(thread, object);
|
||||||
|
@ -269,6 +269,7 @@ JvmtiVTMSTransitionDisabler::disable_VTMS_transitions() {
|
|||||||
ThreadBlockInVM tbivm(thread);
|
ThreadBlockInVM tbivm(thread);
|
||||||
MonitorLocker ml(JvmtiVTMSTransition_lock, Mutex::_no_safepoint_check_flag);
|
MonitorLocker ml(JvmtiVTMSTransition_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
|
||||||
|
assert(!thread->is_in_tmp_VTMS_transition(), "sanity check");
|
||||||
assert(!thread->is_in_VTMS_transition(), "VTMS_transition sanity check");
|
assert(!thread->is_in_VTMS_transition(), "VTMS_transition sanity check");
|
||||||
while (_SR_mode) { // Suspender or resumer is a JvmtiVTMSTransitionDisabler monopolist.
|
while (_SR_mode) { // Suspender or resumer is a JvmtiVTMSTransitionDisabler monopolist.
|
||||||
ml.wait(10); // Wait while there is an active suspender or resumer.
|
ml.wait(10); // Wait while there is an active suspender or resumer.
|
||||||
|
@ -421,6 +421,7 @@ JavaThread::JavaThread() :
|
|||||||
#if INCLUDE_JVMTI
|
#if INCLUDE_JVMTI
|
||||||
_carrier_thread_suspended(false),
|
_carrier_thread_suspended(false),
|
||||||
_is_in_VTMS_transition(false),
|
_is_in_VTMS_transition(false),
|
||||||
|
_is_in_tmp_VTMS_transition(false),
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
_is_VTMS_transition_disabler(false),
|
_is_VTMS_transition_disabler(false),
|
||||||
#endif
|
#endif
|
||||||
|
@ -310,6 +310,7 @@ class JavaThread: public Thread {
|
|||||||
#if INCLUDE_JVMTI
|
#if INCLUDE_JVMTI
|
||||||
volatile bool _carrier_thread_suspended; // Carrier thread is externally suspended
|
volatile bool _carrier_thread_suspended; // Carrier thread is externally suspended
|
||||||
bool _is_in_VTMS_transition; // thread is in virtual thread mount state transition
|
bool _is_in_VTMS_transition; // thread is in virtual thread mount state transition
|
||||||
|
bool _is_in_tmp_VTMS_transition; // thread is in temporary virtual thread mount state transition
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
bool _is_VTMS_transition_disabler; // thread currently disabled VTMS transitions
|
bool _is_VTMS_transition_disabler; // thread currently disabled VTMS transitions
|
||||||
#endif
|
#endif
|
||||||
@ -643,7 +644,12 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool is_in_VTMS_transition() const { return _is_in_VTMS_transition; }
|
bool is_in_VTMS_transition() const { return _is_in_VTMS_transition; }
|
||||||
|
bool is_in_tmp_VTMS_transition() const { return _is_in_tmp_VTMS_transition; }
|
||||||
|
bool is_in_any_VTMS_transition() const { return _is_in_VTMS_transition || _is_in_tmp_VTMS_transition; }
|
||||||
|
|
||||||
void set_is_in_VTMS_transition(bool val);
|
void set_is_in_VTMS_transition(bool val);
|
||||||
|
void toggle_is_in_tmp_VTMS_transition() { _is_in_tmp_VTMS_transition = !_is_in_tmp_VTMS_transition; };
|
||||||
|
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
bool is_VTMS_transition_disabler() const { return _is_VTMS_transition_disabler; }
|
bool is_VTMS_transition_disabler() const { return _is_VTMS_transition_disabler; }
|
||||||
void set_is_VTMS_transition_disabler(bool val);
|
void set_is_VTMS_transition_disabler(bool val);
|
||||||
|
@ -354,6 +354,39 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
carrier.clearInterrupt();
|
carrier.clearInterrupt();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the current thread to the current carrier thread.
|
||||||
|
* @return true if JVMTI was notified
|
||||||
|
*/
|
||||||
|
@ChangesCurrentThread
|
||||||
|
@JvmtiMountTransition
|
||||||
|
private boolean switchToCarrierThread() {
|
||||||
|
boolean notifyJvmti = notifyJvmtiEvents;
|
||||||
|
if (notifyJvmti) {
|
||||||
|
notifyJvmtiHideFrames(true);
|
||||||
|
}
|
||||||
|
Thread carrier = this.carrierThread;
|
||||||
|
assert Thread.currentThread() == this
|
||||||
|
&& carrier == Thread.currentCarrierThread();
|
||||||
|
carrier.setCurrentThread(carrier);
|
||||||
|
return notifyJvmti;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the current thread to the given virtual thread.
|
||||||
|
* If {@code notifyJvmti} is true then JVMTI is notified.
|
||||||
|
*/
|
||||||
|
@ChangesCurrentThread
|
||||||
|
@JvmtiMountTransition
|
||||||
|
private void switchToVirtualThread(VirtualThread vthread, boolean notifyJvmti) {
|
||||||
|
Thread carrier = vthread.carrierThread;
|
||||||
|
assert carrier == Thread.currentCarrierThread();
|
||||||
|
carrier.setCurrentThread(vthread);
|
||||||
|
if (notifyJvmti) {
|
||||||
|
notifyJvmtiHideFrames(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Unmounts this virtual thread, invokes Continuation.yield, and re-mounts the
|
* Unmounts this virtual thread, invokes Continuation.yield, and re-mounts the
|
||||||
* thread when continued. When enabled, JVMTI must be notified from this method.
|
* thread when continued. When enabled, JVMTI must be notified from this method.
|
||||||
@ -526,7 +559,7 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
long startTime = System.nanoTime();
|
long startTime = System.nanoTime();
|
||||||
|
|
||||||
boolean yielded;
|
boolean yielded;
|
||||||
Future<?> unparker = scheduleUnpark(nanos);
|
Future<?> unparker = scheduleUnpark(this::unpark, nanos);
|
||||||
setState(PARKING);
|
setState(PARKING);
|
||||||
try {
|
try {
|
||||||
yielded = yieldContinuation();
|
yielded = yieldContinuation();
|
||||||
@ -579,17 +612,16 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Schedules this thread to be unparked after the given delay.
|
* Schedule an unpark task to run after a given delay.
|
||||||
*/
|
*/
|
||||||
@ChangesCurrentThread
|
@ChangesCurrentThread
|
||||||
private Future<?> scheduleUnpark(long nanos) {
|
private Future<?> scheduleUnpark(Runnable unparker, long nanos) {
|
||||||
Thread carrier = this.carrierThread;
|
// need to switch to current carrier thread to avoid nested parking
|
||||||
// need to switch to current platform thread to avoid nested parking
|
boolean notifyJvmti = switchToCarrierThread();
|
||||||
carrier.setCurrentThread(carrier);
|
|
||||||
try {
|
try {
|
||||||
return UNPARKER.schedule(() -> unpark(), nanos, NANOSECONDS);
|
return UNPARKER.schedule(unparker, nanos, NANOSECONDS);
|
||||||
} finally {
|
} finally {
|
||||||
carrier.setCurrentThread(this);
|
switchToVirtualThread(this, notifyJvmti);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -599,13 +631,12 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
@ChangesCurrentThread
|
@ChangesCurrentThread
|
||||||
private void cancel(Future<?> future) {
|
private void cancel(Future<?> future) {
|
||||||
if (!future.isDone()) {
|
if (!future.isDone()) {
|
||||||
Thread carrier = this.carrierThread;
|
// need to switch to current carrier thread to avoid nested parking
|
||||||
// need to switch to current platform thread to avoid nested parking
|
boolean notifyJvmti = switchToCarrierThread();
|
||||||
carrier.setCurrentThread(carrier);
|
|
||||||
try {
|
try {
|
||||||
future.cancel(false);
|
future.cancel(false);
|
||||||
} finally {
|
} finally {
|
||||||
carrier.setCurrentThread(this);
|
switchToVirtualThread(this, notifyJvmti);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -625,12 +656,11 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
int s = state();
|
int s = state();
|
||||||
if (s == PARKED && compareAndSetState(PARKED, RUNNABLE)) {
|
if (s == PARKED && compareAndSetState(PARKED, RUNNABLE)) {
|
||||||
if (currentThread instanceof VirtualThread vthread) {
|
if (currentThread instanceof VirtualThread vthread) {
|
||||||
Thread carrier = vthread.carrierThread;
|
boolean notifyJvmti = vthread.switchToCarrierThread();
|
||||||
carrier.setCurrentThread(carrier);
|
|
||||||
try {
|
try {
|
||||||
submitRunContinuation();
|
submitRunContinuation();
|
||||||
} finally {
|
} finally {
|
||||||
carrier.setCurrentThread(vthread);
|
switchToVirtualThread(vthread, notifyJvmti);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
submitRunContinuation();
|
submitRunContinuation();
|
||||||
@ -1005,6 +1035,9 @@ final class VirtualThread extends BaseVirtualThread {
|
|||||||
@JvmtiMountTransition
|
@JvmtiMountTransition
|
||||||
private native void notifyJvmtiUnmountEnd(boolean lastUnmount);
|
private native void notifyJvmtiUnmountEnd(boolean lastUnmount);
|
||||||
|
|
||||||
|
@JvmtiMountTransition
|
||||||
|
private native void notifyJvmtiHideFrames(boolean hide);
|
||||||
|
|
||||||
private static native void registerNatives();
|
private static native void registerNatives();
|
||||||
static {
|
static {
|
||||||
registerNatives();
|
registerNatives();
|
||||||
|
@ -36,6 +36,7 @@ static JNINativeMethod methods[] = {
|
|||||||
{ "notifyJvmtiMountEnd", "(Z)V", (void *)&JVM_VirtualThreadMountEnd },
|
{ "notifyJvmtiMountEnd", "(Z)V", (void *)&JVM_VirtualThreadMountEnd },
|
||||||
{ "notifyJvmtiUnmountBegin", "(Z)V", (void *)&JVM_VirtualThreadUnmountBegin },
|
{ "notifyJvmtiUnmountBegin", "(Z)V", (void *)&JVM_VirtualThreadUnmountBegin },
|
||||||
{ "notifyJvmtiUnmountEnd", "(Z)V", (void *)&JVM_VirtualThreadUnmountEnd },
|
{ "notifyJvmtiUnmountEnd", "(Z)V", (void *)&JVM_VirtualThreadUnmountEnd },
|
||||||
|
{ "notifyJvmtiHideFrames", "(Z)V", (void *)&JVM_VirtualThreadHideFrames },
|
||||||
};
|
};
|
||||||
|
|
||||||
JNIEXPORT void JNICALL
|
JNIEXPORT void JNICALL
|
||||||
|
Loading…
Reference in New Issue
Block a user