Currently for any of the data columns (like Overhead/Period..) in
stdio ui, there's separate code to print header/dots/value scattered
along the display code path.

Adding hists_stdio_column struct to centralize all info needed
to print column header/dots/value.

This change eases up addition for new columns, which is now mostly
matter only of adding new hists_stdio_column struct.

Cc: Arnaldo Carvalho de Melo <a...@ghostprotocols.net>
Cc: Peter Zijlstra <a.p.zijls...@chello.nl>
Cc: Ingo Molnar <mi...@elte.hu>
Cc: Paul Mackerras <pau...@samba.org>
Cc: Corey Ashford <cjash...@linux.vnet.ibm.com>
Cc: Frederic Weisbecker <fweis...@gmail.com>
Cc: Paul E. McKenney <paul...@linux.vnet.ibm.com>
Cc: Andi Kleen <a...@firstfloor.org>
Cc: David Ahern <dsah...@gmail.com>
Cc: Namhyung Kim <namhy...@kernel.org>
Signed-off-by: Jiri Olsa <jo...@redhat.com>
---
 tools/perf/builtin-diff.c   |  14 +-
 tools/perf/builtin-report.c |   6 +-
 tools/perf/builtin-top.c    |   6 +-
 tools/perf/ui/stdio/hist.c  | 506 +++++++++++++++++++++++++-------------------
 tools/perf/ui/stdio/hist.h  |  19 ++
 tools/perf/util/hist.c      |   5 +-
 tools/perf/util/hist.h      |  19 +-
 7 files changed, 345 insertions(+), 230 deletions(-)
 create mode 100644 tools/perf/ui/stdio/hist.h

diff --git a/tools/perf/builtin-diff.c b/tools/perf/builtin-diff.c
index 4e63979..282fd5e 100644
--- a/tools/perf/builtin-diff.c
+++ b/tools/perf/builtin-diff.c
@@ -16,6 +16,7 @@
 #include "util/sort.h"
 #include "util/symbol.h"
 #include "util/util.h"
+#include "ui/stdio/hist.h"
 
 #include <stdlib.h>
 
@@ -214,8 +215,7 @@ static int __cmd_diff(void)
                first = false;
 
                hists__match(&evsel_old->hists, &evsel->hists);
-               hists__fprintf(&evsel->hists, &evsel_old->hists,
-                              show_displacement, true, 0, 0, stdout);
+               hists__fprintf(&evsel->hists, true, 0, 0, stdout);
        }
 
 out_delete:
@@ -257,6 +257,15 @@ static const struct option options[] = {
        OPT_END()
 };
 
