Skip to content
Snippets Groups Projects
funclatency.py 8.3 KiB
Newer Older
# @lint-avoid-python-3-compatibility-imports
Brendan Gregg's avatar
Brendan Gregg committed
#
# funclatency   Time functions and print latency as a histogram.
#               For Linux, uses BCC, eBPF.
Brendan Gregg's avatar
Brendan Gregg committed
#
# USAGE: funclatency [-h] [-p PID] [-i INTERVAL] [-T] [-u] [-m] [-F] [-r] [-v]
#                    pattern
Brendan Gregg's avatar
Brendan Gregg committed
#
# Run "funclatency -h" for full usage.
#
# The pattern is a string with optional '*' wildcards, similar to file
# globbing. If you'd prefer to use regular expressions, use the -r option.
Brendan Gregg's avatar
Brendan Gregg committed
#
Brendan Gregg's avatar
Brendan Gregg committed
# Currently nested or recursive functions are not supported properly, and
# timestamps will be overwritten, creating dubious output. Try to match single
# functions, or groups of functions that run at the same stack layer, and
# don't ultimately call each other.
#
Brendan Gregg's avatar
Brendan Gregg committed
# Copyright (c) 2015 Brendan Gregg.
# Licensed under the Apache License, Version 2.0 (the "License")
#
# 20-Sep-2015   Brendan Gregg       Created this.
# 06-Oct-2016   Sasha Goldshtein    Added user function support.
Brendan Gregg's avatar
Brendan Gregg committed

from __future__ import print_function
from bcc import BPF
from time import sleep, strftime
import argparse
import signal

# arguments
examples = """examples:
    ./funclatency do_sys_open       # time the do_sys_open() kernel function
    ./funclatency c:read            # time the read() C library function
Brendan Gregg's avatar
Brendan Gregg committed
    ./funclatency -u vfs_read       # time vfs_read(), in microseconds
    ./funclatency -m do_nanosleep   # time do_nanosleep(), in milliseconds
    ./funclatency -i 2 -d 10 c:open # output every 2 seconds, for duration 10s
Brendan Gregg's avatar
Brendan Gregg committed
    ./funclatency -mTi 5 vfs_read   # output every 5 seconds, with timestamps
    ./funclatency -p 181 vfs_read   # time process 181 only
    ./funclatency 'vfs_fstat*'      # time both vfs_fstat() and vfs_fstatat()
    ./funclatency 'c:*printf'       # time the *printf family of functions
Brendan Gregg's avatar
Brendan Gregg committed
    ./funclatency -F 'vfs_r*'       # show one histogram per matched function
Brendan Gregg's avatar
Brendan Gregg committed
"""
parser = argparse.ArgumentParser(
    description="Time functions and print latency as a histogram",
    formatter_class=argparse.RawDescriptionHelpFormatter,
    epilog=examples)
parser.add_argument("-p", "--pid", type=int,
    help="trace this PID only")
parser.add_argument("-i", "--interval", type=int,
    help="summary interval, in seconds")
parser.add_argument("-d", "--duration", type=int,
    help="total duration of trace, in seconds")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("-T", "--timestamp", action="store_true",
    help="include timestamp on output")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("-u", "--microseconds", action="store_true",
    help="microsecond histogram")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("-m", "--milliseconds", action="store_true",
    help="millisecond histogram")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("-F", "--function", action="store_true",
    help="show a separate histogram per function")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("-r", "--regexp", action="store_true",
    help="use regular expressions. Default is \"*\" wildcards only.")
parser.add_argument("-v", "--verbose", action="store_true",
    help="print the BPF program (for debugging purposes)")
Brendan Gregg's avatar
Brendan Gregg committed
parser.add_argument("pattern",
    help="search expression for functions")
parser.add_argument("--ebpf", action="store_true",
    help=argparse.SUPPRESS)
Brendan Gregg's avatar
Brendan Gregg committed
args = parser.parse_args()
if args.duration and not args.interval:
    args.interval = args.duration
if not args.interval:
    args.interval = 99999999

def bail(error):
    print("Error: " + error)
    exit(1)

parts = args.pattern.split(':')
if len(parts) == 1:
    library = None
    pattern = args.pattern
elif len(parts) == 2:
    library = parts[0]
    libpath = BPF.find_library(library) or BPF.find_exe(library)
    if not libpath:
        bail("can't resolve library %s" % library)
    library = libpath
    pattern = parts[1]
else:
    bail("unrecognized pattern format '%s'" % pattern)

Brendan Gregg's avatar
Brendan Gregg committed
if not args.regexp:
    pattern = pattern.replace('*', '.*')
    pattern = '^' + pattern + '$'
Brendan Gregg's avatar
Brendan Gregg committed

# define BPF program
bpf_text = """
#include <uapi/linux/ptrace.h>

typedef struct ip_pid {
    u64 pid;
} ip_pid_t;

typedef struct hist_key {
    ip_pid_t key;
Brendan Gregg's avatar
Brendan Gregg committed

Brendan Gregg's avatar
Brendan Gregg committed
BPF_HASH(start, u32);
BPF_ARRAY(avg, u64, 2);
Brendan Gregg's avatar
Brendan Gregg committed
STORAGE
Brendan Gregg's avatar
Brendan Gregg committed

int trace_func_entry(struct pt_regs *ctx)
{
    u64 pid_tgid = bpf_get_current_pid_tgid();
    u32 pid = pid_tgid;
    u32 tgid = pid_tgid >> 32;
    u64 ts = bpf_ktime_get_ns();
Brendan Gregg's avatar
Brendan Gregg committed

    FILTER
    ENTRYSTORE
    start.update(&pid, &ts);
Brendan Gregg's avatar
Brendan Gregg committed

Brendan Gregg's avatar
Brendan Gregg committed
}

int trace_func_return(struct pt_regs *ctx)
{
    u64 pid_tgid = bpf_get_current_pid_tgid();
    u32 pid = pid_tgid;
    u32 tgid = pid_tgid >> 32;

    // calculate delta time
    tsp = start.lookup(&pid);
    if (tsp == 0) {
        return 0;   // missed start
    }
    delta = bpf_ktime_get_ns() - *tsp;
    start.delete(&pid);

    u32 lat = 0;
    u32 cnt = 1;
    u64 *sum = avg.lookup(&lat);
    if (sum) lock_xadd(sum, delta);
    u64 *cnts = avg.lookup(&cnt);
    if (cnts) lock_xadd(cnts, 1);

    FACTOR

    // store as histogram
    STORE

    return 0;
Brendan Gregg's avatar
Brendan Gregg committed
}
"""
Brendan Gregg's avatar
Brendan Gregg committed

# do we need to store the IP and pid for each invocation?
need_key = args.function or (library and not args.pid)

Brendan Gregg's avatar
Brendan Gregg committed
# code substitutions
Brendan Gregg's avatar
Brendan Gregg committed
if args.pid:
    bpf_text = bpf_text.replace('FILTER',
        'if (tgid != %d) { return 0; }' % args.pid)
Brendan Gregg's avatar
Brendan Gregg committed
else:
    bpf_text = bpf_text.replace('FILTER', '')
Brendan Gregg's avatar
Brendan Gregg committed
if args.milliseconds:
    bpf_text = bpf_text.replace('FACTOR', 'delta /= 1000000;')
    label = "msecs"
Brendan Gregg's avatar
Brendan Gregg committed
elif args.microseconds:
    bpf_text = bpf_text.replace('FACTOR', 'delta /= 1000;')
    label = "usecs"
Brendan Gregg's avatar
Brendan Gregg committed
else:
    bpf_text = bpf_text.replace('FACTOR', '')
    label = "nsecs"
    bpf_text = bpf_text.replace('STORAGE', 'BPF_HASH(ipaddr, u32);\n' +
        'BPF_HISTOGRAM(dist, hist_key_t);')
    # stash the IP on entry, as on return it's kretprobe_trampoline:
    bpf_text = bpf_text.replace('ENTRYSTORE',
        'u64 ip = PT_REGS_IP(ctx); ipaddr.update(&pid, &ip);')
    pid = '-1' if not library else 'tgid'
    bpf_text = bpf_text.replace('STORE',
        """
    u64 ip, *ipp = ipaddr.lookup(&pid);
    if (ipp) {
        ip = *ipp;
        hist_key_t key;
        key.key.ip = ip;
        key.key.pid = %s;
        key.slot = bpf_log2l(delta);
        dist.increment(key);
        ipaddr.delete(&pid);
    }
        """ % pid)
Brendan Gregg's avatar
Brendan Gregg committed
else:
    bpf_text = bpf_text.replace('STORAGE', 'BPF_HISTOGRAM(dist);')
    bpf_text = bpf_text.replace('ENTRYSTORE', '')
    bpf_text = bpf_text.replace('STORE',
        'dist.increment(bpf_log2l(delta));')
if args.verbose or args.ebpf:
Brendan Gregg's avatar
Brendan Gregg committed

# signal handler
def signal_ignore(signal, frame):
Brendan Gregg's avatar
Brendan Gregg committed

# load BPF program
b = BPF(text=bpf_text)

# attach probes
if not library:
    b.attach_kprobe(event_re=pattern, fn_name="trace_func_entry")
    b.attach_kretprobe(event_re=pattern, fn_name="trace_func_return")
    matched = b.num_open_kprobes()
else:
    b.attach_uprobe(name=library, sym_re=pattern, fn_name="trace_func_entry",
                    pid=args.pid or -1)
    b.attach_uretprobe(name=library, sym_re=pattern,
                       fn_name="trace_func_return", pid=args.pid or -1)
    matched = b.num_open_uprobes()

if matched == 0:
    print("0 functions matched by \"%s\". Exiting." % args.pattern)
    exit()
Brendan Gregg's avatar
Brendan Gregg committed

# header
print("Tracing %d functions for \"%s\"... Hit Ctrl-C to end." %
    (matched / 2, args.pattern))
Brendan Gregg's avatar
Brendan Gregg committed

# output
def print_section(key):
    if not library:
        return BPF.sym(key[0], -1)
    else:
        return "%s [%d]" % (BPF.sym(key[0], key[1]), key[1])

Brendan Gregg's avatar
Brendan Gregg committed
exiting = 0 if args.interval else 1
Brendan Gregg's avatar
Brendan Gregg committed
dist = b.get_table("dist")
while (1):
        sleep(args.interval)
        seconds += args.interval
    except KeyboardInterrupt:
        exiting = 1
        # as cleanup can take many seconds, trap Ctrl-C:
        signal.signal(signal.SIGINT, signal_ignore)
    if args.duration and seconds >= args.duration:
        exiting = 1

    print()
    if args.timestamp:
        print("%-8s\n" % strftime("%H:%M:%S"), end="")

    if need_key:
        dist.print_log2_hist(label, "Function", section_print_fn=print_section,
            bucket_fn=lambda k: (k.ip, k.pid))
    else:
        dist.print_log2_hist(label)
    dist.clear()

    total  = b['avg'][0].value
    counts = b['avg'][1].value
    if counts > 0:
        if label == 'msecs':
            total /= 1000000
        elif label == 'usecs':
            total /= 1000
        avg = total/counts
        print("\navg = %ld %s, total: %ld %s, count: %ld\n" %(total/counts, label, total, label, counts))

    if exiting:
        print("Detaching...")
        exit()