How do I terminate a process tree?
Asked Answered
D

7

19

I am using Runtime.getRuntime().exec() command to start a batch file which in turn starts another process for windows platform.

javaw.exe(Process1)
 |___xyz.bat(Process2)
        |___javaw.exe(Process3)

Runtime.getRuntime().exec() returns a Process object which has a destroy method, but when I use destroy(), it kills only the xyz.bat and leaves the batch file's sub-process dangling.

Is there a clean way in to destroy the process tree starting with the batch process as root?

For the record I cannot use any custom libraries to get rid of the batch file to by-pass the issue.

Drainage answered 12/4, 2012 at 13:12 Comment(1)
Can I ask why the no custom library requirement is in place? In my experience, such requirements usually have a very poor reason for existing, and may be negotiable with explanation of the reason a library is needed (in this case, a required feature that is missing from the Java platform, namely a way of enumerating the subprocesses of a parent process).Novel
N
24

This is not possible using the standard Java API (see edit at end of the post for an update that changes this). You will need some native code of some variety. Using JNA, I've used code that looks like this:

public class Win32Process
{
    WinNT.HANDLE handle;
    int pid;

    Win32Process (int pid) throws IOException
    {
        handle = Kernel32.INSTANCE.OpenProcess ( 
                0x0400| /* PROCESS_QUERY_INFORMATION */
                0x0800| /* PROCESS_SUSPEND_RESUME */
                0x0001| /* PROCESS_TERMINATE */
                0x00100000 /* SYNCHRONIZE */,
                false,
                pid);
        if (handle == null) 
            throw new IOException ("OpenProcess failed: " + 
                    Kernel32Util.formatMessageFromLastErrorCode (Kernel32.INSTANCE.GetLastError ()));
        this.pid = pid;
    }

    @Override
    protected void finalize () throws Throwable
    {
        Kernel32.INSTANCE.CloseHandle (handle);
    }

    public void terminate ()
    {
        Kernel32.INSTANCE.TerminateProcess (handle, 0);
    }

    public List<Win32Process> getChildren () throws IOException
    {
        ArrayList<Win32Process> result = new ArrayList<Win32Process> ();
        WinNT.HANDLE hSnap = KernelExtra.INSTANCE.CreateToolhelp32Snapshot (KernelExtra.TH32CS_SNAPPROCESS, new DWORD(0));
        KernelExtra.PROCESSENTRY32.ByReference ent = new KernelExtra.PROCESSENTRY32.ByReference ();
        if (!KernelExtra.INSTANCE.Process32First (hSnap, ent)) return result;
        do {
            if (ent.th32ParentProcessID.intValue () == pid) result.add (new Win32Process (ent.th32ProcessID.intValue ()));
        } while (KernelExtra.INSTANCE.Process32Next (hSnap, ent));
        Kernel32.INSTANCE.CloseHandle (hSnap);
        return result;
    }

}

This code uses the following JNA declarations that are not included in the standard JNA library:

public interface KernelExtra extends StdCallLibrary {

    /**
     * Includes all heaps of the process specified in th32ProcessID in the snapshot. To enumerate the heaps, see
     * Heap32ListFirst.
     */
    WinDef.DWORD TH32CS_SNAPHEAPLIST = new WinDef.DWORD(0x00000001);

    /**
     * Includes all processes in the system in the snapshot. To enumerate the processes, see Process32First.
     */
    WinDef.DWORD TH32CS_SNAPPROCESS  = new WinDef.DWORD(0x00000002);

    /**
     * Includes all threads in the system in the snapshot. To enumerate the threads, see Thread32First.
     */
    WinDef.DWORD TH32CS_SNAPTHREAD   = new WinDef.DWORD(0x00000004);

    /**
     * Includes all modules of the process specified in th32ProcessID in the snapshot. To enumerate the modules, see
     * Module32First. If the function fails with ERROR_BAD_LENGTH, retry the function until it succeeds.
     */
    WinDef.DWORD TH32CS_SNAPMODULE   = new WinDef.DWORD(0x00000008);

