perf bpf:

Arnaldo Carvalho de Melo:
 
   - Automatically add BTF ELF markers to 'perf trace' BPF programs, so that
     tools such as 'bpftool map dump' can pretty print map keys and values.
 
 perf c2c:
 
   Jiri Olsa:
 
   - Fix report for empty NUMA node.
 
 perf diff:
 
   Jin Yao:
 
   - Support --time, --cpu, --pid and --tid filter options.
 
 perf probe:
 
   Arnaldo Carvalho de Melo:
 
   - Clarify error message about not finding kernel modules debuginfo.
 
 perf record:
 
   Jiri Olsa:
 
   - Fixup probing for max attr.precise_ip.
 
 perf trace:
 
   Arnaldo Carvalho de Melo:
 
   - Add missing %s lost in the 'msg_flags' recvmmsg arg when adding prefix suppression logic.
 
 perf annotate:
 
   Arnaldo Carvalho de Melo:
 
   - Calculate the max instruction name, align column to that, removing the
     hardcoded max 6 chars and cope with instructions with names longer than that,
     such as vpmovmskb, vpcmpeqb, etc.
 
 kernel:
 
   Song Liu:
 
   - Consider events with attr.bpf_event set as side-band.
 
   Gustavo A. R. Silva:
 
   - Mark expected switch fall-through in perf_event_parse_addr_filter().
 
 Libraries:
 
   Jiri Olsa:
 
   - Fix leaks and double frees on error paths.
 
 libtraceevent:
 
   Tony Jones:
 
   - Fix buffer overflow in arg_eval().
 
 python scripting:
 
   Tony Jones:
 
   - More python3 fixes.
 
 Trivial:
 
   Yang Wei:
 
   - Remove needless extra semicolon in clang C++ glue code.
 
 Intel PT/BTS:
 
   Adrian Hunter:
 
   - Improve auxtrace address filter error message when there is no DSO.
 
   - Fix divide by zero when TSC is not available.
 
   - Further improvements to the export to sqlite/posgresql python scripts
     and to the GUI sqlviewer, exporting 'parent_id' so that we have enable
     the creation of call trees.
 
   Andi Kleen:
 
   - Generalize function to copy from thread addr space from intel-bts code.
 
 Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
 -----BEGIN PGP SIGNATURE-----
 
 iHUEABYIAB0WIQR2GiIUctdOfX2qHhGyPKLppCJ+JwUCXIFXsgAKCRCyPKLppCJ+
 Jz++AQDVDXs1rKyZ5JDmnDpJ1tvVPZM1tTAU+6C/GnnoSDgX/AD+L3smvLoPihbu
 msd3TpSroXuQ7nZ4BQ894jHyX3STqQE=
 =MN9Q
 -----END PGP SIGNATURE-----

Merge tag 'perf-core-for-mingo-5.1-20190307' of git://git.kernel.org/pub/scm/linux/kernel/git/acme/linux into perf/urgent

Pull perf/core changes from Arnaldo Carvalho de Melo:

perf bpf:

  Arnaldo Carvalho de Melo:

  - Automatically add BTF ELF markers to 'perf trace' BPF programs, so that
    tools such as 'bpftool map dump' can pretty print map keys and values.

perf c2c:

  Jiri Olsa:

  - Fix report for empty NUMA node.

perf diff:

  Jin Yao:

  - Support --time, --cpu, --pid and --tid filter options.

perf probe:

  Arnaldo Carvalho de Melo:

  - Clarify error message about not finding kernel modules debuginfo.

perf record:

  Jiri Olsa:

  - Fixup probing for max attr.precise_ip.

perf trace:

  Arnaldo Carvalho de Melo:

  - Add missing %s lost in the 'msg_flags' recvmmsg arg when adding prefix suppression logic.

perf annotate:

  Arnaldo Carvalho de Melo:

  - Calculate the max instruction name, align column to that, removing the
    hardcoded max 6 chars and cope with instructions with names longer than that,
    such as vpmovmskb, vpcmpeqb, etc.

kernel:

  Song Liu:

  - Consider events with attr.bpf_event set as side-band.

  Gustavo A. R. Silva:

  - Mark expected switch fall-through in perf_event_parse_addr_filter().

Libraries:

  Jiri Olsa:

  - Fix leaks and double frees on error paths.

libtraceevent:

  Tony Jones:

  - Fix buffer overflow in arg_eval().

python scripting:

  Tony Jones:

  - More python3 fixes.

Trivial:

  Yang Wei:

  - Remove needless extra semicolon in clang C++ glue code.

Intel PT/BTS:

  Adrian Hunter:

  - Improve auxtrace address filter error message when there is no DSO.

  - Fix divide by zero when TSC is not available.

  - Further improvements to the export to sqlite/posgresql python scripts
    and to the GUI sqlviewer, exporting 'parent_id' so that we have enable
    the creation of call trees.

  Andi Kleen:

  - Generalize function to copy from thread addr space from intel-bts code.

Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Signed-off-by: Ingo Molnar <mingo@kernel.org>
This commit is contained in:
Ingo Molnar 2019-03-09 17:00:17 +01:00
commit b339da4803
51 changed files with 970 additions and 441 deletions

View File

@ -4238,7 +4238,8 @@ static bool is_sb_event(struct perf_event *event)
if (attr->mmap || attr->mmap_data || attr->mmap2 || if (attr->mmap || attr->mmap_data || attr->mmap2 ||
attr->comm || attr->comm_exec || attr->comm || attr->comm_exec ||
attr->task || attr->ksymbol || attr->task || attr->ksymbol ||
attr->context_switch) attr->context_switch ||
attr->bpf_event)
return true; return true;
return false; return false;
} }

View File

