libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
ptrace.hxx File Reference
#include <optional>
#include <stdint.h>
#include <elf.h>
#include <linux/audit.h>
#include <sys/ptrace.h>
#include <linux/ptrace.h>
#include <cosmos/BitMask.hxx>
#include <cosmos/dso_export.h>
#include <cosmos/memory.hxx>
#include <cosmos/proc/types.hxx>

Go to the source code of this file.

Classes

struct  cosmos::ptrace::SyscallInfo
 Wrapper around data structure used with ptrace::Request::GET_SYSCALL_INFO. More...
 
class  cosmos::ptrace::PeekSigInfo
 Wrapper around data structure used with ptrace::Request::PEEKSIGINFO. More...
 

Typedefs

using cosmos::ptrace::Opts = BitMask<Opt>
 

Enumerations

enum class  cosmos::ptrace::Opt : intptr_t {
  EXITKILL = PTRACE_O_EXITKILL , TRACECLONE = PTRACE_O_TRACECLONE , TRACEEXEC = PTRACE_O_TRACEEXEC , TRACEEXIT = PTRACE_O_TRACEEXIT ,
  TRACEFORK = PTRACE_O_TRACEFORK , TRACEVFORK = PTRACE_O_TRACEVFORK , TRACEVFORKDONE = PTRACE_O_TRACEVFORKDONE , TRACESYSGOOD = PTRACE_O_TRACESYSGOOD ,
  TRACESECCOMP = PTRACE_O_TRACESECCOMP , SUSPENDSECCOMP = PTRACE_O_SUSPEND_SECCOMP
}
 Different options which can be set for a tracee. More...
 
enum class  cosmos::ptrace::Event {
  VFORK = PTRACE_EVENT_VFORK , FORK = PTRACE_EVENT_FORK , CLONE = PTRACE_EVENT_CLONE , VFORK_DONE = PTRACE_EVENT_VFORK_DONE ,
  EXEC = PTRACE_EVENT_EXEC , EXIT = PTRACE_EVENT_EXIT , STOP = PTRACE_EVENT_STOP , SECCOMP = PTRACE_EVENT_SECCOMP
}
 Different events that can occur in a tracee leading to ptrace-event-stop. More...
 
enum class  cosmos::ptrace::RegisterType { GENERAL_PURPOSE = NT_PRSTATUS , FLOATING_POINT = NT_FPREGSET }
 Different types of register sets that can be read from a tracee via Request::GETREGSET. More...
 
enum class  cosmos::ptrace::Request {
  TRACEME = PTRACE_TRACEME , PEEKDATA = PTRACE_PEEKDATA , PEEKTEXT = PTRACE_PEEKTEXT , PEEKUSER = PTRACE_PEEKUSER ,
  POKEDATA = PTRACE_POKEDATA , POKEUSER = PTRACE_POKEUSER , GETREGS = PTRACE_GETREGS , GETFPREGS = PTRACE_GETFPREGS ,
  GETREGSET = PTRACE_GETREGSET , SETREGS = PTRACE_SETREGS , SETFPREGS = PTRACE_SETFPREGS , SETREGSET = PTRACE_SETREGSET ,
  GETSIGINFO = PTRACE_GETSIGINFO , SETSIGINFO = PTRACE_SETSIGINFO , PEEKSIGINFO = PTRACE_PEEKSIGINFO , GETSIGMASK = PTRACE_GETSIGMASK ,
  SETSIGMASK = PTRACE_SETSIGMASK , SETOPTIONS = PTRACE_SETOPTIONS , GETEVENTMSG = PTRACE_GETEVENTMSG , CONT = PTRACE_CONT ,
  SYSCALL = PTRACE_SYSCALL , SINGLESTEP = PTRACE_SINGLESTEP , LISTEN = PTRACE_LISTEN , KILL = PTRACE_KILL ,
  INTERRUPT = PTRACE_INTERRUPT , ATTACH = PTRACE_ATTACH , SEIZE = PTRACE_SEIZE , SECCOMP_GET_FILTER = PTRACE_SECCOMP_GET_FILTER ,
  DETACH = PTRACE_DETACH , GET_THREAD_AREA = PTRACE_GET_THREAD_AREA , GET_SYSCALL_INFO = PTRACE_GET_SYSCALL_INFO
}
 Basic requests that can be passed to the ptrace() system call. More...
 