    /**
     * Includes all 32-bit modules of the process specified in th32ProcessID in the snapshot when called from a 64-bit
     * process. This flag can be combined with TH32CS_SNAPMODULE or TH32CS_SNAPALL. If the function fails with
     * ERROR_BAD_LENGTH, retry the function until it succeeds.
     */
    WinDef.DWORD TH32CS_SNAPMODULE32 = new WinDef.DWORD(0x00000010);

    /**
     * Includes all processes and threads in the system, plus the heaps and modules of the process specified in th32ProcessID.
     */
    WinDef.DWORD TH32CS_SNAPALL      = new WinDef.DWORD((TH32CS_SNAPHEAPLIST.intValue() |
            TH32CS_SNAPPROCESS.intValue() | TH32CS_SNAPTHREAD.intValue() | TH32CS_SNAPMODULE.intValue()));

    /**
     * Indicates that the snapshot handle is to be inheritable.
     */
    WinDef.DWORD TH32CS_INHERIT      = new WinDef.DWORD(0x80000000);

    /**
     * Describes an entry from a list of the processes residing in the system address space when a snapshot was taken.
     */
    public static class PROCESSENTRY32 extends Structure {

        public static class ByReference extends PROCESSENTRY32 implements Structure.ByReference {
            public ByReference() {
            }

            public ByReference(Pointer memory) {
                super(memory);
            }
        }

        public PROCESSENTRY32() {
            dwSize = new WinDef.DWORD(size());
        }

        public PROCESSENTRY32(Pointer memory) {
            useMemory(memory);
            read();
        }

        /**
         * The size of the structure, in bytes. Before calling the Process32First function, set this member to
         * sizeof(PROCESSENTRY32). If you do not initialize dwSize, Process32First fails.
         */
        public WinDef.DWORD dwSize;

        /**
         * This member is no longer used and is always set to zero.
         */
        public WinDef.DWORD cntUsage;

        /**
         * The process identifier.
         */
        public WinDef.DWORD th32ProcessID;

        /**
         * This member is no longer used and is always set to zero.
         */
        public BaseTSD.ULONG_PTR th32DefaultHeapID;

        /**
         * This member is no longer used and is always set to zero.
         */
        public WinDef.DWORD th32ModuleID;

        /**
         * The number of execution threads started by the process.
         */
        public WinDef.DWORD cntThreads;

        /**
         * The identifier of the process that created this process (its parent process).
         */
        public WinDef.DWORD th32ParentProcessID;

        /**
         * The base priority of any threads created by this process.
         */
        public WinDef.LONG pcPriClassBase;

        /**
         * This member is no longer used, and is always set to zero.
         */
        public WinDef.DWORD dwFlags;

        /**
         * The name of the executable file for the process. To retrieve the full path to the executable file, call the
         * Module32First function and check the szExePath member of the MODULEENTRY32 structure that is returned.
         * However, if the calling process is a 32-bit process, you must call the QueryFullProcessImageName function to
         * retrieve the full path of the executable file for a 64-bit process.
         */
        public char[] szExeFile = new char[WinDef.MAX_PATH];
    }


    // the following methods are in kernel32.dll, but not declared there in the current version of Kernel32:

    /**
     * Takes a snapshot of the specified processes, as well as the heaps, modules, and threads used by these processes.
     *  
     * @param dwFlags
     *   The portions of the system to be included in the snapshot.
     * 
     * @param th32ProcessID
     *   The process identifier of the process to be included in the snapshot. This parameter can be zero to indicate
     *   the current process. This parameter is used when the TH32CS_SNAPHEAPLIST, TH32CS_SNAPMODULE,
     *   TH32CS_SNAPMODULE32, or TH32CS_SNAPALL value is specified. Otherwise, it is ignored and all processes are
     *   included in the snapshot.
     *
     *   If the specified process is the Idle process or one of the CSRSS processes, this function fails and the last
     *   error code is ERROR_ACCESS_DENIED because their access restrictions prevent user-level code from opening them.
     *
     *   If the specified process is a 64-bit process and the caller is a 32-bit process, this function fails and the
     *   last error code is ERROR_PARTIAL_COPY (299).
     *
     * @return
     *   If the function succeeds, it returns an open handle to the specified snapshot.
     *
     *   If the function fails, it returns INVALID_HANDLE_VALUE. To get extended error information, call GetLastError.
     *   Possible error codes include ERROR_BAD_LENGTH.
     */
    public WinNT.HANDLE CreateToolhelp32Snapshot(WinDef.DWORD dwFlags, WinDef.DWORD th32ProcessID);

