Debugging Multiprocess Applications
Since version 0.8 Chutoro Nigiri
, libdebug supports debugging multiprocess applications. This feature allows you to attach to multiple processes and debug them simultaneously. This document explains how to use this feature and provides examples to help you get started.
A Child Process is Born
By default, libdebug will monitor all new children processes created by the tracee process. Of course, it will not retrieve past forked processes that have been created before an attach.
A new process is a big deal. For this reason, libdebug will provide you with a brand new Debugger object for each new child process. This object will be available in the list children
attribute of the parent Debugger object.
Usage Example
from libdebug import debugger
d = debugger("test")
d.run()
[...]
print(f"The process has spawned {len(d.children)} children")
for child in d.children: # (1)!
print(f"Child PID: {child.pid}")
- The
children
attribute is a regular list. Indexing, slicing, and iterating are all supported.
Inherited Properties
When a child process is spawned, it inherits the properties of the parent debugger. This includes whether ASLR is enabled, fast memory reading, and [other properties}../../basics/libdebug101/#what-else-can-i-do). However, the child debugger from that moment on will act independently. As such, any property changes made to the parent debugger will not affect the child debugger, and vice versa.
In terms of registered Stopping Events, the new debugger will be a blank slate. This means the debugger will not inherit breakpoints, watchpoints, syscall handlers, or signal catchers.
Focusing on the Main Process
Some applications may spawn a large number of children processes, and you may only be interested in debugging the main process. In this case, you can disable the automatic monitoring of children processes by setting the follow_children
parameter to False
when creating the Debugger object.
Usage Example
In this example, libdebug will only monitor the main process and ignore any child processes spawned by the tracee.However, you can also decide to stop monitoring child processes at any time during debugging by setting the follow_children
attribute to False
in a certain Debugger object.
Snapshot Behavior
When creating a snapshot of a process from the corresponding Debugger object, the snapshot will not include children processes, but only children threads. Read more about snapshots in the Save States section.
Pipe Redirection
By default, libdebug will redirect the standard input, output, and error of the child processes to pipes. This is how you can interact with these file descriptors using I/O commands. If you keep this parameter enabled, you will be able to interact with the child processes's standard I/O using the same PipeManager object that is provided upon creation of the root Debugger object. This is consistent with limitations of forking in the POSIX standard, where the child process inherits the file descriptors of the parent process.
Read more about disabling pipe redirection in the dedicated section.