#[allow(unused)] use { anyhow::{Context, Error, Result}, jlogger::{jdebug, jerror, jinfo, jwarn, JloggerBuilder}, log::{debug, error, info, warn}, }; use { clap::Parser, libbpf_rs::{set_print, PerfBufferBuilder, PrintLevel}, plain::Plain, std::sync::{ atomic::{AtomicBool, Ordering}, Arc, }, tracelib::{bump_memlock_rlimit, bytes_to_string}, }; #[path = "bpf/execsnoop_pb.skel.rs"] mod execsnoop_pb; use execsnoop_pb::*; fn print_to_log(level: PrintLevel, msg: String) { match level { PrintLevel::Debug => log::debug!("{}", msg.trim_end_matches('\n')), PrintLevel::Info => log::info!("{}", msg.trim_end_matches('\n')), PrintLevel::Warn => log::warn!("{}", msg.trim_end_matches('\n')), } } #[derive(Parser, Debug)] struct Cli { ///Trace process lives at least <DURATION> ms. #[clap(short, default_value_t = 0_u64)] duration: u64, ///Verbose #[clap(short, long, parse(from_occurrences))] verbose: usize, ///Use timestamp instead of date time. #[clap(short = 'r', long)] timestamp: bool, ///Show fork trace. #[clap(short = 'f', long)] fork_event: bool, ///Show thread trace. #[clap(short = 't', long)] thread: bool, ///Show exit trace. #[clap(short = 'e', long)] exit_event: bool, ///show PPID info. #[clap(long)] ppid: bool, } type Event = execsnoop_pb_bss_types::event; unsafe impl Plain for Event {} fn main() -> Result<()> { let cli = Cli::parse(); let max_level = match cli.verbose { 0 => log::LevelFilter::Off, 1 => log::LevelFilter::Error, 2 => log::LevelFilter::Warn, 3 => log::LevelFilter::Info, 4 => log::LevelFilter::Debug, _ => log::LevelFilter::Off, }; JloggerBuilder::new() .max_level(max_level) .log_time(false) .log_runtime(false) .build(); bump_memlock_rlimit(); let skel_builder = ExecsnoopPbSkelBuilder::default(); set_print(Some((PrintLevel::Debug, print_to_log))); let mut open_skel = skel_builder.open().with_context(|| "Failed to open bpf.")?; open_skel.rodata().min_duration_ns = cli.duration * 1000000_u64; let mut skel = open_skel.load().with_context(|| "Faild to load bpf")?; let start = chrono::Local::now(); let show_timestamp = cli.timestamp; let handle_event = move |_cpu: i32, data: &[u8]| { let mut event = Event::default(); plain::copy_from_bytes(&mut event, data).expect("Corrupted event data"); let now = chrono::Local::now(); let timestamp_us = (now.timestamp_nanos() - start.timestamp_nanos()) / 1000; let signals = "HUP INT QUIT ILL TRAP ABRT BUS FPE KILL USR1 SEGV USR2 PIPE ALRM TERM STKFLT CHLD CONT STOP TSTP TTIN TTOU URG XCPU XFSZ VTALRM PROF WINCH POLL PWR SYS" .split(' ').collect::<Vec<&str>>(); let print_timestamp = || { if show_timestamp { print!("{:<10.6}", timestamp_us as f64 / 1000000_f64); } else { print!("{:<20}", now.format("%Y-%m-%d %H:%M:%S")); } }; let print_mark = |mark: &str| { print!(" {:<7}", mark); }; let print_comm = || { print!(" {:<16}", unsafe { bytes_to_string(event.comm.as_ptr()) }); }; let print_id = |id: i32| { print!(" {:<8}", id); }; if event.event_type == 1 { if !cli.exit_event { return; } if !cli.thread && event.tid != event.pid { return; } let signal_info = { if event.last_sig != -1 { format!( "[{}={} <- {}({})]", signals[event.last_sig as usize - 1], event.last_sig, unsafe { bytes_to_string(event.last_sig_comm.as_ptr()) }, event.last_signal_pid ) } else { String::new() } }; let fork_info = { if event.flag & 0x1 == 0x1 { unsafe { format!("[{}]", bytes_to_string(event.comm2.as_ptr())) } } else { String::new() } }; print_timestamp(); print_mark("EXIT"); print_comm(); if cli.thread { print_id(event.tid); } print_id(event.pid); if cli.ppid { print_id(event.ppid); } print!(" [{}]", event.exit_code); if !signal_info.is_empty() { print!(" {}", signal_info); } if !fork_info.is_empty() { print!(" {}", fork_info); } if event.duration_ns > 0 { print!(" ({}ms)", event.duration_ns / 1000000_u64); } println!(); } else if event.event_type == 0 { if !cli.thread && event.tid != event.pid { return; } let args_raw = String::from_utf8_lossy(&event.args[..event.args_size as usize]).to_string(); let mut args = args_raw.split('\0').collect::<Vec<&str>>(); let args_str = args[0..event.args_count as usize].join(" "); let filename = unsafe { bytes_to_string(event.filename.as_ptr()) }; if args.is_empty() { if !filename.is_empty() { args.push(&filename); } } let fork_info = { if event.flag & 0x1 == 0x1 { unsafe { format!("[{}]", bytes_to_string(event.comm2.as_ptr())) } } else { String::new() } }; print_timestamp(); print_mark("EXEC"); print_comm(); if cli.thread { print_id(event.tid); } print_id(event.pid); if cli.ppid { print_id(event.ppid); } if !fork_info.is_empty() { print!(" {}", fork_info); } if !args_str.is_empty() { print!(" {}", args_str); } if event.duration_ns > 0 { print!(" ({}ms)", event.duration_ns / 1000000_u64); } println!(); } else if event.event_type == 2 { if !cli.fork_event { return; } print_timestamp(); print_mark("FORK"); print_comm(); if cli.thread { print_id(event.tid); } print_id(event.pid); if cli.ppid { print_id(event.ppid); } if event.duration_ns > 0 { print!(" ({}ms)", event.duration_ns / 1000000_u64); } println!(); }; }; let perbuf = PerfBufferBuilder::new(skel.maps().pb()) .sample_cb(handle_event) .pages(16) // 4k * 16 .build() .with_context(|| "Failed to create perf buffer")?; skel.attach().with_context(|| "Faild to load bpf")?; let print_timestamp_str = || { if cli.timestamp { print!("{:<10}", "TIME"); } else { print!("{:<20}", "TIME"); } }; let print_mark_str = |mark: &str| { print!(" {:<7}", mark); }; let print_comm_str = || { print!(" {:<16}", "COMM"); }; let print_id_str = |id: &str| { print!(" {:<8}", id); }; print_timestamp_str(); print_mark_str("EVENT"); print_comm_str(); if cli.thread { print_id_str("TID"); } print_id_str("PID"); if cli.ppid { print_id_str("PPID"); } println!(" INFO"); let running = Arc::new(AtomicBool::new(true)); let r = running.clone(); ctrlc::set_handler(move || { r.store(false, Ordering::SeqCst); })?; while running.load(Ordering::SeqCst) { // ctrl-c will fail perbuf.poll() let _ = perbuf.poll(std::time::Duration::from_millis(100)); } Ok(()) }