Skip to content

Commit 091f65f

Browse files
committedOct 31, 2024
Improve JFR event for pinning
1 parent a97c765 commit 091f65f

File tree

12 files changed

+157
-83
lines changed

12 files changed

+157
-83
lines changed
 

‎src/hotspot/share/include/jvm.h

+4-1
Original file line numberDiff line numberDiff line change
@@ -1149,7 +1149,10 @@ JNIEXPORT void JNICALL
11491149
JVM_VirtualThreadDisableSuspend(JNIEnv* env, jclass clazz, jboolean enter);
11501150

11511151
JNIEXPORT void JNICALL
1152-
JVM_VirtualThreadPinnedEvent(jint reasonCode, jstring reasonString);
1152+
JVM_VirtualThreadPinnedStart(JNIEnv* env, jclass clazz);
1153+
1154+
JNIEXPORT void JNICALL
1155+
JVM_VirtualThreadPinnedEnd(JNIEnv* env, jclass clazz, jstring reason);
11531156

11541157
JNIEXPORT jobject JNICALL
11551158
JVM_TakeVirtualThreadListToUnblock(JNIEnv* env, jclass ignored);

‎src/hotspot/share/jfr/metadata/metadata.xml

+1-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,7 @@
156156
<Field type="ulong" name="id" label="Continuation ID" />
157157
</Event>
158158

159-
<Event name="VirtualThreadPinned" category="Java Virtual Machine, Runtime" label="Virtual Thread Pinned" thread="true" stackTrace="true" startTime="false">
159+
<Event name="VirtualThreadPinned" category="Java Virtual Machine, Runtime" label="Virtual Thread Pinned" thread="true" stackTrace="true">
160160
<Field type="string" name="pinnedReason" label="Pinned Reason" />
161161
<Field type="Thread" name="carrierThread" label="Carrier Thread" />
162162
</Event>

‎src/hotspot/share/oops/instanceKlass.cpp

+31-16
Original file line numberDiff line numberDiff line change
@@ -175,6 +175,35 @@ static inline bool is_stack_chunk_class(const Symbol* class_name,
175175
loader_data->is_the_null_class_loader_data());
176176
}
177177

178+
class WaitForClassInitializer : public StackObj {
179+
private:
180+
InstanceKlass* _ik;
181+
Ticks _start_time;
182+
public:
183+
WaitForClassInitializer(InstanceKlass* ik) : _ik(ik) {
184+
_start_time = Ticks::now();
185+
JavaThread* current = JavaThread::current();
186+
current->set_class_to_be_initialized(ik);
187+
}
188+
~WaitForClassInitializer() {
189+
JavaThread* current = JavaThread::current();
190+
current->set_class_to_be_initialized(nullptr);
191+
#if INCLUDE_JFR
192+
ContinuationEntry* ce = current->last_continuation();
193+
if (ce != nullptr && ce->is_virtual_thread()) {
194+
EventVirtualThreadPinned event;
195+
event.set_starttime(_start_time);
196+
if (event.should_commit()) {
197+
ResourceMark rm(current);
198+
char reason[256];
199+
jio_snprintf(reason, sizeof reason, "Waited for initialization of klass %s", _ik->external_name());
200+
current->post_vthread_pinned_event(&event, reason);
201+
}
202+
}
203+
#endif
204+
}
205+
};
206+
178207
// private: called to verify that k is a static member of this nest.
179208
// We know that k is an instance class in the same package and hence the
180209
// same classloader.
@@ -1104,25 +1133,10 @@ void InstanceKlass::initialize_impl(TRAPS) {
11041133
jt->name(), external_name(), init_thread_name());
11051134
}
11061135
wait = true;
1107-
jt->set_class_to_be_initialized(this);
11081136

1109-
#if INCLUDE_JFR
1110-
ContinuationEntry* ce = jt->last_continuation();
1111-
if (ce != nullptr && ce->is_virtual_thread()) {
1112-
EventVirtualThreadPinned e;
1113-
if (e.should_commit()) {
1114-
ResourceMark rm(jt);
1115-
char reason[256];
1116-
jio_snprintf(reason, sizeof reason, "Waiting for initialization of klass %s", external_name());
1117-
e.set_pinnedReason(reason);
1118-
e.set_carrierThread(JFR_JVM_THREAD_ID(THREAD));
1119-
e.commit();
1120-
}
1121-
}
1122-
#endif
1137+
WaitForClassInitializer wfcl(this);
11231138

11241139
ol.wait_uninterruptibly(jt);
1125-
jt->set_class_to_be_initialized(nullptr);
11261140
}
11271141

