summaryrefslogtreecommitdiff
path: root/src/trace.rs
blob: 0031fcd3944f542ab621c8d5061bc9d97120530a (plain)
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(())
}