@ -2457,7 +2457,7 @@ static int arg_num_eval(struct tep_print_arg *arg, long long *val)
static char *arg_eval (struct tep_print_arg *arg) static char *arg_eval (struct tep_print_arg *arg)
{ {
long long val; long long val;
static char buf[20]; static char buf[24];
switch (arg->type) { switch (arg->type) {
case TEP_PRINT_ATOM: case TEP_PRINT_ATOM:

View File

@ -118,6 +118,62 @@ OPTIONS
sum of shown entries will be always 100%. "absolute" means it retains sum of shown entries will be always 100%. "absolute" means it retains
the original value before and after the filter is applied. the original value before and after the filter is applied.
--time::
Analyze samples within given time window. It supports time
percent with multiple time ranges. Time string is 'a%/n,b%/m,...'
or 'a%-b%,c%-%d,...'.
For example:
Select the second 10% time slice to diff:
perf diff --time 10%/2
Select from 0% to 10% time slice to diff:
perf diff --time 0%-10%
Select the first and the second 10% time slices to diff:
perf diff --time 10%/1,10%/2
Select from 0% to 10% and 30% to 40% slices to diff:
perf diff --time 0%-10%,30%-40%
It also supports analyzing samples within a given time window
<start>,<stop>. Times have the format seconds.microseconds. If 'start'
is not given (i.e., time string is ',x.y') then analysis starts at
the beginning of the file. If stop time is not given (i.e, time
string is 'x.y,') then analysis goes to the end of the file. Time string is
'a1.b1,c1.d1:a2.b2,c2.d2'. Use ':' to separate timestamps for different
perf.data files.
For example, we get the timestamp information from 'perf script'.
perf script -i perf.data.old
mgen 13940 [000] 3946.361400: ...
perf script -i perf.data
mgen 13940 [000] 3971.150589 ...
perf diff --time 3946.361400,:3971.150589,
It analyzes the perf.data.old from the timestamp 3946.361400 to
the end of perf.data.old and analyzes the perf.data from the
timestamp 3971.150589 to the end of perf.data.
--cpu:: Only diff samples for the list of CPUs provided. Multiple CPUs can
be provided as a comma-separated list with no space: 0,1. Ranges of
CPUs are specified with -: 0-2. Default is to report samples on all
CPUs.
--pid=::
Only diff samples for given process ID (comma separated list).
--tid=::
Only diff samples for given thread ID (comma separated list).
COMPARISON COMPARISON
---------- ----------
The comparison is governed by the baseline file. The baseline perf.data The comparison is governed by the baseline file. The baseline perf.data

View File

@ -58,7 +58,7 @@ out_free_source:
} }
static int mov__scnprintf(struct ins *ins, char *bf, size_t size, static int mov__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops); struct ins_operands *ops, int max_ins_name);
static struct ins_ops arm64_mov_ops = { static struct ins_ops arm64_mov_ops = {
.parse = arm64_mov__parse, .parse = arm64_mov__parse,

View File

@ -46,7 +46,7 @@ static int s390_call__parse(struct arch *arch, struct ins_operands *ops,
} }
static int call__scnprintf(struct ins *ins, char *bf, size_t size, static int call__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops); struct ins_operands *ops, int max_ins_name);
static struct ins_ops s390_call_ops = { static struct ins_ops s390_call_ops = {
.parse = s390_call__parse, .parse = s390_call__parse,

View File

@ -2056,6 +2056,12 @@ static int setup_nodes(struct perf_session *session)
if (!set) if (!set)
return -ENOMEM; return -ENOMEM;
nodes[node] = set;
/* empty node, skip */
if (cpu_map__empty(map))
continue;
for (cpu = 0; cpu < map->nr; cpu++) { for (cpu = 0; cpu < map->nr; cpu++) {
set_bit(map->map[cpu], set); set_bit(map->map[cpu], set);
@ -2064,8 +2070,6 @@ static int setup_nodes(struct perf_session *session)
cpu2node[map->map[cpu]] = node; cpu2node[map->map[cpu]] = node;
} }
nodes[node] = set;
} }
setup_nodes_header(); setup_nodes_header();

View File

@ -19,12 +19,21 @@
#include "util/util.h" #include "util/util.h"
#include "util/data.h" #include "util/data.h"
#include "util/config.h" #include "util/config.h"
#include "util/time-utils.h"
#include <errno.h> #include <errno.h>
#include <inttypes.h> #include <inttypes.h>
#include <stdlib.h> #include <stdlib.h>
#include <math.h> #include <math.h>
struct perf_diff {
struct perf_tool tool;
const char *time_str;
struct perf_time_interval *ptime_range;
int range_size;
int range_num;
};
/* Diff command specific HPP columns. */ /* Diff command specific HPP columns. */
enum { enum {
PERF_HPP_DIFF__BASELINE, PERF_HPP_DIFF__BASELINE,
@ -74,6 +83,9 @@ static unsigned int sort_compute = 1;
static s64 compute_wdiff_w1; static s64 compute_wdiff_w1;
static s64 compute_wdiff_w2; static s64 compute_wdiff_w2;
static const char *cpu_list;
static DECLARE_BITMAP(cpu_bitmap, MAX_NR_CPUS);
enum { enum {
COMPUTE_DELTA, COMPUTE_DELTA,
COMPUTE_RATIO, COMPUTE_RATIO,
@ -323,22 +335,33 @@ static int formula_fprintf(struct hist_entry *he, struct hist_entry *pair,
return -1; return -1;
} }
static int diff__process_sample_event(struct perf_tool *tool __maybe_unused, static int diff__process_sample_event(struct perf_tool *tool,
union perf_event *event, union perf_event *event,
struct perf_sample *sample, struct perf_sample *sample,
struct perf_evsel *evsel, struct perf_evsel *evsel,
struct machine *machine) struct machine *machine)
{ {
struct perf_diff *pdiff = container_of(tool, struct perf_diff, tool);
struct addr_location al; struct addr_location al;
struct hists *hists = evsel__hists(evsel); struct hists *hists = evsel__hists(evsel);
int ret = -1; int ret = -1;
if (perf_time__ranges_skip_sample(pdiff->ptime_range, pdiff->range_num,
sample->time)) {
return 0;
}
if (machine__resolve(machine, &al, sample) < 0) { if (machine__resolve(machine, &al, sample) < 0) {
pr_warning("problem processing %d event, skipping it.\n", pr_warning("problem processing %d event, skipping it.\n",
event->header.type); event->header.type);
return -1; return -1;
} }
if (cpu_list && !test_bit(sample->cpu, cpu_bitmap)) {
ret = 0;
goto out_put;
}
if (!hists__add_entry(hists, &al, NULL, NULL, NULL, sample, true)) { if (!hists__add_entry(hists, &al, NULL, NULL, NULL, sample, true)) {
pr_warning("problem incrementing symbol period, skipping event\n"); pr_warning("problem incrementing symbol period, skipping event\n");
goto out_put; goto out_put;
@ -359,7 +382,8 @@ out_put:
return ret; return ret;
} }
static struct perf_tool tool = { static struct perf_diff pdiff = {
.tool = {
.sample = diff__process_sample_event, .sample = diff__process_sample_event,
.mmap = perf_event__process_mmap, .mmap = perf_event__process_mmap,
.mmap2 = perf_event__process_mmap2, .mmap2 = perf_event__process_mmap2,
@ -370,6 +394,7 @@ static struct perf_tool tool = {
.namespaces = perf_event__process_namespaces, .namespaces = perf_event__process_namespaces,
.ordered_events = true, .ordered_events = true,
.ordering_requires_timestamps = true, .ordering_requires_timestamps = true,
},
}; };
static struct perf_evsel *evsel_match(struct perf_evsel *evsel, static struct perf_evsel *evsel_match(struct perf_evsel *evsel,
@ -771,19 +796,117 @@ static void data__free(struct data__file *d)
} }
} }
static int abstime_str_dup(char **pstr)
{
char *str = NULL;
if (pdiff.time_str && strchr(pdiff.time_str, ':')) {
str = strdup(pdiff.time_str);
if (!str)
return -ENOMEM;
}
*pstr = str;
return 0;
}
static int parse_absolute_time(struct data__file *d, char **pstr)
{
char *p = *pstr;
int ret;
/*
* Absolute timestamp for one file has the format: a.b,c.d
* For multiple files, the format is: a.b,c.d:a.b,c.d
*/
p = strchr(*pstr, ':');
if (p) {
if (p == *pstr) {
pr_err("Invalid time string\n");
return -EINVAL;
}
*p = 0;
p++;
if (*p == 0) {
pr_err("Invalid time string\n");
return -EINVAL;
}
}
ret = perf_time__parse_for_ranges(*pstr, d->session,
&pdiff.ptime_range,
&pdiff.range_size,
&pdiff.range_num);
if (ret < 0)
return ret;
if (!p || *p == 0)
*pstr = NULL;
else
*pstr = p;
return ret;
}
static int parse_percent_time(struct data__file *d)
{
int ret;
ret = perf_time__parse_for_ranges(pdiff.time_str, d->session,
&pdiff.ptime_range,
&pdiff.range_size,
&pdiff.range_num);
return ret;
}
static int parse_time_str(struct data__file *d, char *abstime_ostr,
char **pabstime_tmp)
{
int ret = 0;
if (abstime_ostr)
ret = parse_absolute_time(d, pabstime_tmp);
else if (pdiff.time_str)
ret = parse_percent_time(d);
return ret;
}
static int __cmd_diff(void) static int __cmd_diff(void)
{ {
struct data__file *d; struct data__file *d;
int ret = -EINVAL, i; int ret, i;
char *abstime_ostr, *abstime_tmp;
ret = abstime_str_dup(&abstime_ostr);
if (ret)
return ret;
abstime_tmp = abstime_ostr;
ret = -EINVAL;
data__for_each_file(i, d) { data__for_each_file(i, d) {
d->session = perf_session__new(&d->data, false, &tool); d->session = perf_session__new(&d->data, false, &pdiff.tool);
if (!d->session) { if (!d->session) {
pr_err("Failed to open %s\n", d->data.path); pr_err("Failed to open %s\n", d->data.path);
ret = -1; ret = -1;
goto out_delete; goto out_delete;
} }
if (pdiff.time_str) {
ret = parse_time_str(d, abstime_ostr, &abstime_tmp);
if (ret < 0)
goto out_delete;
}
if (cpu_list) {
ret = perf_session__cpu_bitmap(d->session, cpu_list,
cpu_bitmap);
if (ret < 0)
goto out_delete;
}
ret = perf_session__process_events(d->session); ret = perf_session__process_events(d->session);
if (ret) { if (ret) {
pr_err("Failed to process %s\n", d->data.path); pr_err("Failed to process %s\n", d->data.path);
@ -791,6 +914,9 @@ static int __cmd_diff(void)
} }
perf_evlist__collapse_resort(d->session->evlist); perf_evlist__collapse_resort(d->session->evlist);
if (pdiff.ptime_range)
zfree(&pdiff.ptime_range);
} }
data_process(); data_process();
@ -802,6 +928,13 @@ static int __cmd_diff(void)
} }
free(data__files); free(data__files);
if (pdiff.ptime_range)
zfree(&pdiff.ptime_range);
if (abstime_ostr)
free(abstime_ostr);
return ret; return ret;
} }
@ -849,6 +982,13 @@ static const struct option options[] = {
OPT_UINTEGER('o', "order", &sort_compute, "Specify compute sorting."), OPT_UINTEGER('o', "order", &sort_compute, "Specify compute sorting."),
OPT_CALLBACK(0, "percentage", NULL, "relative|absolute", OPT_CALLBACK(0, "percentage", NULL, "relative|absolute",
"How to display percentage of filtered entries", parse_filter_percentage), "How to display percentage of filtered entries", parse_filter_percentage),
OPT_STRING(0, "time", &pdiff.time_str, "str",
"Time span (time percent or absolute timestamp)"),
OPT_STRING(0, "cpu", &cpu_list, "cpu", "list of cpus to profile"),
OPT_STRING(0, "pid", &symbol_conf.pid_list_str, "pid[,pid...]",
"only consider symbols in these pids"),
OPT_STRING(0, "tid", &symbol_conf.tid_list_str, "tid[,tid...]",
"only consider symbols in these tids"),
OPT_END() OPT_END()
}; };

View File

@ -1375,38 +1375,15 @@ repeat:
if (symbol__init(&session->header.env) < 0) if (symbol__init(&session->header.env) < 0)
goto error; goto error;
report.ptime_range = perf_time__range_alloc(report.time_str, if (report.time_str) {
&report.range_size); ret = perf_time__parse_for_ranges(report.time_str, session,
if (!report.ptime_range) { &report.ptime_range,
ret = -ENOMEM; &report.range_size,
&report.range_num);
if (ret < 0)
goto error; goto error;
} }
if (perf_time__parse_str(report.ptime_range, report.time_str) != 0) {
if (session->evlist->first_sample_time == 0 &&
session->evlist->last_sample_time == 0) {
pr_err("HINT: no first/last sample time found in perf data.\n"
"Please use latest perf binary to execute 'perf record'\n"
"(if '--buildid-all' is enabled, please set '--timestamp-boundary').\n");
ret = -EINVAL;
goto error;
}
report.range_num = perf_time__percent_parse_str(
report.ptime_range, report.range_size,
report.time_str,
session->evlist->first_sample_time,
session->evlist->last_sample_time);
if (report.range_num < 0) {
pr_err("Invalid time string\n");
ret = -EINVAL;
goto error;
}
} else {
report.range_num = 1;
}
if (session->tevent.pevent && if (session->tevent.pevent &&
tep_set_function_resolver(session->tevent.pevent, tep_set_function_resolver(session->tevent.pevent,
machine__resolve_kernel_addr, machine__resolve_kernel_addr,
@ -1426,6 +1403,7 @@ repeat:
ret = 0; ret = 0;
error: error:
if (report.ptime_range)
zfree(&report.ptime_range); zfree(&report.ptime_range);
perf_session__delete(session); perf_session__delete(session);

View File

@ -3699,44 +3699,21 @@ int cmd_script(int argc, const char **argv)
if (err < 0) if (err < 0)
goto out_delete; goto out_delete;
script.ptime_range = perf_time__range_alloc(script.time_str, if (script.time_str) {
&script.range_size); err = perf_time__parse_for_ranges(script.time_str, session,
if (!script.ptime_range) { &script.ptime_range,
err = -ENOMEM; &script.range_size,
&script.range_num);
if (err < 0)
goto out_delete; goto out_delete;
} }
/* needs to be parsed after looking up reference time */
if (perf_time__parse_str(script.ptime_range, script.time_str) != 0) {
if (session->evlist->first_sample_time == 0 &&
session->evlist->last_sample_time == 0) {
pr_err("HINT: no first/last sample time found in perf data.\n"
"Please use latest perf binary to execute 'perf record'\n"
"(if '--buildid-all' is enabled, please set '--timestamp-boundary').\n");
err = -EINVAL;
goto out_delete;
}
script.range_num = perf_time__percent_parse_str(
script.ptime_range, script.range_size,
script.time_str,
session->evlist->first_sample_time,
session->evlist->last_sample_time);
if (script.range_num < 0) {
pr_err("Invalid time string\n");
err = -EINVAL;
goto out_delete;
}
} else {
script.range_num = 1;
}
err = __cmd_script(&script); err = __cmd_script(&script);
flush_scripting(); flush_scripting();
out_delete: out_delete:
if (script.ptime_range)
zfree(&script.ptime_range); zfree(&script.ptime_range);
perf_evlist__free_stats(session->evlist); perf_evlist__free_stats(session->evlist);

View File

@ -24,7 +24,13 @@ struct bpf_map SEC("maps") name = { \
.key_size = sizeof(type_key), \ .key_size = sizeof(type_key), \
.value_size = sizeof(type_val), \ .value_size = sizeof(type_val), \
.max_entries = _max_entries, \ .max_entries = _max_entries, \
} }; \
struct ____btf_map_##name { \
type_key key; \
type_val value; \
}; \
struct ____btf_map_##name __attribute__((section(".maps." #name), used)) \
____btf_map_##name = { }
/* /*
* FIXME: this should receive .max_entries as a parameter, as careful * FIXME: this should receive .max_entries as a parameter, as careful

View File

@ -7,6 +7,8 @@
# events, etc. Basically, if this script runs successfully and # events, etc. Basically, if this script runs successfully and
# displays expected results, Python scripting support should be ok. # displays expected results, Python scripting support should be ok.
from __future__ import print_function
import os import os
import sys import sys
@ -19,7 +21,7 @@ from perf_trace_context import *
unhandled = autodict() unhandled = autodict()
def trace_begin(): def trace_begin():
print "trace_begin" print("trace_begin")
pass pass
def trace_end(): def trace_end():
@ -33,8 +35,7 @@ def irq__softirq_entry(event_name, context, common_cpu,
print_uncommon(context) print_uncommon(context)
print "vec=%s\n" % \ print("vec=%s" % (symbol_str("irq__softirq_entry", "vec", vec)))
(symbol_str("irq__softirq_entry", "vec", vec)),
def kmem__kmalloc(event_name, context, common_cpu, def kmem__kmalloc(event_name, context, common_cpu,
common_secs, common_nsecs, common_pid, common_comm, common_secs, common_nsecs, common_pid, common_comm,
@ -45,11 +46,10 @@ def kmem__kmalloc(event_name, context, common_cpu,
print_uncommon(context) print_uncommon(context)
print "call_site=%u, ptr=%u, bytes_req=%u, " \ print("call_site=%u, ptr=%u, bytes_req=%u, "
"bytes_alloc=%u, gfp_flags=%s\n" % \ "bytes_alloc=%u, gfp_flags=%s" %
(call_site, ptr, bytes_req, bytes_alloc, (call_site, ptr, bytes_req, bytes_alloc,
flag_str("kmem__kmalloc", "gfp_flags", gfp_flags)))
flag_str("kmem__kmalloc", "gfp_flags", gfp_flags)),
def trace_unhandled(event_name, context, event_fields_dict): def trace_unhandled(event_name, context, event_fields_dict):
try: try:
@ -58,25 +58,27 @@ def trace_unhandled(event_name, context, event_fields_dict):
unhandled[event_name] = 1 unhandled[event_name] = 1
def print_header(event_name, cpu, secs, nsecs, pid, comm): def print_header(event_name, cpu, secs, nsecs, pid, comm):
print "%-20s %5u %05u.%09u %8u %-20s " % \ print("%-20s %5u %05u.%09u %8u %-20s " %
(event_name, cpu, secs, nsecs, pid, comm), (event_name, cpu, secs, nsecs, pid, comm),
end=' ')
# print trace fields not included in handler args # print trace fields not included in handler args
def print_uncommon(context): def print_uncommon(context):
print "common_preempt_count=%d, common_flags=%s, common_lock_depth=%d, " \ print("common_preempt_count=%d, common_flags=%s, "
% (common_pc(context), trace_flag_str(common_flags(context)), \ "common_lock_depth=%d, " %
common_lock_depth(context)) (common_pc(context), trace_flag_str(common_flags(context)),
common_lock_depth(context)))
def print_unhandled(): def print_unhandled():
keys = unhandled.keys() keys = unhandled.keys()
if not keys: if not keys:
return return
print "\nunhandled events:\n\n", print("\nunhandled events:\n")
print "%-40s %10s\n" % ("event", "count"), print("%-40s %10s" % ("event", "count"))
print "%-40s %10s\n" % ("----------------------------------------", \ print("%-40s %10s" % ("----------------------------------------",
"-----------"), "-----------"))
for event_name in keys: for event_name in keys:
print "%-40s %10d\n" % (event_name, unhandled[event_name]) print("%-40s %10d\n" % (event_name, unhandled[event_name]))

View File

@ -15,6 +15,8 @@
# for a x86 HW PMU event: PEBS with load latency data. # for a x86 HW PMU event: PEBS with load latency data.
# #
from __future__ import print_function
import os import os
import sys import sys
import math import math
@ -37,7 +39,7 @@ con = sqlite3.connect("/dev/shm/perf.db")
con.isolation_level = None con.isolation_level = None
def trace_begin(): def trace_begin():
print "In trace_begin:\n" print("In trace_begin:\n")
# #
# Will create several tables at the start, pebs_ll is for PEBS data with # Will create several tables at the start, pebs_ll is for PEBS data with
@ -76,12 +78,12 @@ def process_event(param_dict):
name = param_dict["ev_name"] name = param_dict["ev_name"]
# Symbol and dso info are not always resolved # Symbol and dso info are not always resolved
if (param_dict.has_key("dso")): if ("dso" in param_dict):
dso = param_dict["dso"] dso = param_dict["dso"]
else: else:
dso = "Unknown_dso" dso = "Unknown_dso"
if (param_dict.has_key("symbol")): if ("symbol" in param_dict):
symbol = param_dict["symbol"] symbol = param_dict["symbol"]
else: else:
symbol = "Unknown_symbol" symbol = "Unknown_symbol"
@ -102,7 +104,7 @@ def insert_db(event):
event.ip, event.status, event.dse, event.dla, event.lat)) event.ip, event.status, event.dse, event.dla, event.lat))
def trace_end(): def trace_end():
print "In trace_end:\n" print("In trace_end:\n")
# We show the basic info for the 2 type of event classes # We show the basic info for the 2 type of event classes
show_general_events() show_general_events()
show_pebs_ll() show_pebs_ll()
@ -123,29 +125,29 @@ def show_general_events():
# Check the total record number in the table # Check the total record number in the table
count = con.execute("select count(*) from gen_events") count = con.execute("select count(*) from gen_events")
for t in count: for t in count:
print "There is %d records in gen_events table" % t[0] print("There is %d records in gen_events table" % t[0])
if t[0] == 0: if t[0] == 0:
return return
print "Statistics about the general events grouped by thread/symbol/dso: \n" print("Statistics about the general events grouped by thread/symbol/dso: \n")
# Group by thread # Group by thread
commq = con.execute("select comm, count(comm) from gen_events group by comm order by -count(comm)") commq = con.execute("select comm, count(comm) from gen_events group by comm order by -count(comm)")
print "\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42) print("\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42))
for row in commq: for row in commq:
print "%16s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%16s %8d %s" % (row[0], row[1], num2sym(row[1])))
# Group by symbol # Group by symbol
print "\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58) print("\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58))
symbolq = con.execute("select symbol, count(symbol) from gen_events group by symbol order by -count(symbol)") symbolq = con.execute("select symbol, count(symbol) from gen_events group by symbol order by -count(symbol)")
for row in symbolq: for row in symbolq:
print "%32s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%32s %8d %s" % (row[0], row[1], num2sym(row[1])))
# Group by dso # Group by dso
print "\n%40s %8s %16s\n%s" % ("dso", "number", "histogram", "="*74) print("\n%40s %8s %16s\n%s" % ("dso", "number", "histogram", "="*74))
dsoq = con.execute("select dso, count(dso) from gen_events group by dso order by -count(dso)") dsoq = con.execute("select dso, count(dso) from gen_events group by dso order by -count(dso)")
for row in dsoq: for row in dsoq:
print "%40s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%40s %8d %s" % (row[0], row[1], num2sym(row[1])))
# #
# This function just shows the basic info, and we could do more with the # This function just shows the basic info, and we could do more with the
@ -156,35 +158,35 @@ def show_pebs_ll():
count = con.execute("select count(*) from pebs_ll") count = con.execute("select count(*) from pebs_ll")
for t in count: for t in count:
print "There is %d records in pebs_ll table" % t[0] print("There is %d records in pebs_ll table" % t[0])
if t[0] == 0: if t[0] == 0:
return return
print "Statistics about the PEBS Load Latency events grouped by thread/symbol/dse/latency: \n" print("Statistics about the PEBS Load Latency events grouped by thread/symbol/dse/latency: \n")
# Group by thread # Group by thread
commq = con.execute("select comm, count(comm) from pebs_ll group by comm order by -count(comm)") commq = con.execute("select comm, count(comm) from pebs_ll group by comm order by -count(comm)")
print "\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42) print("\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42))
for row in commq: for row in commq:
print "%16s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%16s %8d %s" % (row[0], row[1], num2sym(row[1])))
# Group by symbol # Group by symbol
print "\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58) print("\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58))
symbolq = con.execute("select symbol, count(symbol) from pebs_ll group by symbol order by -count(symbol)") symbolq = con.execute("select symbol, count(symbol) from pebs_ll group by symbol order by -count(symbol)")
for row in symbolq: for row in symbolq:
print "%32s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%32s %8d %s" % (row[0], row[1], num2sym(row[1])))
# Group by dse # Group by dse
dseq = con.execute("select dse, count(dse) from pebs_ll group by dse order by -count(dse)") dseq = con.execute("select dse, count(dse) from pebs_ll group by dse order by -count(dse)")
print "\n%32s %8s %16s\n%s" % ("dse", "number", "histogram", "="*58) print("\n%32s %8s %16s\n%s" % ("dse", "number", "histogram", "="*58))
for row in dseq: for row in dseq:
print "%32s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%32s %8d %s" % (row[0], row[1], num2sym(row[1])))
# Group by latency # Group by latency
latq = con.execute("select lat, count(lat) from pebs_ll group by lat order by lat") latq = con.execute("select lat, count(lat) from pebs_ll group by lat order by lat")
print "\n%32s %8s %16s\n%s" % ("latency", "number", "histogram", "="*58) print("\n%32s %8s %16s\n%s" % ("latency", "number", "histogram", "="*58))
for row in latq: for row in latq:
print "%32s %8d %s" % (row[0], row[1], num2sym(row[1])) print("%32s %8d %s" % (row[0], row[1], num2sym(row[1])))
def trace_unhandled(event_name, context, event_fields_dict): def trace_unhandled(event_name, context, event_fields_dict):
print ' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]) print (' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]))

View File

@ -394,7 +394,8 @@ if perf_db_export_calls:
'call_id bigint,' 'call_id bigint,'
'return_id bigint,' 'return_id bigint,'
'parent_call_path_id bigint,' 'parent_call_path_id bigint,'
'flags integer)') 'flags integer,'
'parent_id bigint)')
do_query(query, 'CREATE VIEW machines_view AS ' do_query(query, 'CREATE VIEW machines_view AS '
'SELECT ' 'SELECT '
@ -478,8 +479,9 @@ if perf_db_export_calls:
'branch_count,' 'branch_count,'
'call_id,' 'call_id,'
'return_id,' 'return_id,'
'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE flags END AS flags,' 'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE CAST ( flags AS VARCHAR(6) ) END AS flags,'
'parent_call_path_id' 'parent_call_path_id,'
'calls.parent_id'
' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id') ' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id')
do_query(query, 'CREATE VIEW samples_view AS ' do_query(query, 'CREATE VIEW samples_view AS '
@ -575,6 +577,7 @@ def trace_begin():
sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0) sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)
if perf_db_export_calls or perf_db_export_callchains: if perf_db_export_calls or perf_db_export_callchains:
call_path_table(0, 0, 0, 0) call_path_table(0, 0, 0, 0)
call_return_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)
unhandled_count = 0 unhandled_count = 0
@ -657,6 +660,7 @@ def trace_end():
'ADD CONSTRAINT returnfk FOREIGN KEY (return_id) REFERENCES samples (id),' 'ADD CONSTRAINT returnfk FOREIGN KEY (return_id) REFERENCES samples (id),'
'ADD CONSTRAINT parent_call_pathfk FOREIGN KEY (parent_call_path_id) REFERENCES call_paths (id)') 'ADD CONSTRAINT parent_call_pathfk FOREIGN KEY (parent_call_path_id) REFERENCES call_paths (id)')
do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)') do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)')
do_query(query, 'CREATE INDEX pid_idx ON calls (parent_id)')
if (unhandled_count): if (unhandled_count):
print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events" print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events"
@ -728,7 +732,7 @@ def call_path_table(cp_id, parent_id, symbol_id, ip, *x):
value = struct.pack(fmt, 4, 8, cp_id, 8, parent_id, 8, symbol_id, 8, ip) value = struct.pack(fmt, 4, 8, cp_id, 8, parent_id, 8, symbol_id, 8, ip)
call_path_file.write(value) call_path_file.write(value)
def call_return_table(cr_id, thread_id, comm_id, call_path_id, call_time, return_time, branch_count, call_id, return_id, parent_call_path_id, flags, *x): def call_return_table(cr_id, thread_id, comm_id, call_path_id, call_time, return_time, branch_count, call_id, return_id, parent_call_path_id, flags, parent_id, *x):
fmt = "!hiqiqiqiqiqiqiqiqiqiqii" fmt = "!hiqiqiqiqiqiqiqiqiqiqiiiq"
value = struct.pack(fmt, 11, 8, cr_id, 8, thread_id, 8, comm_id, 8, call_path_id, 8, call_time, 8, return_time, 8, branch_count, 8, call_id, 8, return_id, 8, parent_call_path_id, 4, flags) value = struct.pack(fmt, 12, 8, cr_id, 8, thread_id, 8, comm_id, 8, call_path_id, 8, call_time, 8, return_time, 8, branch_count, 8, call_id, 8, return_id, 8, parent_call_path_id, 4, flags, 8, parent_id)
call_file.write(value) call_file.write(value)

View File

@ -222,7 +222,8 @@ if perf_db_export_calls:
'call_id bigint,' 'call_id bigint,'
'return_id bigint,' 'return_id bigint,'
'parent_call_path_id bigint,' 'parent_call_path_id bigint,'
'flags integer)') 'flags integer,'
'parent_id bigint)')
# printf was added to sqlite in version 3.8.3 # printf was added to sqlite in version 3.8.3
sqlite_has_printf = False sqlite_has_printf = False
@ -321,7 +322,8 @@ if perf_db_export_calls:
'call_id,' 'call_id,'
'return_id,' 'return_id,'
'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE flags END AS flags,' 'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE flags END AS flags,'
'parent_call_path_id' 'parent_call_path_id,'
'parent_id'
' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id') ' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id')
do_query(query, 'CREATE VIEW samples_view AS ' do_query(query, 'CREATE VIEW samples_view AS '
@ -373,7 +375,7 @@ if perf_db_export_calls or perf_db_export_callchains:
call_path_query.prepare("INSERT INTO call_paths VALUES (?, ?, ?, ?)") call_path_query.prepare("INSERT INTO call_paths VALUES (?, ?, ?, ?)")
if perf_db_export_calls: if perf_db_export_calls:
call_query = QSqlQuery(db) call_query = QSqlQuery(db)
call_query.prepare("INSERT INTO calls VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)") call_query.prepare("INSERT INTO calls VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)")
def trace_begin(): def trace_begin():
print datetime.datetime.today(), "Writing records..." print datetime.datetime.today(), "Writing records..."
@ -388,6 +390,7 @@ def trace_begin():
sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0) sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)
if perf_db_export_calls or perf_db_export_callchains: if perf_db_export_calls or perf_db_export_callchains:
call_path_table(0, 0, 0, 0) call_path_table(0, 0, 0, 0)
call_return_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)
unhandled_count = 0 unhandled_count = 0
@ -397,6 +400,7 @@ def trace_end():
print datetime.datetime.today(), "Adding indexes" print datetime.datetime.today(), "Adding indexes"
if perf_db_export_calls: if perf_db_export_calls:
do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)') do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)')
do_query(query, 'CREATE INDEX pid_idx ON calls (parent_id)')
if (unhandled_count): if (unhandled_count):
print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events" print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events"
@ -452,4 +456,4 @@ def call_path_table(*x):
bind_exec(call_path_query, 4, x) bind_exec(call_path_query, 4, x)
def call_return_table(*x): def call_return_table(*x):
bind_exec(call_query, 11, x) bind_exec(call_query, 12, x)

View File

@ -167,9 +167,10 @@ class Thread(QThread):
class TreeModel(QAbstractItemModel): class TreeModel(QAbstractItemModel):
def __init__(self, root, parent=None): def __init__(self, glb, parent=None):
super(TreeModel, self).__init__(parent) super(TreeModel, self).__init__(parent)
self.root = root self.glb = glb
self.root = self.GetRoot()
self.last_row_read = 0 self.last_row_read = 0
def Item(self, parent): def Item(self, parent):
@ -557,24 +558,12 @@ class CallGraphRootItem(CallGraphLevelItemBase):
self.child_items.append(child_item) self.child_items.append(child_item)
self.child_count += 1 self.child_count += 1
# Context-sensitive call graph data model # Context-sensitive call graph data model base
class CallGraphModel(TreeModel): class CallGraphModelBase(TreeModel):
def __init__(self, glb, parent=None): def __init__(self, glb, parent=None):
super(CallGraphModel, self).__init__(CallGraphRootItem(glb), parent) super(CallGraphModelBase, self).__init__(glb, parent)
self.glb = glb
def columnCount(self, parent=None):
return 7
def columnHeader(self, column):
headers = ["Call Path", "Object", "Count ", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "]
return headers[column]
def columnAlignment(self, column):
alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ]
return alignment[column]
def FindSelect(self, value, pattern, query): def FindSelect(self, value, pattern, query):
if pattern: if pattern:
@ -594,34 +583,7 @@ class CallGraphModel(TreeModel):
match = " GLOB '" + str(value) + "'" match = " GLOB '" + str(value) + "'"
else: else:
match = " = '" + str(value) + "'" match = " = '" + str(value) + "'"
QueryExec(query, "SELECT call_path_id, comm_id, thread_id" self.DoFindSelect(query, match)
" FROM calls"
" INNER JOIN call_paths ON calls.call_path_id = call_paths.id"
" INNER JOIN symbols ON call_paths.symbol_id = symbols.id"
" WHERE symbols.name" + match +
" GROUP BY comm_id, thread_id, call_path_id"
" ORDER BY comm_id, thread_id, call_path_id")
def FindPath(self, query):
# Turn the query result into a list of ids that the tree view can walk
# to open the tree at the right place.
ids = []
parent_id = query.value(0)
while parent_id:
ids.insert(0, parent_id)
q2 = QSqlQuery(self.glb.db)
QueryExec(q2, "SELECT parent_id"
" FROM call_paths"
" WHERE id = " + str(parent_id))
if not q2.next():
break
parent_id = q2.value(0)
# The call path root is not used
if ids[0] == 1:
del ids[0]
ids.insert(0, query.value(2))
ids.insert(0, query.value(1))
return ids
def Found(self, query, found): def Found(self, query, found):
if found: if found:
@ -675,6 +637,201 @@ class CallGraphModel(TreeModel):
def FindDone(self, thread, callback, ids): def FindDone(self, thread, callback, ids):
callback(ids) callback(ids)
# Context-sensitive call graph data model
class CallGraphModel(CallGraphModelBase):
def __init__(self, glb, parent=None):
super(CallGraphModel, self).__init__(glb, parent)
def GetRoot(self):
return CallGraphRootItem(self.glb)
def columnCount(self, parent=None):
return 7
def columnHeader(self, column):
headers = ["Call Path", "Object", "Count ", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "]
return headers[column]
def columnAlignment(self, column):
alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ]
return alignment[column]
def DoFindSelect(self, query, match):
QueryExec(query, "SELECT call_path_id, comm_id, thread_id"
" FROM calls"
" INNER JOIN call_paths ON calls.call_path_id = call_paths.id"
" INNER JOIN symbols ON call_paths.symbol_id = symbols.id"
" WHERE symbols.name" + match +
" GROUP BY comm_id, thread_id, call_path_id"
" ORDER BY comm_id, thread_id, call_path_id")
def FindPath(self, query):
# Turn the query result into a list of ids that the tree view can walk
# to open the tree at the right place.
ids = []
parent_id = query.value(0)
while parent_id:
ids.insert(0, parent_id)
q2 = QSqlQuery(self.glb.db)
QueryExec(q2, "SELECT parent_id"
" FROM call_paths"
" WHERE id = " + str(parent_id))
if not q2.next():
break
parent_id = q2.value(0)
# The call path root is not used
if ids[0] == 1:
del ids[0]
ids.insert(0, query.value(2))
ids.insert(0, query.value(1))
return ids
# Call tree data model level 2+ item base
class CallTreeLevelTwoPlusItemBase(CallGraphLevelItemBase):
def __init__(self, glb, row, comm_id, thread_id, calls_id, time, branch_count, parent_item):
super(CallTreeLevelTwoPlusItemBase, self).__init__(glb, row, parent_item)
self.comm_id = comm_id
self.thread_id = thread_id
self.calls_id = calls_id
self.branch_count = branch_count
self.time = time
def Select(self):
self.query_done = True;
if self.calls_id == 0:
comm_thread = " AND comm_id = " + str(self.comm_id) + " AND thread_id = " + str(self.thread_id)
else:
comm_thread = ""
query = QSqlQuery(self.glb.db)
QueryExec(query, "SELECT calls.id, name, short_name, call_time, return_time - call_time, branch_count"
" FROM calls"
" INNER JOIN call_paths ON calls.call_path_id = call_paths.id"
" INNER JOIN symbols ON call_paths.symbol_id = symbols.id"
" INNER JOIN dsos ON symbols.dso_id = dsos.id"
" WHERE calls.parent_id = " + str(self.calls_id) + comm_thread +
" ORDER BY call_time, calls.id")
while query.next():
child_item = CallTreeLevelThreeItem(self.glb, self.child_count, self.comm_id, self.thread_id, query.value(0), query.value(1), query.value(2), query.value(3), int(query.value(4)), int(query.value(5)), self)
self.child_items.append(child_item)
self.child_count += 1
# Call tree data model level three item
class CallTreeLevelThreeItem(CallTreeLevelTwoPlusItemBase):
def __init__(self, glb, row, comm_id, thread_id, calls_id, name, dso, count, time, branch_count, parent_item):
super(CallTreeLevelThreeItem, self).__init__(glb, row, comm_id, thread_id, calls_id, time, branch_count, parent_item)
dso = dsoname(dso)
self.data = [ name, dso, str(count), str(time), PercentToOneDP(time, parent_item.time), str(branch_count), PercentToOneDP(branch_count, parent_item.branch_count) ]
self.dbid = calls_id
# Call tree data model level two item
class CallTreeLevelTwoItem(CallTreeLevelTwoPlusItemBase):
def __init__(self, glb, row, comm_id, thread_id, pid, tid, parent_item):
super(CallTreeLevelTwoItem, self).__init__(glb, row, comm_id, thread_id, 0, 0, 0, parent_item)
self.data = [str(pid) + ":" + str(tid), "", "", "", "", "", ""]
self.dbid = thread_id
def Select(self):
super(CallTreeLevelTwoItem, self).Select()
for child_item in self.child_items:
self.time += child_item.time
self.branch_count += child_item.branch_count
for child_item in self.child_items:
child_item.data[4] = PercentToOneDP(child_item.time, self.time)
child_item.data[6] = PercentToOneDP(child_item.branch_count, self.branch_count)
# Call tree data model level one item
class CallTreeLevelOneItem(CallGraphLevelItemBase):
def __init__(self, glb, row, comm_id, comm, parent_item):
super(CallTreeLevelOneItem, self).__init__(glb, row, parent_item)
self.data = [comm, "", "", "", "", "", ""]
self.dbid = comm_id
def Select(self):
self.query_done = True;
query = QSqlQuery(self.glb.db)
QueryExec(query, "SELECT thread_id, pid, tid"
" FROM comm_threads"
" INNER JOIN threads ON thread_id = threads.id"
" WHERE comm_id = " + str(self.dbid))
while query.next():
child_item = CallTreeLevelTwoItem(self.glb, self.child_count, self.dbid, query.value(0), query.value(1), query.value(2), self)
self.child_items.append(child_item)
self.child_count += 1
# Call tree data model root item
class CallTreeRootItem(CallGraphLevelItemBase):
def __init__(self, glb):
super(CallTreeRootItem, self).__init__(glb, 0, None)
self.dbid = 0
self.query_done = True;
query = QSqlQuery(glb.db)
QueryExec(query, "SELECT id, comm FROM comms")
while query.next():
if not query.value(0):
continue
child_item = CallTreeLevelOneItem(glb, self.child_count, query.value(0), query.value(1), self)
self.child_items.append(child_item)
self.child_count += 1
# Call Tree data model
class CallTreeModel(CallGraphModelBase):
def __init__(self, glb, parent=None):
super(CallTreeModel, self).__init__(glb, parent)
def GetRoot(self):
return CallTreeRootItem(self.glb)
def columnCount(self, parent=None):
return 7
def columnHeader(self, column):
headers = ["Call Path", "Object", "Call Time", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "]
return headers[column]
def columnAlignment(self, column):
alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ]
return alignment[column]
def DoFindSelect(self, query, match):
QueryExec(query, "SELECT calls.id, comm_id, thread_id"
" FROM calls"
" INNER JOIN call_paths ON calls.call_path_id = call_paths.id"
" INNER JOIN symbols ON call_paths.symbol_id = symbols.id"
" WHERE symbols.name" + match +
" ORDER BY comm_id, thread_id, call_time, calls.id")
def FindPath(self, query):
# Turn the query result into a list of ids that the tree view can walk
# to open the tree at the right place.
ids = []
parent_id = query.value(0)
while parent_id:
ids.insert(0, parent_id)
q2 = QSqlQuery(self.glb.db)
QueryExec(q2, "SELECT parent_id"
" FROM calls"
" WHERE id = " + str(parent_id))
if not q2.next():
break
parent_id = q2.value(0)
ids.insert(0, query.value(2))
ids.insert(0, query.value(1))
return ids
# Vertical widget layout # Vertical widget layout
class VBox(): class VBox():
@ -693,28 +850,16 @@ class VBox():
def Widget(self): def Widget(self):
return self.vbox return self.vbox
# Context-sensitive call graph window # Tree window base
class CallGraphWindow(QMdiSubWindow): class TreeWindowBase(QMdiSubWindow):
def __init__(self, glb, parent=None): def __init__(self, parent=None):
super(CallGraphWindow, self).__init__(parent) super(TreeWindowBase, self).__init__(parent)
self.model = LookupCreateModel("Context-Sensitive Call Graph", lambda x=glb: CallGraphModel(x)) self.model = None
self.view = None
self.view = QTreeView() self.find_bar = None
self.view.setModel(self.model)
for c, w in ((0, 250), (1, 100), (2, 60), (3, 70), (4, 70), (5, 100)):
self.view.setColumnWidth(c, w)
self.find_bar = FindBar(self, self)
self.vbox = VBox(self.view, self.find_bar.Widget())
self.setWidget(self.vbox.Widget())
AddSubWindow(glb.mainwindow.mdi_area, self, "Context-Sensitive Call Graph")
def DisplayFound(self, ids): def DisplayFound(self, ids):
if not len(ids): if not len(ids):
@ -747,6 +892,53 @@ class CallGraphWindow(QMdiSubWindow):
if not found: if not found:
self.find_bar.NotFound() self.find_bar.NotFound()
# Context-sensitive call graph window
class CallGraphWindow(TreeWindowBase):
def __init__(self, glb, parent=None):
super(CallGraphWindow, self).__init__(parent)
self.model = LookupCreateModel("Context-Sensitive Call Graph", lambda x=glb: CallGraphModel(x))
self.view = QTreeView()
self.view.setModel(self.model)
for c, w in ((0, 250), (1, 100), (2, 60), (3, 70), (4, 70), (5, 100)):
self.view.setColumnWidth(c, w)
self.find_bar = FindBar(self, self)
self.vbox = VBox(self.view, self.find_bar.Widget())
self.setWidget(self.vbox.Widget())
AddSubWindow(glb.mainwindow.mdi_area, self, "Context-Sensitive Call Graph")
# Call tree window
class CallTreeWindow(TreeWindowBase):
def __init__(self, glb, parent=None):
super(CallTreeWindow, self).__init__(parent)
self.model = LookupCreateModel("Call Tree", lambda x=glb: CallTreeModel(x))
self.view = QTreeView()
self.view.setModel(self.model)
for c, w in ((0, 230), (1, 100), (2, 100), (3, 70), (4, 70), (5, 100)):
self.view.setColumnWidth(c, w)
self.find_bar = FindBar(self, self)
self.vbox = VBox(self.view, self.find_bar.Widget())
self.setWidget(self.vbox.Widget())
AddSubWindow(glb.mainwindow.mdi_area, self, "Call Tree")
# Child data item finder # Child data item finder
class ChildDataItemFinder(): class ChildDataItemFinder():
@ -1327,8 +1519,7 @@ class BranchModel(TreeModel):
progress = Signal(object) progress = Signal(object)
def __init__(self, glb, event_id, where_clause, parent=None): def __init__(self, glb, event_id, where_clause, parent=None):
super(BranchModel, self).__init__(BranchRootItem(), parent) super(BranchModel, self).__init__(glb, parent)
self.glb = glb
self.event_id = event_id self.event_id = event_id
self.more = True self.more = True
self.populated = 0 self.populated = 0
@ -1352,6 +1543,9 @@ class BranchModel(TreeModel):
self.fetcher.done.connect(self.Update) self.fetcher.done.connect(self.Update)
self.fetcher.Fetch(glb_chunk_sz) self.fetcher.Fetch(glb_chunk_sz)
def GetRoot(self):
return BranchRootItem()
def columnCount(self, parent=None): def columnCount(self, parent=None):
return 8 return 8
@ -1863,10 +2057,10 @@ def GetEventList(db):
# Is a table selectable # Is a table selectable
def IsSelectable(db, table): def IsSelectable(db, table, sql = ""):
query = QSqlQuery(db) query = QSqlQuery(db)
try: try:
QueryExec(query, "SELECT * FROM " + table + " LIMIT 1") QueryExec(query, "SELECT * FROM " + table + " " + sql + " LIMIT 1")
except: except:
return False return False
return True return True
@ -2275,9 +2469,10 @@ p.c2 {
</style> </style>
<p class=c1><a href=#reports>1. Reports</a></p> <p class=c1><a href=#reports>1. Reports</a></p>
<p class=c2><a href=#callgraph>1.1 Context-Sensitive Call Graph</a></p> <p class=c2><a href=#callgraph>1.1 Context-Sensitive Call Graph</a></p>
<p class=c2><a href=#allbranches>1.2 All branches</a></p> <p class=c2><a href=#calltree>1.2 Call Tree</a></p>
<p class=c2><a href=#selectedbranches>1.3 Selected branches</a></p> <p class=c2><a href=#allbranches>1.3 All branches</a></p>
<p class=c2><a href=#topcallsbyelapsedtime>1.4 Top calls by elapsed time</a></p> <p class=c2><a href=#selectedbranches>1.4 Selected branches</a></p>
<p class=c2><a href=#topcallsbyelapsedtime>1.5 Top calls by elapsed time</a></p>
<p class=c1><a href=#tables>2. Tables</a></p> <p class=c1><a href=#tables>2. Tables</a></p>
<h1 id=reports>1. Reports</h1> <h1 id=reports>1. Reports</h1>
<h2 id=callgraph>1.1 Context-Sensitive Call Graph</h2> <h2 id=callgraph>1.1 Context-Sensitive Call Graph</h2>
@ -2313,7 +2508,10 @@ v- ls
<h3>Find</h3> <h3>Find</h3>
Ctrl-F displays a Find bar which finds function names by either an exact match or a pattern match. Ctrl-F displays a Find bar which finds function names by either an exact match or a pattern match.
The pattern matching symbols are ? for any character and * for zero or more characters. The pattern matching symbols are ? for any character and * for zero or more characters.
<h2 id=allbranches>1.2 All branches</h2> <h2 id=calltree>1.2 Call Tree</h2>
The Call Tree report is very similar to the Context-Sensitive Call Graph, but the data is not aggregated.
Also the 'Count' column, which would be always 1, is replaced by the 'Call Time'.
<h2 id=allbranches>1.3 All branches</h2>
The All branches report displays all branches in chronological order. The All branches report displays all branches in chronological order.
Not all data is fetched immediately. More records can be fetched using the Fetch bar provided. Not all data is fetched immediately. More records can be fetched using the Fetch bar provided.
<h3>Disassembly</h3> <h3>Disassembly</h3>
@ -2339,10 +2537,10 @@ sudo ldconfig
Ctrl-F displays a Find bar which finds substrings by either an exact match or a regular expression match. Ctrl-F displays a Find bar which finds substrings by either an exact match or a regular expression match.
Refer to Python documentation for the regular expression syntax. Refer to Python documentation for the regular expression syntax.
All columns are searched, but only currently fetched rows are searched. All columns are searched, but only currently fetched rows are searched.
<h2 id=selectedbranches>1.3 Selected branches</h2> <h2 id=selectedbranches>1.4 Selected branches</h2>
This is the same as the <a href=#allbranches>All branches</a> report but with the data reduced This is the same as the <a href=#allbranches>All branches</a> report but with the data reduced
by various selection criteria. A dialog box displays available criteria which are AND'ed together. by various selection criteria. A dialog box displays available criteria which are AND'ed together.
<h3>1.3.1 Time ranges</h3> <h3>1.4.1 Time ranges</h3>
The time ranges hint text shows the total time range. Relative time ranges can also be entered in The time ranges hint text shows the total time range. Relative time ranges can also be entered in
ms, us or ns. Also, negative values are relative to the end of trace. Examples: ms, us or ns. Also, negative values are relative to the end of trace. Examples:
<pre> <pre>
@ -2353,7 +2551,7 @@ ms, us or ns. Also, negative values are relative to the end of trace. Examples:
-10ms- The last 10ms -10ms- The last 10ms
</pre> </pre>
N.B. Due to the granularity of timestamps, there could be no branches in any given time range. N.B. Due to the granularity of timestamps, there could be no branches in any given time range.
<h2 id=topcallsbyelapsedtime>1.4 Top calls by elapsed time</h2> <h2 id=topcallsbyelapsedtime>1.5 Top calls by elapsed time</h2>
The Top calls by elapsed time report displays calls in descending order of time elapsed between when the function was called and when it returned. The Top calls by elapsed time report displays calls in descending order of time elapsed between when the function was called and when it returned.
The data is reduced by various selection criteria. A dialog box displays available criteria which are AND'ed together. The data is reduced by various selection criteria. A dialog box displays available criteria which are AND'ed together.
If not all data is fetched, a Fetch bar is provided. Ctrl-F displays a Find bar. If not all data is fetched, a Fetch bar is provided. Ctrl-F displays a Find bar.
@ -2489,6 +2687,9 @@ class MainWindow(QMainWindow):
if IsSelectable(glb.db, "calls"): if IsSelectable(glb.db, "calls"):
reports_menu.addAction(CreateAction("Context-Sensitive Call &Graph", "Create a new window containing a context-sensitive call graph", self.NewCallGraph, self)) reports_menu.addAction(CreateAction("Context-Sensitive Call &Graph", "Create a new window containing a context-sensitive call graph", self.NewCallGraph, self))
if IsSelectable(glb.db, "calls", "WHERE parent_id >= 0"):
reports_menu.addAction(CreateAction("Call &Tree", "Create a new window containing a call tree", self.NewCallTree, self))
self.EventMenu(GetEventList(glb.db), reports_menu) self.EventMenu(GetEventList(glb.db), reports_menu)
if IsSelectable(glb.db, "calls"): if IsSelectable(glb.db, "calls"):
@ -2549,6 +2750,9 @@ class MainWindow(QMainWindow):
def NewCallGraph(self): def NewCallGraph(self):
CallGraphWindow(self.glb, self) CallGraphWindow(self.glb, self)
def NewCallTree(self):
CallTreeWindow(self.glb, self)
def NewTopCalls(self): def NewTopCalls(self):
dialog = TopCallsDialog(self.glb, self) dialog = TopCallsDialog(self.glb, self)
ret = dialog.exec_() ret = dialog.exec_()

View File

@ -10,6 +10,8 @@
# #
# Measures futex contention # Measures futex contention
from __future__ import print_function
import os, sys import os, sys
sys.path.append(os.environ['PERF_EXEC_PATH'] + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') sys.path.append(os.environ['PERF_EXEC_PATH'] + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace')
from Util import * from Util import *
@ -33,18 +35,18 @@ def syscalls__sys_enter_futex(event, ctxt, cpu, s, ns, tid, comm, callchain,
def syscalls__sys_exit_futex(event, ctxt, cpu, s, ns, tid, comm, callchain, def syscalls__sys_exit_futex(event, ctxt, cpu, s, ns, tid, comm, callchain,
nr, ret): nr, ret):
if thread_blocktime.has_key(tid): if tid in thread_blocktime:
elapsed = nsecs(s, ns) - thread_blocktime[tid] elapsed = nsecs(s, ns) - thread_blocktime[tid]
add_stats(lock_waits, (tid, thread_thislock[tid]), elapsed) add_stats(lock_waits, (tid, thread_thislock[tid]), elapsed)
del thread_blocktime[tid] del thread_blocktime[tid]
del thread_thislock[tid] del thread_thislock[tid]
def trace_begin(): def trace_begin():
print "Press control+C to stop and show the summary" print("Press control+C to stop and show the summary")
def trace_end(): def trace_end():
for (tid, lock) in lock_waits: for (tid, lock) in lock_waits:
min, max, avg, count = lock_waits[tid, lock] min, max, avg, count = lock_waits[tid, lock]
print "%s[%d] lock %x contended %d times, %d avg ns" % \ print("%s[%d] lock %x contended %d times, %d avg ns" %
(process_names[tid], tid, lock, count, avg) (process_names[tid], tid, lock, count, avg))

View File

@ -10,6 +10,8 @@
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
# more details. # more details.
from __future__ import print_function
import os import os
import sys import sys
import struct import struct
@ -22,34 +24,34 @@ sys.path.append(os.environ['PERF_EXEC_PATH'] + \
#from Core import * #from Core import *
def trace_begin(): def trace_begin():
print "Intel PT Power Events and PTWRITE" print("Intel PT Power Events and PTWRITE")
def trace_end(): def trace_end():
print "End" print("End")
def trace_unhandled(event_name, context, event_fields_dict): def trace_unhandled(event_name, context, event_fields_dict):
print ' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]) print(' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]))
def print_ptwrite(raw_buf): def print_ptwrite(raw_buf):
data = struct.unpack_from("<IQ", raw_buf) data = struct.unpack_from("<IQ", raw_buf)
flags = data[0] flags = data[0]
payload = data[1] payload = data[1]
exact_ip = flags & 1 exact_ip = flags & 1
print "IP: %u payload: %#x" % (exact_ip, payload), print("IP: %u payload: %#x" % (exact_ip, payload), end=' ')
def print_cbr(raw_buf): def print_cbr(raw_buf):
data = struct.unpack_from("<BBBBII", raw_buf) data = struct.unpack_from("<BBBBII", raw_buf)
cbr = data[0] cbr = data[0]
f = (data[4] + 500) / 1000 f = (data[4] + 500) / 1000
p = ((cbr * 1000 / data[2]) + 5) / 10 p = ((cbr * 1000 / data[2]) + 5) / 10
print "%3u freq: %4u MHz (%3u%%)" % (cbr, f, p), print("%3u freq: %4u MHz (%3u%%)" % (cbr, f, p), end=' ')
def print_mwait(raw_buf): def print_mwait(raw_buf):
data = struct.unpack_from("<IQ", raw_buf) data = struct.unpack_from("<IQ", raw_buf)
payload = data[1] payload = data[1]
hints = payload & 0xff hints = payload & 0xff
extensions = (payload >> 32) & 0x3 extensions = (payload >> 32) & 0x3
print "hints: %#x extensions: %#x" % (hints, extensions), print("hints: %#x extensions: %#x" % (hints, extensions), end=' ')
def print_pwre(raw_buf): def print_pwre(raw_buf):
data = struct.unpack_from("<IQ", raw_buf) data = struct.unpack_from("<IQ", raw_buf)
@ -57,13 +59,14 @@ def print_pwre(raw_buf):
hw = (payload >> 7) & 1 hw = (payload >> 7) & 1
cstate = (payload >> 12) & 0xf cstate = (payload >> 12) & 0xf
subcstate = (payload >> 8) & 0xf subcstate = (payload >> 8) & 0xf
print "hw: %u cstate: %u sub-cstate: %u" % (hw, cstate, subcstate), print("hw: %u cstate: %u sub-cstate: %u" % (hw, cstate, subcstate),
end=' ')
def print_exstop(raw_buf): def print_exstop(raw_buf):
data = struct.unpack_from("<I", raw_buf) data = struct.unpack_from("<I", raw_buf)
flags = data[0] flags = data[0]
exact_ip = flags & 1 exact_ip = flags & 1
print "IP: %u" % (exact_ip), print("IP: %u" % (exact_ip), end=' ')
def print_pwrx(raw_buf): def print_pwrx(raw_buf):
data = struct.unpack_from("<IQ", raw_buf) data = struct.unpack_from("<IQ", raw_buf)
@ -71,18 +74,21 @@ def print_pwrx(raw_buf):
deepest_cstate = payload & 0xf deepest_cstate = payload & 0xf
last_cstate = (payload >> 4) & 0xf last_cstate = (payload >> 4) & 0xf
wake_reason = (payload >> 8) & 0xf wake_reason = (payload >> 8) & 0xf
print "deepest cstate: %u last cstate: %u wake reason: %#x" % (deepest_cstate, last_cstate, wake_reason), print("deepest cstate: %u last cstate: %u wake reason: %#x" %
(deepest_cstate, last_cstate, wake_reason), end=' ')
def print_common_start(comm, sample, name): def print_common_start(comm, sample, name):
ts = sample["time"] ts = sample["time"]
cpu = sample["cpu"] cpu = sample["cpu"]
pid = sample["pid"] pid = sample["pid"]
tid = sample["tid"] tid = sample["tid"]
print "%16s %5u/%-5u [%03u] %9u.%09u %7s:" % (comm, pid, tid, cpu, ts / 1000000000, ts %1000000000, name), print("%16s %5u/%-5u [%03u] %9u.%09u %7s:" %
(comm, pid, tid, cpu, ts / 1000000000, ts %1000000000, name),
end=' ')
def print_common_ip(sample, symbol, dso): def print_common_ip(sample, symbol, dso):
ip = sample["ip"] ip = sample["ip"]
print "%16x %s (%s)" % (ip, symbol, dso) print("%16x %s (%s)" % (ip, symbol, dso))
def process_event(param_dict): def process_event(param_dict):
event_attr = param_dict["attr"] event_attr = param_dict["attr"]
@ -92,12 +98,12 @@ def process_event(param_dict):
name = param_dict["ev_name"] name = param_dict["ev_name"]
# Symbol and dso info are not always resolved # Symbol and dso info are not always resolved
if (param_dict.has_key("dso")): if "dso" in param_dict:
dso = param_dict["dso"] dso = param_dict["dso"]
else: else:
dso = "[unknown]" dso = "[unknown]"
if (param_dict.has_key("symbol")): if "symbol" in param_dict:
symbol = param_dict["symbol"] symbol = param_dict["symbol"]
else: else:
symbol = "[unknown]" symbol = "[unknown]"

View File

@ -48,7 +48,8 @@ def print_memory_type():
total = sum(load_mem_type_cnt.values()) total = sum(load_mem_type_cnt.values())
for mem_type, count in sorted(load_mem_type_cnt.most_common(), \ for mem_type, count in sorted(load_mem_type_cnt.most_common(), \
key = lambda kv: (kv[1], kv[0]), reverse = True): key = lambda kv: (kv[1], kv[0]), reverse = True):
print("%-40s %10d %10.1f%%\n" % (mem_type, count, 100 * count / total), print("%-40s %10d %10.1f%%\n" %
(mem_type, count, 100 * count / total),
end='') end='')
def trace_begin(): def trace_begin():

View File

@ -124,14 +124,16 @@ def print_receive(hunk):
event = event_list[i] event = event_list[i]
if event['event_name'] == 'napi_poll': if event['event_name'] == 'napi_poll':
print(PF_NAPI_POLL % print(PF_NAPI_POLL %
(diff_msec(base_t, event['event_t']), event['dev'])) (diff_msec(base_t, event['event_t']),
event['dev']))
if i == len(event_list) - 1: if i == len(event_list) - 1:
print("") print("")
else: else:
print(PF_JOINT) print(PF_JOINT)
else: else:
print(PF_NET_RECV % print(PF_NET_RECV %
(diff_msec(base_t, event['event_t']), event['skbaddr'], (diff_msec(base_t, event['event_t']),
event['skbaddr'],
event['len'])) event['len']))
if 'comm' in event.keys(): if 'comm' in event.keys():
print(PF_WJOINT) print(PF_WJOINT)

View File

@ -78,7 +78,8 @@ def print_syscall_totals(interval):
("----------------------------------------", ("----------------------------------------",
"----------")) "----------"))
for id, val in sorted(syscalls.items(), key = lambda kv: (kv[1], kv[0]), \ for id, val in sorted(syscalls.items(),
key = lambda kv: (kv[1], kv[0]),
reverse = True): reverse = True):
try: try:
print("%-40s %10d" % (syscall_name(id), val)) print("%-40s %10d" % (syscall_name(id), val))

View File

@ -41,7 +41,6 @@ def trace_end():
def raw_syscalls__sys_enter(event_name, context, common_cpu, def raw_syscalls__sys_enter(event_name, context, common_cpu,
common_secs, common_nsecs, common_pid, common_comm, common_secs, common_nsecs, common_pid, common_comm,
common_callchain, id, args): common_callchain, id, args):
if (for_comm and common_comm != for_comm) or \ if (for_comm and common_comm != for_comm) or \
(for_pid and common_pid != for_pid ): (for_pid and common_pid != for_pid ):
return return
@ -71,6 +70,6 @@ def print_syscall_totals():
for pid in pid_keys: for pid in pid_keys:
print("\n%s [%d]" % (comm, pid)) print("\n%s [%d]" % (comm, pid))
id_keys = syscalls[comm][pid].keys() id_keys = syscalls[comm][pid].keys()
for id, val in sorted(syscalls[comm][pid].items(), \ for id, val in sorted(syscalls[comm][pid].items(),
key = lambda kv: (kv[1], kv[0]), reverse = True): key = lambda kv: (kv[1], kv[0]), reverse = True):
print(" %-38s %10d" % (syscall_name(id), val)) print(" %-38s %10d" % (syscall_name(id), val))

View File

@ -47,8 +47,7 @@ def raw_syscalls__sys_enter(event_name, context, common_cpu,
syscalls[id] = 1 syscalls[id] = 1
def syscalls__sys_enter(event_name, context, common_cpu, def syscalls__sys_enter(event_name, context, common_cpu,
common_secs, common_nsecs, common_pid, common_comm, common_secs, common_nsecs, common_pid, common_comm, id, args):
id, args):
raw_syscalls__sys_enter(**locals()) raw_syscalls__sys_enter(**locals())
def print_syscall_totals(): def print_syscall_totals():
@ -61,6 +60,6 @@ def print_syscall_totals():
print("%-40s %10s" % ("----------------------------------------", print("%-40s %10s" % ("----------------------------------------",
"-----------")) "-----------"))
for id, val in sorted(syscalls.items(), key = lambda kv: (kv[1], kv[0]), \ for id, val in sorted(syscalls.items(),
reverse = True): key = lambda kv: (kv[1], kv[0]), reverse = True):
print("%-40s %10d" % (syscall_name(id), val)) print("%-40s %10d" % (syscall_name(id), val))

View File

@ -29,7 +29,7 @@ static size_t syscall_arg__scnprintf_msg_flags(char *bf, size_t size,
return scnprintf(bf, size, "NONE"); return scnprintf(bf, size, "NONE");
#define P_MSG_FLAG(n) \ #define P_MSG_FLAG(n) \
if (flags & MSG_##n) { \ if (flags & MSG_##n) { \
printed += scnprintf(bf + printed, size - printed, "%s%s", printed ? "|" : "", show_prefix ? prefix : "", #n); \ printed += scnprintf(bf + printed, size - printed, "%s%s%s", printed ? "|" : "", show_prefix ? prefix : "", #n); \
flags &= ~MSG_##n; \ flags &= ~MSG_##n; \
} }

View File

@ -198,18 +198,18 @@ static void ins__delete(struct ins_operands *ops)
} }
static int ins__raw_scnprintf(struct ins *ins, char *bf, size_t size, static int ins__raw_scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
return scnprintf(bf, size, "%-6s %s", ins->name, ops->raw); return scnprintf(bf, size, "%-*s %s", max_ins_name, ins->name, ops->raw);
} }
int ins__scnprintf(struct ins *ins, char *bf, size_t size, int ins__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
if (ins->ops->scnprintf) if (ins->ops->scnprintf)
return ins->ops->scnprintf(ins, bf, size, ops); return ins->ops->scnprintf(ins, bf, size, ops, max_ins_name);
return ins__raw_scnprintf(ins, bf, size, ops); return ins__raw_scnprintf(ins, bf, size, ops, max_ins_name);
} }
bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2) bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2)
@ -273,18 +273,18 @@ indirect_call:
} }
static int call__scnprintf(struct ins *ins, char *bf, size_t size, static int call__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
if (ops->target.sym) if (ops->target.sym)
return scnprintf(bf, size, "%-6s %s", ins->name, ops->target.sym->name); return scnprintf(bf, size, "%-*s %s", max_ins_name, ins->name, ops->target.sym->name);
if (ops->target.addr == 0) if (ops->target.addr == 0)
return ins__raw_scnprintf(ins, bf, size, ops); return ins__raw_scnprintf(ins, bf, size, ops, max_ins_name);
if (ops->target.name) if (ops->target.name)
return scnprintf(bf, size, "%-6s %s", ins->name, ops->target.name); return scnprintf(bf, size, "%-*s %s", max_ins_name, ins->name, ops->target.name);
return scnprintf(bf, size, "%-6s *%" PRIx64, ins->name, ops->target.addr); return scnprintf(bf, size, "%-*s *%" PRIx64, max_ins_name, ins->name, ops->target.addr);
} }
static struct ins_ops call_ops = { static struct ins_ops call_ops = {
@ -388,15 +388,15 @@ static int jump__parse(struct arch *arch, struct ins_operands *ops, struct map_s
} }
static int jump__scnprintf(struct ins *ins, char *bf, size_t size, static int jump__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
const char *c; const char *c;
if (!ops->target.addr || ops->target.offset < 0) if (!ops->target.addr || ops->target.offset < 0)
return ins__raw_scnprintf(ins, bf, size, ops); return ins__raw_scnprintf(ins, bf, size, ops, max_ins_name);
if (ops->target.outside && ops->target.sym != NULL) if (ops->target.outside && ops->target.sym != NULL)
return scnprintf(bf, size, "%-6s %s", ins->name, ops->target.sym->name); return scnprintf(bf, size, "%-*s %s", max_ins_name, ins->name, ops->target.sym->name);
c = strchr(ops->raw, ','); c = strchr(ops->raw, ',');
c = validate_comma(c, ops); c = validate_comma(c, ops);
@ -415,7 +415,7 @@ static int jump__scnprintf(struct ins *ins, char *bf, size_t size,
c++; c++;
} }
return scnprintf(bf, size, "%-6s %.*s%" PRIx64, return scnprintf(bf, size, "%-*s %.*s%" PRIx64, max_ins_name,
ins->name, c ? c - ops->raw : 0, ops->raw, ins->name, c ? c - ops->raw : 0, ops->raw,
ops->target.offset); ops->target.offset);
} }
@ -483,16 +483,16 @@ out_free_ops:
} }
static int lock__scnprintf(struct ins *ins, char *bf, size_t size, static int lock__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
int printed; int printed;
if (ops->locked.ins.ops == NULL) if (ops->locked.ins.ops == NULL)
return ins__raw_scnprintf(ins, bf, size, ops); return ins__raw_scnprintf(ins, bf, size, ops, max_ins_name);
printed = scnprintf(bf, size, "%-6s ", ins->name); printed = scnprintf(bf, size, "%-*s ", max_ins_name, ins->name);
return printed + ins__scnprintf(&ops->locked.ins, bf + printed, return printed + ins__scnprintf(&ops->locked.ins, bf + printed,
size - printed, ops->locked.ops); size - printed, ops->locked.ops, max_ins_name);
} }
static void lock__delete(struct ins_operands *ops) static void lock__delete(struct ins_operands *ops)
@ -564,9 +564,9 @@ out_free_source:
} }
static int mov__scnprintf(struct ins *ins, char *bf, size_t size, static int mov__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
return scnprintf(bf, size, "%-6s %s,%s", ins->name, return scnprintf(bf, size, "%-*s %s,%s", max_ins_name, ins->name,
ops->source.name ?: ops->source.raw, ops->source.name ?: ops->source.raw,
ops->target.name ?: ops->target.raw); ops->target.name ?: ops->target.raw);
} }
@ -604,9 +604,9 @@ static int dec__parse(struct arch *arch __maybe_unused, struct ins_operands *ops
} }
static int dec__scnprintf(struct ins *ins, char *bf, size_t size, static int dec__scnprintf(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops) struct ins_operands *ops, int max_ins_name)
{ {
return scnprintf(bf, size, "%-6s %s", ins->name, return scnprintf(bf, size, "%-*s %s", max_ins_name, ins->name,
ops->target.name ?: ops->target.raw); ops->target.name ?: ops->target.raw);
} }
@ -616,9 +616,9 @@ static struct ins_ops dec_ops = {
}; };
static int nop__scnprintf(struct ins *ins __maybe_unused, char *bf, size_t size, static int nop__scnprintf(struct ins *ins __maybe_unused, char *bf, size_t size,
struct ins_operands *ops __maybe_unused) struct ins_operands *ops __maybe_unused, int max_ins_name)
{ {
return scnprintf(bf, size, "%-6s", "nop"); return scnprintf(bf, size, "%-*s", max_ins_name, "nop");
} }
static struct ins_ops nop_ops = { static struct ins_ops nop_ops = {
@ -1232,12 +1232,12 @@ void disasm_line__free(struct disasm_line *dl)
annotation_line__delete(&dl->al); annotation_line__delete(&dl->al);
} }
int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw) int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw, int max_ins_name)
{ {
if (raw || !dl->ins.ops) if (raw || !dl->ins.ops)
return scnprintf(bf, size, "%-6s %s", dl->ins.name, dl->ops.raw); return scnprintf(bf, size, "%-*s %s", max_ins_name, dl->ins.name, dl->ops.raw);
return ins__scnprintf(&dl->ins, bf, size, &dl->ops); return ins__scnprintf(&dl->ins, bf, size, &dl->ops, max_ins_name);
} }
static void annotation_line__add(struct annotation_line *al, struct list_head *head) static void annotation_line__add(struct annotation_line *al, struct list_head *head)
@ -2414,12 +2414,30 @@ static inline int width_jumps(int n)
return 1; return 1;
} }
static int annotation__max_ins_name(struct annotation *notes)
{
int max_name = 0, len;
struct annotation_line *al;
list_for_each_entry(al, &notes->src->source, node) {
if (al->offset == -1)
continue;
len = strlen(disasm_line(al)->ins.name);
if (max_name < len)
max_name = len;
}
return max_name;
}
void annotation__init_column_widths(struct annotation *notes, struct symbol *sym) void annotation__init_column_widths(struct annotation *notes, struct symbol *sym)
{ {
notes->widths.addr = notes->widths.target = notes->widths.addr = notes->widths.target =
notes->widths.min_addr = hex_width(symbol__size(sym)); notes->widths.min_addr = hex_width(symbol__size(sym));
notes->widths.max_addr = hex_width(sym->end); notes->widths.max_addr = hex_width(sym->end);
notes->widths.jumps = width_jumps(notes->max_jump_sources); notes->widths.jumps = width_jumps(notes->max_jump_sources);
notes->widths.max_ins_name = annotation__max_ins_name(notes);
} }
void annotation__update_column_widths(struct annotation *notes) void annotation__update_column_widths(struct annotation *notes)
@ -2583,7 +2601,7 @@ call_like:
obj__printf(obj, " "); obj__printf(obj, " ");
} }
disasm_line__scnprintf(dl, bf, size, !notes->options->use_offset); disasm_line__scnprintf(dl, bf, size, !notes->options->use_offset, notes->widths.max_ins_name);
} }
static void ipc_coverage_string(char *bf, int size, struct annotation *notes) static void ipc_coverage_string(char *bf, int size, struct annotation *notes)

View File

@ -59,14 +59,14 @@ struct ins_ops {
void (*free)(struct ins_operands *ops); void (*free)(struct ins_operands *ops);
int (*parse)(struct arch *arch, struct ins_operands *ops, struct map_symbol *ms); int (*parse)(struct arch *arch, struct ins_operands *ops, struct map_symbol *ms);
int (*scnprintf)(struct ins *ins, char *bf, size_t size, int (*scnprintf)(struct ins *ins, char *bf, size_t size,
struct ins_operands *ops); struct ins_operands *ops, int max_ins_name);
}; };
bool ins__is_jump(const struct ins *ins); bool ins__is_jump(const struct ins *ins);
bool ins__is_call(const struct ins *ins); bool ins__is_call(const struct ins *ins);
bool ins__is_ret(const struct ins *ins); bool ins__is_ret(const struct ins *ins);
bool ins__is_lock(const struct ins *ins); bool ins__is_lock(const struct ins *ins);
int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops); int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops, int max_ins_name);
bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2); bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2);
#define ANNOTATION__IPC_WIDTH 6 #define ANNOTATION__IPC_WIDTH 6
@ -219,7 +219,7 @@ int __annotation__scnprintf_samples_period(struct annotation *notes,
struct perf_evsel *evsel, struct perf_evsel *evsel,
bool show_freq); bool show_freq);
int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw); int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw, int max_ins_name);
size_t disasm__fprintf(struct list_head *head, FILE *fp); size_t disasm__fprintf(struct list_head *head, FILE *fp);
void symbol__calc_percent(struct symbol *sym, struct perf_evsel *evsel); void symbol__calc_percent(struct symbol *sym, struct perf_evsel *evsel);
@ -289,6 +289,7 @@ struct annotation {
u8 target; u8 target;
u8 min_addr; u8 min_addr;
u8 max_addr; u8 max_addr;
u8 max_ins_name;
} widths; } widths;
bool have_cycles; bool have_cycles;
struct annotated_source *src; struct annotated_source *src;

View File

@ -1918,7 +1918,8 @@ static struct dso *load_dso(const char *name)
if (!map) if (!map)
return NULL; return NULL;
map__load(map); if (map__load(map) < 0)
pr_err("File '%s' not found or has no symbols.\n", name);
dso = dso__get(map->dso); dso = dso__get(map->dso);

View File

@ -156,7 +156,7 @@ getBPFObjectFromModule(llvm::Module *Module)
#endif #endif
if (NotAdded) { if (NotAdded) {
llvm::errs() << "TargetMachine can't emit a file of this type\n"; llvm::errs() << "TargetMachine can't emit a file of this type\n";
return std::unique_ptr<llvm::SmallVectorImpl<char>>(nullptr);; return std::unique_ptr<llvm::SmallVectorImpl<char>>(nullptr);
} }
PM.run(*Module); PM.run(*Module);

View File

@ -237,7 +237,7 @@ static int open_file(struct perf_data *data)
open_file_read(data) : open_file_write(data); open_file_read(data) : open_file_write(data);
if (fd < 0) { if (fd < 0) {
free(data->file.path); zfree(&data->file.path);
return -1; return -1;
} }
@ -270,7 +270,7 @@ int perf_data__open(struct perf_data *data)
void perf_data__close(struct perf_data *data) void perf_data__close(struct perf_data *data)
{ {
free(data->file.path); zfree(&data->file.path);
close(data->file.fd); close(data->file.fd);
} }

View File

@ -510,19 +510,24 @@ int db_export__call_path(struct db_export *dbe, struct call_path *cp)
return 0; return 0;
} }
int db_export__call_return(struct db_export *dbe, struct call_return *cr) int db_export__call_return(struct db_export *dbe, struct call_return *cr,
u64 *parent_db_id)
{ {
int err; int err;
if (cr->db_id)
return 0;
err = db_export__call_path(dbe, cr->cp); err = db_export__call_path(dbe, cr->cp);
if (err) if (err)
return err; return err;
if (!cr->db_id)
cr->db_id = ++dbe->call_return_last_db_id; cr->db_id = ++dbe->call_return_last_db_id;
if (parent_db_id) {
if (!*parent_db_id)
*parent_db_id = ++dbe->call_return_last_db_id;
cr->parent_db_id = *parent_db_id;
}
if (dbe->export_call_return) if (dbe->export_call_return)
return dbe->export_call_return(dbe, cr); return dbe->export_call_return(dbe, cr);

View File

@ -104,6 +104,7 @@ int db_export__sample(struct db_export *dbe, union perf_event *event,
int db_export__branch_types(struct db_export *dbe); int db_export__branch_types(struct db_export *dbe);
int db_export__call_path(struct db_export *dbe, struct call_path *cp); int db_export__call_path(struct db_export *dbe, struct call_path *cp);
int db_export__call_return(struct db_export *dbe, struct call_return *cr); int db_export__call_return(struct db_export *dbe, struct call_return *cr,
u64 *parent_db_id);
#endif #endif

View File

@ -230,18 +230,33 @@ void perf_evlist__set_leader(struct perf_evlist *evlist)
} }
} }
void perf_event_attr__set_max_precise_ip(struct perf_event_attr *attr) void perf_event_attr__set_max_precise_ip(struct perf_event_attr *pattr)
{ {
attr->precise_ip = 3; struct perf_event_attr attr = {
.type = PERF_TYPE_HARDWARE,
.config = PERF_COUNT_HW_CPU_CYCLES,
.exclude_kernel = 1,
.precise_ip = 3,
};
while (attr->precise_ip != 0) { event_attr_init(&attr);
int fd = sys_perf_event_open(attr, 0, -1, -1, 0);
/*
* Unnamed union member, not supported as struct member named
* initializer in older compilers such as gcc 4.4.7
*/
attr.sample_period = 1;
while (attr.precise_ip != 0) {
int fd = sys_perf_event_open(&attr, 0, -1, -1, 0);
if (fd != -1) { if (fd != -1) {
close(fd); close(fd);
break; break;
} }
--attr->precise_ip; --attr.precise_ip;
} }
pattr->precise_ip = attr.precise_ip;
} }
int __perf_evlist__add_default(struct perf_evlist *evlist, bool precise) int __perf_evlist__add_default(struct perf_evlist *evlist, bool precise)

View File

@ -294,20 +294,12 @@ struct perf_evsel *perf_evsel__new_cycles(bool precise)
if (!precise) if (!precise)
goto new_event; goto new_event;
/*
* Unnamed union member, not supported as struct member named
* initializer in older compilers such as gcc 4.4.7
*
* Just for probing the precise_ip:
*/
attr.sample_period = 1;
perf_event_attr__set_max_precise_ip(&attr); perf_event_attr__set_max_precise_ip(&attr);
/* /*
* Now let the usual logic to set up the perf_event_attr defaults * Now let the usual logic to set up the perf_event_attr defaults
* to kick in when we return and before perf_evsel__open() is called. * to kick in when we return and before perf_evsel__open() is called.
*/ */
attr.sample_period = 0;
new_event: new_event:
evsel = perf_evsel__new(&attr); evsel = perf_evsel__new(&attr);
if (evsel == NULL) if (evsel == NULL)

View File

@ -396,11 +396,8 @@ static int hist_entry__init(struct hist_entry *he,
* adding new entries. So we need to save a copy. * adding new entries. So we need to save a copy.
*/ */
he->branch_info = malloc(sizeof(*he->branch_info)); he->branch_info = malloc(sizeof(*he->branch_info));
if (he->branch_info == NULL) { if (he->branch_info == NULL)
map__zput(he->ms.map); goto err;
free(he->stat_acc);
return -ENOMEM;
}
memcpy(he->branch_info, template->branch_info, memcpy(he->branch_info, template->branch_info,
sizeof(*he->branch_info)); sizeof(*he->branch_info));
@ -419,22 +416,16 @@ static int hist_entry__init(struct hist_entry *he,
if (he->raw_data) { if (he->raw_data) {
he->raw_data = memdup(he->raw_data, he->raw_size); he->raw_data = memdup(he->raw_data, he->raw_size);
if (he->raw_data == NULL)
goto err_infos;
}
if (he->raw_data == NULL) { if (he->srcline) {
map__put(he->ms.map); he->srcline = strdup(he->srcline);
if (he->branch_info) { if (he->srcline == NULL)
map__put(he->branch_info->from.map); goto err_rawdata;
map__put(he->branch_info->to.map);
free(he->branch_info);
}
if (he->mem_info) {
map__put(he->mem_info->iaddr.map);
map__put(he->mem_info->daddr.map);
}
free(he->stat_acc);
return -ENOMEM;
}
} }
INIT_LIST_HEAD(&he->pairs.node); INIT_LIST_HEAD(&he->pairs.node);
thread__get(he->thread); thread__get(he->thread);
he->hroot_in = RB_ROOT_CACHED; he->hroot_in = RB_ROOT_CACHED;
@ -444,6 +435,24 @@ static int hist_entry__init(struct hist_entry *he,
he->leaf = true; he->leaf = true;
return 0; return 0;
err_rawdata:
free(he->raw_data);
err_infos:
if (he->branch_info) {
map__put(he->branch_info->from.map);
map__put(he->branch_info->to.map);
free(he->branch_info);
}
if (he->mem_info) {
map__put(he->mem_info->iaddr.map);
map__put(he->mem_info->daddr.map);
}
err:
map__zput(he->ms.map);
free(he->stat_acc);
return -ENOMEM;
} }
static void *hist_entry__zalloc(size_t size) static void *hist_entry__zalloc(size_t size)
@ -606,7 +615,7 @@ __hists__add_entry(struct hists *hists,
.map = al->map, .map = al->map,
.sym = al->sym, .sym = al->sym,
}, },
.srcline = al->srcline ? strdup(al->srcline) : NULL, .srcline = (char *) al->srcline,
.socket = al->socket, .socket = al->socket,
.cpu = al->cpu, .cpu = al->cpu,
.cpumode = al->cpumode, .cpumode = al->cpumode,
@ -963,7 +972,7 @@ iter_add_next_cumulative_entry(struct hist_entry_iter *iter,
.map = al->map, .map = al->map,
.sym = al->sym, .sym = al->sym,
}, },
.srcline = al->srcline ? strdup(al->srcline) : NULL, .srcline = (char *) al->srcline,
.parent = iter->parent, .parent = iter->parent,
.raw_data = sample->raw_data, .raw_data = sample->raw_data,
.raw_size = sample->raw_size, .raw_size = sample->raw_size,

