4417c276e4
Reviewed-by: cjplummer
241 lines
8.8 KiB
Java
241 lines
8.8 KiB
Java
/*
|
|
* Copyright (c) 2002, 2024, 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.
|
|
*/
|
|
|
|
package nsk.share.jdb;
|
|
|
|
import nsk.share.*;
|
|
|
|
import java.io.*;
|
|
|
|
public abstract class JdbTest {
|
|
public static final int PASSED = 0; // Exit code for passed test
|
|
public static final int FAILED = 2; // Exit code for failed test
|
|
public static final int JCK_STATUS_BASE = 95; // Standard JCK-compatible exit code bias
|
|
|
|
/* Flag if the test passes */
|
|
protected boolean success = true;
|
|
|
|
/* Flag if debuggee should fail in a test */
|
|
protected static boolean debuggeeShouldFail = false;
|
|
|
|
/* Handler of command line arguments. */
|
|
protected static JdbArgumentHandler argumentHandler = null;
|
|
|
|
/* Log class to print log messages. */
|
|
protected static Log log = null;
|
|
|
|
protected static Jdb jdb = null;
|
|
protected static Debuggee debuggee = null;
|
|
protected static Launcher launcher = null;
|
|
protected static String debuggeeClass = "";
|
|
protected static String firstBreak = "";
|
|
protected static String lastBreak = "";
|
|
protected static String compoundPromptIdent = null;
|
|
|
|
/* Constructors */
|
|
protected JdbTest (boolean debuggeeShouldFail) {
|
|
this.debuggeeShouldFail = debuggeeShouldFail;
|
|
}
|
|
|
|
protected JdbTest () {
|
|
this.debuggeeShouldFail = false;
|
|
}
|
|
|
|
abstract protected void runCases();
|
|
|
|
protected boolean shouldPass() {
|
|
return false;
|
|
}
|
|
|
|
protected void failure(String errMessage) {
|
|
success = false;
|
|
log.complain(errMessage);
|
|
}
|
|
|
|
protected void display(String message) {
|
|
log.display(message);
|
|
}
|
|
|
|
protected void launchJdbAndDebuggee(String debuggeeClass) throws Exception {
|
|
launcher = new Launcher(argumentHandler, log);
|
|
launcher.launchJdbAndDebuggee(debuggeeClass);
|
|
jdb = launcher.getJdb();
|
|
|
|
if (jdb == null) {
|
|
throw new Failure("jdb object points to null");
|
|
}
|
|
if (debuggeeClass != null) {
|
|
if (jdb.terminated()) {
|
|
throw new Failure("jdb exited before testing with code " + jdb.waitFor());
|
|
}
|
|
|
|
if (argumentHandler.isAttachingConnector() || argumentHandler.isListeningConnector()) {
|
|
debuggee = launcher.getDebuggee();
|
|
|
|
if (debuggee.terminated()) {
|
|
throw new Failure("Debuggee exited before testing");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
protected void initJdb() {
|
|
String[] reply;
|
|
|
|
jdb.setCompoundPromptIdent(compoundPromptIdent);
|
|
|
|
// wait for prompts after connection established
|
|
if (argumentHandler.isAttachingConnector() || argumentHandler.isListeningConnector()) {
|
|
// wait for two prompts (after connection established and VM_INIT received)
|
|
jdb.waitForPrompt(0, false, 2);
|
|
} else if (argumentHandler.isLaunchingConnector()) {
|
|
// wait for one prompt (after connection established)
|
|
jdb.waitForPrompt(0, false);
|
|
} else {
|
|
throw new TestBug("Unexpected connector kind: " + argumentHandler.getConnectorType());
|
|
}
|
|
|
|
display("Setting first breakpoint");
|
|
jdb.setDeferredBreakpointInMethod(firstBreak);
|
|
|
|
display("Starting debuggee class");
|
|
jdb.startDebuggeeClass();
|
|
}
|
|
|
|
protected void afterJdbExit() {
|
|
}
|
|
|
|
protected void runTest(String argv[]) {
|
|
PrintStream out = System.out;
|
|
try {
|
|
argumentHandler = new JdbArgumentHandler(argv);
|
|
log = new Log(out, argumentHandler);
|
|
|
|
if (shouldPass()) {
|
|
log.println("TEST PASSED");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
launchJdbAndDebuggee(debuggeeClass);
|
|
|
|
try {
|
|
initJdb();
|
|
|
|
/* START OF TEST CASES */
|
|
display("Test cases starts.");
|
|
|
|
runCases();
|
|
|
|
display("Test cases ends.");
|
|
/* END OF TEST CASES */
|
|
|
|
} catch (DebuggeeUncaughtException ex) {
|
|
jdb.quit();
|
|
throw new TestFailure(ex);
|
|
} catch (Exception e) {
|
|
failure("Caught unexpected exception while executing the test: " + e);
|
|
e.printStackTrace(log.getOutStream());
|
|
} finally {
|
|
display("Waiting for jdb exits");
|
|
int code = jdb.waitFor(argumentHandler.getWaitTime() * 60 * 1000);
|
|
if (code == PASSED) {
|
|
display("jdb normally exited");
|
|
afterJdbExit();
|
|
} else if (code == LocalProcess.PROCESS_IS_ALIVE) {
|
|
failure("jdb did not exit after timeout.");
|
|
if (!jdb.terminated()) {
|
|
display("Sending quit command to jdb.");
|
|
jdb.quit();
|
|
} else {
|
|
throw new TestBug("code PROCESS_IS_ALIVE is returned for terminated jdb");
|
|
}
|
|
} else {
|
|
failure("jdb abnormally exited with code: " + code);
|
|
}
|
|
|
|
try {
|
|
jdb.close();
|
|
} catch (Throwable ex) {
|
|
failure("Caught exception/error while closing jdb streams:\n\t" + ex);
|
|
ex.printStackTrace(log.getOutStream());
|
|
}
|
|
|
|
jdb = null;
|
|
|
|
if (debuggee != null
|
|
&& (argumentHandler.isAttachingConnector()
|
|
|| argumentHandler.isListeningConnector())) {
|
|
display("Waiting for debuggee exits");
|
|
code = debuggee.waitForDebuggee();
|
|
if (debuggeeShouldFail) {
|
|
if (code == JCK_STATUS_BASE + PASSED) {
|
|
failure("Debuggee PASSED with exit code: " + code + " but should fail");
|
|
} else {
|
|
display("Debuggee FAILED as expected with exit code: " + code);
|
|
}
|
|
} else {
|
|
if (code == JCK_STATUS_BASE + PASSED) {
|
|
display("Debuggee PASSED with exit code: " + code);
|
|
} else {
|
|
failure("Debuggee FAILED with exit code: " + code);
|
|
}
|
|
}
|
|
// debuggee = null;
|
|
}
|
|
}
|
|
|
|
} catch (Throwable t) {
|
|
failure("Caught unexpected exception: " + t);
|
|
t.printStackTrace(out);
|
|
} finally {
|
|
if (jdb != null) {
|
|
log.complain("jdb reference is not null, check for exception in the logs.");
|
|
try {
|
|
jdb.close();
|
|
} catch (Throwable ex) {
|
|
failure("Caught exception/error while closing jdb streams:\n\t" + ex);
|
|
ex.printStackTrace(log.getOutStream());
|
|
}
|
|
}
|
|
|
|
if (debuggee != null && !debuggee.terminated()) {
|
|
log.complain("debuggee is still running, check for exception in the logs.");
|
|
debuggee.killDebuggee();
|
|
}
|
|
}
|
|
|
|
if (!success) {
|
|
log.complain("TEST FAILED");
|
|
throw new RuntimeException("TEST FAILED");
|
|
}
|
|
|
|
} catch (Throwable t) {
|
|
out.println("Caught unexpected exception while starting the test: " + t);
|
|
t.printStackTrace(out);
|
|
throw new RuntimeException("TEST FAILED", t);
|
|
}
|
|
out.println("TEST PASSED");
|
|
}
|
|
}
|