e1e758a7b4
Reviewed-by: coleenp, dholmes
175 lines
4.9 KiB
Java
175 lines
4.9 KiB
Java
/*
|
|
* Copyright (c) 2002, 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;
|
|
|
|
import nsk.share.*;
|
|
|
|
import java.io.*;
|
|
|
|
/**
|
|
* Wrapper for local process.
|
|
* <p>
|
|
* This class provides abilities to launch such process,
|
|
* redirect standard output streams, wait for process terminates
|
|
* or kill the process, and so on.
|
|
* <p>
|
|
* This object is finalized with <code>nsk.share.Finalizer</code>.
|
|
*
|
|
* @see nsk.share.FinalizableObject
|
|
* @see nsk.share.Finalizer
|
|
*/
|
|
|
|
public class LocalProcess extends FinalizableObject {
|
|
|
|
public final static int PROCESS_IS_ALIVE = 222;
|
|
|
|
private Process process;
|
|
|
|
protected Process getProcess() {
|
|
return process;
|
|
}
|
|
|
|
public void launch (String[] args) throws IOException {
|
|
System.out.println("Launching process by array of args: ");
|
|
for (int mm=0; mm < args.length; mm++) {
|
|
System.out.println(" args[" + Integer.toString(mm) + "]: >" +
|
|
args[mm] + "<");
|
|
|
|
}
|
|
|
|
process = Runtime.getRuntime().exec(args);
|
|
|
|
registerCleanup();
|
|
}
|
|
|
|
public void launch (String cmdLine) throws IOException {
|
|
System.out.println("Launching process by command line: " + cmdLine);
|
|
|
|
process = Runtime.getRuntime().exec(cmdLine);
|
|
|
|
registerCleanup();
|
|
}
|
|
|
|
/** Return exit status. */
|
|
public int getStatus () {
|
|
return process.exitValue();
|
|
}
|
|
|
|
/** Check whether the process has been terminated. */
|
|
public boolean terminated() {
|
|
try {
|
|
int value = process.exitValue();
|
|
return true;
|
|
} catch (IllegalThreadStateException e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/** Wait until the process shutdown or crash. */
|
|
public int waitFor () throws InterruptedException {
|
|
return process.waitFor();
|
|
}
|
|
|
|
/**
|
|
* Wait until the process shutdown or crash for given timeout in milliseconds.
|
|
* Returns <code>LocalProcess.PROCESS_IS_ALIVE</code> if process is not terminated
|
|
* after timeout.
|
|
*/
|
|
|
|
public int waitFor (long timeMillisec) throws InterruptedException {
|
|
final Object waitObject = new Object();
|
|
|
|
class Watcher extends Thread {
|
|
int exitCode = LocalProcess.PROCESS_IS_ALIVE;
|
|
Process process;
|
|
|
|
Watcher (Process process) {
|
|
this.process = process;
|
|
}
|
|
|
|
public void run () {
|
|
try {
|
|
synchronized (this) {
|
|
exitCode = process.waitFor();
|
|
}
|
|
} catch (InterruptedException ie) {
|
|
}
|
|
synchronized (waitObject) {
|
|
waitObject.notifyAll();
|
|
}
|
|
}
|
|
|
|
synchronized public int getExitCode() {
|
|
return exitCode;
|
|
}
|
|
}
|
|
|
|
Watcher watcher;
|
|
// yield control to watcher for timeMillisec time.
|
|
synchronized (waitObject) {
|
|
watcher = new Watcher(process);
|
|
watcher.start();
|
|
|
|
waitObject.wait(timeMillisec);
|
|
}
|
|
|
|
if (watcher.isAlive()) {
|
|
watcher.interrupt();
|
|
}
|
|
|
|
return watcher.getExitCode();
|
|
}
|
|
|
|
// --------------------------------------------------- //
|
|
|
|
/** Get a pipe to write to the process' stdin stream. */
|
|
public OutputStream getStdin () {
|
|
return process.getOutputStream();
|
|
}
|
|
|
|
/** Get a pipe to read the process' stdout stream. */
|
|
public InputStream getStdout () {
|
|
return process.getInputStream();
|
|
}
|
|
|
|
/** Get a pipe to read the process stderr stream. */
|
|
public InputStream getStderr () {
|
|
return process.getErrorStream();
|
|
}
|
|
|
|
/** Kill the process. */
|
|
protected void kill() {
|
|
process.destroy();
|
|
}
|
|
|
|
/**
|
|
* This method is called at finalization and calls <code>kill()</code>.
|
|
*
|
|
*/
|
|
@Override
|
|
public void cleanup() {
|
|
kill();
|
|
}
|
|
}
|