View File

@ -328,35 +328,19 @@ static int intel_bts_get_next_insn(struct intel_bts_queue *btsq, u64 ip)
{ {
struct machine *machine = btsq->bts->machine; struct machine *machine = btsq->bts->machine;
struct thread *thread; struct thread *thread;
struct addr_location al;
unsigned char buf[INTEL_PT_INSN_BUF_SZ]; unsigned char buf[INTEL_PT_INSN_BUF_SZ];
ssize_t len; ssize_t len;
int x86_64; bool x86_64;
uint8_t cpumode;
int err = -1; int err = -1;
if (machine__kernel_ip(machine, ip))
cpumode = PERF_RECORD_MISC_KERNEL;
else
cpumode = PERF_RECORD_MISC_USER;
thread = machine__find_thread(machine, -1, btsq->tid); thread = machine__find_thread(machine, -1, btsq->tid);
if (!thread) if (!thread)
return -1; return -1;
if (!thread__find_map(thread, cpumode, ip, &al) || !al.map->dso) len = thread__memcpy(thread, machine, buf, ip, INTEL_PT_INSN_BUF_SZ, &x86_64);
goto out_put;
len = dso__data_read_addr(al.map->dso, al.map, machine, ip, buf,
INTEL_PT_INSN_BUF_SZ);
if (len <= 0) if (len <= 0)
goto out_put; goto out_put;
/* Load maps to ensure dso->is_64_bit has been updated */
map__load(al.map);
x86_64 = al.map->dso->is_64_bit;
if (intel_pt_get_insn(buf, len, x86_64, &btsq->intel_pt_insn)) if (intel_pt_get_insn(buf, len, x86_64, &btsq->intel_pt_insn))
goto out_put; goto out_put;

View File

@ -2531,6 +2531,8 @@ int intel_pt_process_auxtrace_info(union perf_event *event,
} }
pt->timeless_decoding = intel_pt_timeless_decoding(pt); pt->timeless_decoding = intel_pt_timeless_decoding(pt);
if (pt->timeless_decoding && !pt->tc.time_mult)
pt->tc.time_mult = 1;
pt->have_tsc = intel_pt_have_tsc(pt); pt->have_tsc = intel_pt_have_tsc(pt);
pt->sampling_mode = false; pt->sampling_mode = false;
pt->est_tsc = !pt->timeless_decoding; pt->est_tsc = !pt->timeless_decoding;

View File

@ -752,6 +752,19 @@ perf_pmu__get_default_config(struct perf_pmu *pmu __maybe_unused)
return NULL; return NULL;
} }
static int pmu_max_precise(const char *name)
{
char path[PATH_MAX];
int max_precise = -1;
scnprintf(path, PATH_MAX,
"bus/event_source/devices/%s/caps/max_precise",
name);
sysfs__read_int(path, &max_precise);
return max_precise;
}
static struct perf_pmu *pmu_lookup(const char *name) static struct perf_pmu *pmu_lookup(const char *name)
{ {
struct perf_pmu *pmu; struct perf_pmu *pmu;
@ -784,6 +797,7 @@ static struct perf_pmu *pmu_lookup(const char *name)
pmu->name = strdup(name); pmu->name = strdup(name);
pmu->type = type; pmu->type = type;
pmu->is_uncore = pmu_is_uncore(name); pmu->is_uncore = pmu_is_uncore(name);
pmu->max_precise = pmu_max_precise(name);
pmu_add_cpu_aliases(&aliases, pmu); pmu_add_cpu_aliases(&aliases, pmu);
INIT_LIST_HEAD(&pmu->format); INIT_LIST_HEAD(&pmu->format);

View File

@ -26,6 +26,7 @@ struct perf_pmu {
__u32 type; __u32 type;
bool selectable; bool selectable;
bool is_uncore; bool is_uncore;
int max_precise;
struct perf_event_attr *default_config; struct perf_event_attr *default_config;
struct cpu_map *cpus; struct cpu_map *cpus;
struct list_head format; /* HEAD struct perf_pmu_format -> list */ struct list_head format; /* HEAD struct perf_pmu_format -> list */

View File

@ -472,9 +472,12 @@ static struct debuginfo *open_debuginfo(const char *module, struct nsinfo *nsi,
strcpy(reason, "(unknown)"); strcpy(reason, "(unknown)");
} else } else
dso__strerror_load(dso, reason, STRERR_BUFSIZE); dso__strerror_load(dso, reason, STRERR_BUFSIZE);
if (!silent) if (!silent) {
pr_err("Failed to find the path for %s: %s\n", if (module)
module ?: "kernel", reason); pr_err("Module %s is not loaded, please specify its full path name.\n", module);
else
pr_err("Failed to find the path for the kernel: %s\n", reason);
}
return NULL; return NULL;
} }
path = dso->long_name; path = dso->long_name;

