470 lines
16 KiB
Java

/*
* Copyright (c) 2001, 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.
*/
package nsk.share.jpda;
import jdk.test.lib.JDWP;
import nsk.share.*;
import java.io.*;
import java.net.*;
import java.util.function.Consumer;
/**
* This class is used to control debugee VM process.
* <p>
* Object of this class is constructed by <code>DebugeeBinder</code>
* and is used as a mirror of debugee VM process.
* It provides abilities to launch such process,
* redirect standard output streams, wait for process terminates
* or kill the process, and so on.
* <p>
* This is an abstract class that declares abstract methods to control
* debugee VM process.
* Derived classes should implement these methods corresponding to the mode
* that the process should be started in (locally, remotely or manually).
* <p>
* Particular derived classes <code>nsk.share.jdi.Debugee</code> and
* <code>nsk.share.jdwp.Debugee</code> provides additional abilities
* to control debugee VM using JDI or JDWP specific features.
*
* @see DebugeeBinder
*
* @see nsk.share.jdi.Debugee
* @see nsk.share.jdwp.Debugee
*/
abstract public class DebugeeProcess extends FinalizableObject {
/** Default prefix for log messages. */
public static final String LOG_PREFIX = "binder> ";
public static final String DEBUGEE_STDOUT_LOG_PREFIX = "debugee.stdout> ";
public static final String DEBUGEE_STDERR_LOG_PREFIX = "debugee.stderr> ";
/** Messages prefix. */
protected String prefix = LOG_PREFIX;
/** Binder that creates this debugee process. */
protected DebugeeBinder binder = null;
/** Messages log from binder. */
protected Log log = null;
/** Communicational channel between debuger and debugee. */
protected IOPipe pipe = null;
/** Argument handler from binder. */
protected DebugeeArgumentHandler argumentHandler = null;
/** Need or not to check debuggee process termination at exit. */
protected boolean checkTermination = false;
/** Debugee VM process or <i>null</i> if not available. */
protected Process process = null;
/** Make new <code>Debugee</code> object for the given binder. */
protected DebugeeProcess (DebugeeBinder binder) {
this.binder = binder;
this.log = binder.getLog();
// Register the cleanup() method to be called when this instance becomes unreachable.
registerCleanup();
}
/**
* Return already bound ServerSocket for IOPipe connection or null.
*/
protected ServerSocket getPipeServerSocket() {
return binder.getPipeServerSocket();
}
/** Return <code>DebugeeArgumentHandler</code> of the debugee object. */
public DebugeeArgumentHandler getArgumentHandler() {
return binder.getArgumentHandler();
}
/** Return <code>Log</code> of the debugee object. */
public Log getLog() {
return log;
}
/** Return <code>Process</code> object associated with debugee VM or <i>null</i>. */
public Process getProcess() {
return process;
}
// --------------------------------------------------- //
/** Create and return new IOPipe channel to the debuggee VM.
* The channel should be created before debuggee starts execution,
* i.e. the method assumes debuggee is started, but suspended before
* its main class is loaded.
*/
public IOPipe createIOPipe() {
if (pipe != null) {
throw new TestBug("IOPipe channel is already created");
}
pipe = IOPipe.startDebuggerPipe(binder);
return pipe;
}
/** Return IOPipe channel or null if not yet ctreated. */
public IOPipe getIOPipe() {
return pipe;
}
/** Receive and return signal from the debugee VM via IOPipe channel.
*
* @throws TestBug if IOPipe channel is not created
*/
public String receiveSignal() {
if ( pipe == null )
throw new TestBug("IOPipe channel is not initialized");
return pipe.readln();
}
/** Receive an expected <code>signal</code> from the debugee VM via IOPipe channel.
*
* @throws Failure if received signal is not equal to the expected one
* @throws TestBug if IOPipe channel is not created
*/
public void receiveExpectedSignal(String signal) {
String line = receiveSignal();
if (line == null || !line.equals(signal))
throw new Failure("Received unexpected signal from debugee: " + line);
display("Received expected signal from debugee: " + signal);
}
/** Send <code>signal</code> to the debugee VM via IOPipe channel.
*
* @throws TestBug if IOPipe channel is not defined created.
*/
public void sendSignal(String signal) {
if (pipe == null)
throw new TestBug("IOPipe channel is not initialized");
pipe.println(signal);
}
// --------------------------------------------------- //
/** Wait until the debugee VM shutdown or crash. */
abstract protected int waitForDebugee () throws InterruptedException;
/** Kill the debugee VM. */
abstract protected void killDebugee ();
/** Check whether the debugee VM has been terminated. */
abstract public boolean terminated ();
/** Return the debugee VM exit status. */
abstract public int getStatus ();
/** Get a pipe to write to the debugee's stdin stream. */
abstract protected OutputStream getInPipe ();
/** Get a pipe to read the debugee's stdout stream. */
abstract protected InputStream getOutPipe ();
/** Get a pipe to read the debugee's stderr stream. */
abstract protected InputStream getErrPipe ();
// --------------------------------------------------- //
/**
* Wait until the debugee VM shutdown or crash,
* and let finish its stdout, stderr, and stdin
* redirectors (if any).
*
* @return Debugee process exit code.
* @see #waitForRedirectors(long)
*/
public int waitFor () {
long timeout = binder.getArgumentHandler().getWaitTime() * 60 * 1000;
int exitCode = 0;
try {
exitCode = waitForDebugee();
} catch (InterruptedException ie) {
ie.printStackTrace(log.getOutStream());
throw new Failure("Caught exception while waiting for debuggee process: \n\t" + ie);
}
waitForRedirectors(timeout);
if (process != null) {
process.destroy();
}
return exitCode;
}
/**
* Wait until the debugee VM redirectors to complete for specified <code>timeout</code>.
*
* @see #waitFor()
*/
public void waitForRedirectors (long timeout) {
try {
if (stdinRedirector != null) {
if (stdinRedirector.isAlive()) {
stdinRedirector.join(timeout);
if (stdinRedirector.isAlive()) {
log.complain("Timeout for waiting STDIN redirector exceeded");
stdinRedirector.interrupt();
}
}
stdinRedirector = null;
};
if (stdoutRedirector != null) {
if (stdoutRedirector.isAlive()) {
stdoutRedirector.join(timeout);
if (stdoutRedirector.isAlive()) {
log.complain("Timeout for waiting STDOUT redirector exceeded");
stdoutRedirector.interrupt();
}
}
stdoutRedirector = null;
};
if (stderrRedirector != null) {
if (stderrRedirector.isAlive()) {
stderrRedirector.join(timeout);
if (stderrRedirector.isAlive()) {
log.complain("Timeout for waiting STDERR redirector exceeded");
stderrRedirector.interrupt();
}
}
stderrRedirector = null;
};
} catch (InterruptedException ie) {
ie.printStackTrace(log.getOutStream());
throw new Failure("Caught exception while waiting for debuggee output redirectors: \n\t"
+ ie);
}
}
// --------------------------------------------------- //
/**
* Get a pipe to write to the debugee's stdin stream,
* or throw TestBug exception is redirected.
*/
final public OutputStream getStdin () {
if (stdinRedirector != null)
throw new TestBug("Debugee's stdin is redirected");
return getInPipe();
}
/**
* Get a pipe to read the debugee's stdout stream,
* or throw TestBug exception is redirected.
*/
final public InputStream getStdout () {
if (stdoutRedirector != null)
throw new TestBug("Debugee's stdout is redirected");
return getOutPipe();
}
/**
* Get a pipe to read the debugee's stderr stream,
* or throw TestBug exception is redirected.
*/
final public InputStream getStderr () {
if (stderrRedirector != null)
throw new TestBug("Debugee's stderr is redirected");
return getErrPipe();
}
// --------------------------------------------------- //
private IORedirector stdoutRedirector = null;
private IORedirector stderrRedirector = null;
private IORedirector stdinRedirector = null;
// /**
// * Start a thread redirecting the given <code>in</code> stream
// * to the debugee's stdin. If the debugee's stdin was already
// * redirected, the TestBug exception is thrown.
// */
// final public void redirectStdin(InputStream in) {
// if (stdinRedirector != null)
// throw new TestBug("the debugee's stdin is already redirected");
// stdinRedirector = new IORedirector(in,getInPipe());
// stdinRedirector.setName("IORedirector for stdin");
// stdinRedirector.setDaemon(true);
// stdinRedirector.start();
// }
/**
* Start thread redirecting the debugee's stdout to the
* given <code>out</code> stream.
*
* @deprecated Use redirectStdout(Log, String) instead.
*/
@Deprecated
public void redirectStdout(OutputStream out) {
if (stdoutRedirector != null) {
return;
}
stdoutRedirector = new IORedirector(getOutPipe(), out, DEBUGEE_STDOUT_LOG_PREFIX);
stdoutRedirector.setName("IORedirector for stdout");
stdoutRedirector.setDaemon(true);
stdoutRedirector.start();
}
/**
* Start thread redirecting the debugee's stdout to the
* given <code>Log</code>.
*/
public void redirectStdout(Log log, String prefix) {
redirectStdout(log, prefix, null);
}
private void redirectStdout(Log log, String prefix, Consumer<String> stdoutProcessor) {
if (stdoutRedirector != null) {
return;
}
stdoutRedirector = new IORedirector(new BufferedReader(new InputStreamReader(getOutPipe())), log, prefix);
if (stdoutProcessor != null) {
stdoutRedirector.addProcessor(stdoutProcessor);
}
stdoutRedirector.setName("IORedirector for stdout");
stdoutRedirector.setDaemon(true);
stdoutRedirector.start();
}
/**
* Start thread redirecting the debugee's stderr to the
* given <code>err</code> stream.
*
* @deprecated Use redirectStderr(Log, String) instead.
*/
@Deprecated
public void redirectStderr(OutputStream err) {
if (stderrRedirector != null) {
return;
}
stderrRedirector = new IORedirector(getErrPipe(), err, DEBUGEE_STDERR_LOG_PREFIX);
stdoutRedirector.setName("IORedirector for stderr");
stderrRedirector.setDaemon(true);
stderrRedirector.start();
}
/**
* Start thread redirecting the debugee's stderr to the
* given <code>Log</code>.
*/
public void redirectStderr(Log log, String prefix) {
if (stderrRedirector != null) {
return;
}
stderrRedirector = new IORedirector(new BufferedReader(new InputStreamReader(getErrPipe())), log, prefix);
stdoutRedirector.setName("IORedirector for stderr");
stderrRedirector.setDaemon(true);
stderrRedirector.start();
}
/**
* Start thread redirecting the debugee's stdout/stderr to the
* given <code>Log</code> using standard prefixes.
*/
public void redirectOutput(Log log) {
redirectStdout(log, DEBUGEE_STDOUT_LOG_PREFIX);
redirectStderr(log, DEBUGEE_STDERR_LOG_PREFIX);
}
/**
* Starts redirecting of the debugee's stdout/stderr to the
* given <code>Log</code> using standard prefixes
* and detects listening address from the debuggee stdout.
*/
public JDWP.ListenAddress redirectOutputAndDetectListeningAddress(Log log) {
JDWP.ListenAddress listenAddress[] = new JDWP.ListenAddress[1];
Consumer<String> stdoutProcessor = line -> {
JDWP.ListenAddress addr = JDWP.parseListenAddress(line);
if (addr != null) {
synchronized (listenAddress) {
listenAddress[0] = addr;
listenAddress.notifyAll();
}
}
};
redirectStdout(log, DEBUGEE_STDOUT_LOG_PREFIX, stdoutProcessor);
redirectStderr(log, DEBUGEE_STDERR_LOG_PREFIX);
synchronized (listenAddress) {
while (!terminated() && listenAddress[0] == null) {
try {
listenAddress.wait(500);
} catch (InterruptedException e) {
// ignore
}
}
}
if (terminated()) {
throw new Failure("Failed to detect debuggee listening port");
}
log.display("Debuggee is listening on port " + listenAddress[0].address());
return listenAddress[0];
}
// --------------------------------------------------- //
/**
* Kill the debugee VM if it is not terminated yet.
*
* @throws Throwable if any throwable exception is thrown during finalization
*/
public void close() {
if (checkTermination) {
if (!terminated()) {
complain("Debugee VM has not exited correctly: trying to kill it");
killDebugee();
}
checkTermination = false;
}
}
// --------------------------------------------------- //
/**
* Display log message with prefix.
*/
protected void display(String message) {
log.display(prefix + message);
}
/**
* Complain about error with specified message.
*/
protected void complain(String message) {
log.complain(prefix + message);
}
/**
* Finalize debuggee VM wrapper by invoking <code>close()</code>.
*
* @throws Throwable if any throwable exception is thrown during finalization
*/
public void cleanup() {
close();
}
}