#!/usr/bin/env python # # tplist Display kernel tracepoints or USDT probes and their formats. # # USAGE: tplist [-p PID] [-l LIB] [-v] [filter] # # Licensed under the Apache License, Version 2.0 (the "License") # Copyright (C) 2016 Sasha Goldshtein. import argparse import fnmatch import os import re import sys from bcc import USDT trace_root = "/sys/kernel/debug/tracing" event_root = os.path.join(trace_root, "events") parser = argparse.ArgumentParser( description="Display kernel tracepoints or USDT probes " + "and their formats.", formatter_class=argparse.RawDescriptionHelpFormatter) parser.add_argument("-p", "--pid", type=int, default=None, help="List USDT probes in the specified process") parser.add_argument("-l", "--lib", default="", help="List USDT probes in the specified library or executable") parser.add_argument("-v", dest="verbosity", action="count", default=0, help="Increase verbosity level (print variables, arguments, etc.)") parser.add_argument(dest="filter", nargs="?", help="A filter that specifies which probes/tracepoints to print") args = parser.parse_args() def print_tpoint_format(category, event): fmt = open(os.path.join(event_root, category, event, "format")) \ .readlines() for line in fmt: match = re.search(r'field:([^;]*);', line) if match is None: continue parts = match.group(1).split() field_name = parts[-1:][0] field_type = " ".join(parts[:-1]) if field_name.startswith("common_"): continue print(" %s %s;" % (field_type, field_name)) def print_tpoint(category, event): tpoint = "%s:%s" % (category, event) if not args.filter or fnmatch.fnmatch(tpoint, args.filter): print(tpoint) if args.verbosity > 0: print_tpoint_format(category, event) def print_tracepoints(): for category in os.listdir(event_root): cat_dir = os.path.join(event_root, category) if not os.path.isdir(cat_dir): continue for event in os.listdir(cat_dir): evt_dir = os.path.join(cat_dir, event) if os.path.isdir(evt_dir): print_tpoint(category, event) def print_usdt_argument_details(location): for idx in range(0, location.num_arguments): arg = location.get_argument(idx) print(" argument #%d %s" % (idx + 1, arg)) def print_usdt_details(probe): if args.verbosity > 0: print(probe) if args.verbosity > 1: for idx in range(0, probe.num_locations): loc = probe.get_location(idx) print(" location #%d %s" % (idx + 1, loc)) print_usdt_argument_details(loc) else: print(" %d location(s)" % probe.num_locations) print(" %d argument(s)" % probe.num_arguments) else: print("%s %s" % (probe.bin_path.decode(), probe.short_name())) def print_usdt(pid, lib): reader = USDT(path=lib, pid=pid) probes_seen = [] for probe in reader.enumerate_probes(): probe_name = probe.short_name() if not args.filter or fnmatch.fnmatch(probe_name, args.filter): if probe_name in probes_seen: continue probes_seen.append(probe_name) print_usdt_details(probe) if __name__ == "__main__": try: if args.pid or args.lib != "": print_usdt(args.pid, args.lib) else: print_tracepoints() except: if sys.exc_info()[0] is not SystemExit: print(sys.exc_info()[1])