    /**
     * Retrieves information about the first process encountered in a system snapshot.
     *
     * @param hSnapshot A handle to the snapshot returned from a previous call to the CreateToolhelp32Snapshot function.
     * @param lppe A pointer to a PROCESSENTRY32 structure. It contains process information such as the name of the
     *   executable file, the process identifier, and the process identifier of the parent process.
     * @return
     *   Returns TRUE if the first entry of the process list has been copied to the buffer or FALSE otherwise. The
     *   ERROR_NO_MORE_FILES error value is returned by the GetLastError function if no processes exist or the snapshot
     *   does not contain process information.
     */
    public boolean Process32First(WinNT.HANDLE hSnapshot, KernelExtra.PROCESSENTRY32.ByReference lppe);

    /**
     * Retrieves information about the next process recorded in a system snapshot.
     *
     * @param hSnapshot A handle to the snapshot returned from a previous call to the CreateToolhelp32Snapshot function.
     * @param lppe A pointer to a PROCESSENTRY32 structure.
     * @return
     *   Returns TRUE if the next entry of the process list has been copied to the buffer or FALSE otherwise. The
     *   ERROR_NO_MORE_FILES error value is returned by the GetLastError function if no processes exist or the snapshot
     *   does not contain process information.
     */
    public boolean Process32Next(WinNT.HANDLE hSnapshot, KernelExtra.PROCESSENTRY32.ByReference lppe);


}

You can then use the 'getChildren()' method to get a list of children, terminate the parent, and then recursively terminate the children.

