Commit 8c0e4b9f authored by 4ast's avatar 4ast Committed by GitHub

Merge pull request #1130 from cherusk/master

softirqs: focus CPU as disector
parents f609af25 46186905
...@@ -33,6 +33,8 @@ parser.add_argument("-N", "--nanoseconds", action="store_true", ...@@ -33,6 +33,8 @@ parser.add_argument("-N", "--nanoseconds", action="store_true",
help="output in nanoseconds") help="output in nanoseconds")
parser.add_argument("-d", "--dist", action="store_true", parser.add_argument("-d", "--dist", action="store_true",
help="show distributions as histograms") help="show distributions as histograms")
parser.add_argument("-C", "--bycpu", action="store_true",
help="break down softirqs to individual cpus")
parser.add_argument("interval", nargs="?", default=99999999, parser.add_argument("interval", nargs="?", default=99999999,
help="output interval, in seconds") help="output interval, in seconds")
parser.add_argument("count", nargs="?", default=99999999, parser.add_argument("count", nargs="?", default=99999999,
...@@ -48,58 +50,102 @@ else: ...@@ -48,58 +50,102 @@ else:
debug = 0 debug = 0
# define BPF program # define BPF program
bpf_text = """ bpf_text = ""
#include <uapi/linux/ptrace.h> if args.bycpu:
bpf_text = """
#include <uapi/linux/ptrace.h>
typedef struct irq_key { typedef struct irq_cpu_key {
s64 cpu;
u64 slot;
} irq_key_t;
BPF_HASH(start, u32);
BPF_HISTOGRAM(dist, irq_key_t);
// time IRQ
int trace_start_cpu(struct pt_regs *ctx)
{
int curr_cpu = bpf_get_smp_processor_id();
u64 ts = bpf_ktime_get_ns();
start.update(&curr_cpu, &ts);
return 0;
}
int trace_completion_cpu(struct pt_regs *ctx)
{
u64 *tsp, delta;
int curr_cpu = bpf_get_smp_processor_id();
// fetch timestamp and calculate delta
tsp = start.lookup(&curr_cpu);
COMMON
// store as sum or histogram
irq_key_t key = {.cpu = curr_cpu,
STORE
start.delete(&curr_cpu);
return 0;
}
"""
else:
bpf_text = """
#include <uapi/linux/ptrace.h>
typedef struct irq_key {
u64 ip; u64 ip;
u64 slot; u64 slot;
} irq_key_t; } irq_key_t;
BPF_HASH(start, u32);
BPF_HASH(iptr, u32); BPF_HASH(start, u32);
BPF_HISTOGRAM(dist, irq_key_t); BPF_HASH(iptr, u32);
BPF_HISTOGRAM(dist, irq_key_t);
// time IRQ
int trace_start(struct pt_regs *ctx) // time IRQ
{ int trace_start(struct pt_regs *ctx)
{
u32 pid = bpf_get_current_pid_tgid(); u32 pid = bpf_get_current_pid_tgid();
u64 ip = PT_REGS_IP(ctx), ts = bpf_ktime_get_ns(); u64 ip = PT_REGS_IP(ctx), ts = bpf_ktime_get_ns();
start.update(&pid, &ts); start.update(&pid, &ts);
iptr.update(&pid, &ip); iptr.update(&pid, &ip);
return 0; return 0;
} }
int trace_completion(struct pt_regs *ctx) int trace_completion(struct pt_regs *ctx)
{ {
u64 *tsp, delta, ip, *ipp; u64 *tsp, delta, ip, *ipp;
u32 pid = bpf_get_current_pid_tgid(); u32 pid = bpf_get_current_pid_tgid();
// fetch timestamp and calculate delta // fetch timestamp and calculate delta
tsp = start.lookup(&pid); tsp = start.lookup(&pid);
ipp = iptr.lookup(&pid); ipp = iptr.lookup(&pid);
if (tsp == 0 || ipp == 0) { COMMON
return 0; // missed start
}
delta = bpf_ktime_get_ns() - *tsp;
ip = *ipp;
// store as sum or histogram // store as sum or histogram
irq_key_t key = {
STORE STORE
start.delete(&pid); start.delete(&pid);
iptr.delete(&pid); iptr.delete(&pid);
return 0; return 0;
} }
""" """
# code substitutions # code substitutions
bpf_text = bpf_text.replace('COMMON',
"""if (tsp == 0) {
return 0; // missed start
}
delta = bpf_ktime_get_ns() - *tsp;
""")
if args.dist: if args.dist:
bpf_text = bpf_text.replace('STORE', bpf_text = bpf_text.replace('STORE',
'irq_key_t key = {.ip = ip, .slot = bpf_log2l(delta)};' + '.slot = bpf_log2l(delta)};' +
'dist.increment(key);') 'dist.increment(key);')
else: else:
bpf_text = bpf_text.replace('STORE', bpf_text = bpf_text.replace('STORE',
'irq_key_t key = {.ip = ip, .slot = 0 /* ignore */};' + ' .ip = ip, .slot = 0 /* ignore */};' +
'u64 zero = 0, *vp = dist.lookup_or_init(&key, &zero);' + 'u64 zero = 0, *vp = dist.lookup_or_init(&key, &zero);' +
'(*vp) += delta;') '(*vp) += delta;')
if debug: if debug:
...@@ -115,6 +161,10 @@ for softirqfunc in ("blk_iopoll_softirq", "blk_done_softirq", ...@@ -115,6 +161,10 @@ for softirqfunc in ("blk_iopoll_softirq", "blk_done_softirq",
"rcu_process_callbacks", "run_rebalance_domains", "tasklet_action", "rcu_process_callbacks", "run_rebalance_domains", "tasklet_action",
"tasklet_hi_action", "run_timer_softirq", "net_tx_action", "tasklet_hi_action", "run_timer_softirq", "net_tx_action",
"net_rx_action"): "net_rx_action"):
if args.bycpu:
b.attach_kprobe(event=softirqfunc, fn_name="trace_start_cpu")
b.attach_kretprobe(event=softirqfunc, fn_name="trace_completion_cpu")
else:
b.attach_kprobe(event=softirqfunc, fn_name="trace_start") b.attach_kprobe(event=softirqfunc, fn_name="trace_start")
b.attach_kretprobe(event=softirqfunc, fn_name="trace_completion") b.attach_kretprobe(event=softirqfunc, fn_name="trace_completion")
...@@ -134,7 +184,15 @@ while (1): ...@@ -134,7 +184,15 @@ while (1):
print("%-8s\n" % strftime("%H:%M:%S"), end="") print("%-8s\n" % strftime("%H:%M:%S"), end="")
if args.dist: if args.dist:
if args.bycpu:
dist.print_log2_hist(label, "CPU")
else:
dist.print_log2_hist(label, "softirq", section_print_fn=b.ksym) dist.print_log2_hist(label, "softirq", section_print_fn=b.ksym)
else:
if args.bycpu:
print("%-26s %11s %11s" % ("SOFTIRQ", "CPU", "TOTAL_" + label))
for k, v in sorted(dist.items(), key=lambda dist: dist[1].value):
print("%-26s %11d %11d" % (b.ksym(k.ip), k.cpu, v.value / factor))
else: else:
print("%-26s %11s" % ("SOFTIRQ", "TOTAL_" + label)) print("%-26s %11s" % ("SOFTIRQ", "TOTAL_" + label))
for k, v in sorted(dist.items(), key=lambda dist: dist[1].value): for k, v in sorted(dist.items(), key=lambda dist: dist[1].value):
......
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