libcosmos
Linux C++ System Programming Library
|
Thin wrapper class around the ptrace()
system call.
More...
#include <Tracee.hxx>
Public Types | |
enum class | RestartMode { CONT = to_integral(ptrace::Request::CONT) , DETACH = to_integral(ptrace::Request::DETACH) , SYSCALL = to_integral(ptrace::Request::SYSCALL) , SINGLESTEP = to_integral(ptrace::Request::SINGLESTEP) , LISTEN = to_integral(ptrace::Request::LISTEN) } |
Different ways to restart a tracee. More... | |
Public Member Functions | |
Tracee (const ProcessID pid=ProcessID::INVALID) | |
ProcessID | pid () const |
bool | valid () const |
void | seize (const ptrace::Opts opts) |
Seize a tracee. | |
void | attach () |
Attach to a process, making it a tracee. | |
void | detach () |
Detach from and restart the tracee. | |
void | restart (const RestartMode mode, const std::optional< Signal > signal={}) |
Continues a traced process, optionally delivering signal . | |
void | interrupt () |
Interrupt the tracee. | |
void | setOptions (const ptrace::Opts opts) |
Set tracing options for the given tracee. | |
long | peekData (const long *addr) const |
Read one word of data from the tracee's memory. | |
void | pokeData (const long *addr, long value) |
Write one word of data into the tracee's memory. | |
long | peekUser (const long *offset) const |
Read one word of data from the tracee's user area. | |
void | pokeUser (const long *offset, long value) |
Change one word of data in the tracee's user area. | |
void | getRegisters (struct user_regs_struct &out) const |
Copy the tracee's general purpose registers into the provided structure. | |
void | setRegisters (const struct user_regs_struct &out) |
Modify the tracee's general purpose registers. | |
void | getFloatRegisters (struct user_fpregs_struct &out) const |
Copy the tracee's floating point registers into the provided structure. | |
void | setFloatRegisters (const struct user_fpregs_struct &out) |
Modify the tracee's floating point registers. | |
void | getRegisterSet (const ptrace::RegisterType type, InputMemoryRegion &iovec) const |
Retrieve a set of registers from the tracee. | |
void | setRegisterSet (const ptrace::RegisterType type, OutputMemoryRegion &iovec) |
Modify a set of registers in the tracee. | |
void | getSigInfo (SigInfo &info) const |
Obtain information about the signal that caused the stop. | |
void | setSigInfo (const SigInfo &info) |
Set signal information for the tracee. | |
std::vector< SigInfo > | peekSigInfo (const ptrace::PeekSigInfo &settings) |
Obtains SigInfo structures pending for the tracee. | |
void | getSigMask (SigSet &set) const |
Obtain the tracee's mask of blocked signals. | |
void | setSigMask (const SigSet &set) |
Change the tracee's mask of blocked signals. | |
ProcessID | getPIDEventMsg () const |
Returns the PID of a newly created child of the tracee in the context of a ptrace-event-stop. | |
ExitStatus | getExitEventMsg () const |
Returns the exit code of the tracee in the context of a ptrace-event-stop. | |
uint16_t | getSeccompRetDataEventMsg () const |
Returns the SECCOMP_RET_DATA in the context of a ptrace-event-stop. | |
void | getSeccompFilter (std::vector< struct sock_filter > &instructions, const unsigned long prog_index) const |
Retrieve a classic seccomp BPF program installed in the tracee. | |
void | getSyscallInfo (ptrace::SyscallInfo &info) const |
Returns system call information in the context of the current ptrace stop. | |
Protected Member Functions | |
unsigned long | getEventMsg () const |
Returns the current event message for a ptrace-event-stop. | |
template<typename ADDR = void*, typename DATA = void*> | |
std::optional< long > | request (const ptrace::Request req, ADDR addr=nullptr, DATA data=nullptr) const |
template<typename DATA = void*> | |
std::optional< long > | request (const ptrace::Request req, std::nullptr_t, DATA data=nullptr) const |
Protected Attributes | |
ProcessID | m_pid |
Thin wrapper class around the ptrace()
system call.
This is a type safe wrapper around the ptrace()
system call. An instance of this type always operates on the same process supplied during construction time. There are no resources managed by this class.
The ptrace() API is highly complex and this class can only offer some basic wrappers and documentation about it.
Definition at line 41 of file Tracee.hxx.
|
strong |
Different ways to restart a tracee.
This is a sub-set of the commands from ptrace::Request that deal with restarting the tracee in different ways. All of these requests optionally accept a signal to inject, except for RestartMode::LISTEN.
Definition at line 51 of file Tracee.hxx.
|
inlineexplicit |
Definition at line 65 of file Tracee.hxx.
|
inline |
Attach to a process, making it a tracee.
This is the old method of making a process a tracee. Tracers attached to this way don't support all ptrace() operations and it is not recommended to use this method anymore.
The tracee will be sent a SIGSTOP signal, the tracer needs to wait on the tracee to assert it has entered stop state as a result of the attach operation. The synthetic SIGSTOP event should be suppressed by the tracer.
This method of attaching to the tracee has inherent race conditions. Other signals can concurrently occur while the tracer attempts to attach to it. Events other than SIGSTOP should be reinjected until SIGSTOP is observed. This does not reliably work if SIGSTOP itself is occurring in parallel, making attach() unreliable.
Definition at line 110 of file Tracee.hxx.
|
inline |
Detach from and restart the tracee.
The tracee will be restarted (if currently in a tracing stop), the process will no longer be traced. This method can be used regardless of whether the tracee was seize()'d or attach()'d to.
Definition at line 120 of file Tracee.hxx.
|
inlineprotected |
Returns the current event message for a ptrace-event-stop.
The interpretation of the returned value depends on the ptrace::Event that has been reported. If there is no (matching) ptrace-event, then the return value seems to be undefined.
Definition at line 409 of file Tracee.hxx.
|
inline |
Returns the exit code of the tracee in the context of a ptrace-event-stop.
This request is only valid during a ptrace-event-stop when ptrace::Event::EXIT is reported.
The return value is the exit status of the exited tracee process.
Definition at line 331 of file Tracee.hxx.
|
inline |
Copy the tracee's floating point registers into the provided structure.
This is similar to getRegisters() but provides the floating point registers instead.
Preferably use getRegisterSet() instead.
Definition at line 243 of file Tracee.hxx.
|
inline |
Returns the PID of a newly created child of the tracee in the context of a ptrace-event-stop.
This request is only valid during a ptrace-event-stop and when ptrace::Event::FORK, ptrace::EVENT::VFORK, ptrace::Event::VFORK_DONE or ptrace::Event::CLONE is reported.
The return value is the PID of the newly created child process.
Definition at line 319 of file Tracee.hxx.
|
inline |
Copy the tracee's general purpose registers into the provided structure.
You need to include sys/user.h and check out the data structure found in there for details. This is a low level structure specially designed for GDB and also not available on all architectures.
Preferably use getRegisterSet() instead.
Definition at line 219 of file Tracee.hxx.
|
inline |
Retrieve a set of registers from the tracee.
This retrieves binary data based on an I/O vector. For ptrace::RegisterType::GENERAL_PURPOSE the target data structure is found in elf.h called elf_gregset_t
.
The kernel will update iovec
to reflect the actual amount of data that has been returned.
Definition at line 265 of file Tracee.hxx.
void cosmos::Tracee::getSeccompFilter | ( | std::vector< struct sock_filter > & | instructions, |
const unsigned long | prog_index ) const |
Retrieve a classic seccomp BPF program installed in the tracee.
prog_index
is the index of the program to return, where index 0 is the most recently installed program. If the index is greater than the number of installed programs then an ApiError with Errno::NO_ENTRY is thrown.
If instructions
is empty then the call will first ask the kernel how big the given program is, to dimension instructions
accordingly. In a second call the program is retrieved into the vector.
If instructions
is non-empty then the provided size will be used. Note that there seems to be error handling missing in the kernel to detect when the provided vector is too small. This means a too small vector could lead to memory corruption in the process.
Definition at line 14 of file Tracee.cxx.
|
inline |
Returns the SECCOMP_RET_DATA in the context of a ptrace-event-stop.
This request is only valid during a ptrace-event-stop when ptrace::Event::SECCOMP is reported.
The return value is the 16-bit value known as SECCOMP_RET_DATA, see seccomp(2)
.
Definition at line 344 of file Tracee.hxx.
void cosmos::Tracee::getSigInfo | ( | SigInfo & | info | ) | const |
Obtain information about the signal that caused the stop.
Definition at line 61 of file Tracee.cxx.
|
inline |
Obtain the tracee's mask of blocked signals.
Definition at line 302 of file Tracee.hxx.
void cosmos::Tracee::getSyscallInfo | ( | ptrace::SyscallInfo & | info | ) | const |
Returns system call information in the context of the current ptrace stop.
This request is only valid during syscall-entry-stop, syscall-exit-stop or ptrace-event-stop for ptrace::Event::SECCOMP.
Depending on the type of stop that occurred info
will contain different data, thus only certain parts of the struct are accessible via std::optional return values.
Definition at line 53 of file Tracee.cxx.
|
inline |
Interrupt the tracee.
This works only on tracees attached via seize(). As a result to the interrupt the tracee can enter:
Definition at line 156 of file Tracee.hxx.
|
inline |
Read one word of data from the tracee's memory.
addr
specifies the address in the tracee's memory to read a word from.
The size of the word is defined by the type of operating system and architecture of the system. On Linux no differentiation between TEXT and DATA is made, thus only DATA is offered here.
Definition at line 179 of file Tracee.hxx.
std::vector< SigInfo > cosmos::Tracee::peekSigInfo | ( | const ptrace::PeekSigInfo & | settings | ) |
Obtains SigInfo structures pending for the tracee.
Based on settings
obtain a numer of SigInfo structures pending for the tracee. settings
define how many SigInfo will be retrieved at max and from what position in the signal queue.
There is no way to know how many entries exist currently (this information can also rapidly change). If no more SigInfo structures exist at the given position then a short or zero item count is returned.
Definition at line 69 of file Tracee.cxx.
|
inline |
Read one word of data from the tracee's user area.
The user area refers to the kernel's struct user
which contains data about registers and other information about the process. This data is highly OS and architecture specific and could yield unexpected results.
The given parameter is an offset
into struct user
where to read from. The offset typically needs to be word-aligned.
Definition at line 198 of file Tracee.hxx.
|
inline |
Definition at line 69 of file Tracee.hxx.
|
inline |
Write one word of data into the tracee's memory.
Definition at line 184 of file Tracee.hxx.
|
inline |
Change one word of data in the tracee's user area.
This changes on word of data in the tracee's user area.
offset
typically needs to be word-aligned. Definition at line 207 of file Tracee.hxx.
|
inlineprotected |
Definition at line 416 of file Tracee.hxx.
|
inlineprotected |
Definition at line 431 of file Tracee.hxx.
|
inline |
Continues a traced process, optionally delivering signal
.
If the current ptrace stop state doesn't allow injection of a signal, then none should be specified. Generally only a signal-stop state allows injection of signals.
If signal information has been overwritten by using ptrace::set_siginfo(), then the signal
passed here must match, otherwise the behaviour will be undefined.
Definition at line 134 of file Tracee.hxx.
|
inline |
Seize a tracee.
This is the modern way of making a process a tracee. This does not stop the process. The seize property is inherited to matching child processes of the tracee if one of the options Opt::TRACEFORK, Opt::TRACEVFORK or Opt::TRACECLONE is set on the tracee.
Initial tracing options are set atomically alongside the SEIZE request.
Definition at line 88 of file Tracee.hxx.
|
inline |
Modify the tracee's floating point registers.
Definition at line 252 of file Tracee.hxx.
|
inline |
Set tracing options for the given tracee.
This call completely defines the options in effect for the given tracee. These options can be inherited by new tracees that are auto-attached via the TRACEFORK, TRACEVFORK and TRACECLONE options.
Definition at line 166 of file Tracee.hxx.
|
inline |
Modify the tracee's general purpose registers.
Some register modifications may be disallowed by the kernel to maintain integrity of the tracee.
Definition at line 231 of file Tracee.hxx.
|
inline |
Modify a set of registers in the tracee.
Definition at line 270 of file Tracee.hxx.
void cosmos::Tracee::setSigInfo | ( | const SigInfo & | info | ) |
Set signal information for the tracee.
This will affect only signals that would normally be delivered to the tracee and were caught by the tracer. These signals can be hard to tell from synthetic signals generated by ptrace() itself.
When changing the signal information this way then the signal passed to restart() needs to match, to prevent undefined behaviour.
Definition at line 65 of file Tracee.cxx.
|
inline |
Change the tracee's mask of blocked signals.
Definition at line 307 of file Tracee.hxx.
|
inline |
Definition at line 73 of file Tracee.hxx.
|
protected |
Definition at line 437 of file Tracee.hxx.