I believe you can extra the PID of a java.lang.Process using reflection (I haven't done this, however; I switched to creating the processes myself using the Win32 API so that I would have more control over it).

So putting it together, you'd need something like:

int pid = (some code to extract PID from the process you want to kill);
Win32Process process = new Win32Process(pid);
kill(process);

public void kill(Win32Process target) throws IOException
{
   List<Win32Process> children = target.getChildren ();
   target.terminateProcess ();
   for (Win32Process child : children) kill(child);
}

Edit

It turns out that this particular shortcoming of the Java API is being fixed in Java 9. See the preview of the Java 9 documentation here (if the correct page doesn't load, you need to look at the java.lang.ProcessHandle interface). For the requirement of the question above, the code would now look something like this:

Process child = ...;
kill (child.toHandle());

public void kill (ProcessHandle handle)
{
    handle.descendants().forEach((child) -> kill(child));
    handle.destroy();
}

(Note that this isn't tested - I haven't switched to Java 9 yet, but am actively reading about it)

Novel answered 12/4, 2012 at 13:30 Comment(4)
The "Edit" section worked for me on Java 11, I almost missed it though looking at all the JNI code. Maybe this block of code should be at the top now that Java 11 is currently the LTS release.Linstock
Java9 Documentation link in above post is not available now, Here is updated link. docs.oracle.com/javase/9/docs/api/java/lang/ProcessHandle.htmlTopi
The java.lang.ProcessHandle approach looks like clean Java. It works on Linux, too.Beason
@Novel - Could you put your edit on top ? Its been a lifesaver and I'm afraid by not scrolling to the end, many might miss the solutionAmadis
R
2

You cannot do it with the standard Java API's for Java 8 or lower.

Method 1

If you have Java 9 or above, you can use ProcessHandle

Method 2

Use taskkill with the /t and /f flag with Runtime.getRuntime().exec() or ProcessBuilder class in Java.

/f --> to force kill /t --> to kill all the child processes generated by this process.

ProcessBuilder pb1 = new ProcessBuilder("cmd.exe","/c","taskkill /f /t /pid "+p.pid());
Process p1 = pb1.start();

Detailed Analysis

I had used process builder to do it inside my Java code.

I had reproduced the problem statement. This is a sample batch file that starts another Powershell process in windows to print counter from 1 to 20 each second, which is the child process:

@echo off
echo starting
powershell -command " for ($count=1;$count -le 20;$count++) { Start-Sleep 1; Write-Output $count }"
echo success

ps: you'll need to redirect InputStream to stdout to obtain output on terminal.

Using destroy() method of process object kills only that process and not the child/grandchild processes.

You can run tasklist on terminal and locate powershell.exe to compare before and after you run the java program to see the child processes still running.

To kill the child processes along with parent process, start a new process using ProcessBuilder as below where p is the process that you want to get rid of:

ProcessBuilder pb1 = new ProcessBuilder("cmd.exe","/c","taskkill /f /t /pid "+p.pid());
Process p1 = pb1.start();
Rhiannonrhianon answered 13/7, 2021 at 8:30 Comment(0)
N
1

An alternative solution, if you control the child process as well as the batch file, would be to have the child process create a thread, open a ServerSocket, listen for a connection to it, and call System.exit() if it receives a correct password on it.

There may be complications if you need multiple simultaneous instances; at that point you would need some way of allocating port numbers to them.

Novel answered 12/4, 2012 at 13:47 Comment(0)
C
0

You cannot kill a process tree for windows using JDK. You need to rely on WinAPI.You'll have to resort to native commands or JNI libraries, all of which are platform-dependent and more complex than a pure Java solution would be.

A sample link JNI Example

Clamper answered 12/4, 2012 at 13:17 Comment(6)
Unfortunately i cannot use any external or custom libraries.However, i can change the batch file.Is there a way to catch a term signal in the batch sent by Process.destroy() from Java? and then use it to kill the sub-process?Drainage
You could manage processes using the batch file. Please go through the link for several options available to manage. robvanderwoude.com/processes.phpClamper
I don't believe there is any way of making a Windows batch file catch such a signal -- Java natively uses TerminateProcess, which kills the process directly without sending a signal to it first. Besides, if a child process is currently running, the batch file would have to wait for it to exit before it could do anything else, and there's no way for Java to terminate the child process. You will have to use external libraries if you want to do this.Novel
@Novel We are not trying to catch signal of the process from .bat file, rather we query using the one of the built in commands available and kill the process, which is easy way to go at it provided .bat file can be modified otherwise we need to rely on JNA which requires a lot more effor.Clamper
My understanding of the question is that Process3 is a long-running process that has to be terminated at Process1's request. This leaves two problems with the approach you're suggesting: first, Process1 will need to communicate its intent to terminate to the batch file somehow, which seems problematic. And then the batch file will need to terminate Process3, which is complicated by the fact that as far as I can see, there is no easy way for it to identify the PID of Process3 (considering the possibility, for example, of having multiple instances of the same process).Novel
@Novel You understood right. So, what I'm saying is let java control the process of .bat file but .bat file will make sure the processes created by itself.Clamper
C
0

Here is another option. Use this powershell script to execute your bat script. When you want to kill the tree, terminate your powershell script's process and it will execute taskkill on it's subprocess automatically. I have it calling taskkill twice because in some cases it doesn't take on the first try.

Param(
    [string]$path
)

$p = [Diagnostics.Process]::Start("$path").Id

try {
    while($true) {
        sleep 100000
    }
} finally {
    taskkill /pid $p
    taskkill /pid $p
}
Cocks answered 29/10, 2013 at 2:15 Comment(0)
D
0

With java 9, killing the main process kills the whole process tree. You could do something like this:

Process ptree = Runtime.getRuntime().exec("cmd.exe","/c","xyz.bat");
// wait logic
ptree.destroy();

Please take a look at this blog and check out the Deal with Process Trees example.

Derouen answered 15/7, 2016 at 7:56 Comment(0)
H
0

From Java 9 onwards Process.descendants() (also available on ProcessHandle) returns a stream of descendant process handles recursively, so just destroy each and then destroy the root process (or process handle).

process.descendants().forEach(ProcessHandle::destroy);
process.destroy();
Harhay answered 16/7, 2024 at 15:58 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.