db6d0bb861
It seems not very useful, because it's possible and event more convenient to lookup related symbol by name. Also the output value for both 'baseline' and 'new' data is quite apparent from diff output. And above all it complicates hist code factoring ;) Ditching out PERF_HPP__DISPL column with related output functions. Suggested-by: Arnaldo Carvalho de Melo <acme@redhat.com> Signed-off-by: Jiri Olsa <jolsa@redhat.com> Cc: Corey Ashford <cjashfor@linux.vnet.ibm.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Ingo Molnar <mingo@elte.hu> Cc: Ingo Molnar <mingo@kernel.org> Cc: Namhyung Kim <namhyung.kim@lge.com> Cc: Namhyung Kim <namhyung@kernel.org> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Stephane Eranian <eranian@google.com> Link: http://lkml.kernel.org/r/20121206132228.GB1080@krava.brq.redhat.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
528 lines
13 KiB
C
528 lines
13 KiB
C
#include <math.h>
|
|
|
|
#include "../util/hist.h"
|
|
#include "../util/util.h"
|
|
#include "../util/sort.h"
|
|
|
|
|
|
/* hist period print (hpp) functions */
|
|
static int hpp__header_overhead(struct perf_hpp *hpp)
|
|
{
|
|
return scnprintf(hpp->buf, hpp->size, "Overhead");
|
|
}
|
|
|
|
static int hpp__width_overhead(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 8;
|
|
}
|
|
|
|
static int hpp__color_overhead(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period / hists->stats.total_period;
|
|
|
|
return percent_color_snprintf(hpp->buf, hpp->size, " %6.2f%%", percent);
|
|
}
|
|
|
|
static int hpp__entry_overhead(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period / hists->stats.total_period;
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : " %6.2f%%";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
}
|
|
|
|
static int hpp__header_overhead_sys(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%7s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "sys");
|
|
}
|
|
|
|
static int hpp__width_overhead_sys(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 7;
|
|
}
|
|
|
|
static int hpp__color_overhead_sys(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_sys / hists->stats.total_period;
|
|
|
|
return percent_color_snprintf(hpp->buf, hpp->size, "%6.2f%%", percent);
|
|
}
|
|
|
|
static int hpp__entry_overhead_sys(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_sys / hists->stats.total_period;
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : "%6.2f%%";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
}
|
|
|
|
static int hpp__header_overhead_us(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%7s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "user");
|
|
}
|
|
|
|
static int hpp__width_overhead_us(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 7;
|
|
}
|
|
|
|
static int hpp__color_overhead_us(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_us / hists->stats.total_period;
|
|
|
|
return percent_color_snprintf(hpp->buf, hpp->size, "%6.2f%%", percent);
|
|
}
|
|
|
|
static int hpp__entry_overhead_us(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_us / hists->stats.total_period;
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : "%6.2f%%";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
}
|
|
|
|
static int hpp__header_overhead_guest_sys(struct perf_hpp *hpp)
|
|
{
|
|
return scnprintf(hpp->buf, hpp->size, "guest sys");
|
|
}
|
|
|
|
static int hpp__width_overhead_guest_sys(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 9;
|
|
}
|
|
|
|
static int hpp__color_overhead_guest_sys(struct perf_hpp *hpp,
|
|
struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_guest_sys / hists->stats.total_period;
|
|
|
|
return percent_color_snprintf(hpp->buf, hpp->size, " %6.2f%% ", percent);
|
|
}
|
|
|
|
static int hpp__entry_overhead_guest_sys(struct perf_hpp *hpp,
|
|
struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_guest_sys / hists->stats.total_period;
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : " %6.2f%% ";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
}
|
|
|
|
static int hpp__header_overhead_guest_us(struct perf_hpp *hpp)
|
|
{
|
|
return scnprintf(hpp->buf, hpp->size, "guest usr");
|
|
}
|
|
|
|
static int hpp__width_overhead_guest_us(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 9;
|
|
}
|
|
|
|
static int hpp__color_overhead_guest_us(struct perf_hpp *hpp,
|
|
struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_guest_us / hists->stats.total_period;
|
|
|
|
return percent_color_snprintf(hpp->buf, hpp->size, " %6.2f%% ", percent);
|
|
}
|
|
|
|
static int hpp__entry_overhead_guest_us(struct perf_hpp *hpp,
|
|
struct hist_entry *he)
|
|
{
|
|
struct hists *hists = he->hists;
|
|
double percent = 100.0 * he->stat.period_guest_us / hists->stats.total_period;
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : " %6.2f%% ";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
}
|
|
|
|
static int hpp__header_baseline(struct perf_hpp *hpp)
|
|
{
|
|
return scnprintf(hpp->buf, hpp->size, "Baseline");
|
|
}
|
|
|
|
static int hpp__width_baseline(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 8;
|
|
}
|
|
|
|
static double baseline_percent(struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
struct hists *pair_hists = pair ? pair->hists : NULL;
|
|
double percent = 0.0;
|
|
|
|
if (pair) {
|
|
u64 total_period = pair_hists->stats.total_period;
|
|
u64 base_period = pair->stat.period;
|
|
|
|
percent = 100.0 * base_period / total_period;
|
|
}
|
|
|
|
return percent;
|
|
}
|
|
|
|
static int hpp__color_baseline(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
double percent = baseline_percent(he);
|
|
|
|
if (hist_entry__has_pairs(he))
|
|
return percent_color_snprintf(hpp->buf, hpp->size, " %6.2f%%", percent);
|
|
else
|
|
return scnprintf(hpp->buf, hpp->size, " ");
|
|
}
|
|
|
|
static int hpp__entry_baseline(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
double percent = baseline_percent(he);
|
|
const char *fmt = symbol_conf.field_sep ? "%.2f" : " %6.2f%%";
|
|
|
|
if (hist_entry__has_pairs(he) || symbol_conf.field_sep)
|
|
return scnprintf(hpp->buf, hpp->size, fmt, percent);
|
|
else
|
|
return scnprintf(hpp->buf, hpp->size, " ");
|
|
}
|
|
|
|
static int hpp__header_samples(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%11s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Samples");
|
|
}
|
|
|
|
static int hpp__width_samples(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 11;
|
|
}
|
|
|
|
static int hpp__entry_samples(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%" PRIu64 : "%11" PRIu64;
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, he->stat.nr_events);
|
|
}
|
|
|
|
static int hpp__header_period(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%12s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Period");
|
|
}
|
|
|
|
static int hpp__width_period(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 12;
|
|
}
|
|
|
|
static int hpp__entry_period(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%" PRIu64 : "%12" PRIu64;
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, he->stat.period);
|
|
}
|
|
|
|
static int hpp__header_period_baseline(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%12s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Period Base");
|
|
}
|
|
|
|
static int hpp__width_period_baseline(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 12;
|
|
}
|
|
|
|
static int hpp__entry_period_baseline(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
u64 period = pair ? pair->stat.period : 0;
|
|
const char *fmt = symbol_conf.field_sep ? "%" PRIu64 : "%12" PRIu64;
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, period);
|
|
}
|
|
static int hpp__header_delta(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%7s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Delta");
|
|
}
|
|
|
|
static int hpp__width_delta(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 7;
|
|
}
|
|
|
|
static int hpp__entry_delta(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%7.7s";
|
|
char buf[32] = " ";
|
|
double diff = 0.0;
|
|
|
|
if (pair) {
|
|
if (he->diff.computed)
|
|
diff = he->diff.period_ratio_delta;
|
|
else
|
|
diff = perf_diff__compute_delta(he, pair);
|
|
} else
|
|
diff = perf_diff__period_percent(he, he->stat.period);
|
|
|
|
if (fabs(diff) >= 0.01)
|
|
scnprintf(buf, sizeof(buf), "%+4.2F%%", diff);
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, buf);
|
|
}
|
|
|
|
static int hpp__header_ratio(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%14s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Ratio");
|
|
}
|
|
|
|
static int hpp__width_ratio(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 14;
|
|
}
|
|
|
|
static int hpp__entry_ratio(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%14s";
|
|
char buf[32] = " ";
|
|
double ratio = 0.0;
|
|
|
|
if (pair) {
|
|
if (he->diff.computed)
|
|
ratio = he->diff.period_ratio;
|
|
else
|
|
ratio = perf_diff__compute_ratio(he, pair);
|
|
}
|
|
|
|
if (ratio > 0.0)
|
|
scnprintf(buf, sizeof(buf), "%+14.6F", ratio);
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, buf);
|
|
}
|
|
|
|
static int hpp__header_wdiff(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%14s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Weighted diff");
|
|
}
|
|
|
|
static int hpp__width_wdiff(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 14;
|
|
}
|
|
|
|
static int hpp__entry_wdiff(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%14s";
|
|
char buf[32] = " ";
|
|
s64 wdiff = 0;
|
|
|
|
if (pair) {
|
|
if (he->diff.computed)
|
|
wdiff = he->diff.wdiff;
|
|
else
|
|
wdiff = perf_diff__compute_wdiff(he, pair);
|
|
}
|
|
|
|
if (wdiff != 0)
|
|
scnprintf(buf, sizeof(buf), "%14ld", wdiff);
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, buf);
|
|
}
|
|
|
|
static int hpp__header_formula(struct perf_hpp *hpp)
|
|
{
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%70s";
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, "Formula");
|
|
}
|
|
|
|
static int hpp__width_formula(struct perf_hpp *hpp __maybe_unused)
|
|
{
|
|
return 70;
|
|
}
|
|
|
|
static int hpp__entry_formula(struct perf_hpp *hpp, struct hist_entry *he)
|
|
{
|
|
struct hist_entry *pair = hist_entry__next_pair(he);
|
|
const char *fmt = symbol_conf.field_sep ? "%s" : "%-70s";
|
|
char buf[96] = " ";
|
|
|
|
if (pair)
|
|
perf_diff__formula(he, pair, buf, sizeof(buf));
|
|
|
|
return scnprintf(hpp->buf, hpp->size, fmt, buf);
|
|
}
|
|
|
|
#define HPP__COLOR_PRINT_FNS(_name) \
|
|
{ \
|
|
.header = hpp__header_ ## _name, \
|
|
.width = hpp__width_ ## _name, \
|
|
.color = hpp__color_ ## _name, \
|
|
.entry = hpp__entry_ ## _name \
|
|
}
|
|
|
|
#define HPP__PRINT_FNS(_name) \
|
|
{ \
|
|
.header = hpp__header_ ## _name, \
|
|
.width = hpp__width_ ## _name, \
|
|
.entry = hpp__entry_ ## _name \
|
|
}
|
|
|
|
struct perf_hpp_fmt perf_hpp__format[] = {
|
|
HPP__COLOR_PRINT_FNS(baseline),
|
|
HPP__COLOR_PRINT_FNS(overhead),
|
|
HPP__COLOR_PRINT_FNS(overhead_sys),
|
|
HPP__COLOR_PRINT_FNS(overhead_us),
|
|
HPP__COLOR_PRINT_FNS(overhead_guest_sys),
|
|
HPP__COLOR_PRINT_FNS(overhead_guest_us),
|
|
HPP__PRINT_FNS(samples),
|
|
HPP__PRINT_FNS(period),
|
|
HPP__PRINT_FNS(period_baseline),
|
|
HPP__PRINT_FNS(delta),
|
|
HPP__PRINT_FNS(ratio),
|
|
HPP__PRINT_FNS(wdiff),
|
|
HPP__PRINT_FNS(formula)
|
|
};
|
|
|
|
LIST_HEAD(perf_hpp__list);
|
|
|
|
#undef HPP__COLOR_PRINT_FNS
|
|
#undef HPP__PRINT_FNS
|
|
|
|
void perf_hpp__init(void)
|
|
{
|
|
if (symbol_conf.show_cpu_utilization) {
|
|
perf_hpp__column_enable(PERF_HPP__OVERHEAD_SYS);
|
|
perf_hpp__column_enable(PERF_HPP__OVERHEAD_US);
|
|
|
|
if (perf_guest) {
|
|
perf_hpp__column_enable(PERF_HPP__OVERHEAD_GUEST_SYS);
|
|
perf_hpp__column_enable(PERF_HPP__OVERHEAD_GUEST_US);
|
|
}
|
|
}
|
|
|
|
if (symbol_conf.show_nr_samples)
|
|
perf_hpp__column_enable(PERF_HPP__SAMPLES);
|
|
|
|
if (symbol_conf.show_total_period)
|
|
perf_hpp__column_enable(PERF_HPP__PERIOD);
|
|
}
|
|
|
|
void perf_hpp__column_register(struct perf_hpp_fmt *format)
|
|
{
|
|
list_add_tail(&format->list, &perf_hpp__list);
|
|
}
|
|
|
|
void perf_hpp__column_enable(unsigned col)
|
|
{
|
|
BUG_ON(col >= PERF_HPP__MAX_INDEX);
|
|
perf_hpp__column_register(&perf_hpp__format[col]);
|
|
}
|
|
|
|
static inline void advance_hpp(struct perf_hpp *hpp, int inc)
|
|
{
|
|
hpp->buf += inc;
|
|
hpp->size -= inc;
|
|
}
|
|
|
|
int hist_entry__period_snprintf(struct perf_hpp *hpp, struct hist_entry *he,
|
|
bool color)
|
|
{
|
|
const char *sep = symbol_conf.field_sep;
|
|
struct perf_hpp_fmt *fmt;
|
|
char *start = hpp->buf;
|
|
int ret;
|
|
bool first = true;
|
|
|
|
if (symbol_conf.exclude_other && !he->parent)
|
|
return 0;
|
|
|
|
perf_hpp__for_each_format(fmt) {
|
|
/*
|
|
* If there's no field_sep, we still need
|
|
* to display initial ' '.
|
|
*/
|
|
if (!sep || !first) {
|
|
ret = scnprintf(hpp->buf, hpp->size, "%s", sep ?: " ");
|
|
advance_hpp(hpp, ret);
|
|
} else
|
|
first = false;
|
|
|
|
if (color && fmt->color)
|
|
ret = fmt->color(hpp, he);
|
|
else
|
|
ret = fmt->entry(hpp, he);
|
|
|
|
advance_hpp(hpp, ret);
|
|
}
|
|
|
|
return hpp->buf - start;
|
|
}
|
|
|
|
int hist_entry__sort_snprintf(struct hist_entry *he, char *s, size_t size,
|
|
struct hists *hists)
|
|
{
|
|
const char *sep = symbol_conf.field_sep;
|
|
struct sort_entry *se;
|
|
int ret = 0;
|
|
|
|
list_for_each_entry(se, &hist_entry__sort_list, list) {
|
|
if (se->elide)
|
|
continue;
|
|
|
|
ret += scnprintf(s + ret, size - ret, "%s", sep ?: " ");
|
|
ret += se->se_snprintf(he, s + ret, size - ret,
|
|
hists__col_len(hists, se->se_width_idx));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* See hists__fprintf to match the column widths
|
|
*/
|
|
unsigned int hists__sort_list_width(struct hists *hists)
|
|
{
|
|
struct perf_hpp_fmt *fmt;
|
|
struct sort_entry *se;
|
|
int i = 0, ret = 0;
|
|
|
|
perf_hpp__for_each_format(fmt) {
|
|
if (i)
|
|
ret += 2;
|
|
|
|
ret += fmt->width(NULL);
|
|
}
|
|
|
|
list_for_each_entry(se, &hist_entry__sort_list, list)
|
|
if (!se->elide)
|
|
ret += 2 + hists__col_len(hists, se->se_width_idx);
|
|
|
|
if (verbose) /* Addr + origin */
|
|
ret += 3 + BITS_PER_LONG / 4;
|
|
|
|
return ret;
|
|
}
|