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:
commit
b339da4803
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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:
|
||||||
|
@ -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
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
||||||
|
@ -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();
|
||||||
|
@ -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()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
@ -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
|
||||||
|
@ -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]))
|
||||||
|
@ -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())]))
|
||||||
|
@ -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)
|
||||||
|
@ -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)
|
||||||
|
@ -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_()
|
||||||
|
@ -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))
|
||||||
|
|
||||||
|
@ -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]"
|
||||||
|
@ -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():
|
||||||
|
@ -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)
|
||||||
|
@ -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))
|
||||||
|
@ -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))
|
||||||
|
@ -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))
|
||||||
|
@ -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; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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, ¬es->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)
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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)
|
||||||
|
@ -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)
|
||||||
|
@ -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,
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
@ -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 */
|
||||||
|
@ -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;
|
||||||
|
@ -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,
|
||||||
|
@ -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:
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user