View File

@ -1173,7 +1173,7 @@ static int python_export_call_return(struct db_export *dbe,
u64 comm_db_id = cr->comm ? cr->comm->db_id : 0; u64 comm_db_id = cr->comm ? cr->comm->db_id : 0;
PyObject *t; PyObject *t;
t = tuple_new(11); t = tuple_new(12);
tuple_set_u64(t, 0, cr->db_id); tuple_set_u64(t, 0, cr->db_id);
tuple_set_u64(t, 1, cr->thread->db_id); tuple_set_u64(t, 1, cr->thread->db_id);
@ -1186,6 +1186,7 @@ static int python_export_call_return(struct db_export *dbe,
tuple_set_u64(t, 8, cr->return_ref); tuple_set_u64(t, 8, cr->return_ref);
tuple_set_u64(t, 9, cr->cp->parent->db_id); tuple_set_u64(t, 9, cr->cp->parent->db_id);
tuple_set_s32(t, 10, cr->flags); tuple_set_s32(t, 10, cr->flags);
tuple_set_u64(t, 11, cr->parent_db_id);
call_object(tables->call_return_handler, t, "call_return_table"); call_object(tables->call_return_handler, t, "call_return_table");
@ -1194,11 +1195,12 @@ static int python_export_call_return(struct db_export *dbe,
return 0; return 0;
} }
static int python_process_call_return(struct call_return *cr, void *data) static int python_process_call_return(struct call_return *cr, u64 *parent_db_id,
void *data)
{ {
struct db_export *dbe = data; struct db_export *dbe = data;
return db_export__call_return(dbe, cr); return db_export__call_return(dbe, cr, parent_db_id);
} }
static void python_process_general_event(struct perf_sample *sample, static void python_process_general_event(struct perf_sample *sample,

View File

@ -140,7 +140,7 @@ struct perf_session *perf_session__new(struct perf_data *data,
if (perf_data__is_read(data)) { if (perf_data__is_read(data)) {
if (perf_session__open(session) < 0) if (perf_session__open(session) < 0)
goto out_close; goto out_delete;
/* /*
* set session attributes that are present in perf.data * set session attributes that are present in perf.data
@ -181,8 +181,6 @@ struct perf_session *perf_session__new(struct perf_data *data,
return session; return session;
out_close:
perf_data__close(data);
out_delete: out_delete:
perf_session__delete(session); perf_session__delete(session);
out: out:

View File

@ -49,6 +49,7 @@ enum retpoline_state_t {
* @timestamp: timestamp (if known) * @timestamp: timestamp (if known)
* @ref: external reference (e.g. db_id of sample) * @ref: external reference (e.g. db_id of sample)
* @branch_count: the branch count when the entry was created * @branch_count: the branch count when the entry was created
* @db_id: id used for db-export
* @cp: call path * @cp: call path
* @no_call: a 'call' was not seen * @no_call: a 'call' was not seen
* @trace_end: a 'call' but trace ended * @trace_end: a 'call' but trace ended
@ -59,6 +60,7 @@ struct thread_stack_entry {
u64 timestamp; u64 timestamp;
u64 ref; u64 ref;
u64 branch_count; u64 branch_count;
u64 db_id;
struct call_path *cp; struct call_path *cp;
bool no_call; bool no_call;
bool trace_end; bool trace_end;
@ -280,12 +282,14 @@ static int thread_stack__call_return(struct thread *thread,
.comm = ts->comm, .comm = ts->comm,
.db_id = 0, .db_id = 0,
}; };
u64 *parent_db_id;
tse = &ts->stack[idx]; tse = &ts->stack[idx];
cr.cp = tse->cp; cr.cp = tse->cp;
cr.call_time = tse->timestamp; cr.call_time = tse->timestamp;
cr.return_time = timestamp; cr.return_time = timestamp;
cr.branch_count = ts->branch_count - tse->branch_count; cr.branch_count = ts->branch_count - tse->branch_count;
cr.db_id = tse->db_id;
cr.call_ref = tse->ref; cr.call_ref = tse->ref;
cr.return_ref = ref; cr.return_ref = ref;
if (tse->no_call) if (tse->no_call)
@ -295,7 +299,14 @@ static int thread_stack__call_return(struct thread *thread,
if (tse->non_call) if (tse->non_call)
cr.flags |= CALL_RETURN_NON_CALL; cr.flags |= CALL_RETURN_NON_CALL;
return crp->process(&cr, crp->data); /*
* The parent db_id must be assigned before exporting the child. Note
* it is not possible to export the parent first because its information
* is not yet complete because its 'return' has not yet been processed.
*/
parent_db_id = idx ? &(tse - 1)->db_id : NULL;
return crp->process(&cr, parent_db_id, crp->data);
} }
static int __thread_stack__flush(struct thread *thread, struct thread_stack *ts) static int __thread_stack__flush(struct thread *thread, struct thread_stack *ts)
@ -484,7 +495,7 @@ void thread_stack__sample(struct thread *thread, int cpu,
} }
struct call_return_processor * struct call_return_processor *
call_return_processor__new(int (*process)(struct call_return *cr, void *data), call_return_processor__new(int (*process)(struct call_return *cr, u64 *parent_db_id, void *data),
void *data) void *data)
{ {
struct call_return_processor *crp; struct call_return_processor *crp;
@ -537,6 +548,7 @@ static int thread_stack__push_cp(struct thread_stack *ts, u64 ret_addr,
tse->no_call = no_call; tse->no_call = no_call;
tse->trace_end = trace_end; tse->trace_end = trace_end;
tse->non_call = false; tse->non_call = false;
tse->db_id = 0;
return 0; return 0;
} }

View File

@ -55,6 +55,7 @@ enum {
* @call_ref: external reference to 'call' sample (e.g. db_id) * @call_ref: external reference to 'call' sample (e.g. db_id)
* @return_ref: external reference to 'return' sample (e.g. db_id) * @return_ref: external reference to 'return' sample (e.g. db_id)
* @db_id: id used for db-export * @db_id: id used for db-export
* @parent_db_id: id of parent call used for db-export
* @flags: Call/Return flags * @flags: Call/Return flags
*/ */
struct call_return { struct call_return {
@ -67,6 +68,7 @@ struct call_return {
u64 call_ref; u64 call_ref;
u64 return_ref; u64 return_ref;
u64 db_id; u64 db_id;
u64 parent_db_id;
u32 flags; u32 flags;
}; };
@ -79,7 +81,7 @@ struct call_return {
*/ */
struct call_return_processor { struct call_return_processor {
struct call_path_root *cpr; struct call_path_root *cpr;
int (*process)(struct call_return *cr, void *data); int (*process)(struct call_return *cr, u64 *parent_db_id, void *data);
void *data; void *data;
}; };
@ -93,7 +95,7 @@ void thread_stack__free(struct thread *thread);
size_t thread_stack__depth(struct thread *thread, int cpu); size_t thread_stack__depth(struct thread *thread, int cpu);
struct call_return_processor * struct call_return_processor *
call_return_processor__new(int (*process)(struct call_return *cr, void *data), call_return_processor__new(int (*process)(struct call_return *cr, u64 *parent_db_id, void *data),
void *data); void *data);
void call_return_processor__free(struct call_return_processor *crp); void call_return_processor__free(struct call_return_processor *crp);
int thread_stack__process(struct thread *thread, struct comm *comm, int thread_stack__process(struct thread *thread, struct comm *comm,

View File

@ -12,6 +12,7 @@
#include "debug.h" #include "debug.h"
#include "namespaces.h" #include "namespaces.h"
#include "comm.h" #include "comm.h"
#include "map.h"
#include "symbol.h" #include "symbol.h"
#include "unwind.h" #include "unwind.h"
@ -393,3 +394,25 @@ struct thread *thread__main_thread(struct machine *machine, struct thread *threa
return machine__find_thread(machine, thread->pid_, thread->pid_); return machine__find_thread(machine, thread->pid_, thread->pid_);
} }
int thread__memcpy(struct thread *thread, struct machine *machine,
void *buf, u64 ip, int len, bool *is64bit)
{
u8 cpumode = PERF_RECORD_MISC_USER;
struct addr_location al;
long offset;
if (machine__kernel_ip(machine, ip))
cpumode = PERF_RECORD_MISC_KERNEL;
if (!thread__find_map(thread, cpumode, ip, &al) || !al.map->dso ||
al.map->dso->data.status == DSO_DATA_STATUS_ERROR ||
map__load(al.map) < 0)
return -1;
offset = al.map->map_ip(al.map, ip);
if (is64bit)
*is64bit = al.map->dso->is_64_bit;
return dso__data_read_offset(al.map->dso, machine, offset, buf, len);
}

View File

@ -113,6 +113,9 @@ struct symbol *thread__find_symbol_fb(struct thread *thread, u8 cpumode,
void thread__find_cpumode_addr_location(struct thread *thread, u64 addr, void thread__find_cpumode_addr_location(struct thread *thread, u64 addr,
struct addr_location *al); struct addr_location *al);
int thread__memcpy(struct thread *thread, struct machine *machine,
void *buf, u64 ip, int len, bool *is64bit);
static inline void *thread__priv(struct thread *thread) static inline void *thread__priv(struct thread *thread)
{ {
return thread->priv; return thread->priv;

View File

@ -11,6 +11,8 @@
#include "perf.h" #include "perf.h"
#include "debug.h" #include "debug.h"
#include "time-utils.h" #include "time-utils.h"
#include "session.h"
#include "evlist.h"
int parse_nsec_time(const char *str, u64 *ptime) int parse_nsec_time(const char *str, u64 *ptime)
{ {
@ -374,7 +376,7 @@ bool perf_time__ranges_skip_sample(struct perf_time_interval *ptime_buf,
struct perf_time_interval *ptime; struct perf_time_interval *ptime;
int i; int i;
if ((timestamp == 0) || (num == 0)) if ((!ptime_buf) || (timestamp == 0) || (num == 0))
return false; return false;
if (num == 1) if (num == 1)
@ -396,6 +398,53 @@ bool perf_time__ranges_skip_sample(struct perf_time_interval *ptime_buf,
return (i == num) ? true : false; return (i == num) ? true : false;
} }
int perf_time__parse_for_ranges(const char *time_str,
struct perf_session *session,
struct perf_time_interval **ranges,
int *range_size, int *range_num)
{
struct perf_time_interval *ptime_range;
int size, num, ret;
ptime_range = perf_time__range_alloc(time_str, &size);
if (!ptime_range)
return -ENOMEM;
if (perf_time__parse_str(ptime_range, time_str) != 0) {
if (session->evlist->first_sample_time == 0 &&
session->evlist->last_sample_time == 0) {
pr_err("HINT: no first/last sample time found in perf data.\n"
"Please use latest perf binary to execute 'perf record'\n"
"(if '--buildid-all' is enabled, please set '--timestamp-boundary').\n");
ret = -EINVAL;
goto error;
}
num = perf_time__percent_parse_str(
ptime_range, size,
time_str,
session->evlist->first_sample_time,
session->evlist->last_sample_time);
if (num < 0) {
pr_err("Invalid time string\n");
ret = -EINVAL;
goto error;
}
} else {
num = 1;
}
*range_size = size;
*range_num = num;
*ranges = ptime_range;
return 0;
error:
free(ptime_range);
return ret;
}
int timestamp__scnprintf_usec(u64 timestamp, char *buf, size_t sz) int timestamp__scnprintf_usec(u64 timestamp, char *buf, size_t sz)
{ {
u64 sec = timestamp / NSEC_PER_SEC; u64 sec = timestamp / NSEC_PER_SEC;

View File

@ -23,6 +23,12 @@ bool perf_time__skip_sample(struct perf_time_interval *ptime, u64 timestamp);
bool perf_time__ranges_skip_sample(struct perf_time_interval *ptime_buf, bool perf_time__ranges_skip_sample(struct perf_time_interval *ptime_buf,
int num, u64 timestamp); int num, u64 timestamp);
struct perf_session;
int perf_time__parse_for_ranges(const char *str, struct perf_session *session,
struct perf_time_interval **ranges,
int *range_size, int *range_num);
int timestamp__scnprintf_usec(u64 timestamp, char *buf, size_t sz); int timestamp__scnprintf_usec(u64 timestamp, char *buf, size_t sz);
int fetch_current_timestamp(char *buf, size_t sz); int fetch_current_timestamp(char *buf, size_t sz);