2011-02-04 09:45:46 -02:00
|
|
|
#include "annotate.h"
|
2010-05-13 19:47:16 +02:00
|
|
|
#include "util.h"
|
2010-05-21 12:48:39 +02:00
|
|
|
#include "build-id.h"
|
2009-09-28 15:32:55 +02:00
|
|
|
#include "hist.h"
|
2009-12-14 13:10:39 -02:00
|
|
|
#include "session.h"
|
|
|
|
|
#include "sort.h"
|
2013-01-22 18:09:33 +09:00
|
|
|
#include "evsel.h"
|
2009-12-16 14:31:49 -02:00
|
|
|
#include <math.h>
|
2009-09-28 15:32:55 +02:00
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
static bool hists__filter_entry_by_dso(struct hists *hists,
|
|
|
|
|
struct hist_entry *he);
|
|
|
|
|
static bool hists__filter_entry_by_thread(struct hists *hists,
|
|
|
|
|
struct hist_entry *he);
|
2012-03-16 17:50:51 +09:00
|
|
|
static bool hists__filter_entry_by_symbol(struct hists *hists,
|
|
|
|
|
struct hist_entry *he);
|
2011-10-19 13:09:10 -02:00
|
|
|
|
2010-07-21 09:19:41 -03:00
|
|
|
enum hist_filter {
|
|
|
|
|
HIST_FILTER__DSO,
|
|
|
|
|
HIST_FILTER__THREAD,
|
|
|
|
|
HIST_FILTER__PARENT,
|
2012-03-16 17:50:51 +09:00
|
|
|
HIST_FILTER__SYMBOL,
|
2010-07-21 09:19:41 -03:00
|
|
|
};
|
|
|
|
|
|
2009-09-28 15:32:55 +02:00
|
|
|
struct callchain_param callchain_param = {
|
|
|
|
|
.mode = CHAIN_GRAPH_REL,
|
2011-06-07 23:49:46 +08:00
|
|
|
.min_percent = 0.5,
|
2013-07-18 15:33:57 -07:00
|
|
|
.order = ORDER_CALLEE,
|
|
|
|
|
.key = CCKEY_FUNCTION
|
2009-09-28 15:32:55 +02:00
|
|
|
};
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
u16 hists__col_len(struct hists *hists, enum hist_column col)
|
2010-07-20 14:42:52 -03:00
|
|
|
{
|
2011-09-26 12:33:28 -03:00
|
|
|
return hists->col_len[col];
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
void hists__set_col_len(struct hists *hists, enum hist_column col, u16 len)
|
2010-07-20 14:42:52 -03:00
|
|
|
{
|
2011-09-26 12:33:28 -03:00
|
|
|
hists->col_len[col] = len;
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
bool hists__new_col_len(struct hists *hists, enum hist_column col, u16 len)
|
2010-07-20 14:42:52 -03:00
|
|
|
{
|
2011-09-26 12:33:28 -03:00
|
|
|
if (len > hists__col_len(hists, col)) {
|
|
|
|
|
hists__set_col_len(hists, col, len);
|
2010-07-20 14:42:52 -03:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2012-08-20 13:52:05 +09:00
|
|
|
void hists__reset_col_len(struct hists *hists)
|
2010-07-20 14:42:52 -03:00
|
|
|
{
|
|
|
|
|
enum hist_column col;
|
|
|
|
|
|
|
|
|
|
for (col = 0; col < HISTC_NR_COLS; ++col)
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__set_col_len(hists, col, 0);
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
|
|
|
|
|
2012-02-09 23:21:01 +01:00
|
|
|
static void hists__set_unres_dso_col_len(struct hists *hists, int dso)
|
|
|
|
|
{
|
|
|
|
|
const unsigned int unresolved_col_width = BITS_PER_LONG / 4;
|
|
|
|
|
|
|
|
|
|
if (hists__col_len(hists, dso) < unresolved_col_width &&
|
|
|
|
|
!symbol_conf.col_width_list_str && !symbol_conf.field_sep &&
|
|
|
|
|
!symbol_conf.dso_list)
|
|
|
|
|
hists__set_col_len(hists, dso, unresolved_col_width);
|
|
|
|
|
}
|
|
|
|
|
|
2012-08-20 13:52:05 +09:00
|
|
|
void hists__calc_col_len(struct hists *hists, struct hist_entry *h)
|
2010-07-20 14:42:52 -03:00
|
|
|
{
|
2012-02-09 23:21:01 +01:00
|
|
|
const unsigned int unresolved_col_width = BITS_PER_LONG / 4;
|
2013-01-24 16:10:35 +01:00
|
|
|
int symlen;
|
2010-07-20 14:42:52 -03:00
|
|
|
u16 len;
|
|
|
|
|
|
2013-04-01 20:35:19 +09:00
|
|
|
/*
|
|
|
|
|
* +4 accounts for '[x] ' priv level info
|
|
|
|
|
* +2 accounts for 0x prefix on raw addresses
|
|
|
|
|
* +3 accounts for ' y ' symtab origin info
|
|
|
|
|
*/
|
|
|
|
|
if (h->ms.sym) {
|
|
|
|
|
symlen = h->ms.sym->namelen + 4;
|
|
|
|
|
if (verbose)
|
|
|
|
|
symlen += BITS_PER_LONG / 4 + 2 + 3;
|
|
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL, symlen);
|
|
|
|
|
} else {
|
2013-01-24 16:10:35 +01:00
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL, symlen);
|
2012-02-09 23:21:01 +01:00
|
|
|
hists__set_unres_dso_col_len(hists, HISTC_DSO);
|
2013-01-24 16:10:35 +01:00
|
|
|
}
|
2010-07-20 14:42:52 -03:00
|
|
|
|
|
|
|
|
len = thread__comm_len(h->thread);
|
2011-09-26 12:33:28 -03:00
|
|
|
if (hists__new_col_len(hists, HISTC_COMM, len))
|
|
|
|
|
hists__set_col_len(hists, HISTC_THREAD, len + 6);
|
2010-07-20 14:42:52 -03:00
|
|
|
|
|
|
|
|
if (h->ms.map) {
|
|
|
|
|
len = dso__name_len(h->ms.map->dso);
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__new_col_len(hists, HISTC_DSO, len);
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
2012-02-09 23:21:01 +01:00
|
|
|
|
2012-12-27 18:11:42 +09:00
|
|
|
if (h->parent)
|
|
|
|
|
hists__new_col_len(hists, HISTC_PARENT, h->parent->namelen);
|
|
|
|
|
|
2012-02-09 23:21:01 +01:00
|
|
|
if (h->branch_info) {
|
|
|
|
|
if (h->branch_info->from.sym) {
|
|
|
|
|
symlen = (int)h->branch_info->from.sym->namelen + 4;
|
2013-04-01 20:35:19 +09:00
|
|
|
if (verbose)
|
|
|
|
|
symlen += BITS_PER_LONG / 4 + 2 + 3;
|
2012-02-09 23:21:01 +01:00
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL_FROM, symlen);
|
|
|
|
|
|
|
|
|
|
symlen = dso__name_len(h->branch_info->from.map->dso);
|
|
|
|
|
hists__new_col_len(hists, HISTC_DSO_FROM, symlen);
|
|
|
|
|
} else {
|
|
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL_FROM, symlen);
|
|
|
|
|
hists__set_unres_dso_col_len(hists, HISTC_DSO_FROM);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (h->branch_info->to.sym) {
|
|
|
|
|
symlen = (int)h->branch_info->to.sym->namelen + 4;
|
2013-04-01 20:35:19 +09:00
|
|
|
if (verbose)
|
|
|
|
|
symlen += BITS_PER_LONG / 4 + 2 + 3;
|
2012-02-09 23:21:01 +01:00
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL_TO, symlen);
|
|
|
|
|
|
|
|
|
|
symlen = dso__name_len(h->branch_info->to.map->dso);
|
|
|
|
|
hists__new_col_len(hists, HISTC_DSO_TO, symlen);
|
|
|
|
|
} else {
|
|
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_SYMBOL_TO, symlen);
|
|
|
|
|
hists__set_unres_dso_col_len(hists, HISTC_DSO_TO);
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-01-24 16:10:35 +01:00
|
|
|
|
|
|
|
|
if (h->mem_info) {
|
|
|
|
|
if (h->mem_info->daddr.sym) {
|
|
|
|
|
symlen = (int)h->mem_info->daddr.sym->namelen + 4
|
|
|
|
|
+ unresolved_col_width + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_DADDR_SYMBOL,
|
|
|
|
|
symlen);
|
|
|
|
|
} else {
|
|
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_DADDR_SYMBOL,
|
|
|
|
|
symlen);
|
|
|
|
|
}
|
|
|
|
|
if (h->mem_info->daddr.map) {
|
|
|
|
|
symlen = dso__name_len(h->mem_info->daddr.map->dso);
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_DADDR_DSO,
|
|
|
|
|
symlen);
|
|
|
|
|
} else {
|
|
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__set_unres_dso_col_len(hists, HISTC_MEM_DADDR_DSO);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
symlen = unresolved_col_width + 4 + 2;
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_DADDR_SYMBOL, symlen);
|
|
|
|
|
hists__set_unres_dso_col_len(hists, HISTC_MEM_DADDR_DSO);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_LOCKED, 6);
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_TLB, 22);
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_SNOOP, 12);
|
|
|
|
|
hists__new_col_len(hists, HISTC_MEM_LVL, 21 + 3);
|
|
|
|
|
hists__new_col_len(hists, HISTC_LOCAL_WEIGHT, 12);
|
|
|
|
|
hists__new_col_len(hists, HISTC_GLOBAL_WEIGHT, 12);
|
2013-09-20 07:40:43 -07:00
|
|
|
|
|
|
|
|
if (h->transaction)
|
|
|
|
|
hists__new_col_len(hists, HISTC_TRANSACTION,
|
|
|
|
|
hist_entry__transaction_len());
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
|
|
|
|
|
2012-08-20 13:52:05 +09:00
|
|
|
void hists__output_recalc_col_len(struct hists *hists, int max_rows)
|
|
|
|
|
{
|
|
|
|
|
struct rb_node *next = rb_first(&hists->entries);
|
|
|
|
|
struct hist_entry *n;
|
|
|
|
|
int row = 0;
|
|
|
|
|
|
|
|
|
|
hists__reset_col_len(hists);
|
|
|
|
|
|
|
|
|
|
while (next && row++ < max_rows) {
|
|
|
|
|
n = rb_entry(next, struct hist_entry, rb_node);
|
|
|
|
|
if (!n->filtered)
|
|
|
|
|
hists__calc_col_len(hists, n);
|
|
|
|
|
next = rb_next(&n->rb_node);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-04 12:27:03 -02:00
|
|
|
static void hist_entry__add_cpumode_period(struct hist_entry *he,
|
2010-05-14 14:19:35 -03:00
|
|
|
unsigned int cpumode, u64 period)
|
2010-04-19 13:32:50 +08:00
|
|
|
{
|
2010-05-09 13:02:23 -03:00
|
|
|
switch (cpumode) {
|
2010-04-19 13:32:50 +08:00
|
|
|
case PERF_RECORD_MISC_KERNEL:
|
2012-10-04 21:49:41 +09:00
|
|
|
he->stat.period_sys += period;
|
2010-04-19 13:32:50 +08:00
|
|
|
break;
|
|
|
|
|
case PERF_RECORD_MISC_USER:
|
2012-10-04 21:49:41 +09:00
|
|
|
he->stat.period_us += period;
|
2010-04-19 13:32:50 +08:00
|
|
|
break;
|
|
|
|
|
case PERF_RECORD_MISC_GUEST_KERNEL:
|
2012-10-04 21:49:41 +09:00
|
|
|
he->stat.period_guest_sys += period;
|
2010-04-19 13:32:50 +08:00
|
|
|
break;
|
|
|
|
|
case PERF_RECORD_MISC_GUEST_USER:
|
2012-10-04 21:49:41 +09:00
|
|
|
he->stat.period_guest_us += period;
|
2010-04-19 13:32:50 +08:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-01-24 16:10:29 +01:00
|
|
|
static void he_stat__add_period(struct he_stat *he_stat, u64 period,
|
|
|
|
|
u64 weight)
|
2012-10-04 21:49:43 +09:00
|
|
|
{
|
2013-01-24 16:10:35 +01:00
|
|
|
|
2012-10-04 21:49:43 +09:00
|
|
|
he_stat->period += period;
|
2013-01-24 16:10:29 +01:00
|
|
|
he_stat->weight += weight;
|
2012-10-04 21:49:43 +09:00
|
|
|
he_stat->nr_events += 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void he_stat__add_stat(struct he_stat *dest, struct he_stat *src)
|
|
|
|
|
{
|
|
|
|
|
dest->period += src->period;
|
|
|
|
|
dest->period_sys += src->period_sys;
|
|
|
|
|
dest->period_us += src->period_us;
|
|
|
|
|
dest->period_guest_sys += src->period_guest_sys;
|
|
|
|
|
dest->period_guest_us += src->period_guest_us;
|
|
|
|
|
dest->nr_events += src->nr_events;
|
2013-01-24 16:10:29 +01:00
|
|
|
dest->weight += src->weight;
|
2012-10-04 21:49:43 +09:00
|
|
|
}
|
|
|
|
|
|
2011-10-05 19:16:15 -03:00
|
|
|
static void hist_entry__decay(struct hist_entry *he)
|
|
|
|
|
{
|
2012-10-04 21:49:41 +09:00
|
|
|
he->stat.period = (he->stat.period * 7) / 8;
|
|
|
|
|
he->stat.nr_events = (he->stat.nr_events * 7) / 8;
|
2013-01-24 16:10:29 +01:00
|
|
|
/* XXX need decay for weight too? */
|
2011-10-05 19:16:15 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool hists__decay_entry(struct hists *hists, struct hist_entry *he)
|
|
|
|
|
{
|
2012-10-04 21:49:41 +09:00
|
|
|
u64 prev_period = he->stat.period;
|
2011-10-20 06:45:44 -02:00
|
|
|
|
|
|
|
|
if (prev_period == 0)
|
2011-10-13 08:01:33 -03:00
|
|
|
return true;
|
2011-10-20 06:45:44 -02:00
|
|
|
|
2011-10-05 19:16:15 -03:00
|
|
|
hist_entry__decay(he);
|
2011-10-20 06:45:44 -02:00
|
|
|
|
|
|
|
|
if (!he->filtered)
|
2012-10-04 21:49:41 +09:00
|
|
|
hists->stats.total_period -= prev_period - he->stat.period;
|
2011-10-20 06:45:44 -02:00
|
|
|
|
2012-10-04 21:49:41 +09:00
|
|
|
return he->stat.period == 0;
|
2011-10-05 19:16:15 -03:00
|
|
|
}
|
|
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
void hists__decay_entries(struct hists *hists, bool zap_user, bool zap_kernel)
|
2011-10-05 19:16:15 -03:00
|
|
|
{
|
|
|
|
|
struct rb_node *next = rb_first(&hists->entries);
|
|
|
|
|
struct hist_entry *n;
|
|
|
|
|
|
|
|
|
|
while (next) {
|
|
|
|
|
n = rb_entry(next, struct hist_entry, rb_node);
|
|
|
|
|
next = rb_next(&n->rb_node);
|
2011-10-13 08:01:33 -03:00
|
|
|
/*
|
|
|
|
|
* We may be annotating this, for instance, so keep it here in
|
|
|
|
|
* case some it gets new samples, we'll eventually free it when
|
|
|
|
|
* the user stops browsing and it agains gets fully decayed.
|
|
|
|
|
*/
|
2011-10-17 09:05:04 -02:00
|
|
|
if (((zap_user && n->level == '.') ||
|
|
|
|
|
(zap_kernel && n->level != '.') ||
|
|
|
|
|
hists__decay_entry(hists, n)) &&
|
|
|
|
|
!n->used) {
|
2011-10-05 19:16:15 -03:00
|
|
|
rb_erase(&n->rb_node, &hists->entries);
|
|
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
if (sort__need_collapse)
|
2011-10-05 19:16:15 -03:00
|
|
|
rb_erase(&n->rb_node_in, &hists->entries_collapsed);
|
|
|
|
|
|
|
|
|
|
hist_entry__free(n);
|
|
|
|
|
--hists->nr_entries;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-28 15:32:55 +02:00
|
|
|
/*
|
2010-05-14 14:19:35 -03:00
|
|
|
* histogram, sorted on item, collects periods
|
2009-09-28 15:32:55 +02:00
|
|
|
*/
|
|
|
|
|
|
2010-05-09 13:02:23 -03:00
|
|
|
static struct hist_entry *hist_entry__new(struct hist_entry *template)
|
|
|
|
|
{
|
2010-08-22 20:05:22 +02:00
|
|
|
size_t callchain_size = symbol_conf.use_callchain ? sizeof(struct callchain_root) : 0;
|
2013-01-24 16:10:35 +01:00
|
|
|
struct hist_entry *he = zalloc(sizeof(*he) + callchain_size);
|
2010-05-09 13:02:23 -03:00
|
|
|
|
2012-01-04 12:27:03 -02:00
|
|
|
if (he != NULL) {
|
|
|
|
|
*he = *template;
|
2012-10-04 21:49:42 +09:00
|
|
|
|
2012-01-04 12:27:03 -02:00
|
|
|
if (he->ms.map)
|
|
|
|
|
he->ms.map->referenced = true;
|
2013-01-14 15:02:45 +01:00
|
|
|
|
|
|
|
|
if (he->branch_info) {
|
2013-04-01 20:35:17 +09:00
|
|
|
/*
|
|
|
|
|
* This branch info is (a part of) allocated from
|
|
|
|
|
* machine__resolve_bstack() and will be freed after
|
|
|
|
|
* adding new entries. So we need to save a copy.
|
|
|
|
|
*/
|
|
|
|
|
he->branch_info = malloc(sizeof(*he->branch_info));
|
|
|
|
|
if (he->branch_info == NULL) {
|
|
|
|
|
free(he);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memcpy(he->branch_info, template->branch_info,
|
|
|
|
|
sizeof(*he->branch_info));
|
|
|
|
|
|
2013-01-14 15:02:45 +01:00
|
|
|
if (he->branch_info->from.map)
|
|
|
|
|
he->branch_info->from.map->referenced = true;
|
|
|
|
|
if (he->branch_info->to.map)
|
|
|
|
|
he->branch_info->to.map->referenced = true;
|
|
|
|
|
}
|
|
|
|
|
|
2013-01-24 16:10:35 +01:00
|
|
|
if (he->mem_info) {
|
|
|
|
|
if (he->mem_info->iaddr.map)
|
|
|
|
|
he->mem_info->iaddr.map->referenced = true;
|
|
|
|
|
if (he->mem_info->daddr.map)
|
|
|
|
|
he->mem_info->daddr.map->referenced = true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-09 13:02:23 -03:00
|
|
|
if (symbol_conf.use_callchain)
|
2012-01-04 12:27:03 -02:00
|
|
|
callchain_init(he->callchain);
|
2012-10-25 14:42:45 -02:00
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&he->pairs.node);
|
2010-05-09 13:02:23 -03:00
|
|
|
}
|
|
|
|
|
|
2012-01-04 12:27:03 -02:00
|
|
|
return he;
|
2010-05-09 13:02:23 -03:00
|
|
|
}
|
|
|
|
|
|
2012-12-10 17:29:56 +09:00
|
|
|
void hists__inc_nr_entries(struct hists *hists, struct hist_entry *h)
|
2010-05-10 13:57:51 -03:00
|
|
|
{
|
2010-07-20 14:42:52 -03:00
|
|
|
if (!h->filtered) {
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__calc_col_len(hists, h);
|
|
|
|
|
++hists->nr_entries;
|
2012-10-04 21:49:41 +09:00
|
|
|
hists->stats.total_period += h->stat.period;
|
2010-07-20 14:42:52 -03:00
|
|
|
}
|
2010-05-10 13:57:51 -03:00
|
|
|
}
|
|
|
|
|
|
2010-07-21 09:19:41 -03:00
|
|
|
static u8 symbol__parent_filter(const struct symbol *parent)
|
|
|
|
|
{
|
|
|
|
|
if (symbol_conf.exclude_other && parent == NULL)
|
|
|
|
|
return 1 << HIST_FILTER__PARENT;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-09 23:21:01 +01:00
|
|
|
static struct hist_entry *add_hist_entry(struct hists *hists,
|
|
|
|
|
struct hist_entry *entry,
|
2010-05-10 13:04:11 -03:00
|
|
|
struct addr_location *al,
|
2013-01-24 16:10:29 +01:00
|
|
|
u64 period,
|
|
|
|
|
u64 weight)
|
2009-10-03 10:42:45 -03:00
|
|
|
{
|
2011-10-05 17:50:23 -03:00
|
|
|
struct rb_node **p;
|
2009-10-03 10:42:45 -03:00
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
|
struct hist_entry *he;
|
2013-10-01 07:22:15 -07:00
|
|
|
int64_t cmp;
|
2009-10-03 10:42:45 -03:00
|
|
|
|
2011-10-05 17:50:23 -03:00
|
|
|
p = &hists->entries_in->rb_node;
|
|
|
|
|
|
2009-10-03 10:42:45 -03:00
|
|
|
while (*p != NULL) {
|
|
|
|
|
parent = *p;
|
2011-10-05 17:50:23 -03:00
|
|
|
he = rb_entry(parent, struct hist_entry, rb_node_in);
|
2009-10-03 10:42:45 -03:00
|
|
|
|
2012-12-10 17:29:54 +09:00
|
|
|
/*
|
|
|
|
|
* Make sure that it receives arguments in a same order as
|
|
|
|
|
* hist_entry__collapse() so that we can use an appropriate
|
|
|
|
|
* function when searching an entry regardless which sort
|
|
|
|
|
* keys were used.
|
|
|
|
|
*/
|
|
|
|
|
cmp = hist_entry__cmp(he, entry);
|
2009-10-03 10:42:45 -03:00
|
|
|
|
|
|
|
|
if (!cmp) {
|
2013-01-24 16:10:29 +01:00
|
|
|
he_stat__add_period(&he->stat, period, weight);
|
2012-03-27 03:14:18 -04:00
|
|
|
|
2013-04-01 20:35:18 +09:00
|
|
|
/*
|
|
|
|
|
* This mem info was allocated from machine__resolve_mem
|
|
|
|
|
* and will not be used anymore.
|
|
|
|
|
*/
|
|
|
|
|
free(entry->mem_info);
|
|
|
|
|
|
2012-03-27 03:14:18 -04:00
|
|
|
/* If the map of an existing hist_entry has
|
|
|
|
|
* become out-of-date due to an exec() or
|
|
|
|
|
* similar, update it. Otherwise we will
|
|
|
|
|
* mis-adjust symbol addresses when computing
|
|
|
|
|
* the history counter to increment.
|
|
|
|
|
*/
|
|
|
|
|
if (he->ms.map != entry->ms.map) {
|
|
|
|
|
he->ms.map = entry->ms.map;
|
|
|
|
|
if (he->ms.map)
|
|
|
|
|
he->ms.map->referenced = true;
|
|
|
|
|
}
|
2010-05-09 13:02:23 -03:00
|
|
|
goto out;
|
2009-10-03 10:42:45 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cmp < 0)
|
|
|
|
|
p = &(*p)->rb_left;
|
|
|
|
|
else
|
|
|
|
|
p = &(*p)->rb_right;
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-09 23:21:01 +01:00
|
|
|
he = hist_entry__new(entry);
|
2009-10-03 10:42:45 -03:00
|
|
|
if (!he)
|
2013-05-14 11:09:02 +09:00
|
|
|
return NULL;
|
2011-10-05 17:50:23 -03:00
|
|
|
|
|
|
|
|
rb_link_node(&he->rb_node_in, parent, p);
|
|
|
|
|
rb_insert_color(&he->rb_node_in, hists->entries_in);
|
2010-05-09 13:02:23 -03:00
|
|
|
out:
|
2010-05-14 14:19:35 -03:00
|
|
|
hist_entry__add_cpumode_period(he, al->cpumode, period);
|
2009-10-03 10:42:45 -03:00
|
|
|
return he;
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-22 19:01:31 -03:00
|
|
|
struct hist_entry *__hists__add_mem_entry(struct hists *hists,
|
2013-01-24 16:10:35 +01:00
|
|
|
struct addr_location *al,
|
|
|
|
|
struct symbol *sym_parent,
|
|
|
|
|
struct mem_info *mi,
|
|
|
|
|
u64 period,
|
|
|
|
|
u64 weight)
|
|
|
|
|
{
|
|
|
|
|
struct hist_entry entry = {
|
|
|
|
|
.thread = al->thread,
|
|
|
|
|
.ms = {
|
|
|
|
|
.map = al->map,
|
|
|
|
|
.sym = al->sym,
|
|
|
|
|
},
|
|
|
|
|
.stat = {
|
|
|
|
|
.period = period,
|
|
|
|
|
.weight = weight,
|
|
|
|
|
.nr_events = 1,
|
|
|
|
|
},
|
|
|
|
|
.cpu = al->cpu,
|
|
|
|
|
.ip = al->addr,
|
|
|
|
|
.level = al->level,
|
|
|
|
|
.parent = sym_parent,
|
|
|
|
|
.filtered = symbol__parent_filter(sym_parent),
|
2013-10-22 19:01:31 -03:00
|
|
|
.hists = hists,
|
2013-01-24 16:10:35 +01:00
|
|
|
.mem_info = mi,
|
|
|
|
|
.branch_info = NULL,
|
|
|
|
|
};
|
2013-10-22 19:01:31 -03:00
|
|
|
return add_hist_entry(hists, &entry, al, period, weight);
|
2013-01-24 16:10:35 +01:00
|
|
|
}
|
|
|
|
|
|
2013-10-22 19:01:31 -03:00
|
|
|
struct hist_entry *__hists__add_branch_entry(struct hists *hists,
|
2012-02-09 23:21:01 +01:00
|
|
|
struct addr_location *al,
|
|
|
|
|
struct symbol *sym_parent,
|
|
|
|
|
struct branch_info *bi,
|
2013-01-24 16:10:29 +01:00
|
|
|
u64 period,
|
|
|
|
|
u64 weight)
|
2012-02-09 23:21:01 +01:00
|
|
|
{
|
|
|
|
|
struct hist_entry entry = {
|
|
|
|
|
.thread = al->thread,
|
|
|
|
|
.ms = {
|
|
|
|
|
.map = bi->to.map,
|
|
|
|
|
.sym = bi->to.sym,
|
|
|
|
|
},
|
|
|
|
|
.cpu = al->cpu,
|
|
|
|
|
.ip = bi->to.addr,
|
|
|
|
|
.level = al->level,
|
2012-10-04 21:49:41 +09:00
|
|
|
.stat = {
|
|
|
|
|
.period = period,
|
2012-10-04 21:49:42 +09:00
|
|
|
.nr_events = 1,
|
2013-01-24 16:10:29 +01:00
|
|
|
.weight = weight,
|
2012-10-04 21:49:41 +09:00
|
|
|
},
|
2012-02-09 23:21:01 +01:00
|
|
|
.parent = sym_parent,
|
|
|
|
|
.filtered = symbol__parent_filter(sym_parent),
|
|
|
|
|
.branch_info = bi,
|
2013-10-22 19:01:31 -03:00
|
|
|
.hists = hists,
|
2013-01-24 16:10:35 +01:00
|
|
|
.mem_info = NULL,
|
2012-02-09 23:21:01 +01:00
|
|
|
};
|
|
|
|
|
|
2013-10-22 19:01:31 -03:00
|
|
|
return add_hist_entry(hists, &entry, al, period, weight);
|
2012-02-09 23:21:01 +01:00
|
|
|
}
|
|
|
|
|
|
2013-10-22 19:01:31 -03:00
|
|
|
struct hist_entry *__hists__add_entry(struct hists *hists,
|
2012-02-09 23:21:01 +01:00
|
|
|
struct addr_location *al,
|
2013-01-24 16:10:29 +01:00
|
|
|
struct symbol *sym_parent, u64 period,
|
2013-09-20 07:40:43 -07:00
|
|
|
u64 weight, u64 transaction)
|
2012-02-09 23:21:01 +01:00
|
|
|
{
|
|
|
|
|
struct hist_entry entry = {
|
|
|
|
|
.thread = al->thread,
|
|
|
|
|
.ms = {
|
|
|
|
|
.map = al->map,
|
|
|
|
|
.sym = al->sym,
|
|
|
|
|
},
|
|
|
|
|
.cpu = al->cpu,
|
|
|
|
|
.ip = al->addr,
|
|
|
|
|
.level = al->level,
|
2012-10-04 21:49:41 +09:00
|
|
|
.stat = {
|
|
|
|
|
.period = period,
|
2012-10-04 21:49:42 +09:00
|
|
|
.nr_events = 1,
|
2013-01-24 16:10:29 +01:00
|
|
|
.weight = weight,
|
2012-10-04 21:49:41 +09:00
|
|
|
},
|
2012-02-09 23:21:01 +01:00
|
|
|
.parent = sym_parent,
|
|
|
|
|
.filtered = symbol__parent_filter(sym_parent),
|
2013-10-22 19:01:31 -03:00
|
|
|
.hists = hists,
|
2013-01-24 16:10:35 +01:00
|
|
|
.branch_info = NULL,
|
|
|
|
|
.mem_info = NULL,
|
2013-09-20 07:40:43 -07:00
|
|
|
.transaction = transaction,
|
2012-02-09 23:21:01 +01:00
|
|
|
};
|
|
|
|
|
|
2013-10-22 19:01:31 -03:00
|
|
|
return add_hist_entry(hists, &entry, al, period, weight);
|
2012-02-09 23:21:01 +01:00
|
|
|
}
|
|
|
|
|
|
2009-09-28 15:32:55 +02:00
|
|
|
int64_t
|
|
|
|
|
hist_entry__cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
|
{
|
|
|
|
|
struct sort_entry *se;
|
|
|
|
|
int64_t cmp = 0;
|
|
|
|
|
|
|
|
|
|
list_for_each_entry(se, &hist_entry__sort_list, list) {
|
2010-04-14 19:11:29 +02:00
|
|
|
cmp = se->se_cmp(left, right);
|
2009-09-28 15:32:55 +02:00
|
|
|
if (cmp)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
|
hist_entry__collapse(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
|
{
|
|
|
|
|
struct sort_entry *se;
|
|
|
|
|
int64_t cmp = 0;
|
|
|
|
|
|
|
|
|
|
list_for_each_entry(se, &hist_entry__sort_list, list) {
|
|
|
|
|
int64_t (*f)(struct hist_entry *, struct hist_entry *);
|
|
|
|
|
|
2010-04-14 19:11:29 +02:00
|
|
|
f = se->se_collapse ?: se->se_cmp;
|
2009-09-28 15:32:55 +02:00
|
|
|
|
|
|
|
|
cmp = f(left, right);
|
|
|
|
|
if (cmp)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void hist_entry__free(struct hist_entry *he)
|
|
|
|
|
{
|
2012-11-07 16:27:14 +09:00
|
|
|
free(he->branch_info);
|
2013-01-24 16:10:38 +01:00
|
|
|
free(he->mem_info);
|
2013-09-11 14:09:28 +09:00
|
|
|
free_srcline(he->srcline);
|
2009-09-28 15:32:55 +02:00
|
|
|
free(he);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* collapse the histogram
|
|
|
|
|
*/
|
|
|
|
|
|
2012-09-11 01:15:03 +03:00
|
|
|
static bool hists__collapse_insert_entry(struct hists *hists __maybe_unused,
|
2011-01-14 04:51:58 +01:00
|
|
|
struct rb_root *root,
|
|
|
|
|
struct hist_entry *he)
|
2009-09-28 15:32:55 +02:00
|
|
|
{
|
2009-12-14 11:37:11 -02:00
|
|
|
struct rb_node **p = &root->rb_node;
|
2009-09-28 15:32:55 +02:00
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
|
struct hist_entry *iter;
|
|
|
|
|
int64_t cmp;
|
|
|
|
|
|
|
|
|
|
while (*p != NULL) {
|
|
|
|
|
parent = *p;
|
2011-10-05 17:50:23 -03:00
|
|
|
iter = rb_entry(parent, struct hist_entry, rb_node_in);
|
2009-09-28 15:32:55 +02:00
|
|
|
|
|
|
|
|
cmp = hist_entry__collapse(iter, he);
|
|
|
|
|
|
|
|
|
|
if (!cmp) {
|
2012-10-04 21:49:43 +09:00
|
|
|
he_stat__add_stat(&iter->stat, &he->stat);
|
2012-09-26 16:47:28 +09:00
|
|
|
|
2011-01-14 04:51:58 +01:00
|
|
|
if (symbol_conf.use_callchain) {
|
2012-05-31 14:43:26 +09:00
|
|
|
callchain_cursor_reset(&callchain_cursor);
|
|
|
|
|
callchain_merge(&callchain_cursor,
|
|
|
|
|
iter->callchain,
|
2011-01-14 04:51:58 +01:00
|
|
|
he->callchain);
|
|
|
|
|
}
|
2009-09-28 15:32:55 +02:00
|
|
|
hist_entry__free(he);
|
2010-05-10 13:57:51 -03:00
|
|
|
return false;
|
2009-09-28 15:32:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cmp < 0)
|
|
|
|
|
p = &(*p)->rb_left;
|
|
|
|
|
else
|
|
|
|
|
p = &(*p)->rb_right;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-05 17:50:23 -03:00
|
|
|
rb_link_node(&he->rb_node_in, parent, p);
|
|
|
|
|
rb_insert_color(&he->rb_node_in, root);
|
2010-05-10 13:57:51 -03:00
|
|
|
return true;
|
2009-09-28 15:32:55 +02:00
|
|
|
}
|
|
|
|
|
|
2011-10-05 17:50:23 -03:00
|
|
|
static struct rb_root *hists__get_rotate_entries_in(struct hists *hists)
|
|
|
|
|
{
|
|
|
|
|
struct rb_root *root;
|
|
|
|
|
|
|
|
|
|
pthread_mutex_lock(&hists->lock);
|
|
|
|
|
|
|
|
|
|
root = hists->entries_in;
|
|
|
|
|
if (++hists->entries_in > &hists->entries_in_array[1])
|
|
|
|
|
hists->entries_in = &hists->entries_in_array[0];
|
|
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&hists->lock);
|
|
|
|
|
|
|
|
|
|
return root;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
static void hists__apply_filters(struct hists *hists, struct hist_entry *he)
|
|
|
|
|
{
|
|
|
|
|
hists__filter_entry_by_dso(hists, he);
|
|
|
|
|
hists__filter_entry_by_thread(hists, he);
|
2012-03-16 17:50:51 +09:00
|
|
|
hists__filter_entry_by_symbol(hists, he);
|
2011-10-19 13:09:10 -02:00
|
|
|
}
|
|
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
void hists__collapse_resort(struct hists *hists)
|
2011-10-05 17:50:23 -03:00
|
|
|
{
|
|
|
|
|
struct rb_root *root;
|
|
|
|
|
struct rb_node *next;
|
|
|
|
|
struct hist_entry *n;
|
|
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
if (!sort__need_collapse)
|
2011-10-05 17:50:23 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
root = hists__get_rotate_entries_in(hists);
|
|
|
|
|
next = rb_first(root);
|
|
|
|
|
|
|
|
|
|
while (next) {
|
2013-09-17 16:34:28 -03:00
|
|
|
if (session_done())
|
|
|
|
|
break;
|
2011-10-05 17:50:23 -03:00
|
|
|
n = rb_entry(next, struct hist_entry, rb_node_in);
|
|
|
|
|
next = rb_next(&n->rb_node_in);
|
|
|
|
|
|
|
|
|
|
rb_erase(&n->rb_node_in, root);
|
2011-10-19 13:09:10 -02:00
|
|
|
if (hists__collapse_insert_entry(hists, &hists->entries_collapsed, n)) {
|
|
|
|
|
/*
|
|
|
|
|
* If it wasn't combined with one of the entries already
|
|
|
|
|
* collapsed, we need to apply the filters that may have
|
|
|
|
|
* been set by, say, the hist_browser.
|
|
|
|
|
*/
|
|
|
|
|
hists__apply_filters(hists, n);
|
|
|
|
|
}
|
2011-10-05 17:50:23 -03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-28 15:32:55 +02:00
|
|
|
/*
|
2010-05-14 14:19:35 -03:00
|
|
|
* reverse the map, sort on period.
|
2009-09-28 15:32:55 +02:00
|
|
|
*/
|
|
|
|
|
|
2013-01-22 18:09:33 +09:00
|
|
|
static int period_cmp(u64 period_a, u64 period_b)
|
|
|
|
|
{
|
|
|
|
|
if (period_a > period_b)
|
|
|
|
|
return 1;
|
|
|
|
|
if (period_a < period_b)
|
|
|
|
|
return -1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int hist_entry__sort_on_period(struct hist_entry *a,
|
|
|
|
|
struct hist_entry *b)
|
|
|
|
|
{
|
|
|
|
|
int ret;
|
|
|
|
|
int i, nr_members;
|
|
|
|
|
struct perf_evsel *evsel;
|
|
|
|
|
struct hist_entry *pair;
|
|
|
|
|
u64 *periods_a, *periods_b;
|
|
|
|
|
|
|
|
|
|
ret = period_cmp(a->stat.period, b->stat.period);
|
|
|
|
|
if (ret || !symbol_conf.event_group)
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
evsel = hists_to_evsel(a->hists);
|
|
|
|
|
nr_members = evsel->nr_members;
|
|
|
|
|
if (nr_members <= 1)
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
periods_a = zalloc(sizeof(periods_a) * nr_members);
|
|
|
|
|
periods_b = zalloc(sizeof(periods_b) * nr_members);
|
|
|
|
|
|
|
|
|
|
if (!periods_a || !periods_b)
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
list_for_each_entry(pair, &a->pairs.head, pairs.node) {
|
|
|
|
|
evsel = hists_to_evsel(pair->hists);
|
|
|
|
|
periods_a[perf_evsel__group_idx(evsel)] = pair->stat.period;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list_for_each_entry(pair, &b->pairs.head, pairs.node) {
|
|
|
|
|
evsel = hists_to_evsel(pair->hists);
|
|
|
|
|
periods_b[perf_evsel__group_idx(evsel)] = pair->stat.period;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 1; i < nr_members; i++) {
|
|
|
|
|
ret = period_cmp(periods_a[i], periods_b[i]);
|
|
|
|
|
if (ret)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
free(periods_a);
|
|
|
|
|
free(periods_b);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-10 13:04:11 -03:00
|
|
|
static void __hists__insert_output_entry(struct rb_root *entries,
|
|
|
|
|
struct hist_entry *he,
|
|
|
|
|
u64 min_callchain_hits)
|
2009-09-28 15:32:55 +02:00
|
|
|
{
|
2010-05-10 13:04:11 -03:00
|
|
|
struct rb_node **p = &entries->rb_node;
|
2009-09-28 15:32:55 +02:00
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
|
struct hist_entry *iter;
|
|
|
|
|
|
2009-12-15 20:04:42 -02:00
|
|
|
if (symbol_conf.use_callchain)
|
2010-04-02 09:50:42 -03:00
|
|
|
callchain_param.sort(&he->sorted_chain, he->callchain,
|
2009-09-28 15:32:55 +02:00
|
|
|
min_callchain_hits, &callchain_param);
|
|
|
|
|
|
|
|
|
|
while (*p != NULL) {
|
|
|
|
|
parent = *p;
|
|
|
|
|
iter = rb_entry(parent, struct hist_entry, rb_node);
|
|
|
|
|
|
2013-01-22 18:09:33 +09:00
|
|
|
if (hist_entry__sort_on_period(he, iter) > 0)
|
2009-09-28 15:32:55 +02:00
|
|
|
p = &(*p)->rb_left;
|
|
|
|
|
else
|
|
|
|
|
p = &(*p)->rb_right;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rb_link_node(&he->rb_node, parent, p);
|
2010-05-10 13:04:11 -03:00
|
|
|
rb_insert_color(&he->rb_node, entries);
|
2009-09-28 15:32:55 +02:00
|
|
|
}
|
|
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
void hists__output_resort(struct hists *hists)
|
2009-09-28 15:32:55 +02:00
|
|
|
{
|
2011-10-05 17:50:23 -03:00
|
|
|
struct rb_root *root;
|
2009-09-28 15:32:55 +02:00
|
|
|
struct rb_node *next;
|
|
|
|
|
struct hist_entry *n;
|
|
|
|
|
u64 min_callchain_hits;
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
min_callchain_hits = hists->stats.total_period * (callchain_param.min_percent / 100);
|
2009-09-28 15:32:55 +02:00
|
|
|
|
2013-05-14 11:09:01 +09:00
|
|
|
if (sort__need_collapse)
|
2011-10-05 17:50:23 -03:00
|
|
|
root = &hists->entries_collapsed;
|
|
|
|
|
else
|
|
|
|
|
root = hists->entries_in;
|
|
|
|
|
|
|
|
|
|
next = rb_first(root);
|
|
|
|
|
hists->entries = RB_ROOT;
|
2009-09-28 15:32:55 +02:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists->nr_entries = 0;
|
2011-10-27 09:19:48 -02:00
|
|
|
hists->stats.total_period = 0;
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__reset_col_len(hists);
|
2010-05-10 13:57:51 -03:00
|
|
|
|
2009-09-28 15:32:55 +02:00
|
|
|
while (next) {
|
2011-10-05 17:50:23 -03:00
|
|
|
n = rb_entry(next, struct hist_entry, rb_node_in);
|
|
|
|
|
next = rb_next(&n->rb_node_in);
|
2009-09-28 15:32:55 +02:00
|
|
|
|
2011-10-05 17:50:23 -03:00
|
|
|
__hists__insert_output_entry(&hists->entries, n, min_callchain_hits);
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__inc_nr_entries(hists, n);
|
2009-09-28 15:32:55 +02:00
|
|
|
}
|
2011-10-05 17:50:23 -03:00
|
|
|
}
|
2009-12-14 11:37:11 -02:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
static void hists__remove_entry_filter(struct hists *hists, struct hist_entry *h,
|
2010-07-16 12:35:07 -03:00
|
|
|
enum hist_filter filter)
|
|
|
|
|
{
|
|
|
|
|
h->filtered &= ~(1 << filter);
|
|
|
|
|
if (h->filtered)
|
|
|
|
|
return;
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
++hists->nr_entries;
|
2010-07-26 17:13:40 -03:00
|
|
|
if (h->ms.unfolded)
|
2011-09-26 12:33:28 -03:00
|
|
|
hists->nr_entries += h->nr_rows;
|
2010-07-26 17:13:40 -03:00
|
|
|
h->row_offset = 0;
|
2012-10-04 21:49:41 +09:00
|
|
|
hists->stats.total_period += h->stat.period;
|
|
|
|
|
hists->stats.nr_events[PERF_RECORD_SAMPLE] += h->stat.nr_events;
|
2010-07-16 12:35:07 -03:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__calc_col_len(hists, h);
|
2010-07-16 12:35:07 -03:00
|
|
|
}
|
|
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
|
|
|
|
|
static bool hists__filter_entry_by_dso(struct hists *hists,
|
|
|
|
|
struct hist_entry *he)
|
|
|
|
|
{
|
|
|
|
|
if (hists->dso_filter != NULL &&
|
|
|
|
|
(he->ms.map == NULL || he->ms.map->dso != hists->dso_filter)) {
|
|
|
|
|
he->filtered |= (1 << HIST_FILTER__DSO);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-18 19:07:34 -02:00
|
|
|
void hists__filter_by_dso(struct hists *hists)
|
2010-05-11 11:10:15 -03:00
|
|
|
{
|
|
|
|
|
struct rb_node *nd;
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists->nr_entries = hists->stats.total_period = 0;
|
|
|
|
|
hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
|
|
|
|
|
hists__reset_col_len(hists);
|
2010-05-11 11:10:15 -03:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
|
2010-05-11 11:10:15 -03:00
|
|
|
struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
|
|
|
|
|
|
|
|
|
|
if (symbol_conf.exclude_other && !h->parent)
|
|
|
|
|
continue;
|
|
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
if (hists__filter_entry_by_dso(hists, h))
|
2010-05-11 11:10:15 -03:00
|
|
|
continue;
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__remove_entry_filter(hists, h, HIST_FILTER__DSO);
|
2010-05-11 11:10:15 -03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
static bool hists__filter_entry_by_thread(struct hists *hists,
|
|
|
|
|
struct hist_entry *he)
|
|
|
|
|
{
|
|
|
|
|
if (hists->thread_filter != NULL &&
|
|
|
|
|
he->thread != hists->thread_filter) {
|
|
|
|
|
he->filtered |= (1 << HIST_FILTER__THREAD);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-18 19:07:34 -02:00
|
|
|
void hists__filter_by_thread(struct hists *hists)
|
2010-05-11 11:10:15 -03:00
|
|
|
{
|
|
|
|
|
struct rb_node *nd;
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists->nr_entries = hists->stats.total_period = 0;
|
|
|
|
|
hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
|
|
|
|
|
hists__reset_col_len(hists);
|
2010-05-11 11:10:15 -03:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
|
2010-05-11 11:10:15 -03:00
|
|
|
struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
|
|
|
|
|
|
2011-10-19 13:09:10 -02:00
|
|
|
if (hists__filter_entry_by_thread(hists, h))
|
2010-05-11 11:10:15 -03:00
|
|
|
continue;
|
2010-07-16 12:35:07 -03:00
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
hists__remove_entry_filter(hists, h, HIST_FILTER__THREAD);
|
2010-05-11 11:10:15 -03:00
|
|
|
}
|
|
|
|
|
}
|
2010-05-11 23:18:06 -03:00
|
|
|
|
2012-03-16 17:50:51 +09:00
|
|
|
static bool hists__filter_entry_by_symbol(struct hists *hists,
|
|
|
|
|
struct hist_entry *he)
|
|
|
|
|
{
|
|
|
|
|
if (hists->symbol_filter_str != NULL &&
|
|
|
|
|
(!he->ms.sym || strstr(he->ms.sym->name,
|
|
|
|
|
hists->symbol_filter_str) == NULL)) {
|
|
|
|
|
he->filtered |= (1 << HIST_FILTER__SYMBOL);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void hists__filter_by_symbol(struct hists *hists)
|
|
|
|
|
{
|
|
|
|
|
struct rb_node *nd;
|
|
|
|
|
|
|
|
|
|
hists->nr_entries = hists->stats.total_period = 0;
|
|
|
|
|
hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
|
|
|
|
|
hists__reset_col_len(hists);
|
|
|
|
|
|
|
|
|
|
for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
|
|
|
|
|
struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
|
|
|
|
|
|
|
|
|
|
if (hists__filter_entry_by_symbol(hists, h))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
hists__remove_entry_filter(hists, h, HIST_FILTER__SYMBOL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-04 13:43:24 -02:00
|
|
|
int hist_entry__inc_addr_samples(struct hist_entry *he, int evidx, u64 ip)
|
2010-05-11 23:18:06 -03:00
|
|
|
{
|
2011-02-04 13:43:24 -02:00
|
|
|
return symbol__inc_addr_samples(he->ms.sym, he->ms.map, evidx, ip);
|
2010-05-11 23:18:06 -03:00
|
|
|
}
|
|
|
|
|
|
2011-02-08 13:27:39 -02:00
|
|
|
int hist_entry__annotate(struct hist_entry *he, size_t privsize)
|
2010-05-11 23:18:06 -03:00
|
|
|
{
|
2011-02-08 13:27:39 -02:00
|
|
|
return symbol__annotate(he->ms.sym, he->ms.map, privsize);
|
2010-05-11 23:18:06 -03:00
|
|
|
}
|
2010-05-14 10:36:42 -03:00
|
|
|
|
2012-12-18 16:24:46 -03:00
|
|
|
void events_stats__inc(struct events_stats *stats, u32 type)
|
|
|
|
|
{
|
|
|
|
|
++stats->nr_events[0];
|
|
|
|
|
++stats->nr_events[type];
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-26 12:33:28 -03:00
|
|
|
void hists__inc_nr_events(struct hists *hists, u32 type)
|
2010-05-14 10:36:42 -03:00
|
|
|
{
|
2012-12-18 16:24:46 -03:00
|
|
|
events_stats__inc(&hists->stats, type);
|
2010-05-14 10:36:42 -03:00
|
|
|
}
|
2012-11-08 17:54:33 -03:00
|
|
|
|
2012-11-08 18:03:09 -03:00
|
|
|
static struct hist_entry *hists__add_dummy_entry(struct hists *hists,
|
|
|
|
|
struct hist_entry *pair)
|
|
|
|
|
{
|
2012-12-10 17:29:55 +09:00
|
|
|
struct rb_root *root;
|
|
|
|
|
struct rb_node **p;
|
2012-11-08 18:03:09 -03:00
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
|
struct hist_entry *he;
|
2013-10-01 07:22:15 -07:00
|
|
|
int64_t cmp;
|
2012-11-08 18:03:09 -03:00
|
|
|
|
2012-12-10 17:29:55 +09:00
|
|
|
if (sort__need_collapse)
|
|
|
|
|
root = &hists->entries_collapsed;
|
|
|
|
|
else
|
|
|
|
|
root = hists->entries_in;
|
|
|
|
|
|
|
|
|
|
p = &root->rb_node;
|
|
|
|
|
|
2012-11-08 18:03:09 -03:00
|
|
|
while (*p != NULL) {
|
|
|
|
|
parent = *p;
|
2012-12-10 17:29:55 +09:00
|
|
|
he = rb_entry(parent, struct hist_entry, rb_node_in);
|
2012-11-08 18:03:09 -03:00
|
|
|
|
2012-12-10 17:29:55 +09:00
|
|
|
cmp = hist_entry__collapse(he, pair);
|
2012-11-08 18:03:09 -03:00
|
|
|
|
|
|
|
|
if (!cmp)
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
if (cmp < 0)
|
|
|
|
|
p = &(*p)->rb_left;
|
|
|
|
|
else
|
|
|
|
|
p = &(*p)->rb_right;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
he = hist_entry__new(pair);
|
|
|
|
|
if (he) {
|
2012-11-12 13:20:03 -03:00
|
|
|
memset(&he->stat, 0, sizeof(he->stat));
|
|
|
|
|
he->hists = hists;
|
2012-12-10 17:29:55 +09:00
|
|
|
rb_link_node(&he->rb_node_in, parent, p);
|
|
|
|
|
rb_insert_color(&he->rb_node_in, root);
|
2012-11-08 18:03:09 -03:00
|
|
|
hists__inc_nr_entries(hists, he);
|
2012-12-01 21:18:20 +01:00
|
|
|
he->dummy = true;
|
2012-11-08 18:03:09 -03:00
|
|
|
}
|
|
|
|
|
out:
|
|
|
|
|
return he;
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-08 17:54:33 -03:00
|
|
|
static struct hist_entry *hists__find_entry(struct hists *hists,
|
|
|
|
|
struct hist_entry *he)
|
|
|
|
|
{
|
2012-12-10 17:29:55 +09:00
|
|
|
struct rb_node *n;
|
|
|
|
|
|
|
|
|
|
if (sort__need_collapse)
|
|
|
|
|
n = hists->entries_collapsed.rb_node;
|
|
|
|
|
else
|
|
|
|
|
n = hists->entries_in->rb_node;
|
2012-11-08 17:54:33 -03:00
|
|
|
|
|
|
|
|
while (n) {
|
2012-12-10 17:29:55 +09:00
|
|
|
struct hist_entry *iter = rb_entry(n, struct hist_entry, rb_node_in);
|
|
|
|
|
int64_t cmp = hist_entry__collapse(iter, he);
|
2012-11-08 17:54:33 -03:00
|
|
|
|
|
|
|
|
if (cmp < 0)
|
|
|
|
|
n = n->rb_left;
|
|
|
|
|
else if (cmp > 0)
|
|
|
|
|
n = n->rb_right;
|
|
|
|
|
else
|
|
|
|
|
return iter;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Look for pairs to link to the leader buckets (hist_entries):
|
|
|
|
|
*/
|
|
|
|
|
void hists__match(struct hists *leader, struct hists *other)
|
|
|
|
|
{
|
2012-12-10 17:29:55 +09:00
|
|
|
struct rb_root *root;
|
2012-11-08 17:54:33 -03:00
|
|
|
struct rb_node *nd;
|
|
|
|
|
struct hist_entry *pos, *pair;
|
|
|
|
|
|
2012-12-10 17:29:55 +09:00
|
|
|
if (sort__need_collapse)
|
|
|
|
|
root = &leader->entries_collapsed;
|
|
|
|
|
else
|
|
|
|
|
root = leader->entries_in;
|
|
|
|
|
|
|
|
|
|
for (nd = rb_first(root); nd; nd = rb_next(nd)) {
|
|
|
|
|
pos = rb_entry(nd, struct hist_entry, rb_node_in);
|
2012-11-08 17:54:33 -03:00
|
|
|
pair = hists__find_entry(other, pos);
|
|
|
|
|
|
|
|
|
|
if (pair)
|
2012-11-29 15:38:34 +09:00
|
|
|
hist_entry__add_pair(pair, pos);
|
2012-11-08 17:54:33 -03:00
|
|
|
}
|
|
|
|
|
}
|
2012-11-08 18:03:09 -03:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Look for entries in the other hists that are not present in the leader, if
|
|
|
|
|
* we find them, just add a dummy entry on the leader hists, with period=0,
|
|
|
|
|
* nr_events=0, to serve as the list header.
|
|
|
|
|
*/
|
|
|
|
|
int hists__link(struct hists *leader, struct hists *other)
|
|
|
|
|
{
|
2012-12-10 17:29:55 +09:00
|
|
|
struct rb_root *root;
|
2012-11-08 18:03:09 -03:00
|
|
|
struct rb_node *nd;
|
|
|
|
|
struct hist_entry *pos, *pair;
|
|
|
|
|
|
2012-12-10 17:29:55 +09:00
|
|
|
if (sort__need_collapse)
|
|
|
|
|
root = &other->entries_collapsed;
|
|
|
|
|
else
|
|
|
|
|
root = other->entries_in;
|
|
|
|
|
|
|
|
|
|
for (nd = rb_first(root); nd; nd = rb_next(nd)) {
|
|
|
|
|
pos = rb_entry(nd, struct hist_entry, rb_node_in);
|
2012-11-08 18:03:09 -03:00
|
|
|
|
|
|
|
|
if (!hist_entry__has_pairs(pos)) {
|
|
|
|
|
pair = hists__add_dummy_entry(leader, pos);
|
|
|
|
|
if (pair == NULL)
|
|
|
|
|
return -1;
|
2012-11-29 15:38:34 +09:00
|
|
|
hist_entry__add_pair(pos, pair);
|
2012-11-08 18:03:09 -03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|