You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
106 lines
4.1 KiB
106 lines
4.1 KiB
4 months ago
|
#!/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:%s" %
|
||
|
(probe.bin_path, probe.provider, probe.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])
|