enum class  cosmos::ptrace::Arch : uint32_t { X86_64 = AUDIT_ARCH_X86_64 , I386 = AUDIT_ARCH_I386 }
 System call ABI architecture. More...
 

Functions

std::optional< long > cosmos::ptrace::trace (const ptrace::Request req, const ProcessID pid, void *addr=nullptr, void *data=nullptr)
 Perform a tracer request.
 
void cosmos::ptrace::traceme ()
 Inform the kernel that this process is to be traced by its parent.
 

Detailed Description

Wrappers around data structures for the ptrace() system call.

ptrace() is a complex ioctl() like system call using varargs. To improve readability and type safety, every ptrace command is made available through an individual wrapper found in the Tracee class.

Definition in file ptrace.hxx.

Typedef Documentation

◆ Opts

using cosmos::ptrace::Opts = BitMask<Opt>

Definition at line 55 of file ptrace.hxx.

Enumeration Type Documentation

◆ Arch

enum class cosmos::ptrace::Arch : uint32_t
strong

System call ABI architecture.

This is currently shortened just for x86 ABIs until we add support for more exotics archs.

Definition at line 174 of file ptrace.hxx.

174 : uint32_t {
175 X86_64 = AUDIT_ARCH_X86_64,
176 I386 = AUDIT_ARCH_I386
177};

◆ Event

enum class cosmos::ptrace::Event
strong

Different events that can occur in a tracee leading to ptrace-event-stop.

Enumerator
VFORK 

vfork() (or clone with VFORK flag) is upcoming.

FORK 

fork() (or clone with SIGCHLD as exit signal) is upcoming.

CLONE 

clone() is upcoming.

VFORK_DONE 

vfork() (or clone() with VFORK flag) was finished but not yet returned.

EXEC 

exec() is in progress, the thread ID is the new one already.

EXIT 

exit() is upcoming.

STOP 

Initial tracee stop after SEIZE or on new child creations.

SECCOMP 

Stop triggered by a seccomp rule on tracee syscall entry.

Definition at line 58 of file ptrace.hxx.

58 {
60 VFORK = PTRACE_EVENT_VFORK,
62 FORK = PTRACE_EVENT_FORK,
64 CLONE = PTRACE_EVENT_CLONE,
66 VFORK_DONE = PTRACE_EVENT_VFORK_DONE,
68 EXEC = PTRACE_EVENT_EXEC,
70 EXIT = PTRACE_EVENT_EXIT,
72 STOP = PTRACE_EVENT_STOP,
74 SECCOMP = PTRACE_EVENT_SECCOMP
75};
@ VFORK
The calling process is suspended until the child calls execve() or _exit(), see vfork(); should not b...
@ EXEC
allow execute access
@ SECCOMP
Stop triggered by a seccomp rule on tracee syscall entry.
@ EXIT
exit() is upcoming.
@ FORK
fork() (or clone with SIGCHLD as exit signal) is upcoming.
@ CLONE
clone() is upcoming.
@ VFORK_DONE
vfork() (or clone() with VFORK flag) was finished but not yet returned.

◆ Opt

enum class cosmos::ptrace::Opt : intptr_t
strong

Different options which can be set for a tracee.

Enumerator
EXITKILL 

When the tracer exits, the tracees will be sent SIGKILL.

TRACECLONE 

Stop on clone(2) and automatically trace the newly cloned process.

TRACEEXEC 

Stop on execve(2).

TRACEEXIT 

Stop the tracee at when it exits.

TRACEFORK 

Stop on fork(2) and automatically trace the newly forked process.

TRACEVFORK 

Stop on vfork(2) and automatically trace the newly forked proc.

TRACEVFORKDONE 

Stop tracee at completion of vfork(2).

TRACESYSGOOD 

For better detection of system-call-stops, sets bit 7 in the si_code field (WSTOPSIG() == SIGTRAP|0x80).

TRACESECCOMP 

Stop when a SECCOMP_RET_TRACE rule is triggered.

SUSPENDSECCOMP 

Suspends the tracee's seccomp protections.

