Commit aa6c9168 authored by Mark Drayton's avatar Mark Drayton Committed by 4ast

trace: add pid/tid filtering, fix symbolizing, misc nits (#798)

* support filtering by process ID (-p) or thread ID (-t); previously -p
  actually filtered on thread ID (aka "pid" in kernel-speak)
* include process and thread ID in output
* flip order of user and kernel stacks to flow more naturally
* resolve symbols using process ID instead of thread ID so only one symbol
  cache is instantiated per process
* misc aesthetic fixes here and there
parent 5496aaf3
...@@ -2,7 +2,9 @@ ...@@ -2,7 +2,9 @@
.SH NAME .SH NAME
trace \- Trace a function and print its arguments or return value, optionally evaluating a filter. Uses Linux eBPF/bcc. trace \- Trace a function and print its arguments or return value, optionally evaluating a filter. Uses Linux eBPF/bcc.
.SH SYNOPSIS .SH SYNOPSIS
.B trace [-h] [-p PID] [-v] [-Z STRING_SIZE] [-S] [-M MAX_EVENTS] [-o] [-K] [-U] [-I header] probe [probe ...] .B trace [-h] [-p PID] [-t TID] [-v] [-Z STRING_SIZE] [-S]
[-M MAX_EVENTS] [-o] [-K] [-U] [-I header]
probe [probe ...]
.SH DESCRIPTION .SH DESCRIPTION
trace probes functions you specify and displays trace messages if a particular trace probes functions you specify and displays trace messages if a particular
condition is met. You can control the message format to display function condition is met. You can control the message format to display function
...@@ -19,6 +21,9 @@ Print usage message. ...@@ -19,6 +21,9 @@ Print usage message.
\-p PID \-p PID
Trace only functions in the process PID. Trace only functions in the process PID.
.TP .TP
\-t TID
Trace only functions in the thread TID.
.TP
\-v \-v
Display the generated BPF program, for debugging purposes. Display the generated BPF program, for debugging purposes.
.TP .TP
......
...@@ -3,8 +3,9 @@ ...@@ -3,8 +3,9 @@
# trace Trace a function and print a trace message based on its # trace Trace a function and print a trace message based on its
# parameters, with an optional filter. # parameters, with an optional filter.
# #
# USAGE: trace [-h] [-p PID] [-v] [-Z STRING_SIZE] [-S] [-M MAX_EVENTS] [-o] # usage: trace [-h] [-p PID] [-t TID] [-v] [-Z STRING_SIZE] [-S]
# [-K] [-U] [-I header] probe [probe ...] # [-M MAX_EVENTS] [-o] [-K] [-U] [-I header]
# probe [probe ...]
# #
# Licensed under the Apache License, Version 2.0 (the "License") # Licensed under the Apache License, Version 2.0 (the "License")
# Copyright (C) 2016 Sasha Goldshtein. # Copyright (C) 2016 Sasha Goldshtein.
...@@ -51,6 +52,7 @@ class Probe(object): ...@@ -51,6 +52,7 @@ class Probe(object):
event_count = 0 event_count = 0
first_ts = 0 first_ts = 0
use_localtime = True use_localtime = True
tgid = -1
pid = -1 pid = -1
@classmethod @classmethod
...@@ -58,6 +60,7 @@ class Probe(object): ...@@ -58,6 +60,7 @@ class Probe(object):
cls.max_events = args.max_events cls.max_events = args.max_events
cls.use_localtime = not args.offset cls.use_localtime = not args.offset
cls.first_ts = Time.monotonic_time() cls.first_ts = Time.monotonic_time()
cls.tgid = args.tgid or -1
cls.pid = args.pid or -1 cls.pid = args.pid or -1
def __init__(self, probe, string_size, kernel_stack, user_stack): def __init__(self, probe, string_size, kernel_stack, user_stack):
...@@ -154,7 +157,8 @@ class Probe(object): ...@@ -154,7 +157,8 @@ class Probe(object):
self.function = parts[2] self.function = parts[2]
def _find_usdt_probe(self): def _find_usdt_probe(self):
self.usdt = USDT(path=self.library, pid=Probe.pid) target = Probe.pid if Probe.pid else Probe.tgid
self.usdt = USDT(path=self.library, pid=target)
for probe in self.usdt.enumerate_probes(): for probe in self.usdt.enumerate_probes():
if probe.name == self.usdt_name: if probe.name == self.usdt_name:
return # Found it, will enable later return # Found it, will enable later
...@@ -258,6 +262,7 @@ static inline bool %s(char const *ignored, unsigned long str) { ...@@ -258,6 +262,7 @@ static inline bool %s(char const *ignored, unsigned long str) {
(self._display_function(), self.probe_num) (self._display_function(), self.probe_num)
fields = [ fields = [
("timestamp_ns", ct.c_ulonglong), ("timestamp_ns", ct.c_ulonglong),
("tgid", ct.c_uint),
("pid", ct.c_uint), ("pid", ct.c_uint),
("comm", ct.c_char * 16) # TASK_COMM_LEN ("comm", ct.c_char * 16) # TASK_COMM_LEN
] ]
...@@ -309,6 +314,7 @@ static inline bool %s(char const *ignored, unsigned long str) { ...@@ -309,6 +314,7 @@ static inline bool %s(char const *ignored, unsigned long str) {
struct %s struct %s
{ {
u64 timestamp_ns; u64 timestamp_ns;
u32 tgid;
u32 pid; u32 pid;
char comm[TASK_COMM_LEN]; char comm[TASK_COMM_LEN];
%s %s
...@@ -370,13 +376,15 @@ BPF_PERF_OUTPUT(%s); ...@@ -370,13 +376,15 @@ BPF_PERF_OUTPUT(%s);
# it to the function body: # it to the function body:
if len(self.library) == 0 and Probe.pid != -1: if len(self.library) == 0 and Probe.pid != -1:
pid_filter = """ pid_filter = """
u32 __pid = bpf_get_current_pid_tgid();
if (__pid != %d) { return 0; } if (__pid != %d) { return 0; }
""" % Probe.pid """ % Probe.pid
elif len(self.library) == 0 and Probe.tgid != -1:
pid_filter = """
if (__tgid != %d) { return 0; }
""" % Probe.tgid
elif not include_self: elif not include_self:
pid_filter = """ pid_filter = """
u32 __pid = bpf_get_current_pid_tgid(); if (__tgid == %d) { return 0; }
if (__pid == %d) { return 0; }
""" % os.getpid() """ % os.getpid()
else: else:
pid_filter = "" pid_filter = ""
...@@ -410,6 +418,9 @@ BPF_PERF_OUTPUT(%s); ...@@ -410,6 +418,9 @@ BPF_PERF_OUTPUT(%s);
text = heading + """ text = heading + """
{ {
u64 __pid_tgid = bpf_get_current_pid_tgid();
u32 __tgid = __pid_tgid >> 32;
u32 __pid = __pid_tgid; // implicit cast to u32 for bottom half
%s %s
%s %s
%s %s
...@@ -417,7 +428,8 @@ BPF_PERF_OUTPUT(%s); ...@@ -417,7 +428,8 @@ BPF_PERF_OUTPUT(%s);
struct %s __data = {0}; struct %s __data = {0};
__data.timestamp_ns = bpf_ktime_get_ns(); __data.timestamp_ns = bpf_ktime_get_ns();
__data.pid = bpf_get_current_pid_tgid(); __data.tgid = __tgid;
__data.pid = __pid;
bpf_get_current_comm(&__data.comm, sizeof(__data.comm)); bpf_get_current_comm(&__data.comm, sizeof(__data.comm));
%s %s
%s %s
...@@ -444,17 +456,17 @@ BPF_PERF_OUTPUT(%s); ...@@ -444,17 +456,17 @@ BPF_PERF_OUTPUT(%s);
else: # self.probe_type == 't' else: # self.probe_type == 't'
return self.tp_event return self.tp_event
def print_stack(self, bpf, stack_id, pid): def print_stack(self, bpf, stack_id, tgid):
if stack_id < 0: if stack_id < 0:
print(" %d" % stack_id) print(" %d" % stack_id)
return return
stack = list(bpf.get_table(self.stacks_name).walk(stack_id)) stack = list(bpf.get_table(self.stacks_name).walk(stack_id))
for addr in stack: for addr in stack:
print(" %016x %s" % (addr, bpf.sym(addr, pid))) print(" %016x %s" % (addr, bpf.sym(addr, tgid)))
def _format_message(self, bpf, pid, values): def _format_message(self, bpf, tgid, values):
# Replace each %K with kernel sym and %U with user sym in pid # Replace each %K with kernel sym and %U with user sym in tgid
kernel_placeholders = [i for i in xrange(0, len(self.types)) kernel_placeholders = [i for i in xrange(0, len(self.types))
if self.types[i] == 'K'] if self.types[i] == 'K']
user_placeholders = [i for i in xrange(0, len(self.types)) user_placeholders = [i for i in xrange(0, len(self.types))
...@@ -462,7 +474,7 @@ BPF_PERF_OUTPUT(%s); ...@@ -462,7 +474,7 @@ BPF_PERF_OUTPUT(%s);
for kp in kernel_placeholders: for kp in kernel_placeholders:
values[kp] = bpf.ksymaddr(values[kp]) values[kp] = bpf.ksymaddr(values[kp])
for up in user_placeholders: for up in user_placeholders:
values[up] = bpf.symaddr(values[up], pid) values[up] = bpf.symaddr(values[up], tgid)
return self.python_format % tuple(values) return self.python_format % tuple(values)
def print_event(self, bpf, cpu, data, size): def print_event(self, bpf, cpu, data, size):
...@@ -471,19 +483,17 @@ BPF_PERF_OUTPUT(%s); ...@@ -471,19 +483,17 @@ BPF_PERF_OUTPUT(%s);
event = ct.cast(data, ct.POINTER(self.python_struct)).contents event = ct.cast(data, ct.POINTER(self.python_struct)).contents
values = map(lambda i: getattr(event, "v%d" % i), values = map(lambda i: getattr(event, "v%d" % i),
range(0, len(self.values))) range(0, len(self.values)))
msg = self._format_message(bpf, event.pid, values) msg = self._format_message(bpf, event.tgid, values)
time = strftime("%H:%M:%S") if Probe.use_localtime else \ time = strftime("%H:%M:%S") if Probe.use_localtime else \
Probe._time_off_str(event.timestamp_ns) Probe._time_off_str(event.timestamp_ns)
print("%-8s %-6d %-12s %-16s %s" % print("%-8s %-6d %-6d %-12s %-16s %s" %
(time[:8], event.pid, event.comm[:12], (time[:8], event.tgid, event.pid, event.comm,
self._display_function(), msg)) self._display_function(), msg))
if self.user_stack:
print(" User Stack Trace:")
self.print_stack(bpf, event.user_stack_id, event.pid)
if self.kernel_stack: if self.kernel_stack:
print(" Kernel Stack Trace:")
self.print_stack(bpf, event.kernel_stack_id, -1) self.print_stack(bpf, event.kernel_stack_id, -1)
if self.user_stack:
self.print_stack(bpf, event.user_stack_id, event.tgid)
if self.user_stack or self.kernel_stack: if self.user_stack or self.kernel_stack:
print("") print("")
...@@ -549,9 +559,9 @@ trace 'c:malloc "size = %d", arg1' ...@@ -549,9 +559,9 @@ trace 'c:malloc "size = %d", arg1'
Trace malloc calls and print the size being allocated Trace malloc calls and print the size being allocated
trace 'p:c:write (arg1 == 1) "writing %d bytes to STDOUT", arg3' trace 'p:c:write (arg1 == 1) "writing %d bytes to STDOUT", arg3'
Trace the write() call from libc to monitor writes to STDOUT Trace the write() call from libc to monitor writes to STDOUT
trace 'r::__kmalloc (retval == 0) "kmalloc failed!" trace 'r::__kmalloc (retval == 0) "kmalloc failed!"'
Trace returns from __kmalloc which returned a null pointer Trace returns from __kmalloc which returned a null pointer
trace 'r:c:malloc (retval) "allocated = %x", retval trace 'r:c:malloc (retval) "allocated = %x", retval'
Trace returns from malloc and print non-NULL allocated buffers Trace returns from malloc and print non-NULL allocated buffers
trace 't:block:block_rq_complete "sectors=%d", args->nr_sector' trace 't:block:block_rq_complete "sectors=%d", args->nr_sector'
Trace the block_rq_complete kernel tracepoint and print # of tx sectors Trace the block_rq_complete kernel tracepoint and print # of tx sectors
...@@ -564,8 +574,12 @@ trace 'u:pthread:pthread_create (arg4 != 0)' ...@@ -564,8 +574,12 @@ trace 'u:pthread:pthread_create (arg4 != 0)'
"functions and print trace messages.", "functions and print trace messages.",
formatter_class=argparse.RawDescriptionHelpFormatter, formatter_class=argparse.RawDescriptionHelpFormatter,
epilog=Tool.examples) epilog=Tool.examples)
parser.add_argument("-p", "--pid", type=int, # we'll refer to the userspace concepts of "pid" and "tid" by
help="id of the process to trace (optional)") # their kernel names -- tgid and pid -- inside the script
parser.add_argument("-p", "--pid", type=int, metavar="PID",
dest="tgid", help="id of the process to trace (optional)")
parser.add_argument("-t", "--tid", type=int, metavar="TID",
dest="pid", help="id of the thread to trace (optional)")
parser.add_argument("-v", "--verbose", action="store_true", parser.add_argument("-v", "--verbose", action="store_true",
help="print resulting BPF program code before executing") help="print resulting BPF program code before executing")
parser.add_argument("-Z", "--string-size", type=int, parser.add_argument("-Z", "--string-size", type=int,
...@@ -587,6 +601,8 @@ trace 'u:pthread:pthread_create (arg4 != 0)' ...@@ -587,6 +601,8 @@ trace 'u:pthread:pthread_create (arg4 != 0)'
metavar="header", metavar="header",
help="additional header files to include in the BPF program") help="additional header files to include in the BPF program")
self.args = parser.parse_args() self.args = parser.parse_args()
if self.args.tgid and self.args.pid:
parser.error("only one of -p and -t may be specified")
def _create_probes(self): def _create_probes(self):
Probe.configure(self.args) Probe.configure(self.args)
...@@ -636,8 +652,8 @@ trace 'u:pthread:pthread_create (arg4 != 0)' ...@@ -636,8 +652,8 @@ trace 'u:pthread:pthread_create (arg4 != 0)'
self.probes)) self.probes))
# Print header # Print header
print("%-8s %-6s %-12s %-16s %s" % print("%-8s %-6s %-6s %-12s %-16s %s" %
("TIME", "PID", "COMM", "FUNC", ("TIME", "PID", "TID", "COMM", "FUNC",
"-" if not all_probes_trivial else "")) "-" if not all_probes_trivial else ""))
while True: while True:
......
...@@ -171,7 +171,8 @@ libraries and then accessing the /home/vagrant directory listing. ...@@ -171,7 +171,8 @@ libraries and then accessing the /home/vagrant directory listing.
USAGE message: USAGE message:
# trace -h # trace -h
usage: trace.py [-h] [-p PID] [-v] [-Z STRING_SIZE] [-S] [-M MAX_EVENTS] [-o] usage: trace [-h] [-p PID] [-t TID] [-v] [-Z STRING_SIZE] [-S]
[-M MAX_EVENTS] [-o] [-K] [-U] [-I header]
probe [probe ...] probe [probe ...]
Attach to functions and print trace messages. Attach to functions and print trace messages.
...@@ -182,6 +183,7 @@ positional arguments: ...@@ -182,6 +183,7 @@ positional arguments:
optional arguments: optional arguments:
-h, --help show this help message and exit -h, --help show this help message and exit
-p PID, --pid PID id of the process to trace (optional) -p PID, --pid PID id of the process to trace (optional)
-t TID, --tid TID id of the thread to trace (optional)
-v, --verbose print resulting BPF program code before executing -v, --verbose print resulting BPF program code before executing
-Z STRING_SIZE, --string-size STRING_SIZE -Z STRING_SIZE, --string-size STRING_SIZE
maximum size to read from strings maximum size to read from strings
...@@ -202,8 +204,8 @@ trace 'do_sys_open "%s", arg2' ...@@ -202,8 +204,8 @@ trace 'do_sys_open "%s", arg2'
Trace the open syscall and print the filename being opened Trace the open syscall and print the filename being opened
trace 'sys_read (arg3 > 20000) "read %d bytes", arg3' trace 'sys_read (arg3 > 20000) "read %d bytes", arg3'
Trace the read syscall and print a message for reads >20000 bytes Trace the read syscall and print a message for reads >20000 bytes
trace r::do_sys_return trace 'r::do_sys_return "%llx", retval'
Trace the return from the open syscall Trace the return from the open syscall and print the return value
trace 'c:open (arg2 == 42) "%s %d", arg1, arg2' trace 'c:open (arg2 == 42) "%s %d", arg1, arg2'
Trace the open() call from libc only if the flags (arg2) argument is 42 Trace the open() call from libc only if the flags (arg2) argument is 42
trace 'c:malloc "size = %d", arg1' trace 'c:malloc "size = %d", arg1'
...@@ -218,4 +220,3 @@ trace 't:block:block_rq_complete "sectors=%d", args->nr_sector' ...@@ -218,4 +220,3 @@ trace 't:block:block_rq_complete "sectors=%d", args->nr_sector'
Trace the block_rq_complete kernel tracepoint and print # of tx sectors Trace the block_rq_complete kernel tracepoint and print # of tx sectors
trace 'u:pthread:pthread_create (arg4 != 0)' trace 'u:pthread:pthread_create (arg4 != 0)'
Trace the USDT probe pthread_create when its 4th argument is non-zero Trace the USDT probe pthread_create when its 4th argument is non-zero
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment