38cc0391f3
Reviewed-by: lmesnik, amenkov, sspitsyn
188 lines
6.1 KiB
Java
188 lines
6.1 KiB
Java
/*
|
|
* Copyright (c) 2007, 2023, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/**
|
|
* @test
|
|
* @bug 6517249
|
|
* @summary JDWP: Cannot do an invokeMethod after a popFrames operation
|
|
* @author jjh
|
|
*
|
|
* @run build TestScaffold VMConnection TargetListener TargetAdapter
|
|
* @run compile -g PopAndInvokeTest.java
|
|
* @run driver PopAndInvokeTest
|
|
*/
|
|
import com.sun.jdi.*;
|
|
import com.sun.jdi.event.*;
|
|
import java.util.*;
|
|
|
|
class PopAndInvokeTarg {
|
|
static boolean waiting = false;
|
|
|
|
public static void A() {
|
|
System.out.println(" debuggee: in A");
|
|
}
|
|
|
|
public static void invokeee() {
|
|
System.out.println(" debuggee: invokee");
|
|
}
|
|
|
|
public static void waiter() {
|
|
if (waiting) {
|
|
return;
|
|
}
|
|
System.out.println(" debuggee: in waiter");
|
|
// No printlns or other calls allowed after this point.
|
|
waiting = true;
|
|
while (true) {
|
|
}
|
|
}
|
|
|
|
public static void main(String[] args) {
|
|
System.out.println(" debuggee: Howdy!");
|
|
/*
|
|
* Debugger will bkpt in A, popFrames back to here
|
|
* and then do an invokeMethod on invokeee.
|
|
* This should work.
|
|
*/
|
|
A();
|
|
|
|
/*
|
|
* Debugger will resume and we will enter
|
|
* waiter(). Debugger will then do a suspend,
|
|
* a popFrames back to here, and an invoke
|
|
* which should fail.
|
|
*/
|
|
waiter();
|
|
System.out.println(" debuggee: Goodbye from PopAndInvokeTarg!");
|
|
}
|
|
}
|
|
|
|
|
|
/********** test program **********/
|
|
|
|
public class PopAndInvokeTest extends TestScaffold {
|
|
ClassType targetClass;
|
|
ThreadReference mainThread;
|
|
|
|
PopAndInvokeTest (String args[]) {
|
|
super(args);
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
new PopAndInvokeTest(args).startTests();
|
|
}
|
|
|
|
StackFrame frameFor(String methodName) throws Exception {
|
|
Iterator it = mainThread.frames().iterator();
|
|
|
|
while (it.hasNext()) {
|
|
StackFrame frame = (StackFrame)it.next();
|
|
if (frame.location().method().name().equals(methodName)) {
|
|
return frame;
|
|
}
|
|
}
|
|
failure("FAIL: " + methodName + " not on stack");
|
|
return null;
|
|
}
|
|
|
|
/********** test core **********/
|
|
|
|
protected void runTests() throws Exception {
|
|
/*
|
|
* Get to the top of main()
|
|
* to determine targetClass and mainThread
|
|
*/
|
|
runOnce();
|
|
}
|
|
|
|
void runOnce() throws Exception {
|
|
|
|
BreakpointEvent bpe = startTo("PopAndInvokeTarg", "A", "()V");
|
|
targetClass = (ClassType)bpe.location().declaringType();
|
|
mainThread = bpe.thread();
|
|
|
|
/*
|
|
* Verify that an invokeMethod works ok after a popFrames
|
|
* in a thread suspended by an event.
|
|
*/
|
|
mainThread.popFrames(frameFor("A"));
|
|
|
|
System.out.println("Debugger: Popped back to the call to A()");
|
|
System.out.println("Debugger: Doing invoke");
|
|
|
|
Method invokeeeMethod = (Method)targetClass.methodsByName("invokeee").get(0);
|
|
try {
|
|
targetClass.invokeMethod(mainThread, invokeeeMethod,
|
|
new ArrayList(), 0);
|
|
} catch (Exception ex) {
|
|
failure("failure: invoke got unexpected exception: " + ex);
|
|
ex.printStackTrace();
|
|
}
|
|
System.out.println("Debugger: invoke done");
|
|
|
|
/*
|
|
* Verify that an invokeMethod gets an IncompatibleThreadStateException
|
|
* after a popFrames in a thread that is _not_ suspended by an event.
|
|
*/
|
|
System.out.println("Debugger: Resuming debuggee");
|
|
vm().resume();
|
|
|
|
Field waiting = targetClass.fieldByName("waiting");
|
|
while (true) {
|
|
// Wait until debuggee enters the 'waiting' method.
|
|
BooleanValue bv= (BooleanValue)targetClass.getValue(waiting);
|
|
if (!bv.value()) {
|
|
try {
|
|
Thread.sleep(10);
|
|
} catch (InterruptedException ee) {
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// debuggee has entered the waiting method
|
|
System.out.println("Debugger: Suspending debuggee");
|
|
vm().suspend();
|
|
System.out.println("Debugger: Popping frame for waiter");
|
|
mainThread.popFrames(frameFor("waiter"));
|
|
System.out.println("Debugger: Invoking method");
|
|
try {
|
|
targetClass.invokeMethod(mainThread, invokeeeMethod,
|
|
new ArrayList(), 0);
|
|
} catch (IncompatibleThreadStateException ee) {
|
|
System.out.println("Debugger: Success: Got expected IncompatibleThreadStateException");
|
|
break;
|
|
} catch (Exception ee) {
|
|
failure("FAIL: Got unexpected exception: " + ee);
|
|
break;
|
|
}
|
|
failure("FAIL: Did not get IncompatibleThreadStateException " +
|
|
"when debuggee is not suspended by an event");
|
|
}
|
|
listenUntilVMDisconnect();
|
|
if (testFailed) {
|
|
throw new Exception("PopAndInvokeTest failed");
|
|
}
|
|
System.out.println("Passed:");
|
|
}
|
|
}
|