From e109ea262d2059115c2069f32f420a9d3af71803 Mon Sep 17 00:00:00 2001 From: edef Date: Wed, 26 May 2021 11:16:14 +0000 Subject: ripple/driver.pl: a basic syscall tracing driver This program spawns a child in a new cgroup (using systemd-run) under bpftrace and counts syscall invocations for each (syscall, comm) pair. It outputs a TSV of (syscall, comm, count) tuples to STDERR. This makes it easy to assess which system calls a build process uses, segmented by a useful approximation of the invoking programs. Change-Id: I9ab7a34fda82245d3fa22b47840de16d1decf719 --- ripple/driver.pl | 68 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 68 insertions(+) create mode 100755 ripple/driver.pl (limited to 'ripple') diff --git a/ripple/driver.pl b/ripple/driver.pl new file mode 100755 index 0000000..07be06f --- /dev/null +++ b/ripple/driver.pl @@ -0,0 +1,68 @@ +#! /usr/bin/perl +# SPDX-FileCopyrightText: edef +# SPDX-License-Identifier: OSL-3.0 + +use strict; +use POSIX qw(mkfifo); + +@ARGV or die "Usage: $0 PROGRAM [ARG]... 2> [LOG FILE]"; + +unlink(my $fifo = "tracepipe"); +mkfifo($fifo, 0600) or die; + +my $script = do { local $/; }; +defined(my $pid = fork) or die; + +if (!$pid) { + open(STDERR, ">&", STDOUT) or die; + exec( + 'systemd-run', '--user', '--scope', '--', + # NOTE: this expects bpftrace to be SUID-root, + # and relies on shells dropping euid + 'bpftrace', '-o', $fifo, '-e', $script, '-c', + @ARGV + ) or die; +} + +my %count; + +# TODO(edef): if bpftrace fails, the FIFO will never open +open(TRACE, '<', $fifo) or die; +while () { + chomp; + next if /^Attaching \d+ probes[.][.][.]$/ or /^$/; + /^@\[tracepoint:syscalls:sys_enter_(\w+), (.+)\]: (\d+)$/ or die $_; + $count{$1}{$2} += $3; +} + +waitpid $pid, 0; + +foreach my $comm (sort keys %count) { + die "unhandled: $comm" if $comm =~ /\s/; + my $comm_count = $count{$comm}; + foreach my $sys (sort keys %$comm_count) { + my $n = $comm_count->{$sys}; + print STDERR "$comm\t$sys\t$n\n"; + } +} + +__DATA__ +BEGIN { + @cgroup = cgroup; + @self = pid; +} + +tracepoint:syscalls:sys_enter_* /cgroup == @cgroup && pid != @self/ { + @[probe, comm] = count(); +} + +interval:s:1 { + print(@); + clear(@); +} + +tracepoint:sched:sched_process_exit /tid == cpid/ { + clear(@cgroup); + clear(@self); + exit(); +} -- cgit 1.4.1