1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
use nix::{
sys::{
ptrace,
signal::Signal,
wait::{waitpid, WaitStatus},
},
unistd::Pid,
};
use std::{os::unix::process::CommandExt, process::Command};
pub fn exec(command: &mut Command) -> anyhow::Result<Pid> {
unsafe {
command.pre_exec(|| ptrace::traceme().map_err(|e| e.into()));
}
let child = command.spawn()?;
Ok(Pid::from_raw(child.id() as i32))
}
pub fn trace(pid: Pid) -> anyhow::Result<()> {
let mut have_to_print = true;
loop {
have_to_print ^= true;
ptrace::syscall(pid, None)?;
let status = waitpid(pid, None)?;
match status {
WaitStatus::Exited(_pid, _) => {
break;
}
WaitStatus::Stopped(pid, signal) => {
match signal {
Signal::SIGTRAP => {
let regs = ptrace::getregs(pid)?;
if have_to_print {
println!(
"{}({:x}, {:x}, {:x}, ...) = {:x}",
regs.orig_rax, regs.rdi, regs.rsi, regs.rdx, regs.rax,
);
}
}
_ => {}
};
}
_ => {}
};
}
Ok(())
}
|