Definition at line 32 of file ptrace.hxx.

32 : intptr_t { /* is a void* in ptrace(2), so we need pointer width */
34 EXITKILL = PTRACE_O_EXITKILL,
36 TRACECLONE = PTRACE_O_TRACECLONE,
38 TRACEEXEC = PTRACE_O_TRACEEXEC,
40 TRACEEXIT = PTRACE_O_TRACEEXIT,
42 TRACEFORK = PTRACE_O_TRACEFORK,
44 TRACEVFORK = PTRACE_O_TRACEVFORK,
46 TRACEVFORKDONE = PTRACE_O_TRACEVFORKDONE,
48 TRACESYSGOOD = PTRACE_O_TRACESYSGOOD,
50 TRACESECCOMP = PTRACE_O_TRACESECCOMP,
52 SUSPENDSECCOMP = PTRACE_O_SUSPEND_SECCOMP,
53};
@ TRACEVFORKDONE
Stop tracee at completion of vfork(2).
@ TRACEFORK
Stop on fork(2) and automatically trace the newly forked process.
@ TRACEVFORK
Stop on vfork(2) and automatically trace the newly forked proc.
@ TRACECLONE
Stop on clone(2) and automatically trace the newly cloned process.
@ TRACESECCOMP
Stop when a SECCOMP_RET_TRACE rule is triggered.
@ TRACEEXIT
Stop the tracee at when it exits.
@ TRACEEXEC
Stop on execve(2).
@ TRACESYSGOOD
For better detection of system-call-stops, sets bit 7 in the si_code field (WSTOPSIG() == SIGTRAP|0x8...
@ EXITKILL
When the tracer exits, the tracees will be sent SIGKILL.
@ SUSPENDSECCOMP
Suspends the tracee's seccomp protections.

◆ RegisterType

enum class cosmos::ptrace::RegisterType
strong

Different types of register sets that can be read from a tracee via Request::GETREGSET.

Definition at line 78 of file ptrace.hxx.

78 {
79 GENERAL_PURPOSE = NT_PRSTATUS,
80 FLOATING_POINT = NT_FPREGSET
81};

◆ Request

enum class cosmos::ptrace::Request
strong

Basic requests that can be passed to the ptrace() system call.

Note
On system call level this has an actual enum __ptrace_request type, thus there is no defined underlying type and we keep the compiler's default.
Enumerator
TRACEME 

The tracee asks to be traced by its parent.

PEEKDATA 

Read a word at a given address of the tracee's memory.

PEEKTEXT 

Read a word at a given offset of the tracee's TEXT data (the same as PEEKDATA on Linux).

PEEKUSER 

Read a word at a given offset of the tracee's USER data (holds registers and other metadata).

POKEDATA 

Write a word at the given address into the tracee's memory.

POKEUSER 

Write a word at the given offset into the tracee's USER data.

GETREGS 

Copy the tracee's general-purpose registers to the given address in the tracer.

GETFPREGS 

Copy the tracee's floating-point registers to the given address in the tracer.

GETREGSET 

Reg the tracee's registers in an architecture dependent way.

SETREGS 

Modify the tracee's general-purpose registers (not available on all architectures).

SETFPREGS 

Modify the tracee's floating-point registers.

SETREGSET 

Modify the tracee's registers, analogous to GETREGSET.

GETSIGINFO 

Retrieve information about the signal that cause the tracee to stop. Copies a siginfo_t structure.

SETSIGINFO 

Modify the tracee's signal information (used when the tracer catched a signal that would normally be delivered to the tracee).

PEEKSIGINFO 

Retrieve siginfo_t structures without removing them from the tracee's queue.

GETSIGMASK 

Retrieves a copy of the mask of blocked signals from the tracee.

SETSIGMASK 

Change the tracee's mask of blocked signals.

SETOPTIONS 

Set ptrace options (see ptrace::Opts)

GETEVENTMSG 

Retrieve a message (unsigned long) about the ptrace event that just happened.

CONT 

Restart the stopped tracee. Non-zero data is interpreted as a signal number to be delivered to the tracee.

SYSCALL 

Restart the stopped tracee like CONT, but arrange for the tracee to be stopped at entry/exit to/from a system call.

SINGLESTEP 

Like SYSCALL, but arrange for the tracee to be stopped after a single instruction.

LISTEN 

Restart a stopped tracee, but let it enter a SIGSTOP like state. Works only for SEIZE'd tracees.

KILL 

Send a SIGKILL to the tracee (this is buggy, don't use it).

INTERRUPT 

Stop a tracee. Works only for SIZE'd tracees.

ATTACH 

Attach to the specified process, making it a tracee of the caller.

SEIZE 

Like ATTACH but does not stop the process.

SECCOMP_GET_FILTER 

Dump the tracee's classic BPF filters.

DETACH 

Restart the stopped tracee, but first detach from it.

GET_THREAD_AREA 

Performs an operation similar to get_thread_area().

GET_SYSCALL_INFO 

Retrieve information about the system call that cause the stop.

Definition at line 89 of file ptrace.hxx.

89 {
91 TRACEME = PTRACE_TRACEME,
93 PEEKDATA = PTRACE_PEEKDATA,
95 PEEKTEXT = PTRACE_PEEKTEXT,
97 PEEKUSER = PTRACE_PEEKUSER,
99 POKEDATA = PTRACE_POKEDATA,
101 POKEUSER = PTRACE_POKEUSER,
103 GETREGS = PTRACE_GETREGS,
105 GETFPREGS = PTRACE_GETFPREGS,
107 GETREGSET = PTRACE_GETREGSET,
109 SETREGS = PTRACE_SETREGS,
111 SETFPREGS = PTRACE_SETFPREGS,
113 SETREGSET = PTRACE_SETREGSET,
115 GETSIGINFO = PTRACE_GETSIGINFO,
117 SETSIGINFO = PTRACE_SETSIGINFO,
119 PEEKSIGINFO = PTRACE_PEEKSIGINFO,
121 GETSIGMASK = PTRACE_GETSIGMASK,
123 SETSIGMASK = PTRACE_SETSIGMASK,
125 SETOPTIONS = PTRACE_SETOPTIONS,
127 GETEVENTMSG = PTRACE_GETEVENTMSG,
129 CONT = PTRACE_CONT,
131 SYSCALL = PTRACE_SYSCALL,
133 SINGLESTEP = PTRACE_SINGLESTEP,
134 // When in syscall-enter-stop, change the number of the system call to execute (only supported on arm)
135 // (it seems this has been removed from system headers by now)
136#ifdef PTRACE_SET_SYSCALL
137 SET_SYSCALL = PTRACE_SET_SYSCALL,
138#endif
139#ifdef PTRACE_SYSEMU
141 SYSEMU = PTRACE_SYSEMU,
143 SYSEMU_SINGLESTEP = PTRACE_SYSEMU_SINGLESTEP,
144#endif
146 LISTEN = PTRACE_LISTEN,
148 KILL = PTRACE_KILL,
150 INTERRUPT = PTRACE_INTERRUPT,
152 ATTACH = PTRACE_ATTACH,
154 SEIZE = PTRACE_SEIZE,
156 SECCOMP_GET_FILTER = PTRACE_SECCOMP_GET_FILTER,
158 DETACH = PTRACE_DETACH,
160 GET_THREAD_AREA = PTRACE_GET_THREAD_AREA,
161#if PTRACE_SET_THREAD_AREA
163 SET_THREAD_AREA = PTRACE_SET_THREAD_AREA,
164#endif
166 GET_SYSCALL_INFO = PTRACE_GET_SYSCALL_INFO,
167};
@ SEIZE
Like ATTACH but does not stop the process.
@ PEEKTEXT
Read a word at a given offset of the tracee's TEXT data (the same as PEEKDATA on Linux).
@ GETREGS
Copy the tracee's general-purpose registers to the given address in the tracer.
@ GETSIGINFO
Retrieve information about the signal that cause the tracee to stop. Copies a siginfo_t structure.
@ DETACH
Restart the stopped tracee, but first detach from it.
@ SETFPREGS
Modify the tracee's floating-point registers.
@ PEEKUSER
Read a word at a given offset of the tracee's USER data (holds registers and other metadata).
@ SETSIGINFO
Modify the tracee's signal information (used when the tracer catched a signal that would normally be ...
@ SYSCALL
Restart the stopped tracee like CONT, but arrange for the tracee to be stopped at entry/exit to/from ...
@ POKEUSER
Write a word at the given offset into the tracee's USER data.
@ GETSIGMASK
Retrieves a copy of the mask of blocked signals from the tracee.
@ PEEKDATA
Read a word at a given address of the tracee's memory.
@ GETREGSET
Reg the tracee's registers in an architecture dependent way.
@ GETEVENTMSG
Retrieve a message (unsigned long) about the ptrace event that just happened.
@ SETREGSET
Modify the tracee's registers, analogous to GETREGSET.
@ GET_THREAD_AREA
Performs an operation similar to get_thread_area().
@ SECCOMP_GET_FILTER
Dump the tracee's classic BPF filters.
@ SETOPTIONS
Set ptrace options (see ptrace::Opts)
@ POKEDATA
Write a word at the given address into the tracee's memory.
@ ATTACH
Attach to the specified process, making it a tracee of the caller.
@ SETREGS
Modify the tracee's general-purpose registers (not available on all architectures).
@ LISTEN
Restart a stopped tracee, but let it enter a SIGSTOP like state. Works only for SEIZE'd tracees.
@ GET_SYSCALL_INFO
Retrieve information about the system call that cause the stop.
@ SETSIGMASK
Change the tracee's mask of blocked signals.
@ PEEKSIGINFO
Retrieve siginfo_t structures without removing them from the tracee's queue.
@ TRACEME
The tracee asks to be traced by its parent.
@ SINGLESTEP
Like SYSCALL, but arrange for the tracee to be stopped after a single instruction.
@ GETFPREGS
Copy the tracee's floating-point registers to the given address in the tracer.

Function Documentation

◆ trace()

std::optional< long > COSMOS_API cosmos::ptrace::trace ( const ptrace::Request req,
const ProcessID pid,
void * addr = nullptr,
void * data = nullptr )

Perform a tracer request.

This is only a thin wrapper around the actual ptrace() system call. For a more convenient and safe interface use the Tracee class which splits this call into its individual sub-operations.

On error conditions this throws an ApiError. Depending on req a long value is returned (for PEEK operations, SECCOMP_GET_FILTER and GET_SYSCALL_INFO), otherwise std::nullopt.

The necessity and meaning of addr and data are also depending on the actual req.

Definition at line 8 of file ptrace.cxx.

8 {
9 const auto is_peek =
10 req == Request::PEEKDATA ||
11 req == Request::PEEKTEXT ||
12 req == Request::PEEKUSER;
13
14 if (is_peek) {
15 // PEEK requests have difficult error handling, we need to
16 // explicitly reset the errno.
18 }
19
20 const auto ret = ::ptrace(
21 static_cast<__ptrace_request>(to_integral(req)),
22 to_integral(pid), addr, data);
23
24 if (ret == -1) {
25 if (is_peek && !is_errno_set()) {
26 // we actually peeked a -1
27 return ret;
28 }
29
30 cosmos_throw (ApiError("ptrace()"));
31 }
32
33 // only these ptrace requests return meaningful data, otherwise just
34 // zero on success.
35 if (is_peek || req == Request::SECCOMP_GET_FILTER || req == Request::GET_SYSCALL_INFO) {
36 return ret;
37 }
38
39 return std::nullopt;
40}
void reset_errno()
Resets the currently set errno to indicate no error.
Definition errno.hxx:113
bool is_errno_set()
Checks whether currently an errno is set.
Definition errno.hxx:117

◆ traceme()

void COSMOS_API cosmos::ptrace::traceme ( )

Inform the kernel that this process is to be traced by its parent.

A typical pattern for tracing a child process is to:

  • fork()
  • traceme() and raise(SIGSTOP) in the child process
  • wait for ptrace-signal-stop in the parent

Since this does not rely on ptrace::Request::SEIZE to attach to the tracee, certain features of the ptrace API won't be available when using the TRACEME approach.

Definition at line 42 of file ptrace.cxx.

42 {
43 // all arguments except the request are ignored.
44 // this is not clearly documented but I checked the kernel code to verify.
45 trace(Request::TRACEME, ProcessID{0}, nullptr, nullptr);
46}