2007-12-01 00:00:00 +00:00
|
|
|
/*
|
2010-05-25 15:58:33 -07:00
|
|
|
* Copyright (c) 2001, Oracle and/or its affiliates. All rights reserved.
|
2007-12-01 00:00:00 +00:00
|
|
|
* 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.
|
|
|
|
*
|
2010-05-25 15:58:33 -07:00
|
|
|
* 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.
|
2007-12-01 00:00:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @bug 4467564
|
|
|
|
* @summary Test the popping of frames in synchronous context
|
|
|
|
* (that is, when stopped at an event)
|
|
|
|
*
|
|
|
|
* @author Robert Field
|
|
|
|
*
|
|
|
|
* @run build TestScaffold VMConnection TargetListener TargetAdapter
|
|
|
|
* @run compile -g PopSynchronousTest.java
|
|
|
|
* @run main PopSynchronousTest
|
|
|
|
*/
|
|
|
|
import com.sun.jdi.*;
|
|
|
|
import com.sun.jdi.event.*;
|
|
|
|
import com.sun.jdi.request.*;
|
|
|
|
|
|
|
|
import java.util.*;
|
|
|
|
|
|
|
|
/********** target program **********/
|
|
|
|
|
|
|
|
class PopSynchronousTarg {
|
|
|
|
static String s;
|
|
|
|
static PopSynchronousTarg sole;
|
|
|
|
|
|
|
|
synchronized void a(int y, boolean w) {
|
|
|
|
if (y == 6 && w) {
|
|
|
|
s += "@";
|
|
|
|
} else {
|
|
|
|
s += " aArgFail ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
String b(String h) {
|
|
|
|
if (h.equals("yo")) {
|
|
|
|
s += "[";
|
|
|
|
} else {
|
|
|
|
s += " bArgFail ";
|
|
|
|
}
|
|
|
|
a(6, true);
|
|
|
|
s += "]";
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
long c() {
|
|
|
|
s += "<";
|
|
|
|
synchronized (s) {
|
|
|
|
b("yo");
|
|
|
|
}
|
|
|
|
s += ">";
|
|
|
|
return 17;
|
|
|
|
}
|
|
|
|
static void p() {
|
|
|
|
s += "(";
|
|
|
|
if (sole.c() != 17) {
|
|
|
|
s += " cReturnFail ";
|
|
|
|
}
|
|
|
|
s += ")";
|
|
|
|
}
|
|
|
|
static void report() {
|
|
|
|
}
|
|
|
|
public static void main(String[] args){
|
|
|
|
s = new String();
|
|
|
|
sole = new PopSynchronousTarg();
|
|
|
|
for (int i = 0; i < 100; ++i) {
|
|
|
|
p();
|
|
|
|
// System.out.println(s);
|
|
|
|
report();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/********** test program **********/
|
|
|
|
|
|
|
|
public class PopSynchronousTest extends TestScaffold {
|
|
|
|
ReferenceType targetClass;
|
|
|
|
ThreadReference mainThread;
|
|
|
|
int stateIndex = 0;
|
|
|
|
String expected = "";
|
|
|
|
static final String[] calls = {"a", "b", "c", "p", "main"};
|
|
|
|
static final int popMax = calls.length - 1;
|
|
|
|
static final String[] states =
|
|
|
|
{"main-i", "p-e", "p-i", "c-e", "c-i", "b-e", "b-i", "a-e", "a-l", "b-l", "c-l", "p-l", "main-r", "report-e"};
|
|
|
|
static final String[] output =
|
|
|
|
{"", "", "(", "", "<", "", "[", "", "@", "]", ">", ")", "", ""};
|
|
|
|
|
|
|
|
|
|
|
|
PopSynchronousTest (String args[]) {
|
|
|
|
super(args);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
new PopSynchronousTest(args).startTests();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/********** test assist **********/
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
String actual() throws Exception {
|
|
|
|
Field field = targetClass.fieldByName("s");
|
|
|
|
StringReference sr = (StringReference)(targetClass.getValue(field));
|
|
|
|
return sr.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
void checkExpected() throws Exception {
|
|
|
|
if (!actual().equals(expected)) {
|
|
|
|
failure("FAIL: expected value: " + expected +
|
|
|
|
" got: " + actual());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int methodIndex(String methodName) {
|
|
|
|
for (int i = 0; i < popMax; ++i) {
|
|
|
|
if (methodName.equals(calls[i])) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean isTop(String methodName) throws Exception {
|
|
|
|
return mainThread.frame(0).location().method().name().equals(methodName);
|
|
|
|
}
|
|
|
|
|
|
|
|
void checkTop(String methodName, boolean atStart) throws Exception {
|
|
|
|
Location loc = mainThread.frame(0).location();
|
|
|
|
Method meth = loc.method();
|
|
|
|
String name = meth.name();
|
|
|
|
if (!isTop(methodName)) {
|
|
|
|
failure("FAIL: expected " + methodName +
|
|
|
|
" at top of stack, instead: " + name);
|
|
|
|
} else if ((meth.location().codeIndex() == loc.codeIndex()) != atStart) {
|
|
|
|
failure("FAIL: not at expect position: " + loc.codeIndex());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void checkState() throws Exception {
|
|
|
|
String name = states[stateIndex];
|
|
|
|
int dash = name.indexOf('-');
|
|
|
|
String methodName = name.substring(0,dash);
|
|
|
|
String posName = name.substring(dash+1);
|
|
|
|
checkTop(methodName, posName.equals("e"));
|
|
|
|
checkExpected();
|
|
|
|
}
|
|
|
|
|
|
|
|
void incrementState() {
|
|
|
|
stateIndex = (stateIndex + 1) % (states.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetState(String stateName) {
|
|
|
|
for (int i=0; i < states.length; ++i) {
|
|
|
|
if (states[i].equals(stateName)) {
|
|
|
|
stateIndex = i;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
failure("TEST FAILURE: cannot find state: " + stateName);
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetExpected() throws Exception {
|
|
|
|
println("Current value: " + actual());
|
|
|
|
Field field = targetClass.fieldByName("s");
|
|
|
|
expected = "";
|
|
|
|
((ClassType)targetClass).setValue(field, vm().mirrorOf(expected));
|
|
|
|
}
|
|
|
|
|
|
|
|
void stateTo(String stateName) {
|
|
|
|
do {
|
|
|
|
incrementState();
|
|
|
|
expected += output[stateIndex];
|
|
|
|
} while(!states[stateIndex].equals(stateName));
|
|
|
|
}
|
|
|
|
|
|
|
|
void resumeTo(String methodName) throws Exception {
|
|
|
|
List meths = targetClass.methodsByName(methodName);
|
|
|
|
Method meth = (Method)(meths.get(0));
|
|
|
|
resumeTo(meth.location());
|
|
|
|
stateTo(methodName + "-e");
|
|
|
|
checkState();
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop(String methodName) throws Exception {
|
|
|
|
mainThread.popFrames(frameFor(methodName));
|
|
|
|
resetState(methodName + "-e");
|
|
|
|
--stateIndex;
|
|
|
|
checkState();
|
|
|
|
}
|
|
|
|
|
|
|
|
void reenter(String methodName) throws Exception {
|
|
|
|
pop(methodName);
|
|
|
|
stepIntoInstruction(mainThread);
|
|
|
|
incrementState();
|
|
|
|
checkState();
|
|
|
|
}
|
|
|
|
|
|
|
|
/********** test core **********/
|
|
|
|
|
|
|
|
protected void runTests() throws Exception {
|
|
|
|
/*
|
|
|
|
* Get to the top of main()
|
|
|
|
* to determine targetClass and mainThread
|
|
|
|
*/
|
|
|
|
BreakpointEvent bpe = startToMain("PopSynchronousTarg");
|
|
|
|
targetClass = bpe.location().declaringType();
|
|
|
|
mainThread = bpe.thread();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Testing
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* individual tests */
|
|
|
|
for (int i = 0; i < popMax; ++i) {
|
|
|
|
String from = calls[i];
|
|
|
|
for (int j = i; j < popMax; ++j) {
|
|
|
|
String to = calls[j];
|
|
|
|
String prev = calls[j+1];
|
|
|
|
println("TEST pop from '" + from + "' to '" + to + "'");
|
|
|
|
resumeTo(from);
|
|
|
|
reenter(to);
|
|
|
|
resumeTo("report");
|
|
|
|
resetExpected();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* sequential tests */
|
|
|
|
|
|
|
|
println("TEST pop a b c p");
|
|
|
|
resumeTo("a");
|
|
|
|
pop("a");
|
|
|
|
pop("b");
|
|
|
|
pop("c");
|
|
|
|
pop("p");
|
|
|
|
resumeTo("report");
|
|
|
|
resetExpected();
|
|
|
|
|
|
|
|
println("TEST pop a c p");
|
|
|
|
resumeTo("a");
|
|
|
|
pop("a");
|
|
|
|
pop("c");
|
|
|
|
pop("p");
|
|
|
|
resumeTo("report");
|
|
|
|
resetExpected();
|
|
|
|
|
|
|
|
println("TEST stress a");
|
|
|
|
resumeTo("a");
|
|
|
|
for (int i = 0; i < 100; ++i) {
|
|
|
|
reenter("a");
|
|
|
|
}
|
|
|
|
resumeTo("report");
|
|
|
|
resetExpected();
|
|
|
|
|
|
|
|
println("TEST stress c");
|
|
|
|
resumeTo("c");
|
|
|
|
for (int i = 0; i < 100; ++i) {
|
|
|
|
reenter("c");
|
|
|
|
}
|
|
|
|
resumeTo("report");
|
|
|
|
resetExpected();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* we are done, get rid of target
|
|
|
|
*/
|
|
|
|
vm().dispose();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* deal with results of test
|
|
|
|
* if anything has called failure("foo") testFailed will be true
|
|
|
|
*/
|
|
|
|
if (!testFailed) {
|
|
|
|
println("PopSynchronousTest: passed");
|
|
|
|
} else {
|
|
|
|
throw new Exception("PopSynchronousTest: failed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|