+static void setup_ui_stdio(void)
+{
+       hists_stdio_column__register_idx(HISTC_BASELINE);
+       hists_stdio_column__register_global();
+       hists_stdio_column__register_idx(HISTC_DELTA);
+       if (show_displacement)
+               hists_stdio_column__register_idx(HISTC_DISPLACEMENT);
+}
+
 int cmd_diff(int argc, const char **argv, const char *prefix __used)
 {
        sort_order = diff__default_sort_order;
@@ -280,6 +289,7 @@ int cmd_diff(int argc, const char **argv, const char 
*prefix __used)
                return -1;
 
        setup_sorting(diff_usage, options);
+       setup_ui_stdio();
        setup_pager();
 
        sort_entry__setup_elide(&sort_dso, symbol_conf.dso_list, "dso", NULL);
diff --git a/tools/perf/builtin-report.c b/tools/perf/builtin-report.c
index d618253..c855c9a 100644
--- a/tools/perf/builtin-report.c
+++ b/tools/perf/builtin-report.c
@@ -33,6 +33,7 @@
 #include "util/thread.h"
 #include "util/sort.h"
 #include "util/hist.h"
+#include "ui/stdio/hist.h"
 
 #include <linux/bitmap.h>
 
@@ -315,12 +316,15 @@ static int perf_evlist__tty_browse_hists(struct 
perf_evlist *evlist,
 {
        struct perf_evsel *pos;
 
+       hists_stdio_column__register_idx(HISTC_OVERHEAD);
+       hists_stdio_column__register_global();
+
        list_for_each_entry(pos, &evlist->entries, node) {
                struct hists *hists = &pos->hists;
                const char *evname = perf_evsel__name(pos);
 
                hists__fprintf_nr_sample_events(hists, evname, stdout);
-               hists__fprintf(hists, NULL, false, true, 0, 0, stdout);
+               hists__fprintf(hists, true, 0, 0, stdout);
                fprintf(stdout, "\n\n");
        }
 
diff --git a/tools/perf/builtin-top.c b/tools/perf/builtin-top.c
index 0513aaa..26da442 100644
--- a/tools/perf/builtin-top.c
+++ b/tools/perf/builtin-top.c
@@ -41,6 +41,7 @@
 #include "util/intlist.h"
 
 #include "util/debug.h"
+#include "ui/stdio/hist.h"
 
 #include <assert.h>
 #include <elf.h>
@@ -315,7 +316,7 @@ static void perf_top__print_sym_table(struct perf_top *top)
        hists__output_recalc_col_len(&top->sym_evsel->hists,
                                     top->winsize.ws_row - 3);
        putchar('\n');
-       hists__fprintf(&top->sym_evsel->hists, NULL, false, false,
+       hists__fprintf(&top->sym_evsel->hists, false,
                       top->winsize.ws_row - 4 - printed, win_width, stdout);
 }
 
@@ -607,6 +608,9 @@ static void *display_thread(void *arg)
        struct perf_top *top = arg;
        int delay_msecs, c;
 
+       hists_stdio_column__register_idx(HISTC_OVERHEAD);
+       hists_stdio_column__register_global();
+
        tcgetattr(0, &save);
        tc = save;
        tc.c_lflag &= ~(ICANON | ECHO);
diff --git a/tools/perf/ui/stdio/hist.c b/tools/perf/ui/stdio/hist.c
index 4ae9ee5..48fb0b3 100644
--- a/tools/perf/ui/stdio/hist.c
+++ b/tools/perf/ui/stdio/hist.c
@@ -1,10 +1,180 @@
 #include <stdio.h>
 #include <math.h>
 
+#include "hist.h"
 #include "../../util/util.h"
 #include "../../util/hist.h"
 #include "../../util/sort.h"
 
+static double get_period_percent(struct hist_entry *he, u64 period)
+{
+       u64 total  = he->hists->stats.total_period;
+       return (period * 100.0) / total;
+}
+
+static int
+hists_stdio_column__delta_snprintf(struct hist_entry *he, char *bf,
+                                  size_t size, unsigned int width __used)
+{
+       struct hist_entry *pair = he->pair;
+       double new_percent = get_period_percent(he, he->period);
+       double old_percent = pair ? get_period_percent(pair, pair->period) : 
0.0;
+       double diff = new_percent - old_percent;
+       int ret;
+
+       if (fabs(diff) >= 0.01)
+               ret = percent_color_snprintf(bf, size, "%+7.2F%%", diff);
+       else
+               ret = scnprintf(bf, size, "%8s", " ");
+
+       return ret;
+}
+
+static int
+hists_stdio_column__baseline_snprintf(struct hist_entry *he, char *bf,
+                                     size_t size, unsigned int width __used)
+{
+       struct hist_entry *pair = he->pair;
+       double percent = pair ? get_period_percent(pair, pair->period) : 0.0;
+
+       return percent_color_snprintf(bf, size, "%7.2F%%", percent);
+}
+
+static int
+hists_stdio_column__overhead_snprintf(struct hist_entry *he, char *bf,
+                                     size_t size, unsigned int width __used)
+{
+       double percent = get_period_percent(he, he->period);
+       return percent_color_snprintf(bf, size, "%7.2f%%", percent);
+}
+
+static int
+hists_stdio_column__nr_samples_snprintf(struct hist_entry *he, char *bf,
+                                       size_t size, unsigned int width __used)
+{
+       return scnprintf(bf, size, "%12" PRIu64, he->nr_events);
+}
+
+static int
+hists_stdio_column__total_period_snprintf(struct hist_entry *he, char *bf,
+                                         size_t size, unsigned int width 
__used)
+{
+       return scnprintf(bf, size , "%12" PRIu64, he->period);
+}
+
+static int
+hists_stdio_column__cpu_sys_snprintf(struct hist_entry *he, char *bf,
+                                    size_t size, unsigned int width __used)
+{
+       double percent = get_period_percent(he, he->period_sys);
+       return percent_color_snprintf(bf, size, "%7.2f%%", percent);
+}
+
+static int
+hists_stdio_column__cpu_us_snprintf(struct hist_entry *he, char *bf,
+                                   size_t size, unsigned int width __used)
+{
+       double percent = get_period_percent(he, he->period_us);
+       return percent_color_snprintf(bf, size, "%7.2f%%", percent);
+}
+
+static int
+hists_stdio_column__cpu_guest_sys_snprintf(struct hist_entry *he, char *bf,
+                                          size_t size, unsigned int width 
__used)
+{
+       double percent = get_period_percent(he, he->period_guest_sys);
+       return percent_color_snprintf(bf, size, "%8.2f%%", percent);
+}
+
+static int
+hists_stdio_column__cpu_guest_us_snprintf(struct hist_entry *he, char *bf,
+                                         size_t size, unsigned int width 
__used)
+{
+       double percent = get_period_percent(he, he->period_guest_us);
+       return percent_color_snprintf(bf, size, "%7.2f%%", percent);
+}
+
+static int
+hists_stdio_column__displacement_snprintf(struct hist_entry *he, char *bf,
+                                         size_t size, unsigned int width 
__used)
+{
+       struct hist_entry *pair = he->pair;
+       unsigned long displ = pair ? pair->position - he->position : 0;
+       return scnprintf(bf, size, displ ? "%+5ld" : "     ", displ);
+}
+
+LIST_HEAD(hists_stdio_column__list);
+
+#define DEF_COLUMN(name, c, w, h)                                      \
+[ c ] = {                                                                      
\
+       .list           = LIST_HEAD_INIT((hists_stdio_column__array[ c 
]).list),\
+       .header         = h,                                            \
+       .snprintf       = hists_stdio_column__ ## name ## _snprintf,    \
+       .col_idx        = c,                                            \
+       .col_width      = w,                                            \
+},
+
+struct hists_stdio_column hists_stdio_column__array[HISTC_STDIO_NR_COLS] = {
+DEF_COLUMN(overhead, HISTC_OVERHEAD, 8, "Overhead")
+DEF_COLUMN(baseline, HISTC_BASELINE, 8, "Baseline")
+DEF_COLUMN(cpu_sys, HISTC_CPU_UTILIZATION_SYS, 8, "sys")
+DEF_COLUMN(cpu_us, HISTC_CPU_UTILIZATION_US, 8, "us")
+DEF_COLUMN(cpu_guest_sys, HISTC_CPU_UTILIZATION_GUEST_SYS, 8, "guest sys")
+DEF_COLUMN(cpu_guest_us, HISTC_CPU_UTILIZATION_GUEST_US, 8, "guest us")
+DEF_COLUMN(nr_samples, HISTC_NR_SAMPLES, 12, "Samples")
+DEF_COLUMN(total_period, HISTC_TOTAL_PERIOD, 12, "Period")
+DEF_COLUMN(delta, HISTC_DELTA, 8, "Delta")
+DEF_COLUMN(displacement, HISTC_DISPLACEMENT, 5, "Displ")
+};
+
+int hists_stdio_column__register_idx(int idx)
+{
+       struct hists_stdio_column *col;
+
+       if (idx >= HISTC_STDIO_NR_COLS)
+               return -EINVAL;
+
+       col = &hists_stdio_column__array[idx];
+
+       if (!list_empty(&col->list))
+               return -EBUSY;
+
+       list_add_tail(&col->list, &hists_stdio_column__list);
+       return 0;
+}
+
+void hists_stdio_column__set_width(struct hists *hists)
+{
+       struct hists_stdio_column *col;
+       unsigned i;
+
+       for (i = 0; i < HISTC_STDIO_NR_COLS; i++) {
+               col = &hists_stdio_column__array[i];
+               hists__set_col_len(hists, i, col->col_width);
+       }
+}
+
+void hists_stdio_column__register_global(void)
+{
+#define reg(idx) hists_stdio_column__register_idx(idx)
+
+       if (symbol_conf.show_cpu_utilization) {
+               reg(HISTC_CPU_UTILIZATION_SYS);
+               reg(HISTC_CPU_UTILIZATION_US);
+       }
+
+       if (perf_guest) {
+               reg(HISTC_CPU_UTILIZATION_GUEST_SYS);
+               reg(HISTC_CPU_UTILIZATION_GUEST_US);
+       }
+
+       if (symbol_conf.show_nr_samples)
+               reg(HISTC_NR_SAMPLES);
+
+       if (symbol_conf.show_total_period)
+               reg(HISTC_TOTAL_PERIOD);
+#undef reg
+}
 
 static size_t callchain__fprintf_left_margin(FILE *fp, int left_margin)
 {
@@ -291,114 +461,20 @@ static size_t hist_entry_callchain__fprintf(struct 
hist_entry *he,
        return 0;
 }
 
-static int hist_entry__period_snprintf(struct hist_entry *he, char *s,
-                                    size_t size, struct hists *pair_hists,
-                                    bool show_displacement, long displacement,
-                                    bool color, u64 total_period)
+static int
+hists_stdio_column__snprintf(struct hist_entry *he, char *s, size_t size,
+                           struct hists *hists)
 {
-       u64 period, total, period_sys, period_us, period_guest_sys, 
period_guest_us;
-       u64 nr_events;
        const char *sep = symbol_conf.field_sep;
-       int ret;
-
-       if (symbol_conf.exclude_other && !he->parent)
-               return 0;
-
-       if (pair_hists) {
-               period = he->pair ? he->pair->period : 0;
-               nr_events = he->pair ? he->pair->nr_events : 0;
-               total = pair_hists->stats.total_period;
-               period_sys = he->pair ? he->pair->period_sys : 0;
-               period_us = he->pair ? he->pair->period_us : 0;
-               period_guest_sys = he->pair ? he->pair->period_guest_sys : 0;
-               period_guest_us = he->pair ? he->pair->period_guest_us : 0;
-       } else {
-               period = he->period;
-               nr_events = he->nr_events;
-               total = total_period;
-               period_sys = he->period_sys;
-               period_us = he->period_us;
-               period_guest_sys = he->period_guest_sys;
-               period_guest_us = he->period_guest_us;
-       }
-
-       if (total) {
-               if (color)
-                       ret = percent_color_snprintf(s, size,
-                                                    sep ? "%.2f" : "   
%6.2f%%",
-                                                    (period * 100.0) / total);
-               else
-                       ret = scnprintf(s, size, sep ? "%.2f" : "   %6.2f%%",
-                                      (period * 100.0) / total);
-               if (symbol_conf.show_cpu_utilization) {
-                       ret += percent_color_snprintf(s + ret, size - ret,
-                                       sep ? "%.2f" : "   %6.2f%%",
-                                       (period_sys * 100.0) / total);
-                       ret += percent_color_snprintf(s + ret, size - ret,
-                                       sep ? "%.2f" : "   %6.2f%%",
-                                       (period_us * 100.0) / total);
-                       if (perf_guest) {
-                               ret += percent_color_snprintf(s + ret,
-                                               size - ret,
-                                               sep ? "%.2f" : "   %6.2f%%",
-                                               (period_guest_sys * 100.0) /
-                                                               total);
-                               ret += percent_color_snprintf(s + ret,
-                                               size - ret,
-                                               sep ? "%.2f" : "   %6.2f%%",
-                                               (period_guest_us * 100.0) /
-                                                               total);
-                       }
-               }
-       } else
-               ret = scnprintf(s, size, sep ? "%" PRIu64 : "%12" PRIu64 " ", 
period);
-
-       if (symbol_conf.show_nr_samples) {
-               if (sep)
-                       ret += scnprintf(s + ret, size - ret, "%c%" PRIu64, 
*sep, nr_events);
-               else
-                       ret += scnprintf(s + ret, size - ret, "%11" PRIu64, 
nr_events);
-       }
-
-       if (symbol_conf.show_total_period) {
-               if (sep)
-                       ret += scnprintf(s + ret, size - ret, "%c%" PRIu64, 
*sep, period);
-               else
-                       ret += scnprintf(s + ret, size - ret, " %12" PRIu64, 
period);
-       }
-
-       if (pair_hists) {
-               char bf[32];
-               double old_percent = 0, new_percent = 0, diff;
-
-               if (total > 0)
-                       old_percent = (period * 100.0) / total;
-               if (total_period > 0)
-                       new_percent = (he->period * 100.0) / total_period;
-
-               diff = new_percent - old_percent;
-
-               if (fabs(diff) >= 0.01)
-                       scnprintf(bf, sizeof(bf), "%+4.2F%%", diff);
-               else
-                       scnprintf(bf, sizeof(bf), " ");
-
-               if (sep)
-                       ret += scnprintf(s + ret, size - ret, "%c%s", *sep, bf);
-               else
-                       ret += scnprintf(s + ret, size - ret, "%11.11s", bf);
+       struct hists_stdio_column *col;
+       int ret = 0;
 
-               if (show_displacement) {
-                       if (displacement)
-                               scnprintf(bf, sizeof(bf), "%+4ld", 
displacement);
-                       else
-                               scnprintf(bf, sizeof(bf), " ");
+       ret = snprintf(s, size, "  ");
 
-                       if (sep)
-                               ret += scnprintf(s + ret, size - ret, "%c%s", 
*sep, bf);
-                       else
-                               ret += scnprintf(s + ret, size - ret, "%6.6s", 
bf);
-               }
+       list_for_each_entry(col, &hists_stdio_column__list, list) {
+               ret += scnprintf(s + ret, size - ret, "%s", sep ?: "  ");
+               ret += col->snprintf(he, s + ret, size - ret,
+                                      hists__col_len(hists, col->col_idx));
        }
 
        return ret;
@@ -440,9 +516,8 @@ static size_t hist_entry__callchain_fprintf(struct 
hist_entry *he,
 }
 
 static int hist_entry__fprintf(struct hist_entry *he, size_t size,
-                              struct hists *hists, struct hists *pair_hists,
-                              bool show_displacement, long displacement,
-                              u64 total_period, FILE *fp)
+                              struct hists *hists, u64 total_period,
+                              FILE *fp)
 {
        char bf[512];
        int ret;
@@ -450,9 +525,8 @@ static int hist_entry__fprintf(struct hist_entry *he, 
size_t size,
        if (size == 0 || size > sizeof(bf))
                size = sizeof(bf);
 
-       ret = hist_entry__period_snprintf(he, bf, size, pair_hists,
-                                         show_displacement, displacement,
-                                         true, total_period);
+       ret = hists_stdio_column__snprintf(he, bf, size, hists);
+
        hist_entry__sort_snprintf(he, bf + ret, size - ret, hists);
 
        ret = fprintf(fp, "%s\n", bf);
@@ -464,138 +538,117 @@ static int hist_entry__fprintf(struct hist_entry *he, 
size_t size,
        return ret;
 }
 
-size_t hists__fprintf(struct hists *hists, struct hists *pair,
-                     bool show_displacement, bool show_header, int max_rows,
-                     int max_cols, FILE *fp)
+static void fprintf_header(FILE *fp, struct hists *hists,
+                          const char *header, int col)
 {
-       struct sort_entry *se;
-       struct rb_node *nd;
-       size_t ret = 0;
-       u64 total_period;
-       long displacement = 0;
-       unsigned int width;
-       const char *sep = symbol_conf.field_sep;
        const char *col_width = symbol_conf.col_width_list_str;
-       int nr_rows = 0;
-
-       init_rem_hits();
-
-       if (!show_header)
-               goto print_entries;
+       const char *sep = symbol_conf.field_sep;
+       unsigned int width;
 
-       fprintf(fp, "# %s", pair ? "Baseline" : "Overhead");
+       if (sep) {
+               fprintf(fp, "%c%s", *sep, header);
+               return;
+       }
 
-       if (symbol_conf.show_cpu_utilization) {
-               if (sep) {
-                       ret += fprintf(fp, "%csys", *sep);
-                       ret += fprintf(fp, "%cus", *sep);
-                       if (perf_guest) {
-                               ret += fprintf(fp, "%cguest sys", *sep);
-                               ret += fprintf(fp, "%cguest us", *sep);
-                       }
-               } else {
-                       ret += fprintf(fp, "     sys  ");
-                       ret += fprintf(fp, "      us  ");
-                       if (perf_guest) {
-                               ret += fprintf(fp, "  guest sys  ");
-                               ret += fprintf(fp, "  guest us  ");
-                       }
+       width = strlen(header);
+       if (symbol_conf.col_width_list_str) {
+               if (col_width) {
+                       hists__set_col_len(hists, col,
+                                          atoi(col_width));
+                       col_width = strchr(col_width, ',');
+                       if (col_width)
+                               ++col_width;
                }
        }
 
-       if (symbol_conf.show_nr_samples) {
-               if (sep)
-                       fprintf(fp, "%cSamples", *sep);
-               else
-                       fputs("  Samples  ", fp);
-       }
+       if (!hists__new_col_len(hists, col, width))
+               width = hists__col_len(hists, col);
 
-       if (symbol_conf.show_total_period) {
-               if (sep)
-                       ret += fprintf(fp, "%cPeriod", *sep);
-               else
-                       ret += fprintf(fp, "   Period    ");
-       }
+       fprintf(fp, "  %*s", width, header);
+}
 
-       if (pair) {
-               if (sep)
-                       ret += fprintf(fp, "%cDelta", *sep);
-               else
-                       ret += fprintf(fp, "  Delta    ");
+static void fprintf_dots(FILE *fp, struct hists *hists,
+                        const char *header, int col)
+{
+       unsigned int width;
+       unsigned int i;
 
-               if (show_displacement) {
-                       if (sep)
-                               ret += fprintf(fp, "%cDisplacement", *sep);
-                       else
-                               ret += fprintf(fp, " Displ");
-               }
-       }
+       fprintf(fp, "  ");
+       width = hists__col_len(hists, col);
+
+       if (width == 0)
+               width = strlen(header);
+
+       for (i = 0; i < width; i++)
+               fprintf(fp, ".");
+}
+
+static int hists__fprintf_header(struct hists *hists, FILE *fp,
+                                int max_rows, int max_cols __used)
+{
+       struct sort_entry *se;
+       struct hists_stdio_column *col;
+       int nr_rows = 0;
+
+       /*
+        * TODO Both fprintf_header and fprintf_dots should take
+        * max_cols and output data appropriatelly. Seems there was
+        * no need so far ;)
+        */
+
+       /* First line - headers */
+       fprintf(fp, "# ");
+
+       list_for_each_entry(col, &hists_stdio_column__list, list)
+               fprintf_header(fp, hists, col->header, col->col_idx);
 
        list_for_each_entry(se, &hist_entry__sort_list, list) {
                if (se->elide)
                        continue;
-               if (sep) {
-                       fprintf(fp, "%c%s", *sep, se->se_header);
-                       continue;
-               }
-               width = strlen(se->se_header);
-               if (symbol_conf.col_width_list_str) {
-                       if (col_width) {
-                               hists__set_col_len(hists, se->se_width_idx,
-                                                  atoi(col_width));
-                               col_width = strchr(col_width, ',');
-                               if (col_width)
-                                       ++col_width;
-                       }
-               }
-               if (!hists__new_col_len(hists, se->se_width_idx, width))
-                       width = hists__col_len(hists, se->se_width_idx);
-               fprintf(fp, "  %*s", width, se->se_header);
+
+               fprintf_header(fp, hists, se->se_header, se->se_width_idx);
        }
 
        fprintf(fp, "\n");
+
        if (max_rows && ++nr_rows >= max_rows)
-               goto out;
+               return nr_rows;
 
-       if (sep)
-               goto print_entries;
+       if (symbol_conf.field_sep)
+               return nr_rows;
 
-       fprintf(fp, "# ........");
-       if (symbol_conf.show_cpu_utilization)
-               fprintf(fp, "   .......   .......");
-       if (symbol_conf.show_nr_samples)
-               fprintf(fp, " ..........");
-       if (symbol_conf.show_total_period)
-               fprintf(fp, " ............");
-       if (pair) {
-               fprintf(fp, " ..........");
-               if (show_displacement)
-                       fprintf(fp, " .....");
-       }
-       list_for_each_entry(se, &hist_entry__sort_list, list) {
-               unsigned int i;
+       /* Second line - dots */
+       fprintf(fp, "# ");
+
+       list_for_each_entry(col, &hists_stdio_column__list, list)
+               fprintf_dots(fp, hists, col->header, col->col_idx);
 
+       list_for_each_entry(se, &hist_entry__sort_list, list) {
                if (se->elide)
                        continue;
 
-               fprintf(fp, "  ");
-               width = hists__col_len(hists, se->se_width_idx);
-               if (width == 0)
-                       width = strlen(se->se_header);
-               for (i = 0; i < width; i++)
-                       fprintf(fp, ".");
+               fprintf_dots(fp, hists, se->se_header, se->se_width_idx);
        }
 
        fprintf(fp, "\n");
        if (max_rows && ++nr_rows >= max_rows)
-               goto out;
+               return nr_rows;
 
        fprintf(fp, "#\n");
-       if (max_rows && ++nr_rows >= max_rows)
-               goto out;
+       if (max_rows)
+                ++nr_rows;
 
-print_entries:
-       total_period = hists->stats.total_period;
+       return nr_rows;
+}
+
+static size_t
+hists__fprint_data(struct hists *hists, FILE *fp, int nr_rows,
+                  int max_rows, int max_cols)
+{
+       struct rb_node *nd;
+       size_t ret = 0;
+
+       init_rem_hits();
 
        for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
                struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
@@ -603,18 +656,12 @@ print_entries:
                if (h->filtered)
                        continue;
 
-               if (show_displacement) {
-                       if (h->pair != NULL)
-                               displacement = ((long)h->pair->position -
-                                               (long)h->position);
-                       else
-                               displacement = 0;
-               }
-               ret += hist_entry__fprintf(h, max_cols, hists, pair, 
show_displacement,
-                                          displacement, total_period, fp);
+               ret += hist_entry__fprintf(h, max_cols, hists,
+                                          hists->stats.total_period,
+                                          fp);
 
                if (max_rows && ++nr_rows >= max_rows)
-                       goto out;
+                       break;
 
                if (h->ms.map == NULL && verbose > 1) {
                        __map_groups__fprintf_maps(&h->thread->mg,
@@ -622,12 +669,25 @@ print_entries:
                        fprintf(fp, "%.10s end\n", graph_dotted_line);
                }
        }
-out:
-       free(rem_sq_bracket);
 
+       free(rem_sq_bracket);
        return ret;
 }
 
+size_t hists__fprintf(struct hists *hists, bool show_header,
+                     int max_rows, int max_cols, FILE *fp)
+{
+       int nr_rows = 0;
+
+       if (show_header)
+               nr_rows = hists__fprintf_header(hists, fp, max_rows, max_cols);
+
+       if (max_rows && nr_rows >= max_rows)
+               return 0;
+
+       return hists__fprint_data(hists, fp, nr_rows, max_rows, max_cols);
+}
+
 size_t hists__fprintf_nr_events(struct hists *hists, FILE *fp)
 {
        int i;
diff --git a/tools/perf/ui/stdio/hist.h b/tools/perf/ui/stdio/hist.h
new file mode 100644
index 0000000..8e15d88
--- /dev/null
+++ b/tools/perf/ui/stdio/hist.h
@@ -0,0 +1,19 @@
+#ifndef __PERF_UI_STDIO_HIST_H
+#define __PERF_UI_STDIO_HIST_H
+
+#include <linux/list.h>
+#include "../../util/hist.h"
+
+struct hists_stdio_column {
+       struct list_head list;
+       const char      *header;
+       int              col_idx;
+       int              col_width;
+       int             (*snprintf)(struct hist_entry *self, char *bf,
+                                   size_t size, unsigned int width);
+};
+
+int hists_stdio_column__register_idx(int idx);
+void hists_stdio_column__register_global(void);
+void hists_stdio_column__set_width(struct hists *hists);
+#endif /* __PERF_UI_STDIO_HIST_H */
diff --git a/tools/perf/util/hist.c b/tools/perf/util/hist.c
index 0ed0683..5fb5cd8 100644
--- a/tools/perf/util/hist.c
+++ b/tools/perf/util/hist.c
@@ -5,6 +5,7 @@
 #include "session.h"
 #include "sort.h"
 #include <math.h>
+#include "../ui/stdio/hist.h"
 
 static bool hists__filter_entry_by_dso(struct hists *hists,
                                       struct hist_entry *he);
@@ -49,7 +50,9 @@ void hists__reset_col_len(struct hists *hists)
 {
        enum hist_column col;
 
-       for (col = 0; col < HISTC_NR_COLS; ++col)
+       hists_stdio_column__set_width(hists);
+
+       for (col = HISTC_SYMBOL; col < HISTC_NR_COLS; ++col)
                hists__set_col_len(hists, col, 0);
 }
 
diff --git a/tools/perf/util/hist.h b/tools/perf/util/hist.h
index 2e650ff..36ff4c5 100644
--- a/tools/perf/util/hist.h
+++ b/tools/perf/util/hist.h
@@ -36,6 +36,19 @@ struct events_stats {
 };
 
 enum hist_column {
+       /* stdio (hists_stdio_column__list) */
+       HISTC_OVERHEAD,
+       HISTC_BASELINE,
+       HISTC_CPU_UTILIZATION_SYS,
+       HISTC_CPU_UTILIZATION_US,
+       HISTC_CPU_UTILIZATION_GUEST_SYS,
+       HISTC_CPU_UTILIZATION_GUEST_US,
+       HISTC_NR_SAMPLES,
+       HISTC_TOTAL_PERIOD,
+       HISTC_DELTA,
+       HISTC_DISPLACEMENT,
+
+       /* sorted (hist_entry__sort_list) */
        HISTC_SYMBOL,
        HISTC_DSO,
        HISTC_THREAD,
@@ -49,6 +62,9 @@ enum hist_column {
        HISTC_DSO_TO,
        HISTC_SRCLINE,
        HISTC_NR_COLS, /* Last entry */
+
+       /* Last stdio ui data column */
+       HISTC_STDIO_NR_COLS = HISTC_DISPLACEMENT + 1,
 };
 
 struct thread;
@@ -98,8 +114,7 @@ void hists__output_recalc_col_len(struct hists *hists, int 
max_rows);
 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(struct hists *self, struct hists *pair,
-                     bool show_displacement, bool show_header,
+size_t hists__fprintf(struct hists *self, bool show_header,
                      int max_rows, int max_cols, FILE *fp);
 
 int hist_entry__inc_addr_samples(struct hist_entry *self, int evidx, u64 addr);
-- 
1.7.11.4

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to