Skip to content

Commit 6a9145b

Browse files
committedMar 11, 2022
Proper error handling of JVMTI OPAQUE_FRAME errors in JDWP and JDI. Get at least one JDI test running in vthread mode to detect OPAQUE_FRAME for each of PopFrame, ForceEarlyReturn, and SetLocalXXX.
1 parent 1207895 commit 6a9145b

File tree

8 files changed

+78
-40
lines changed

8 files changed

+78
-40
lines changed
 

‎src/jdk.jdi/share/classes/com/sun/tools/jdi/ThreadReferenceImpl.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
import com.sun.jdi.MonitorInfo;
4242
import com.sun.jdi.NativeMethodException;
4343
import com.sun.jdi.ObjectReference;
44+
import com.sun.jdi.OpaqueFrameException;
4445
import com.sun.jdi.ReferenceType;
4546
import com.sun.jdi.StackFrame;
4647
import com.sun.jdi.ThreadGroupReference;
@@ -585,7 +586,12 @@ public void forceEarlyReturn(Value returnValue) throws InvalidTypeException,
585586
} catch (JDWPException exc) {
586587
switch (exc.errorCode()) {
587588
case JDWP.Error.OPAQUE_FRAME:
588-
throw new NativeMethodException();
589+
if (meth.isNative()) {
590+
throw new NativeMethodException();
591+
} else {
592+
assert isVirtual(); // can only happen with virtual threads
593+
throw new OpaqueFrameException();
594+
}
589595
case JDWP.Error.THREAD_NOT_SUSPENDED:
590596
throw new IncompatibleThreadStateException(
591597
"Thread not suspended");

‎src/jdk.jdwp.agent/share/native/libjdwp/StackFrameImpl.c

+1-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2022, 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
@@ -447,13 +447,6 @@ popFrames(PacketInputStream *in, PacketOutputStream *out)
447447
return JNI_TRUE;
448448
}
449449

450-
/* "popFrames" is not supported on a vthread. Although JVMTI will produce INVALID_THREAD,
451-
we check here and force the error to make it obvious to the reader. */
452-
if (isVThread(thread)) {
453-
outStream_setError(out, JDWP_ERROR(INVALID_THREAD));
454-
return JNI_TRUE;
455-
}
456-
457450
fnum = getFrameNumber(frame);
458451
error = threadControl_popFrames(thread, fnum);
459452
if (error != JVMTI_ERROR_NONE) {

‎src/jdk.jdwp.agent/share/native/libjdwp/ThreadReferenceImpl.c

-7
Original file line numberDiff line numberDiff line change
@@ -595,13 +595,6 @@ forceEarlyReturn(PacketInputStream *in, PacketOutputStream *out)
595595
return JNI_TRUE;
596596
}
597597

598-
/* "forceEarlyReturn" is not supported on a vthread. Although JVMTI will produce INVALID_THREAD,
599-
we check here and force the error to make it obvious to the reader. */
600-
if (isVThread(thread)) {
601-
outStream_setError(out, JDWP_ERROR(INVALID_THREAD));
602-
return JNI_TRUE;
603-
}
604-
605598
typeKey = inStream_readByte(in);
606599
if (inStream_error(in)) {
607600
return JNI_TRUE;

‎test/hotspot/jtreg/ProblemList-vthread.txt

+6-7
Original file line numberDiff line numberDiff line change
@@ -541,13 +541,12 @@ vmTestbase/nsk/jdb/repeat/repeat001/repeat001.java
541541

542542
vmTestbase/nsk/jdi/ExceptionEvent/catchLocation/location002/TestDescription.java 8278470 generic-all
543543

544-
####
545-
## JVMTI SetLocalXXX() is returning OPAQUE_FRAME because vthreads are not except with topmost farme.
546-
547-
vmTestbase/nsk/jdi/StackFrame/_bounds_/bounds002/TestDescription.java
548544

549545
####
550546
## JVMTI PopFrame() is returning OPAQUE_FRAME because vthreads are not supported.
547+
## Note: vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes001 was converted
548+
## to support vthreads and expect the OPAQUE_FRAME error. The others were
549+
## not because they don't add any additional value.
551550

552551
vmTestbase/nsk/jdi/Scenarios/invokeMethod/popframes001/TestDescription.java
553552
vmTestbase/nsk/jdi/BScenarios/hotswap/tc01x002/TestDescription.java
@@ -559,15 +558,15 @@ vmTestbase/nsk/jdi/BScenarios/hotswap/tc06x001/TestDescription.java
559558
vmTestbase/nsk/jdi/BScenarios/hotswap/tc08x001/TestDescription.java
560559
vmTestbase/nsk/jdi/BScenarios/hotswap/tc10x002/TestDescription.java
561560

562-
vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes001/TestDescription.java
563561
vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes002/TestDescription.java
564562
vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes003/TestDescription.java
565563
vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes004/TestDescription.java
566564

567565
####
568-
## JVMTI ForceEarlyReturn not supported for vthreads (JVMTI_ERROR_INVALID_THREAD)
566+
## JVMTI ForceEarlyReturn not supported for vthreads (JVMTI_ERROR_OPAQUE_FRAME)
567+
## Note forceEarlyReturn002 was converted to support vthreads. The rest were not
568+
## since there is no added value (JVMTI_ERROR_OPAQUE_FRAME is expected).
569569

570-
vmTestbase/nsk/jdi/ThreadReference/forceEarlyReturn/forceEarlyReturn002/forceEarlyReturn002.java
571570
vmTestbase/nsk/jdi/ThreadReference/forceEarlyReturn/forceEarlyReturn014/forceEarlyReturn014.java
572571
vmTestbase/nsk/jdi/stress/serial/forceEarlyReturn001/TestDescription.java
573572
vmTestbase/nsk/jdi/stress/serial/forceEarlyReturn002/TestDescription.java

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/StackFrame/_bounds_/bounds002.java

+15-3
Original file line numberDiff line numberDiff line change
@@ -184,12 +184,24 @@ private void execTest() {
184184
complain("Unexpected " + e);
185185
exitStatus = Consts.TEST_FAILED;
186186
}
187+
boolean vthreadMode = "Virtual".equals(System.getProperty("main.wrapper"));
187188
try {
188189
stackFrame.setValue(var, null);
189-
display("OK");
190+
if (vthreadMode) {
191+
// should have failed because stackFrame is not the topmost frame
192+
complain("Expected OpaqueFrameException");
193+
exitStatus = Consts.TEST_FAILED;
194+
} else {
195+
display("OK");
196+
}
190197
} catch(Exception e) {
191-
complain("Unexpected " + e);
192-
exitStatus = Consts.TEST_FAILED;
198+
if (vthreadMode && e instanceof OpaqueFrameException) {
199+
// pass
200+
display("OK");
201+
} else {
202+
complain("Unexpected " + e);
203+
exitStatus = Consts.TEST_FAILED;
204+
}
193205
}
194206
display("");
195207

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ThreadReference/forceEarlyReturn/forceEarlyReturn002/forceEarlyReturn002.java

+20-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2007, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2007, 2022, 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
@@ -147,17 +147,30 @@ public void doTest() {
147147

148148
// get value for early return
149149
ObjectReference returnValue = (ObjectReference) referenceType.getValue(referenceType.fieldByName("expectedValue"));
150-
150+
boolean vthreadMode = "Virtual".equals(System.getProperty("main.wrapper"));
151151
try {
152-
// don't expect any exception
152+
// don't expect any exception, except for vthreads expect OpaqueFrameException
153153
threadReference.forceEarlyReturn(returnValue);
154+
if (vthreadMode) {
155+
setSuccess(false);
156+
log.complain("Expected OpaqueFrameException");
157+
}
154158
} catch (Exception e) {
155-
setSuccess(false);
156-
log.complain("Unexpected exception: " + e);
157-
e.printStackTrace(log.getOutStream());
159+
if (vthreadMode && (e instanceof OpaqueFrameException)) {
160+
// pass
161+
} else {
162+
setSuccess(false);
163+
log.complain("Unexpected exception: " + e);
164+
e.printStackTrace(log.getOutStream());
165+
}
158166
}
159167

160-
testMethodExitEvent(threadReference, ClassUsingTestClass.breakpointMethodName);
168+
if (vthreadMode) {
169+
// MethodExit event won't be as expected if using vthreads, so just resume
170+
threadReference.resume();
171+
} else {
172+
testMethodExitEvent(threadReference, ClassUsingTestClass.breakpointMethodName);
173+
}
161174

162175
if (!isDebuggeeReady())
163176
return;

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ThreadReference/forceEarlyReturn/forceEarlyReturn002/forceEarlyReturn002a.java

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2007, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2007, 2022, 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
@@ -83,7 +83,9 @@ public boolean parseCommand(String command) {
8383
classUsingTestClass.createExpectedValue();
8484
Object value = classUsingTestClass.testClassMethod();
8585

86-
if (ClassUsingTestClass.expectedValue != value) {
86+
boolean vthreadMode = "Virtual".equals(System.getProperty("main.wrapper"));
87+
// expectedValue should be set as expected unless in vthread mode
88+
if (vthreadMode == (ClassUsingTestClass.expectedValue == value)) {
8789
setSuccess(false);
8890
log.complain("Unexpected result of testClassMethod: " + value);
8991
}

‎test/hotspot/jtreg/vmTestbase/nsk/jdi/ThreadReference/popFrames/popframes001.java

+25-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2001, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2001, 2022, 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
@@ -93,6 +93,8 @@
9393

9494
public class popframes001 extends JDIBase {
9595

96+
static boolean vthreadMode = "Virtual".equals(System.getProperty("main.wrapper"));
97+
9698
public static void main (String argv[]) {
9799

98100
int result = run(argv, System.out);
@@ -219,6 +221,10 @@ private int runTest() {
219221
try {
220222
testRun();
221223

224+
if (vthreadMode) {
225+
return 0; // just exit. we already got the expected OpaqueFrameException
226+
}
227+
222228
log2("waiting for VMDeathEvent");
223229
getEventSet();
224230
if (eventIterator.nextEvent() instanceof VMDeathEvent)
@@ -350,10 +356,24 @@ private void testRun()
350356
log2("......thread2Ref.popFrames(stackFrame);");
351357
try {
352358
thread2Ref.popFrames(stackFrame);
353-
} catch ( IncompatibleThreadStateException e ) {
354-
log3("ERROR: IncompatibleThreadStateException");
355-
testExitCode = FAILED;
356-
break;
359+
if (vthreadMode) {
360+
log3("ERROR: Expected OpaqueFrameException");
361+
testExitCode = FAILED;
362+
}
363+
} catch ( Exception e ) {
364+
if (vthreadMode && (e instanceof OpaqueFrameException)) {
365+
// pass. resume thread and exit
366+
log2("......got expected OpaqueFrameException");
367+
log2("......thread2Ref.resume();");
368+
thread2Ref.resume();
369+
breakpointForCommunication();
370+
vm.resume();
371+
break;
372+
} else {
373+
log3("ERROR: " + e.getClass().getSimpleName());
374+
testExitCode = FAILED;
375+
throw e;
376+
}
357377
}
358378

359379
log2("......thread2Ref.resume();");

0 commit comments

Comments
 (0)
Please sign in to comment.