Skip to content

Commit b890336

Browse files
author
Serguei Spitsyn
committedMay 23, 2024
8328083: degrade virtual thread support for GetObjectMonitorUsage
Reviewed-by: cjplummer, alanb
1 parent 4e6d851 commit b890336

File tree

12 files changed

+186
-57
lines changed

12 files changed

+186
-57
lines changed
 

‎src/hotspot/share/prims/jvmti.xml

+17-6
Original file line numberDiff line numberDiff line change
@@ -8249,44 +8249,55 @@ class C2 extends C1 implements I2 {
82498249
<field id="owner">
82508250
<jthread/>
82518251
<description>
8252-
The thread owning this monitor, or <code>nullptr</code> if unused
8252+
The platform thread owning this monitor, or <code>nullptr</code> if owned
8253+
by a virtual thread or not owned
82538254
</description>
82548255
</field>
82558256
<field id="entry_count">
82568257
<jint/>
82578258
<description>
8258-
The number of times the owning thread has entered the monitor
8259+
The number of times the platform thread owning this monitor has entered it,
8260+
or <code>0</code> if owned by a virtual thread or not owned
82598261
</description>
82608262
</field>
82618263
<field id="waiter_count">
82628264
<jint/>
82638265
<description>
8264-
The number of threads waiting to own this monitor
8266+
The number of platform threads waiting to own this monitor, or <code>0</code>
8267+
if only virtual threads are waiting or no threads are waiting
82658268
</description>
82668269
</field>
82678270
<field id="waiters">
82688271
<allocfieldbuf><jthread/></allocfieldbuf>
82698272
<description>
8270-
The <code>waiter_count</code> waiting threads
8273+
The <code>waiter_count</code> waiting platform threads
82718274
</description>
82728275
</field>
82738276
<field id="notify_waiter_count">
82748277
<jint/>
82758278
<description>
8276-
The number of threads waiting to be notified by this monitor
8279+
The number of platform threads waiting to own this monitor, or <code>0</code>
8280+
if only virtual threads are waiting to be notified or no threads are waiting
8281+
to be notified
82778282
</description>
82788283
</field>
82798284
<field id="notify_waiters">
82808285
<allocfieldbuf><jthread/></allocfieldbuf>
82818286
<description>
8282-
The <code>notify_waiter_count</code> threads waiting to be notified
8287+
The <code>notify_waiter_count</code> platform threads waiting to be notified
82838288
</description>
82848289
</field>
82858290
</typedef>
82868291
<description>
82878292
Get information about the object's monitor.
82888293
The fields of the <functionlink id="jvmtiMonitorUsage"></functionlink> structure
82898294
are filled in with information about usage of the monitor.
8295+
<p/>
8296+
<b> This function does not support getting information about an object's monitor
8297+
when it is owned by a virtual thread. It also does not support returning a
8298+
reference to virtual threads that are waiting to own a monitor or waiting to
8299+
be notified.
8300+
</b>
82908301
<todo>
82918302
Decide and then clarify suspend requirements.
82928303
</todo>

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

+25-7
Original file line numberDiff line numberDiff line change
@@ -1482,14 +1482,19 @@ JvmtiEnvBase::get_object_monitor_usage(JavaThread* calling_thread, jobject objec
14821482
// first derive the object's owner and entry_count (if any)
14831483
owning_thread = ObjectSynchronizer::get_lock_owner(tlh.list(), hobj);
14841484
if (owning_thread != nullptr) {
1485-
Handle th(current_thread, get_vthread_or_thread_oop(owning_thread));
1485+
oop thread_oop = get_vthread_or_thread_oop(owning_thread);
1486+
bool is_virtual = java_lang_VirtualThread::is_instance(thread_oop);
1487+
if (is_virtual) {
1488+
thread_oop = nullptr;
1489+
}
1490+
Handle th(current_thread, thread_oop);
14861491
ret.owner = (jthread)jni_reference(calling_thread, th);
14871492

14881493
// The recursions field of a monitor does not reflect recursions
14891494
// as lightweight locks before inflating the monitor are not included.
14901495
// We have to count the number of recursive monitor entries the hard way.
14911496
// We pass a handle to survive any GCs along the way.
1492-
ret.entry_count = count_locked_objects(owning_thread, hobj);
1497+
ret.entry_count = is_virtual ? 0 : count_locked_objects(owning_thread, hobj);
14931498
}
14941499
// implied else: entry_count == 0
14951500

@@ -1513,6 +1518,7 @@ JvmtiEnvBase::get_object_monitor_usage(JavaThread* calling_thread, jobject objec
15131518
// this object has a lightweight monitor
15141519
}
15151520

1521+
jint skipped = 0;
15161522
if (mon != nullptr) {
15171523
// Robustness: the actual waiting list can be smaller.
15181524
// The nWait count we got from the mon->waiters() may include the re-entering
@@ -1522,11 +1528,16 @@ JvmtiEnvBase::get_object_monitor_usage(JavaThread* calling_thread, jobject objec
15221528
for (ObjectWaiter* waiter = mon->first_waiter();
15231529
waiter != nullptr && (nWait == 0 || waiter != mon->first_waiter());
15241530
waiter = mon->next_waiter(waiter)) {
1531+
JavaThread *w = mon->thread_of_waiter(waiter);
1532+
oop thread_oop = get_vthread_or_thread_oop(w);
1533+
if (java_lang_VirtualThread::is_instance(thread_oop)) {
1534+
skipped++;
1535+
}
15251536
nWait++;
15261537
}
15271538
}
15281539
ret.waiter_count = nWant;
1529-
ret.notify_waiter_count = nWait;
1540+
ret.notify_waiter_count = nWait - skipped;
15301541

15311542
// Allocate memory for heavyweight and lightweight monitor.
15321543
jvmtiError err;
@@ -1561,13 +1572,20 @@ JvmtiEnvBase::get_object_monitor_usage(JavaThread* calling_thread, jobject objec
15611572
}
15621573
if (ret.notify_waiter_count > 0) { // we have threads waiting to be notified in Object.wait()
15631574
ObjectWaiter *waiter = mon->first_waiter();
1575+
jint skipped = 0;
15641576
for (int i = 0; i < nWait; i++) {
15651577
JavaThread *w = mon->thread_of_waiter(waiter);
1578+
oop thread_oop = get_vthread_or_thread_oop(w);
1579+
bool is_virtual = java_lang_VirtualThread::is_instance(thread_oop);
15661580
assert(w != nullptr, "sanity check");
1567-
// If the thread was found on the ObjectWaiter list, then
1568-
// it has not been notified.
1569-
Handle th(current_thread, get_vthread_or_thread_oop(w));
1570-
ret.notify_waiters[i] = (jthread)jni_reference(calling_thread, th);
1581+
if (java_lang_VirtualThread::is_instance(thread_oop)) {
1582+
skipped++;
1583+
} else {
1584+
// If the thread was found on the ObjectWaiter list, then
1585+
// it has not been notified.
1586+
Handle th(current_thread, get_vthread_or_thread_oop(w));
1587+
ret.notify_waiters[i - skipped] = (jthread)jni_reference(calling_thread, th);
1588+
}
15711589
waiter = mon->next_waiter(waiter);
15721590
}
15731591
}

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

+8-5
Original file line numberDiff line numberDiff line change
@@ -1183,7 +1183,8 @@ void Threads::metadata_handles_do(void f(Metadata*)) {
11831183
}
11841184

11851185
#if INCLUDE_JVMTI
1186-
// Get count of Java threads that are waiting to enter or re-enter the specified monitor.
1186+
// Get Java threads that are waiting to enter or re-enter the specified monitor.
1187+
// Java threads that are executing mounted virtual threads are not included.
11871188
GrowableArray<JavaThread*>* Threads::get_pending_threads(ThreadsList * t_list,
11881189
int count,
11891190
address monitor) {
@@ -1194,14 +1195,16 @@ GrowableArray<JavaThread*>* Threads::get_pending_threads(ThreadsList * t_list,
11941195
for (JavaThread* p : *t_list) {
11951196
if (!p->can_call_java()) continue;
11961197

1198+
oop thread_oop = JvmtiEnvBase::get_vthread_or_thread_oop(p);
1199+
if (java_lang_VirtualThread::is_instance(thread_oop)) {
1200+
continue;
1201+
}
11971202
// The first stage of async deflation does not affect any field
11981203
// used by this comparison so the ObjectMonitor* is usable here.
11991204
address pending = (address)p->current_pending_monitor();
12001205
address waiting = (address)p->current_waiting_monitor();
1201-
oop thread_oop = JvmtiEnvBase::get_vthread_or_thread_oop(p);
1202-
bool is_virtual = java_lang_VirtualThread::is_instance(thread_oop);
1203-
jint state = is_virtual ? JvmtiEnvBase::get_vthread_state(thread_oop, p)
1204-
: JvmtiEnvBase::get_thread_state(thread_oop, p);
1206+
// do not include virtual threads to the list
1207+
jint state = JvmtiEnvBase::get_thread_state(thread_oop, p);
12051208
if (pending == monitor || (waiting == monitor &&
12061209
(state & JVMTI_THREAD_STATE_BLOCKED_ON_MONITOR_ENTER))
12071210
) { // found a match

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -129,7 +129,8 @@ class Threads: AllStatic {
129129
// Print threads busy compiling, and returns the number of printed threads.
130130
static unsigned print_threads_compiling(outputStream* st, char* buf, int buflen, bool short_form = false);
131131

132-
// Get count of Java threads that are waiting to enter or re-enter the specified monitor.
132+
// Get Java threads that are waiting to enter or re-enter the specified monitor.
133+
// Java threads that are executing mounted virtual threads are not included.
133134
static GrowableArray<JavaThread*>* get_pending_threads(ThreadsList * t_list,
134135
int count, address monitor);
135136

‎src/java.se/share/data/jdwp/jdwp.spec

+9-6
Original file line numberDiff line numberDiff line change
@@ -1617,11 +1617,14 @@ JDWP "Java(tm) Debug Wire Protocol"
16171617
(object object "The object ID")
16181618
)
16191619
(Reply
1620-
(threadObject owner "The monitor owner, or null if it is not currently owned.")
1621-
(int entryCount "The number of times the monitor has been entered.")
1622-
(Repeat waiters "The total number of threads that are waiting to enter or re-enter "
1623-
"the monitor, or waiting to be notified by the monitor."
1624-
(threadObject thread "A thread waiting for this monitor.")
1620+
(threadObject owner "The platform thread owning this monitor, or null "
1621+
"if owned by a virtual thread or not owned.")
1622+
(int entryCount "The number of times the owning platform thread has entered the monitor, "
1623+
"or 0 if owned by a virtual thread or not owned.")
1624+
(Repeat waiters "The total number of platform threads that are waiting to enter or re-enter "
1625+
"the monitor, or waiting to be notified by the monitor, or 0 if "
1626+
"only virtual threads are waiting or no threads are waiting."
1627+
(threadObject thread "A platform thread waiting for this monitor.")
16251628
)
16261629
)
16271630
(ErrorSet
@@ -2871,7 +2874,7 @@ JDWP "Java(tm) Debug Wire Protocol"
28712874
"if not explicitly requested."
28722875

28732876
(int requestID
2874-
"Request that generated event (or 0 if this "
2877+
"Request that generated event, or 0 if this "
28752878
"event is automatically generated.")
28762879
(threadObject thread "Initial thread")
28772880
)

‎src/jdk.jdi/share/classes/com/sun/jdi/ObjectReference.java

+10-7
Original file line numberDiff line numberDiff line change
@@ -345,7 +345,7 @@ Value invokeMethod(ThreadReference thread, Method method,
345345

346346
/**
347347
* Returns a List containing a {@link ThreadReference} for
348-
* each thread currently waiting for this object's monitor.
348+
* each platform thread currently waiting for this object's monitor.
349349
* See {@link ThreadReference#currentContendedMonitor} for
350350
* information about when a thread is considered to be waiting
351351
* for a monitor.
@@ -355,7 +355,8 @@ Value invokeMethod(ThreadReference thread, Method method,
355355
* operation is supported.
356356
*
357357
* @return a List of {@link ThreadReference} objects. The list
358-
* has zero length if no threads are waiting for the monitor.
358+
* has zero length if no threads are waiting for the monitor,
359+
* or only virtual threads are waiting for the monitor.
359360
* @throws java.lang.UnsupportedOperationException if the
360361
* target VM does not support this operation.
361362
* @throws IncompatibleThreadStateException if any
@@ -366,7 +367,7 @@ List<ThreadReference> waitingThreads()
366367
throws IncompatibleThreadStateException;
367368

368369
/**
369-
* Returns an {@link ThreadReference} for the thread, if any,
370+
* Returns a {@link ThreadReference} for the platform thread, if any,
370371
* which currently owns this object's monitor.
371372
* See {@link ThreadReference#ownedMonitors} for a definition
372373
* of ownership.
@@ -375,8 +376,9 @@ List<ThreadReference> waitingThreads()
375376
* {@link VirtualMachine#canGetMonitorInfo} to determine if the
376377
* operation is supported.
377378
*
378-
* @return the {@link ThreadReference} which currently owns the
379-
* monitor, or null if it is unowned.
379+
* @return the {@link ThreadReference} of the platform thread which
380+
* currently owns the monitor, or null if the monitor is owned
381+
* by a virtual thread or not owned.
380382
*
381383
* @throws java.lang.UnsupportedOperationException if the
382384
* target VM does not support this operation.
@@ -386,8 +388,9 @@ List<ThreadReference> waitingThreads()
386388
ThreadReference owningThread() throws IncompatibleThreadStateException;
387389

388390
/**
389-
* Returns the number times this object's monitor has been
390-
* entered by the current owning thread.
391+
* Returns the number of times this object's monitor has been entered by
392+
* the current owning thread if the owning thread is platform thread;
393+
* Returns 0 if not owned by a platform thread.
391394
* See {@link ThreadReference#ownedMonitors} for a definition
392395
* of ownership.
393396
* <p>

‎test/hotspot/jtreg/serviceability/jvmti/ObjectMonitorUsage/ObjectMonitorUsage.java

+60-15
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,13 @@ static void joinThreads(Thread[] threads) {
114114
throw new Error("Unexpected " + e);
115115
}
116116
}
117+
static Thread expOwnerThread() {
118+
return Thread.currentThread().isVirtual() ? null : Thread.currentThread();
119+
}
120+
121+
static int expEntryCount() {
122+
return Thread.currentThread().isVirtual() ? 0 : 1;
123+
}
117124

118125
/* Scenario #0:
119126
* - owning: 0
@@ -127,14 +134,18 @@ static void test0(boolean isVirtual) {
127134

128135
setTestedMonitor(lockCheck);
129136
Thread[] wThreads = startWaitingThreads(isVirtual);
137+
final int expWaitingCount = isVirtual ? 0 : NUMBER_OF_WAITING_THREADS;
130138

139+
// The numbers below describe the testing scenario, not the expected results.
140+
// The expected numbers are different for virtual threads because
141+
// they are not supported by JVMTI GetObjectMonitorUsage.
131142
// entry count: 0
132143
// count of threads waiting to enter: 0
133144
// count of threads waiting to re-enter: 0
134145
// count of threads waiting to be notified: NUMBER_OF_WAITING_THREADS
135146
check(lockCheck, null, 0, // no owner thread
136147
0, // count of threads waiting to enter: 0
137-
NUMBER_OF_WAITING_THREADS);
148+
expWaitingCount);
138149

139150
synchronized (lockCheck) {
140151
lockCheck.notifyAll();
@@ -158,20 +169,30 @@ static void test1(boolean isVirtual) {
158169
Thread[] eThreads = null;
159170

160171
synchronized (lockCheck) {
172+
// Virtual threads are not supported by GetObjectMonitorUsage.
173+
// Correct the expected values for the virtual thread case.
174+
int expEnteringCount = isVirtual ? 0 : NUMBER_OF_ENTERING_THREADS;
175+
176+
// The numbers below describe the testing scenario, not the expected results.
177+
// The expected numbers are different for virtual threads because
178+
// they are not supported by JVMTI GetObjectMonitorUsage.
161179
// entry count: 1
162180
// count of threads waiting to enter: 0
163181
// count of threads waiting to re-enter: 0
164182
// count of threads waiting to be notified: 0
165-
check(lockCheck, Thread.currentThread(), 1, 0, 0);
183+
check(lockCheck, expOwnerThread(), expEntryCount(), 0, 0);
166184

167185
eThreads = startEnteringThreads(isVirtual);
168186

187+
// The numbers below describe the testing scenario, not the expected results.
188+
// The expected numbers are different for virtual threads because
189+
// they are not supported by JVMTI GetObjectMonitorUsage.
169190
// entry count: 1
170191
// count of threads waiting to enter: NUMBER_OF_ENTERING_THREADS
171192
// count of threads waiting to re-enter: 0
172193
// count of threads waiting to be notified: 0
173-
check(lockCheck, Thread.currentThread(), 1,
174-
NUMBER_OF_ENTERING_THREADS,
194+
check(lockCheck, expOwnerThread(), expEntryCount(),
195+
expEnteringCount,
175196
0 /* count of threads waiting to be notified: 0 */);
176197

177198
}
@@ -195,15 +216,23 @@ static void test2(boolean isVirtual) throws Error {
195216
Thread[] eThreads = null;
196217

197218
synchronized (lockCheck) {
219+
// Virtual threads are not supported by the GetObjectMonitorUsage.
220+
// Correct the expected values for the virtual thread case.
221+
int expEnteringCount = isVirtual ? 0 : NUMBER_OF_ENTERING_THREADS;
222+
int expWaitingCount = isVirtual ? 0 : NUMBER_OF_WAITING_THREADS;
223+
198224
eThreads = startEnteringThreads(isVirtual);
199225

226+
// The numbers below describe the testing scenario, not the expected results.
227+
// The expected numbers are different for virtual threads because
228+
// they are not supported by JVMTI GetObjectMonitorUsage.
200229
// entry count: 1
201230
// count of threads waiting to enter: NUMBER_OF_ENTERING_THREADS
202231
// count of threads waiting to re-enter: 0
203232
// count of threads waiting to be notified: NUMBER_OF_WAITING_THREADS
204-
check(lockCheck, Thread.currentThread(), 1,
205-
NUMBER_OF_ENTERING_THREADS,
206-
NUMBER_OF_WAITING_THREADS);
233+
check(lockCheck, expOwnerThread(), expEntryCount(),
234+
expEnteringCount,
235+
expWaitingCount);
207236

208237
lockCheck.notifyAll();
209238
}
@@ -234,35 +263,51 @@ static void test3(boolean isVirtual) throws Error {
234263
Thread[] eThreads = null;
235264

236265
synchronized (lockCheck) {
266+
// Virtual threads are not supported by GetObjectMonitorUsage.
267+
// Correct the expected values for the virtual thread case.
268+
int expEnteringCount = isVirtual ? 0 : NUMBER_OF_ENTERING_THREADS;
269+
int expWaitingCount = isVirtual ? 0 : NUMBER_OF_WAITING_THREADS;
270+
271+
// The numbers below describe the testing scenario, not the expected results.
272+
// The expected numbers are different for virtual threads because
273+
// they are not supported by JVMTI GetObjectMonitorUsage.
237274
// entry count: 1
238275
// count of threads waiting to enter: 0
239276
// count of threads waiting to re-enter: 0
240277
// count of threads waiting to be notified: NUMBER_OF_WAITING_THREADS
241-
check(lockCheck, Thread.currentThread(), 1,
278+
check(lockCheck, expOwnerThread(), expEntryCount(),
242279
0, // number of threads waiting to enter or re-enter
243-
NUMBER_OF_WAITING_THREADS);
280+
expWaitingCount);
244281

245282
eThreads = startEnteringThreads(isVirtual);
246283

284+
// The numbers below describe the testing scenario, not the expected results.
285+
// The expected numbers are different for virtual threads because
286+
// they are not supported by JVMTI GetObjectMonitorUsage.
247287
// entry count: 1
248288
// count of threads waiting to enter: NUMBER_OF_ENTERING_THREADS
249289
// count of threads waiting to re-enter: 0
250290
// count of threads waiting to be notified: NUMBER_OF_WAITING_THREADS
251-
check(lockCheck, Thread.currentThread(), 1,
252-
NUMBER_OF_ENTERING_THREADS,
253-
NUMBER_OF_WAITING_THREADS);
291+
check(lockCheck, expOwnerThread(), expEntryCount(),
292+
expEnteringCount,
293+
expWaitingCount);
254294

255295
for (int i = 0; i < NUMBER_OF_WAITING_THREADS; i++) {
296+
expEnteringCount = isVirtual ? 0 : NUMBER_OF_ENTERING_THREADS + i + 1;
297+
expWaitingCount = isVirtual ? 0 : NUMBER_OF_WAITING_THREADS - i - 1;
256298
lockCheck.notify(); // notify waiting threads one by one
257299
// now the notified WaitingTask has to be blocked on the lockCheck re-enter
258300

301+
// The numbers below describe the testing scenario, not the expected results.
302+
// The expected numbers are different for virtual threads because
303+
// they are not supported by JVMTI GetObjectMonitorUsage.
259304
// entry count: 1
260305
// count of threads waiting to enter: NUMBER_OF_ENTERING_THREADS
261306
// count of threads waiting to re-enter: i + 1
262307
// count of threads waiting to be notified: NUMBER_OF_WAITING_THREADS - i - 1
263-
check(lockCheck, Thread.currentThread(), 1,
264-
NUMBER_OF_ENTERING_THREADS + i + 1,
265-
NUMBER_OF_WAITING_THREADS - i - 1);
308+
check(lockCheck, expOwnerThread(), expEntryCount(),
309+
expEnteringCount,
310+
expWaitingCount);
266311
}
267312
}
268313
joinThreads(wThreads);

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ObjectReference/entryCount/entrycount002.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -197,8 +197,11 @@ private void execTest() {
197197

198198
display("Checking entryCount for iteration : " + i);
199199
try {
200+
// The lockRef.entryCount() is expected to return 0 if the owner thread is virtual.
201+
int expEntryCount = mainThread.isVirtual() ? 0 : i;
200202
int entryCount = lockRef.entryCount();
201-
if (entryCount != i) {
203+
204+
if (entryCount != expEntryCount) {
202205
exitCode = Consts.TEST_FAILED;
203206
complain("entry count method returned unexpected value : " + entryCount +
204207
"\n\t expected one : " + i);

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ObjectReference/owningThread/owningthread002.java

+9
Original file line numberDiff line numberDiff line change
@@ -200,6 +200,15 @@ private void execTest() {
200200
try {
201201
ThreadReference thread = lockRef.owningThread();
202202

203+
// The lockRef.owningThread() is expected to return null if tested threads are virtual.
204+
if (eventThread.isVirtual()) {
205+
if (thread == null) {
206+
display("expected null is returned` by owningThread method on virtual thread: " + eventThread.name());
207+
} else {
208+
complain("owningThread returned ThreadReference of virtual thread instead of null: " + thread.name());
209+
}
210+
continue;
211+
}
203212
if (thread.name().indexOf(owningthread002a.threadNamePrefix) < 0) {
204213
exitCode = Consts.TEST_FAILED;
205214
complain("owningThread returned ThreadReference with unexpected name: " + thread.name());

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ObjectReference/waitingThreads/waitingthreads002.java

+6-1
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,9 @@ private static void execTest() {
122122
if (thread.name().indexOf(waitingthreads002a.threadNamePrefix) >= 0 &&
123123
thread.status() == ThreadReference.THREAD_STATUS_MONITOR ) {
124124
waitingCount++;
125+
// Virtual threads are not present in result returned by objRef.waitingThreads().
126+
if (!thread.isVirtual()) {
127+
}
125128
}
126129
}
127130
}
@@ -159,7 +162,9 @@ private static void execTest() {
159162
objRef = (ObjectReference) debuggeeClass.getValue(debuggeeClass.fieldByName(fieldName));
160163
try {
161164
List waitingThreads = objRef.waitingThreads();
162-
if (waitingThreads.size() != waitingthreads002a.threadCount) {
165+
final boolean vthreadMode = "Virtual".equals(System.getProperty("test.thread.factory"));
166+
final int expWaitingCount = vthreadMode ? 0 : waitingthreads002a.threadCount;
167+
if (waitingThreads.size() != expWaitingCount) {
163168
exitStatus = Consts.TEST_FAILED;
164169
complain("waitingThreads method returned list with unexpected size for " + fieldName +
165170
"\n\t expected value : " + waitingthreads002a.threadCount + "; got one : " + waitingThreads.size());

‎test/hotspot/jtreg/vmTestbase/nsk/jvmti/GetObjectMonitorUsage/objmonusage001.java

+21-5
Original file line numberDiff line numberDiff line change
@@ -61,8 +61,16 @@ public static int run(String argv[], PrintStream out) {
6161
syncObject[i] = new Object();
6262
runn[i] = new objmonusage001a(mainThread, i, syncObject[i]);
6363
}
64+
// Virtual threads are not supported by GetObjectMonitorUsage.
65+
// Correct the expected values if the test is executed with
66+
// JTREG_TEST_THREAD_FACTORY=Virtual.
67+
Thread expOwner = mainThread.isVirtual() ? null : mainThread;
68+
int expEntryCount = mainThread.isVirtual() ? 0 : 1;
6469

6570
for (int i = 0; i < NUMBER_OF_THREADS; i++) {
71+
Thread expNotifyWaiter = runn[i].isVirtual() ? null : runn[i];
72+
int expNotifyWaitingCount = runn[i].isVirtual() ? 0 : 1;
73+
6674
synchronized (syncObject[i]) {
6775
runn[i].start();
6876
try {
@@ -92,8 +100,8 @@ public static int run(String argv[], PrintStream out) {
92100
// This is a stable verification point because the worker thread is in wait()
93101
// and is not notified and the main thread is doing the verification.
94102
//
95-
check(NUMBER_OF_THREADS + i, syncObject[i], mainThread, 1,
96-
null, 0, runn[i], 1);
103+
check(NUMBER_OF_THREADS + i, syncObject[i], expOwner, expEntryCount,
104+
null, 0, expNotifyWaiter, expNotifyWaitingCount);
97105
}
98106

99107
// Check #3:
@@ -117,7 +125,7 @@ public static int run(String argv[], PrintStream out) {
117125
// and is not notified and the main thread is doing the verification.
118126
//
119127
check((NUMBER_OF_THREADS * 2) + i, syncObject[i], null, 0,
120-
null, 0, runn[i], 1);
128+
null, 0, expNotifyWaiter, expNotifyWaitingCount);
121129
}
122130

123131
for (int i = 0; i < NUMBER_OF_THREADS; i++) {
@@ -147,6 +155,14 @@ public objmonusage001a(Thread mt, int i, Object s) {
147155
}
148156

149157
public void run() {
158+
// Virtual threads are not supported by GetObjectMonitorUsage.
159+
// Correct the expected values if the test is executed with
160+
// JTREG_TEST_THREAD_FACTORY=Virtual.
161+
Thread expOwner = this.isVirtual() ? null : this;
162+
Thread expNotifyWaiter = mainThread.isVirtual() ? null : mainThread;
163+
int expEntryCount = this.isVirtual() ? 0 : 1;
164+
int expNotifyWaitingCount = mainThread.isVirtual() ? 0 : 1;
165+
150166
synchronized (syncObject) {
151167
// Check #1:
152168
// - owner == this_thread:
@@ -166,8 +182,8 @@ public void run() {
166182
// This is a stable verification point because the main thread is in wait()
167183
// and is not notified and this worker thread is doing the verification.
168184
//
169-
objmonusage001.check(index, syncObject, this, 1,
170-
null, 0, mainThread, 1);
185+
objmonusage001.check(index, syncObject, expOwner, expEntryCount,
186+
null, 0, expNotifyWaiter, expNotifyWaitingCount);
171187
syncObject.notify();
172188

173189
try {

‎test/hotspot/jtreg/vmTestbase/nsk/jvmti/GetObjectMonitorUsage/objmonusage004.java

+15-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2003, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -60,11 +60,23 @@ public static int run(String args[], PrintStream out) {
6060
Thread currThread = Thread.currentThread();
6161
ContendThread thr[] = new ContendThread[NUMBER_OF_THREADS];
6262
synchronized (lockCheck) {
63+
// Virtual threads are not supported by GetObjectMonitorUsage.
64+
// Correct the expected values if the test is executed with
65+
// JTREG_TEST_THREAD_FACTORY=Virtual.
66+
Thread expOwner = currThread.isVirtual() ? null : currThread;
67+
int expEntryCount = currThread.isVirtual() ? 0 : 2;
68+
6369
synchronized (lockCheck) {
64-
check(lockCheck, currThread, 2, 0);
70+
check(lockCheck, expOwner, expEntryCount, 0);
6571
}
72+
expEntryCount = currThread.isVirtual() ? 0 : 1;
73+
int expWaiterCount = 0;
74+
6675
for (int i = 0; i < NUMBER_OF_THREADS; i++) {
6776
thr[i] = new ContendThread();
77+
if (!thr[i].isVirtual()) {
78+
expWaiterCount++;
79+
}
6880
synchronized (lockStart) {
6981
thr[i].start();
7082
try {
@@ -74,7 +86,7 @@ public static int run(String args[], PrintStream out) {
7486
throw new Error("Unexpected " + e);
7587
}
7688
}
77-
check(lockCheck, currThread, 1, i + 1);
89+
check(lockCheck, expOwner, expEntryCount, expWaiterCount);
7890
}
7991
}
8092

0 commit comments

Comments
 (0)
Please sign in to comment.