perf hists: Threaded addition and sorting of entries
By using a mutex just for inserting and rotating two hist_entry rb trees, so that when sorting we can get the last batch of entries created from the ring buffer, merge it with whatever we have processed so far and show the output while new entries are being added. The 'report' tool continues, for now, to do it without threading, but will use this in the future to allow visualization of results in long perf.data sessions while the entries are being processed. The new 'top' tool will be the first user. Cc: David Ahern <dsahern@gmail.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Mike Galbraith <efault@gmx.de> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Stephane Eranian <eranian@google.com> Link: http://lkml.kernel.org/n/tip-9b05atsn0q6m7fqgrug8fk2i@git.kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
Родитель
3f2728bdb6
Коммит
1980c2ebd7
|
@ -39,6 +39,7 @@ void perf_evsel__init(struct perf_evsel *evsel,
|
||||||
evsel->idx = idx;
|
evsel->idx = idx;
|
||||||
evsel->attr = *attr;
|
evsel->attr = *attr;
|
||||||
INIT_LIST_HEAD(&evsel->node);
|
INIT_LIST_HEAD(&evsel->node);
|
||||||
|
hists__init(&evsel->hists);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct perf_evsel *perf_evsel__new(struct perf_event_attr *attr, int idx)
|
struct perf_evsel *perf_evsel__new(struct perf_event_attr *attr, int idx)
|
||||||
|
|
|
@ -118,6 +118,7 @@ static void hists__inc_nr_entries(struct hists *hists, struct hist_entry *h)
|
||||||
if (!h->filtered) {
|
if (!h->filtered) {
|
||||||
hists__calc_col_len(hists, h);
|
hists__calc_col_len(hists, h);
|
||||||
++hists->nr_entries;
|
++hists->nr_entries;
|
||||||
|
hists->stats.total_period += h->period;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -132,7 +133,7 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
|
||||||
struct addr_location *al,
|
struct addr_location *al,
|
||||||
struct symbol *sym_parent, u64 period)
|
struct symbol *sym_parent, u64 period)
|
||||||
{
|
{
|
||||||
struct rb_node **p = &hists->entries.rb_node;
|
struct rb_node **p;
|
||||||
struct rb_node *parent = NULL;
|
struct rb_node *parent = NULL;
|
||||||
struct hist_entry *he;
|
struct hist_entry *he;
|
||||||
struct hist_entry entry = {
|
struct hist_entry entry = {
|
||||||
|
@ -150,9 +151,13 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
|
||||||
};
|
};
|
||||||
int cmp;
|
int cmp;
|
||||||
|
|
||||||
|
pthread_mutex_lock(&hists->lock);
|
||||||
|
|
||||||
|
p = &hists->entries_in->rb_node;
|
||||||
|
|
||||||
while (*p != NULL) {
|
while (*p != NULL) {
|
||||||
parent = *p;
|
parent = *p;
|
||||||
he = rb_entry(parent, struct hist_entry, rb_node);
|
he = rb_entry(parent, struct hist_entry, rb_node_in);
|
||||||
|
|
||||||
cmp = hist_entry__cmp(&entry, he);
|
cmp = hist_entry__cmp(&entry, he);
|
||||||
|
|
||||||
|
@ -170,12 +175,14 @@ struct hist_entry *__hists__add_entry(struct hists *hists,
|
||||||
|
|
||||||
he = hist_entry__new(&entry);
|
he = hist_entry__new(&entry);
|
||||||
if (!he)
|
if (!he)
|
||||||
return NULL;
|
goto out_unlock;
|
||||||
rb_link_node(&he->rb_node, parent, p);
|
|
||||||
rb_insert_color(&he->rb_node, &hists->entries);
|
rb_link_node(&he->rb_node_in, parent, p);
|
||||||
hists__inc_nr_entries(hists, he);
|
rb_insert_color(&he->rb_node_in, hists->entries_in);
|
||||||
out:
|
out:
|
||||||
hist_entry__add_cpumode_period(he, al->cpumode, period);
|
hist_entry__add_cpumode_period(he, al->cpumode, period);
|
||||||
|
out_unlock:
|
||||||
|
pthread_mutex_unlock(&hists->lock);
|
||||||
return he;
|
return he;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -233,7 +240,7 @@ static bool hists__collapse_insert_entry(struct hists *hists,
|
||||||
|
|
||||||
while (*p != NULL) {
|
while (*p != NULL) {
|
||||||
parent = *p;
|
parent = *p;
|
||||||
iter = rb_entry(parent, struct hist_entry, rb_node);
|
iter = rb_entry(parent, struct hist_entry, rb_node_in);
|
||||||
|
|
||||||
cmp = hist_entry__collapse(iter, he);
|
cmp = hist_entry__collapse(iter, he);
|
||||||
|
|
||||||
|
@ -254,35 +261,57 @@ static bool hists__collapse_insert_entry(struct hists *hists,
|
||||||
p = &(*p)->rb_right;
|
p = &(*p)->rb_right;
|
||||||
}
|
}
|
||||||
|
|
||||||
rb_link_node(&he->rb_node, parent, p);
|
rb_link_node(&he->rb_node_in, parent, p);
|
||||||
rb_insert_color(&he->rb_node, root);
|
rb_insert_color(&he->rb_node_in, root);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void __hists__collapse_resort(struct hists *hists, bool threaded)
|
||||||
|
{
|
||||||
|
struct rb_root *root;
|
||||||
|
struct rb_node *next;
|
||||||
|
struct hist_entry *n;
|
||||||
|
|
||||||
|
if (!sort__need_collapse && !threaded)
|
||||||
|
return;
|
||||||
|
|
||||||
|
root = hists__get_rotate_entries_in(hists);
|
||||||
|
next = rb_first(root);
|
||||||
|
hists->stats.total_period = 0;
|
||||||
|
|
||||||
|
while (next) {
|
||||||
|
n = rb_entry(next, struct hist_entry, rb_node_in);
|
||||||
|
next = rb_next(&n->rb_node_in);
|
||||||
|
|
||||||
|
rb_erase(&n->rb_node_in, root);
|
||||||
|
if (hists__collapse_insert_entry(hists, &hists->entries_collapsed, n))
|
||||||
|
hists__inc_nr_entries(hists, n);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void hists__collapse_resort(struct hists *hists)
|
void hists__collapse_resort(struct hists *hists)
|
||||||
{
|
{
|
||||||
struct rb_root tmp;
|
return __hists__collapse_resort(hists, false);
|
||||||
struct rb_node *next;
|
}
|
||||||
struct hist_entry *n;
|
|
||||||
|
|
||||||
if (!sort__need_collapse)
|
void hists__collapse_resort_threaded(struct hists *hists)
|
||||||
return;
|
{
|
||||||
|
return __hists__collapse_resort(hists, true);
|
||||||
tmp = RB_ROOT;
|
|
||||||
next = rb_first(&hists->entries);
|
|
||||||
hists->nr_entries = 0;
|
|
||||||
hists__reset_col_len(hists);
|
|
||||||
|
|
||||||
while (next) {
|
|
||||||
n = rb_entry(next, struct hist_entry, rb_node);
|
|
||||||
next = rb_next(&n->rb_node);
|
|
||||||
|
|
||||||
rb_erase(&n->rb_node, &hists->entries);
|
|
||||||
if (hists__collapse_insert_entry(hists, &tmp, n))
|
|
||||||
hists__inc_nr_entries(hists, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
hists->entries = tmp;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -315,31 +344,43 @@ static void __hists__insert_output_entry(struct rb_root *entries,
|
||||||
rb_insert_color(&he->rb_node, entries);
|
rb_insert_color(&he->rb_node, entries);
|
||||||
}
|
}
|
||||||
|
|
||||||
void hists__output_resort(struct hists *hists)
|
static void __hists__output_resort(struct hists *hists, bool threaded)
|
||||||
{
|
{
|
||||||
struct rb_root tmp;
|
struct rb_root *root;
|
||||||
struct rb_node *next;
|
struct rb_node *next;
|
||||||
struct hist_entry *n;
|
struct hist_entry *n;
|
||||||
u64 min_callchain_hits;
|
u64 min_callchain_hits;
|
||||||
|
|
||||||
min_callchain_hits = hists->stats.total_period * (callchain_param.min_percent / 100);
|
min_callchain_hits = hists->stats.total_period * (callchain_param.min_percent / 100);
|
||||||
|
|
||||||
tmp = RB_ROOT;
|
if (sort__need_collapse || threaded)
|
||||||
next = rb_first(&hists->entries);
|
root = &hists->entries_collapsed;
|
||||||
|
else
|
||||||
|
root = hists->entries_in;
|
||||||
|
|
||||||
|
next = rb_first(root);
|
||||||
|
hists->entries = RB_ROOT;
|
||||||
|
|
||||||
hists->nr_entries = 0;
|
hists->nr_entries = 0;
|
||||||
hists__reset_col_len(hists);
|
hists__reset_col_len(hists);
|
||||||
|
|
||||||
while (next) {
|
while (next) {
|
||||||
n = rb_entry(next, struct hist_entry, rb_node);
|
n = rb_entry(next, struct hist_entry, rb_node_in);
|
||||||
next = rb_next(&n->rb_node);
|
next = rb_next(&n->rb_node_in);
|
||||||
|
|
||||||
rb_erase(&n->rb_node, &hists->entries);
|
__hists__insert_output_entry(&hists->entries, n, min_callchain_hits);
|
||||||
__hists__insert_output_entry(&tmp, n, min_callchain_hits);
|
|
||||||
hists__inc_nr_entries(hists, n);
|
hists__inc_nr_entries(hists, n);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
hists->entries = tmp;
|
void hists__output_resort(struct hists *hists)
|
||||||
|
{
|
||||||
|
return __hists__output_resort(hists, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
void hists__output_resort_threaded(struct hists *hists)
|
||||||
|
{
|
||||||
|
return __hists__output_resort(hists, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static size_t callchain__fprintf_left_margin(FILE *fp, int left_margin)
|
static size_t callchain__fprintf_left_margin(FILE *fp, int left_margin)
|
||||||
|
@ -1043,3 +1084,13 @@ size_t hists__fprintf_nr_events(struct hists *hists, FILE *fp)
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void hists__init(struct hists *hists)
|
||||||
|
{
|
||||||
|
memset(hists, 0, sizeof(*hists));
|
||||||
|
hists->entries_in_array[0] = hists->entries_in_array[1] = RB_ROOT;
|
||||||
|
hists->entries_in = &hists->entries_in_array[0];
|
||||||
|
hists->entries_collapsed = RB_ROOT;
|
||||||
|
hists->entries = RB_ROOT;
|
||||||
|
pthread_mutex_init(&hists->lock, NULL);
|
||||||
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
#define __PERF_HIST_H
|
#define __PERF_HIST_H
|
||||||
|
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
#include <pthread.h>
|
||||||
#include "callchain.h"
|
#include "callchain.h"
|
||||||
|
|
||||||
extern struct callchain_param callchain_param;
|
extern struct callchain_param callchain_param;
|
||||||
|
@ -43,8 +44,12 @@ enum hist_column {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct hists {
|
struct hists {
|
||||||
|
struct rb_root entries_in_array[2];
|
||||||
|
struct rb_root *entries_in;
|
||||||
struct rb_root entries;
|
struct rb_root entries;
|
||||||
|
struct rb_root entries_collapsed;
|
||||||
u64 nr_entries;
|
u64 nr_entries;
|
||||||
|
pthread_mutex_t lock;
|
||||||
struct events_stats stats;
|
struct events_stats stats;
|
||||||
u64 event_stream;
|
u64 event_stream;
|
||||||
u16 col_len[HISTC_NR_COLS];
|
u16 col_len[HISTC_NR_COLS];
|
||||||
|
@ -52,6 +57,8 @@ struct hists {
|
||||||
struct callchain_cursor callchain_cursor;
|
struct callchain_cursor callchain_cursor;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void hists__init(struct hists *hists);
|
||||||
|
|
||||||
struct hist_entry *__hists__add_entry(struct hists *self,
|
struct hist_entry *__hists__add_entry(struct hists *self,
|
||||||
struct addr_location *al,
|
struct addr_location *al,
|
||||||
struct symbol *parent, u64 period);
|
struct symbol *parent, u64 period);
|
||||||
|
@ -67,7 +74,9 @@ int hist_entry__snprintf(struct hist_entry *self, char *bf, size_t size,
|
||||||
void hist_entry__free(struct hist_entry *);
|
void hist_entry__free(struct hist_entry *);
|
||||||
|
|
||||||
void hists__output_resort(struct hists *self);
|
void hists__output_resort(struct hists *self);
|
||||||
|
void hists__output_resort_threaded(struct hists *hists);
|
||||||
void hists__collapse_resort(struct hists *self);
|
void hists__collapse_resort(struct hists *self);
|
||||||
|
void hists__collapse_resort_threaded(struct hists *hists);
|
||||||
|
|
||||||
void hists__inc_nr_events(struct hists *self, u32 type);
|
void hists__inc_nr_events(struct hists *self, u32 type);
|
||||||
size_t hists__fprintf_nr_events(struct hists *self, FILE *fp);
|
size_t hists__fprintf_nr_events(struct hists *self, FILE *fp);
|
||||||
|
|
|
@ -45,6 +45,7 @@ extern enum sort_type sort__first_dimension;
|
||||||
* @nr_rows - rows expanded in callchain, recalculated on folding/unfolding
|
* @nr_rows - rows expanded in callchain, recalculated on folding/unfolding
|
||||||
*/
|
*/
|
||||||
struct hist_entry {
|
struct hist_entry {
|
||||||
|
struct rb_node rb_node_in;
|
||||||
struct rb_node rb_node;
|
struct rb_node rb_node;
|
||||||
u64 period;
|
u64 period;
|
||||||
u64 period_sys;
|
u64 period_sys;
|
||||||
|
|
Загрузка…
Ссылка в новой задаче