Commit a0b51af3 authored by Namhyung Kim's avatar Namhyung Kim Committed by Jiri Olsa

perf hists: Check if accumulated when adding a hist entry

To support callchain accumulation, @entry should be recognized if it's
accumulated or not when add_hist_entry() called.  The period of an
accumulated entry should be added to ->stat_acc but not ->stat. Add
@sample_self arg for that.
Signed-off-by: default avatarNamhyung Kim <namhyung@kernel.org>
Tested-by: default avatarArun Sharma <asharma@fb.com>
Tested-by: default avatarRodrigo Campos <rodrigo@sdfg.com.ar>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Link: http://lkml.kernel.org/r/1401335910-16832-5-git-send-email-namhyung@kernel.orgSigned-off-by: default avatarJiri Olsa <jolsa@kernel.org>
parent f8be1c8c
...@@ -65,7 +65,8 @@ static int perf_evsel__add_sample(struct perf_evsel *evsel, ...@@ -65,7 +65,8 @@ static int perf_evsel__add_sample(struct perf_evsel *evsel,
return 0; return 0;
} }
he = __hists__add_entry(&evsel->hists, al, NULL, NULL, NULL, 1, 1, 0); he = __hists__add_entry(&evsel->hists, al, NULL, NULL, NULL, 1, 1, 0,
true);
if (he == NULL) if (he == NULL)
return -ENOMEM; return -ENOMEM;
......
...@@ -315,7 +315,7 @@ static int hists__add_entry(struct hists *hists, ...@@ -315,7 +315,7 @@ static int hists__add_entry(struct hists *hists,
u64 weight, u64 transaction) u64 weight, u64 transaction)
{ {
if (__hists__add_entry(hists, al, NULL, NULL, NULL, period, weight, if (__hists__add_entry(hists, al, NULL, NULL, NULL, period, weight,
transaction) != NULL) transaction, true) != NULL)
return 0; return 0;
return -ENOMEM; return -ENOMEM;
} }
......
...@@ -247,7 +247,7 @@ static struct hist_entry *perf_evsel__add_hist_entry(struct perf_evsel *evsel, ...@@ -247,7 +247,7 @@ static struct hist_entry *perf_evsel__add_hist_entry(struct perf_evsel *evsel,
pthread_mutex_lock(&evsel->hists.lock); pthread_mutex_lock(&evsel->hists.lock);
he = __hists__add_entry(&evsel->hists, al, NULL, NULL, NULL, he = __hists__add_entry(&evsel->hists, al, NULL, NULL, NULL,
sample->period, sample->weight, sample->period, sample->weight,
sample->transaction); sample->transaction, true);
pthread_mutex_unlock(&evsel->hists.lock); pthread_mutex_unlock(&evsel->hists.lock);
if (he == NULL) if (he == NULL)
return NULL; return NULL;
......
...@@ -88,7 +88,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine) ...@@ -88,7 +88,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine)
goto out; goto out;
he = __hists__add_entry(&evsel->hists, &al, NULL, he = __hists__add_entry(&evsel->hists, &al, NULL,
NULL, NULL, 1, 1, 0); NULL, NULL, 1, 1, 0, true);
if (he == NULL) if (he == NULL)
goto out; goto out;
...@@ -112,7 +112,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine) ...@@ -112,7 +112,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine)
goto out; goto out;
he = __hists__add_entry(&evsel->hists, &al, NULL, he = __hists__add_entry(&evsel->hists, &al, NULL,
NULL, NULL, 1, 1, 0); NULL, NULL, 1, 1, 0, true);
if (he == NULL) if (he == NULL)
goto out; goto out;
......
...@@ -279,7 +279,8 @@ void hists__decay_entries(struct hists *hists, bool zap_user, bool zap_kernel) ...@@ -279,7 +279,8 @@ void hists__decay_entries(struct hists *hists, bool zap_user, bool zap_kernel)
* histogram, sorted on item, collects periods * histogram, sorted on item, collects periods
*/ */
static struct hist_entry *hist_entry__new(struct hist_entry *template) static struct hist_entry *hist_entry__new(struct hist_entry *template,
bool sample_self)
{ {
size_t callchain_size = 0; size_t callchain_size = 0;
struct hist_entry *he; struct hist_entry *he;
...@@ -299,6 +300,8 @@ static struct hist_entry *hist_entry__new(struct hist_entry *template) ...@@ -299,6 +300,8 @@ static struct hist_entry *hist_entry__new(struct hist_entry *template)
return NULL; return NULL;
} }
memcpy(he->stat_acc, &he->stat, sizeof(he->stat)); memcpy(he->stat_acc, &he->stat, sizeof(he->stat));
if (!sample_self)
memset(&he->stat, 0, sizeof(he->stat));
} }
if (he->ms.map) if (he->ms.map)
...@@ -351,7 +354,8 @@ static u8 symbol__parent_filter(const struct symbol *parent) ...@@ -351,7 +354,8 @@ static u8 symbol__parent_filter(const struct symbol *parent)
static struct hist_entry *add_hist_entry(struct hists *hists, static struct hist_entry *add_hist_entry(struct hists *hists,
struct hist_entry *entry, struct hist_entry *entry,
struct addr_location *al) struct addr_location *al,
bool sample_self)
{ {
struct rb_node **p; struct rb_node **p;
struct rb_node *parent = NULL; struct rb_node *parent = NULL;
...@@ -375,6 +379,7 @@ static struct hist_entry *add_hist_entry(struct hists *hists, ...@@ -375,6 +379,7 @@ static struct hist_entry *add_hist_entry(struct hists *hists,
cmp = hist_entry__cmp(he, entry); cmp = hist_entry__cmp(he, entry);
if (!cmp) { if (!cmp) {
if (sample_self)
he_stat__add_period(&he->stat, period, weight); he_stat__add_period(&he->stat, period, weight);
if (symbol_conf.cumulate_callchain) if (symbol_conf.cumulate_callchain)
he_stat__add_period(he->stat_acc, period, weight); he_stat__add_period(he->stat_acc, period, weight);
...@@ -405,13 +410,14 @@ static struct hist_entry *add_hist_entry(struct hists *hists, ...@@ -405,13 +410,14 @@ static struct hist_entry *add_hist_entry(struct hists *hists,
p = &(*p)->rb_right; p = &(*p)->rb_right;
} }
he = hist_entry__new(entry); he = hist_entry__new(entry, sample_self);
if (!he) if (!he)
return NULL; return NULL;
rb_link_node(&he->rb_node_in, parent, p); rb_link_node(&he->rb_node_in, parent, p);
rb_insert_color(&he->rb_node_in, hists->entries_in); rb_insert_color(&he->rb_node_in, hists->entries_in);
out: out:
if (sample_self)
he_stat__add_cpumode_period(&he->stat, al->cpumode, period); he_stat__add_cpumode_period(&he->stat, al->cpumode, period);
if (symbol_conf.cumulate_callchain) if (symbol_conf.cumulate_callchain)
he_stat__add_cpumode_period(he->stat_acc, al->cpumode, period); he_stat__add_cpumode_period(he->stat_acc, al->cpumode, period);
...@@ -423,7 +429,8 @@ struct hist_entry *__hists__add_entry(struct hists *hists, ...@@ -423,7 +429,8 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
struct symbol *sym_parent, struct symbol *sym_parent,
struct branch_info *bi, struct branch_info *bi,
struct mem_info *mi, struct mem_info *mi,
u64 period, u64 weight, u64 transaction) u64 period, u64 weight, u64 transaction,
bool sample_self)
{ {
struct hist_entry entry = { struct hist_entry entry = {
.thread = al->thread, .thread = al->thread,
...@@ -448,7 +455,7 @@ struct hist_entry *__hists__add_entry(struct hists *hists, ...@@ -448,7 +455,7 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
.transaction = transaction, .transaction = transaction,
}; };
return add_hist_entry(hists, &entry, al); return add_hist_entry(hists, &entry, al, sample_self);
} }
static int static int
...@@ -501,7 +508,7 @@ iter_add_single_mem_entry(struct hist_entry_iter *iter, struct addr_location *al ...@@ -501,7 +508,7 @@ iter_add_single_mem_entry(struct hist_entry_iter *iter, struct addr_location *al
* and the he_stat__add_period() function. * and the he_stat__add_period() function.
*/ */
he = __hists__add_entry(&iter->evsel->hists, al, iter->parent, NULL, mi, he = __hists__add_entry(&iter->evsel->hists, al, iter->parent, NULL, mi,
cost, cost, 0); cost, cost, 0, true);
if (!he) if (!he)
return -ENOMEM; return -ENOMEM;
...@@ -608,7 +615,7 @@ iter_add_next_branch_entry(struct hist_entry_iter *iter, struct addr_location *a ...@@ -608,7 +615,7 @@ iter_add_next_branch_entry(struct hist_entry_iter *iter, struct addr_location *a
* and not events sampled. Thus we use a pseudo period of 1. * and not events sampled. Thus we use a pseudo period of 1.
*/ */
he = __hists__add_entry(&evsel->hists, al, iter->parent, &bi[i], NULL, he = __hists__add_entry(&evsel->hists, al, iter->parent, &bi[i], NULL,
1, 1, 0); 1, 1, 0, true);
if (he == NULL) if (he == NULL)
return -ENOMEM; return -ENOMEM;
...@@ -657,7 +664,7 @@ iter_add_single_normal_entry(struct hist_entry_iter *iter, struct addr_location ...@@ -657,7 +664,7 @@ iter_add_single_normal_entry(struct hist_entry_iter *iter, struct addr_location
he = __hists__add_entry(&evsel->hists, al, iter->parent, NULL, NULL, he = __hists__add_entry(&evsel->hists, al, iter->parent, NULL, NULL,
sample->period, sample->weight, sample->period, sample->weight,
sample->transaction); sample->transaction, true);
if (he == NULL) if (he == NULL)
return -ENOMEM; return -ENOMEM;
...@@ -1161,7 +1168,7 @@ static struct hist_entry *hists__add_dummy_entry(struct hists *hists, ...@@ -1161,7 +1168,7 @@ static struct hist_entry *hists__add_dummy_entry(struct hists *hists,
p = &(*p)->rb_right; p = &(*p)->rb_right;
} }
he = hist_entry__new(pair); he = hist_entry__new(pair, true);
if (he) { if (he) {
memset(&he->stat, 0, sizeof(he->stat)); memset(&he->stat, 0, sizeof(he->stat));
he->hists = hists; he->hists = hists;
......
...@@ -130,7 +130,8 @@ struct hist_entry *__hists__add_entry(struct hists *hists, ...@@ -130,7 +130,8 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
struct symbol *parent, struct symbol *parent,
struct branch_info *bi, struct branch_info *bi,
struct mem_info *mi, u64 period, struct mem_info *mi, u64 period,
u64 weight, u64 transaction); u64 weight, u64 transaction,
bool sample_self);
int hist_entry_iter__add(struct hist_entry_iter *iter, struct addr_location *al, int hist_entry_iter__add(struct hist_entry_iter *iter, struct addr_location *al,
struct perf_evsel *evsel, struct perf_sample *sample, struct perf_evsel *evsel, struct perf_sample *sample,
int max_stack_depth); int max_stack_depth);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment