-
Notifications
You must be signed in to change notification settings - Fork 2.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add epbf program to trace podman cleanup errors in CI #23487
Merged
Merged
Changes from all commits
Commits
Show all changes
2 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,151 @@ | ||
#!/usr/bin/env bpftrace | ||
|
||
// Traces podman and conmon commands for there exit status and all process that | ||
// send signals to them. Also prints stderr for all podman container cleanup processes. | ||
// The goal is to be able to trace all podman cleanup errors, while there is an | ||
// option to log the errors to the syslog that only works of the process if the | ||
// process got there and not if it was killed before. This program here will trace | ||
// the exit status as well so we know even if it was killed by a signal. | ||
// | ||
// This is script uses /~https://github.com/bpftrace/bpftrace to compile this into | ||
// kernel ebpf. | ||
// | ||
// Usage: sudo ./podman_cleanup_tracer.bt | ||
// see below for output format | ||
|
||
BEGIN { | ||
print("Output format is one of:"); | ||
print("exec TIME PID PPID COMM ARGV(truncated)"); | ||
print("cmd TIME PID PPID ARGV(full)"); | ||
print("exit TIME PID PPID COMM EXIT_CODE EXIT_SIGNAL"); | ||
print("kill TIME PID PPID COMM SIGNAL TPID RET"); | ||
print("stderr TIME PID PPID COMM OUTPUT"); | ||
} | ||
|
||
// Trace all exec calls to find all podman + conmon processes, it will also catch | ||
// netavark and aardvark-dns as they have podman in the path as well but this is | ||
// good so we can see any errors there as well. | ||
tracepoint:syscalls:sys_enter_exec* | ||
/ strcontains(str(args.argv[0]),"podman") || strcontains(str(args.argv[0]), "conmon") / | ||
{ | ||
|
||
// create entry in pid map so we can check the pid later | ||
@pids[pid] = 1; | ||
|
||
// Find the podman cleanup process spawned by conmon. | ||
// I tried matching argv but there seems to be no way to iterate over it. | ||
// In practise parent name conmon and argv0 podman should contain all the | ||
// cleanup processes we care about. | ||
if (comm == "conmon" && strcontains(str(args.argv[0]), "podman")) { | ||
@cleanupPids[pid] = 1; | ||
} | ||
|
||
printf("%-6s %s %-8d %-8d %-12s ", | ||
"exec", | ||
strftime("%H:%M:%S.%f", nsecs), | ||
pid, | ||
curtask->real_parent->tgid, | ||
comm | ||
); | ||
|
||
// Unfortunately this doesn't print the full argv0 as there is some limit, | ||
// as such we use the /proc/<>/cmdline reading hack below on exec exit. | ||
join(args->argv); | ||
} | ||
|
||
|
||
// Print the full cmdline | ||
tracepoint:syscalls:sys_exit_exec* | ||
/ @pids[pid] / | ||
{ | ||
printf("%-6s %s %-8d %-8d ", | ||
"cmd", | ||
strftime("%H:%M:%S.%f", nsecs), | ||
pid, | ||
curtask->real_parent->tgid | ||
); | ||
// This can fail to open the file it is done in user space and | ||
// thus racy if the process exits quickly. | ||
cat("/proc/%d/cmdline", pid); | ||
print(""); | ||
} | ||
|
||
|
||
// Trace all exits for the pids we matched above, | ||
// pid == tid is used to only match the main exit | ||
// and not all thread exits which we do not care about. | ||
tracepoint:sched:sched_process_exit | ||
/ @pids[pid] && pid == tid / | ||
{ | ||
printf("%-6s %s %-8d %-8d %-12s %d %d\n", | ||
"exit", | ||
strftime("%H:%M:%S.%f", nsecs), | ||
pid, | ||
curtask->real_parent->tgid, | ||
comm, | ||
curtask->exit_code >> 8, // actual exit code | ||
curtask->exit_code & 0xFF // signal number if killed | ||
); | ||
|
||
// process is done remove pid from map | ||
delete(@pids[pid]); | ||
delete(@cleanupPids[pid]); | ||
} | ||
|
||
// Trace all kill calls that target our pids. | ||
// This will not catch signals send via pidfd_send_signal because | ||
// I don't see a way to translate the pidfd to our target pids and | ||
// I don't want to log all send signals on the system. | ||
tracepoint:syscalls:sys_enter_kill | ||
/ @pids[args.pid] / | ||
{ | ||
@tpid[tid] = args.pid; | ||
@tsig[tid] = args.sig; | ||
} | ||
|
||
tracepoint:syscalls:sys_exit_kill | ||
/ @tpid[tid] / | ||
{ | ||
printf("%-6s %s %-8d %-8d %-12s %d %-8d %d\n", | ||
"kill", | ||
strftime("%H:%M:%S.%f", nsecs), | ||
pid, | ||
curtask->real_parent->tgid, | ||
comm, | ||
@tsig[tid], | ||
@tpid[tid], | ||
args.ret | ||
); | ||
delete(@tpid[tid]); | ||
delete(@tsig[tid]); | ||
} | ||
|
||
|
||
// Print anything written on stderr for the podman container cleanup process. | ||
tracepoint:syscalls:sys_enter_write | ||
/ @cleanupPids[pid] && args.fd == 2 / | ||
{ | ||
printf("%-6s %s %-8d %-8d %-12s", | ||
"stderr", | ||
strftime("%H:%M:%S.%f", nsecs), | ||
pid, | ||
curtask->real_parent->tgid, | ||
comm | ||
); | ||
|
||
// String size limit is is 64 by default, this includes the 0 byte and when we | ||
// hit the string limit it also adds "..." when using 63 so we use 63 as len here. | ||
// While upstream fixed these low string limits (/~https://github.com/bpftrace/bpftrace/issues/305) | ||
// it is not yet in older distro version we use so we cannot use that yet. | ||
// Thus manually print several times. | ||
$len = 62; | ||
$offset = 62; | ||
printf("%s", str(args.buf, $len)); | ||
|
||
unroll(10) { | ||
if ((int64)args.count > $offset ) { | ||
printf("%s", str(args.buf + $offset, $len)); | ||
} | ||
$offset += $len | ||
} | ||
} |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why the asterisk? (nonblocking)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh that was a leftover from a previous iteration where I wrote to a
*.tmp
file and then calledtr
later but then I figured I can just pipe into tr and do not need a extra file so we can drop it in a future cleanup I guess