libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
ptrace.hxx
Go to the documentation of this file.
1#pragma once
2
13// C++
14#include <optional>
15#include <stdint.h>
16
17// Linux
18#include <elf.h>
19#include <linux/audit.h>
20#include <sys/ptrace.h>
21#include <linux/ptrace.h> // ptrace_syscall_info is only found in here?
22
23// cosmos
24#include <cosmos/BitMask.hxx>
25#include <cosmos/dso_export.h>
26#include <cosmos/memory.hxx>
27#include <cosmos/proc/types.hxx>
28
29namespace cosmos::ptrace {
30
32enum class Opt : 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};
54
55using Opts = BitMask<Opt>;
56
58enum class Event {
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};
76
78enum class RegisterType {
79 GENERAL_PURPOSE = NT_PRSTATUS,
80 FLOATING_POINT = NT_FPREGSET
81};
82
84
89enum class Request {
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};
168
170
174enum class Arch : uint32_t {
175 X86_64 = AUDIT_ARCH_X86_64,
176 I386 = AUDIT_ARCH_I386
177};
178
181public: // types
182
184 enum class Type : uint8_t {
185 ENTRY = PTRACE_SYSCALL_INFO_ENTRY,
186 EXIT = PTRACE_SYSCALL_INFO_EXIT,
187 SECCOMP = PTRACE_SYSCALL_INFO_SECCOMP,
188 NONE = PTRACE_SYSCALL_INFO_NONE
189 };
190
191 using EntryInfo = decltype(ptrace_syscall_info::entry);
192 using ExitInfo = decltype(ptrace_syscall_info::exit);
193 using SeccompInfo = decltype(ptrace_syscall_info::seccomp);
194
195public: // functions
196
197 Type type() const {
198 return Type{m_info.op};
199 }
200
202 Arch arch() const {
203 return Arch{m_info.arch};
204 }
205
207 uint64_t instructionPtr() const {
208 return m_info.instruction_pointer;
209 }
210
212 uint64_t stackPtr() const {
213 return m_info.stack_pointer;
214 }
215
217 std::optional<EntryInfo> entryInfo() const {
218 return type() == Type::ENTRY ? std::make_optional(m_info.entry) : std::nullopt;
219 }
220
222 std::optional<ExitInfo> exitInfo() const {
223 return type() == Type::EXIT ? std::make_optional(m_info.exit) : std::nullopt;
224 }
225
227 std::optional<SeccompInfo> seccompInfo() const {
228 return type() == Type::SECCOMP ? std::make_optional(m_info.seccomp) : std::nullopt;
229 }
230
231 auto raw() { return &m_info; }
232 auto raw() const { return &m_info; }
233
234protected: // data
235 struct ptrace_syscall_info m_info;
236};
237
240public: // types
241
242 enum class Flag : uint32_t {
243 PEEK_SHARED = PTRACE_PEEKSIGINFO_SHARED,
244 };
245
246 using Flags = BitMask<Flag>;
247
248public: // functions
249
251
256 clear();
257 setFlags(Flags{Flag::PEEK_SHARED});
258 }
259
260 void clear() {
261 zero_object(m_args);
262 }
263
264 void setFlags(Flags flags) {
265 m_args.flags = flags.raw();
266 }
267
269 void setOffset(uint64_t off) {
270 m_args.off = off;
271 }
272
274 void setAmount(int32_t nr) {
275 m_args.nr = nr;
276 }
277
278 auto raw() const {
279 return &m_args;
280 }
281
282protected: // data
283
284 struct ptrace_peeksiginfo_args m_args;
285};
286
288
300std::optional<long> COSMOS_API trace(const ptrace::Request req, const ProcessID pid,
301 void *addr = nullptr, void *data = nullptr);
302
304
315void COSMOS_API traceme();
316
317} // end ns
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
Wrapper around data structure used with ptrace::Request::PEEKSIGINFO.
Definition ptrace.hxx:239
PeekSigInfo()
Creates an object with default settings.
Definition ptrace.hxx:255
@ PEEK_SHARED
dump signals from the process-wide queue, otherwise from the per-thread queue.
void setOffset(uint64_t off)
Sets the offset into the queue from which SigInfo structures should be obtained.
Definition ptrace.hxx:269
void setAmount(int32_t nr)
Sets the maximum amount of SigInfo structures to copy.
Definition ptrace.hxx:274
@ VFORK
The calling process is suspended until the child calls execve() or _exit(), see vfork(); should not b...
void zero_object(T &obj)
Completely overwrites the given object with zeroes.
Definition memory.hxx:23
Request
Basic requests that can be passed to the ptrace() system call.
Definition ptrace.hxx:89
@ 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.
RegisterType
Different types of register sets that can be read from a tracee via Request::GETREGSET.
Definition ptrace.hxx:78
Event
Different events that can occur in a tracee leading to ptrace-event-stop.
Definition ptrace.hxx:58
@ 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
Different options which can be set for a tracee.
Definition ptrace.hxx:32
@ 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.
Arch
System call ABI architecture.
Definition ptrace.hxx:174
Wrapper around data structure used with ptrace::Request::GET_SYSCALL_INFO.
Definition ptrace.hxx:180
uint64_t stackPtr() const
Returns the CPU stack pointer value.
Definition ptrace.hxx:212
Type
Type of the system call information provided.
Definition ptrace.hxx:184
@ SECCOMP
ptrace-event-stop for ptrace::Event::SECCOMP.
@ ENTRY
system-call-entry stop.
@ EXIT
system-call-exit-stop.
@ NONE
no meaningful information placed into struct.
Arch arch() const
Returns the system call ABI in effect for the current system call.
Definition ptrace.hxx:202
std::optional< ExitInfo > exitInfo() const
If available return the syscall-exit-stop information from the struct.
Definition ptrace.hxx:222
uint64_t instructionPtr() const
Returns the CPU instruction pointer value.
Definition ptrace.hxx:207
std::optional< SeccompInfo > seccompInfo() const
If available return the ptrace-event seccomp info from the struct.
Definition ptrace.hxx:227
std::optional< EntryInfo > entryInfo() const
If available return the syscall-entry-stop information from the struct.
Definition ptrace.hxx:217