11281142
// Step 3
@@ -1619,6 +1633,7 @@ void InstanceKlass::call_class_initializer(TRAPS) {
16191633
THREAD->name());
16201634
}
16211635
if (h_method() != nullptr) {
1636+
ThreadInClassInitializer ticl(this);
16221637
JavaCallArguments args; // No arguments
16231638
JavaValue result(T_VOID);
16241639
JavaCalls::call(&result, h_method, &args, CHECK); // Static call (no args)

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

+12-8
Original file line numberDiff line numberDiff line change
@@ -3975,16 +3975,20 @@ JVM_ENTRY(void, JVM_VirtualThreadDisableSuspend(JNIEnv* env, jclass clazz, jbool
39753975
#endif
39763976
JVM_END
39773977

3978-
JVM_ENTRY_NO_ENV(void, JVM_VirtualThreadPinnedEvent(jint reasonCode, jstring reasonString))
3978+
JVM_ENTRY_NO_ENV(void, JVM_VirtualThreadPinnedStart(JNIEnv* env, jclass ignored))
39793979
#if INCLUDE_JFR
3980-
EventVirtualThreadPinned e;
3981-
if (e.should_commit()) {
3980+
THREAD->start_vthread_pinned();
3981+
#endif
3982+
JVM_END
3983+
3984+
JVM_ENTRY_NO_ENV(void, JVM_VirtualThreadPinnedEnd(JNIEnv* env, jclass ignored, jstring reason))
3985+
#if INCLUDE_JFR
3986+
EventVirtualThreadPinned event;
3987+
event.set_starttime(THREAD->vthread_pinned_start_time());
3988+
if (event.should_commit()) {
39823989
ResourceMark rm(THREAD);
3983-
// ignore reason code for now
3984-
const char *reason = java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(reasonString));
3985-
e.set_pinnedReason(reason);
3986-
e.set_carrierThread(JFR_JVM_THREAD_ID(THREAD));
3987-
e.commit();
3990+
const char *str = java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(reason));
3991+
THREAD->post_vthread_pinned_event(&event, str);
39883992
}
39893993
#endif
39903994
JVM_END

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

+19
Original file line numberDiff line numberDiff line change
@@ -516,6 +516,7 @@ JavaThread::JavaThread(MemTag mem_tag) :
516516
_parker(),
517517

518518
_class_to_be_initialized(nullptr),
519+
_class_being_initialized(nullptr),
519520

520521
_SleepEvent(ParkEvent::Allocate(this)),
521522

@@ -2335,3 +2336,21 @@ void JavaThread::add_oop_handles_for_release() {
23352336
_oop_handle_list = new_head;
23362337
Service_lock->notify_all();
23372338
}
2339+
2340+
#if INCLUDE_JFR
2341+
void JavaThread::post_vthread_pinned_event(EventVirtualThreadPinned* event, const char* reason) {
2342+
if (event->should_commit()) {
2343+
char long_reason[256];
2344+
InstanceKlass* ik = class_being_initialized();
2345+
if (ik != nullptr) {
2346+
ResourceMark rm(this);
2347+
jio_snprintf(long_reason, sizeof(long_reason), "%s when initializing %s", reason, ik->external_name());
2348+
} else {
2349+
jio_snprintf(long_reason, sizeof(long_reason), "%s when pinned", reason);
2350+
}
2351+
event->set_pinnedReason(long_reason);
2352+
event->set_carrierThread(JFR_JVM_THREAD_ID(this));
2353+
event->commit();
2354+
}
2355+
}
2356+
#endif

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

+34
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@
3737
#include "runtime/javaFrameAnchor.hpp"
3838
#include "runtime/lockStack.hpp"
3939
#include "runtime/park.hpp"
40+
#include "utilities/ticks.hpp"
4041
#include "runtime/safepointMechanism.hpp"
4142
#include "runtime/stackWatermarkSet.hpp"
4243
#include "runtime/stackOverflow.hpp"
@@ -78,6 +79,8 @@ class javaVFrame;
7879
class JavaThread;
7980
typedef void (*ThreadFunction)(JavaThread*, TRAPS);
8081

82+
class EventVirtualThreadPinned;
83+
8184
class JavaThread: public Thread {
8285
friend class VMStructs;
8386
friend class JVMCIVMStructs;
@@ -1221,11 +1224,22 @@ class JavaThread: public Thread {
12211224
void set_class_to_be_initialized(InstanceKlass* k);
12221225
InstanceKlass* class_to_be_initialized() const;
12231226

1227+
// Track executing class initializer, see ThreadInClassInitializer
1228+
void set_class_being_initialized(InstanceKlass* k);
1229+
InstanceKlass* class_being_initialized() const;
1230+
12241231
private:
12251232
InstanceKlass* _class_to_be_initialized;
1233+
InstanceKlass* _class_being_initialized;
12261234

12271235
// java.lang.Thread.sleep support
12281236
ParkEvent * _SleepEvent;
1237+
1238+
#if INCLUDE_JFR
1239+
// Support for jdk.VirtualThreadPinned event
1240+
Ticks _pinned_start_time;
1241+
#endif
1242+
12291243
public:
12301244
bool sleep(jlong millis);
12311245
bool sleep_nanos(jlong nanos);
@@ -1234,6 +1248,12 @@ class JavaThread: public Thread {
12341248
void interrupt();
12351249
bool is_interrupted(bool clear_interrupted);
12361250

1251+
#if INCLUDE_JFR
1252+
Ticks& vthread_pinned_start_time() { return _pinned_start_time; }
1253+
void start_vthread_pinned() { _pinned_start_time = Ticks::now(); }
1254+
void post_vthread_pinned_event(EventVirtualThreadPinned* event, const char* reason);
1255+
#endif
1256+
12371257
// This is only for use by JVMTI RawMonitorWait. It emulates the actions of
12381258
// the Java code in Object::wait which are not present in RawMonitorWait.
12391259
bool get_and_clear_interrupted();
@@ -1340,4 +1360,18 @@ class ThreadOnMonitorWaitedEvent {
13401360
~ThreadOnMonitorWaitedEvent() { JVMTI_ONLY(_thread->set_on_monitor_waited_event(false);) }
13411361
};
13421362

1363+
class ThreadInClassInitializer : public StackObj {
1364+
InstanceKlass* _previous;
1365+
public:
1366+
ThreadInClassInitializer(InstanceKlass* ik) {
1367+
JavaThread* current = JavaThread::current();
1368+
_previous = current->class_being_initialized();
1369+
current->set_class_being_initialized(ik);
1370+
}
1371+
~ThreadInClassInitializer() {
1372+
JavaThread* current = JavaThread::current();
1373+
current->set_class_being_initialized(_previous);
1374+
}
1375+
};
1376+
13431377
#endif // SHARE_RUNTIME_JAVATHREAD_HPP

‎src/hotspot/share/runtime/javaThread.inline.hpp

+9
Original file line numberDiff line numberDiff line change
@@ -241,6 +241,15 @@ inline InstanceKlass* JavaThread::class_to_be_initialized() const {
241241
return _class_to_be_initialized;
242242
}
243243

244+
inline void JavaThread::set_class_being_initialized(InstanceKlass* k) {
245+
assert(this == Thread::current(), "Only the current thread can set this field");
246+
_class_being_initialized = k;
247+
}
248+
249+
inline InstanceKlass* JavaThread::class_being_initialized() const {
250+
return _class_being_initialized;
251+
}
252+
244253
inline void JavaThread::om_set_monitor_cache(ObjectMonitor* monitor) {
245254
assert(UseObjectMonitorTable, "must be");
246255
assert(monitor != nullptr, "use om_clear_monitor_cache to clear");

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

+27-39
Original file line numberDiff line numberDiff line change
@@ -119,15 +119,6 @@ OopStorage* ObjectMonitor::_oop_storage = nullptr;
119119
OopHandle ObjectMonitor::_vthread_cxq_head;
120120
ParkEvent* ObjectMonitor::_vthread_unparker_ParkEvent = nullptr;
121121

122-
static void post_virtual_thread_pinned_event(JavaThread* current, const char* reason) {
123-
EventVirtualThreadPinned e;
124-
if (e.should_commit()) {
125-
e.set_pinnedReason(reason);
126-
e.set_carrierThread(JFR_JVM_THREAD_ID(current));
127-
e.commit();
128-
}
129-
}
130-
131122
// -----------------------------------------------------------------------------
132123
// Theory of operations -- Monitors lists, thread residency, etc:
133124
//
@@ -480,14 +471,15 @@ void ObjectMonitor::enter_with_contention_mark(JavaThread *current, ObjectMonito
480471
assert(!is_being_async_deflated(), "must be");
481472

482473
JFR_ONLY(JfrConditionalFlush<EventJavaMonitorEnter> flush(current);)
483-
EventJavaMonitorEnter event;
484-
if (event.is_started()) {
485-
event.set_monitorClass(object()->klass());
474+
EventJavaMonitorEnter enter_event;
475+
if (enter_event.is_started()) {
476+
enter_event.set_monitorClass(object()->klass());
486477
// Set an address that is 'unique enough', such that events close in
487478
// time and with the same address are likely (but not guaranteed) to
488479
// belong to the same object.
489-
event.set_address((uintptr_t)this);
480+
enter_event.set_address((uintptr_t)this);
490481
}
482+
EventVirtualThreadPinned vthread_pinned_event;
491483

492484
{ // Change java thread status to indicate blocked on monitor enter.
493485
JavaThreadBlockedOnMonitorEnterState jtbmes(current, this);
@@ -528,11 +520,6 @@ void ObjectMonitor::enter_with_contention_mark(JavaThread *current, ObjectMonito
528520
(!acquired && !current->preemption_cancelled() && state == java_lang_VirtualThread::BLOCKING), "invariant");
529521
return;
530522
}
531-
if (result == freeze_pinned_native) {
532-
post_virtual_thread_pinned_event(current, "Native frame or <clinit> on stack");
533-
} else if (result == freeze_unsupported) {
534-
post_virtual_thread_pinned_event(current, "Native frame or <clinit> or monitors on stack");
535-
}
536523
}
537524

538525
OSThreadContendState osts(current->osthread());
@@ -594,10 +581,16 @@ void ObjectMonitor::enter_with_contention_mark(JavaThread *current, ObjectMonito
594581
// event handler consumed an unpark() issued by the thread that
595582
// just exited the monitor.
596583
}
597-
if (event.should_commit()) {
598-
event.set_previousOwner(_previous_owner_tid);
599-
event.commit();
584+
if (enter_event.should_commit()) {
585+
enter_event.set_previousOwner(_previous_owner_tid);
586+
enter_event.commit();
587+
}
588+
589+
ContinuationEntry* ce = current->last_continuation();
590+
if (ce != nullptr && ce->is_virtual_thread()) {
591+
current->post_vthread_pinned_event(&vthread_pinned_event, "Blocked on monitor enter");
600592
}
593+
601594
OM_PERFDATA_OP(ContendedLockAttempts, inc());
602595
}
603596

@@ -1646,7 +1639,8 @@ void ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS) {
16461639

16471640
CHECK_OWNER(); // Throws IMSE if not owner.
16481641

1649-
EventJavaMonitorWait event;
1642+
EventJavaMonitorWait wait_event;
1643+
EventVirtualThreadPinned vthread_pinned_event;
16501644

16511645
// check for a pending interrupt
16521646
if (interruptible && current->is_interrupted(true) && !HAS_PENDING_EXCEPTION) {
@@ -1664,8 +1658,8 @@ void ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS) {
16641658
// consume an unpark() meant for the ParkEvent associated with
16651659
// this ObjectMonitor.
16661660
}
1667-
if (event.should_commit()) {
1668-
post_monitor_wait_event(&event, this, 0, millis, false);
1661+
if (wait_event.should_commit()) {
1662+
post_monitor_wait_event(&wait_event, this, 0, millis, false);
16691663
}
16701664
THROW(vmSymbols::java_lang_InterruptedException());
16711665
return;
@@ -1681,16 +1675,6 @@ void ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS) {
16811675
current->set_current_waiting_monitor(nullptr);
16821676
return;
16831677
}
1684-
if (result == freeze_pinned_native || result == freeze_unsupported) {
1685-
const Klass* monitor_klass = object()->klass();
1686-
if (!is_excluded(monitor_klass)) {
1687-
if (result == freeze_pinned_native) {
1688-
post_virtual_thread_pinned_event(current,"Native frame or <clinit> on stack");
1689-
} else if (result == freeze_unsupported) {
1690-
post_virtual_thread_pinned_event(current, "Native frame or <clinit> or monitors on stack");
1691-
}
1692-
}
1693-
}
16941678
}
16951679

16961680
// create a node to be put into the queue
@@ -1816,8 +1800,12 @@ void ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS) {
18161800
}
18171801
}
18181802

1819-
if (event.should_commit()) {
1820-
post_monitor_wait_event(&event, this, node._notifier_tid, millis, ret == OS_TIMEOUT);
1803+
if (wait_event.should_commit()) {
1804+
post_monitor_wait_event(&wait_event, this, node._notifier_tid, millis, ret == OS_TIMEOUT);
1805+
}
1806+
1807+
if (ce != nullptr && ce->is_virtual_thread()) {
1808+
current->post_vthread_pinned_event(&vthread_pinned_event, "Waited in Object.wait");
18211809
}
18221810

18231811
OrderAccess::fence();
@@ -2049,9 +2037,9 @@ bool ObjectMonitor::VThreadWaitReenter(JavaThread* current, ObjectWaiter* node,
20492037
node->_interrupted = !was_notified && current->is_interrupted(false);
20502038

20512039
// Post JFR and JVMTI events.
2052-
EventJavaMonitorWait event;
2053-
if (event.should_commit() || JvmtiExport::should_post_monitor_waited()) {
2054-
vthread_monitor_waited_event(current, node, cont, &event, !was_notified && !node->_interrupted);
2040+
EventJavaMonitorWait wait_event;
2041+
if (wait_event.should_commit() || JvmtiExport::should_post_monitor_waited()) {
2042+
vthread_monitor_waited_event(current, node, cont, &wait_event, !was_notified && !node->_interrupted);
20552043
}
20562044

20572045
// Mark that we are at reenter so that we don't call this method again.

0 commit comments

Comments
 (0)
Please sign in to comment.