|
@@ -26,6 +26,7 @@ struct hist_browser {
|
|
|
struct map_symbol *selection;
|
|
|
int print_seq;
|
|
|
bool show_dso;
|
|
|
+ bool show_headers;
|
|
|
float min_pcnt;
|
|
|
u64 nr_non_filtered_entries;
|
|
|
u64 nr_callchain_rows;
|
|
@@ -56,11 +57,42 @@ static u32 hist_browser__nr_entries(struct hist_browser *hb)
|
|
|
return nr_entries + hb->nr_callchain_rows;
|
|
|
}
|
|
|
|
|
|
-static void hist_browser__refresh_dimensions(struct hist_browser *browser)
|
|
|
+static void hist_browser__update_rows(struct hist_browser *hb)
|
|
|
{
|
|
|
+ struct ui_browser *browser = &hb->b;
|
|
|
+ u16 header_offset = hb->show_headers ? 1 : 0, index_row;
|
|
|
+
|
|
|
+ browser->rows = browser->height - header_offset;
|
|
|
+ /*
|
|
|
+ * Verify if we were at the last line and that line isn't
|
|
|
+ * visibe because we now show the header line(s).
|
|
|
+ */
|
|
|
+ index_row = browser->index - browser->top_idx;
|
|
|
+ if (index_row >= browser->rows)
|
|
|
+ browser->index -= index_row - browser->rows + 1;
|
|
|
+}
|
|
|
+
|
|
|
+static void hist_browser__refresh_dimensions(struct ui_browser *browser)
|
|
|
+{
|
|
|
+ struct hist_browser *hb = container_of(browser, struct hist_browser, b);
|
|
|
+
|
|
|
/* 3 == +/- toggle symbol before actual hist_entry rendering */
|
|
|
- browser->b.width = 3 + (hists__sort_list_width(browser->hists) +
|
|
|
- sizeof("[k]"));
|
|
|
+ browser->width = 3 + (hists__sort_list_width(hb->hists) + sizeof("[k]"));
|
|
|
+ /*
|
|
|
+ * FIXME: Just keeping existing behaviour, but this really should be
|
|
|
+ * before updating browser->width, as it will invalidate the
|
|
|
+ * calculation above. Fix this and the fallout in another
|
|
|
+ * changeset.
|
|
|
+ */
|
|
|
+ ui_browser__refresh_dimensions(browser);
|
|
|
+ hist_browser__update_rows(hb);
|
|
|
+}
|
|
|
+
|
|
|
+static void hist_browser__gotorc(struct hist_browser *browser, int row, int column)
|
|
|
+{
|
|
|
+ u16 header_offset = browser->show_headers ? 1 : 0;
|
|
|
+
|
|
|
+ ui_browser__gotorc(&browser->b, row + header_offset, column);
|
|
|
}
|
|
|
|
|
|
static void hist_browser__reset(struct hist_browser *browser)
|
|
@@ -73,7 +105,7 @@ static void hist_browser__reset(struct hist_browser *browser)
|
|
|
|
|
|
hist_browser__update_nr_entries(browser);
|
|
|
browser->b.nr_entries = hist_browser__nr_entries(browser);
|
|
|
- hist_browser__refresh_dimensions(browser);
|
|
|
+ hist_browser__refresh_dimensions(&browser->b);
|
|
|
ui_browser__reset_index(&browser->b);
|
|
|
}
|
|
|
|
|
@@ -355,7 +387,6 @@ static int hist_browser__run(struct hist_browser *browser,
|
|
|
browser->b.entries = &browser->hists->entries;
|
|
|
browser->b.nr_entries = hist_browser__nr_entries(browser);
|
|
|
|
|
|
- hist_browser__refresh_dimensions(browser);
|
|
|
hists__browser_title(browser->hists, title, sizeof(title));
|
|
|
|
|
|
if (ui_browser__show(&browser->b, title,
|
|
@@ -392,10 +423,10 @@ static int hist_browser__run(struct hist_browser *browser,
|
|
|
struct hist_entry *h = rb_entry(browser->b.top,
|
|
|
struct hist_entry, rb_node);
|
|
|
ui_helpline__pop();
|
|
|
- ui_helpline__fpush("%d: nr_ent=(%d,%d), height=%d, idx=%d, fve: idx=%d, row_off=%d, nrows=%d",
|
|
|
+ ui_helpline__fpush("%d: nr_ent=(%d,%d), rows=%d, idx=%d, fve: idx=%d, row_off=%d, nrows=%d",
|
|
|
seq++, browser->b.nr_entries,
|
|
|
browser->hists->nr_entries,
|
|
|
- browser->b.height,
|
|
|
+ browser->b.rows,
|
|
|
browser->b.index,
|
|
|
browser->b.top_idx,
|
|
|
h->row_offset, h->nr_rows);
|
|
@@ -409,6 +440,10 @@ static int hist_browser__run(struct hist_browser *browser,
|
|
|
/* Expand the whole world. */
|
|
|
hist_browser__set_folding(browser, true);
|
|
|
break;
|
|
|
+ case 'H':
|
|
|
+ browser->show_headers = !browser->show_headers;
|
|
|
+ hist_browser__update_rows(browser);
|
|
|
+ break;
|
|
|
case K_ENTER:
|
|
|
if (hist_browser__toggle_fold(browser))
|
|
|
break;
|
|
@@ -508,13 +543,13 @@ static int hist_browser__show_callchain_node_rb_tree(struct hist_browser *browse
|
|
|
}
|
|
|
|
|
|
ui_browser__set_color(&browser->b, color);
|
|
|
- ui_browser__gotorc(&browser->b, row, 0);
|
|
|
+ hist_browser__gotorc(browser, row, 0);
|
|
|
slsmg_write_nstring(" ", offset + extra_offset);
|
|
|
slsmg_printf("%c ", folded_sign);
|
|
|
slsmg_write_nstring(str, width);
|
|
|
free(alloc_str);
|
|
|
|
|
|
- if (++row == browser->b.height)
|
|
|
+ if (++row == browser->b.rows)
|
|
|
goto out;
|
|
|
do_next:
|
|
|
if (folded_sign == '+')
|
|
@@ -527,7 +562,7 @@ do_next:
|
|
|
new_level, row, row_offset,
|
|
|
is_current_entry);
|
|
|
}
|
|
|
- if (row == browser->b.height)
|
|
|
+ if (row == browser->b.rows)
|
|
|
goto out;
|
|
|
node = next;
|
|
|
}
|
|
@@ -567,13 +602,13 @@ static int hist_browser__show_callchain_node(struct hist_browser *browser,
|
|
|
|
|
|
s = callchain_list__sym_name(chain, bf, sizeof(bf),
|
|
|
browser->show_dso);
|
|
|
- ui_browser__gotorc(&browser->b, row, 0);
|
|
|
+ hist_browser__gotorc(browser, row, 0);
|
|
|
ui_browser__set_color(&browser->b, color);
|
|
|
slsmg_write_nstring(" ", offset);
|
|
|
slsmg_printf("%c ", folded_sign);
|
|
|
slsmg_write_nstring(s, width - 2);
|
|
|
|
|
|
- if (++row == browser->b.height)
|
|
|
+ if (++row == browser->b.rows)
|
|
|
goto out;
|
|
|
}
|
|
|
|
|
@@ -602,7 +637,7 @@ static int hist_browser__show_callchain(struct hist_browser *browser,
|
|
|
row += hist_browser__show_callchain_node(browser, node, level,
|
|
|
row, row_offset,
|
|
|
is_current_entry);
|
|
|
- if (row == browser->b.height)
|
|
|
+ if (row == browser->b.rows)
|
|
|
break;
|
|
|
}
|
|
|
|
|
@@ -732,7 +767,7 @@ static int hist_browser__show_entry(struct hist_browser *browser,
|
|
|
.ptr = &arg,
|
|
|
};
|
|
|
|
|
|
- ui_browser__gotorc(&browser->b, row, 0);
|
|
|
+ hist_browser__gotorc(browser, row, 0);
|
|
|
|
|
|
perf_hpp__for_each_format(fmt) {
|
|
|
if (perf_hpp__should_skip(fmt))
|
|
@@ -776,7 +811,7 @@ static int hist_browser__show_entry(struct hist_browser *browser,
|
|
|
} else
|
|
|
--row_offset;
|
|
|
|
|
|
- if (folded_sign == '-' && row != browser->b.height) {
|
|
|
+ if (folded_sign == '-' && row != browser->b.rows) {
|
|
|
printed += hist_browser__show_callchain(browser, &entry->sorted_chain,
|
|
|
1, row, &row_offset,
|
|
|
¤t_entry);
|
|
@@ -787,6 +822,56 @@ static int hist_browser__show_entry(struct hist_browser *browser,
|
|
|
return printed;
|
|
|
}
|
|
|
|
|
|
+static int advance_hpp_check(struct perf_hpp *hpp, int inc)
|
|
|
+{
|
|
|
+ advance_hpp(hpp, inc);
|
|
|
+ return hpp->size <= 0;
|
|
|
+}
|
|
|
+
|
|
|
+static int hists__scnprintf_headers(char *buf, size_t size, struct hists *hists)
|
|
|
+{
|
|
|
+ struct perf_hpp dummy_hpp = {
|
|
|
+ .buf = buf,
|
|
|
+ .size = size,
|
|
|
+ };
|
|
|
+ struct perf_hpp_fmt *fmt;
|
|
|
+ size_t ret = 0;
|
|
|
+
|
|
|
+ if (symbol_conf.use_callchain) {
|
|
|
+ ret = scnprintf(buf, size, " ");
|
|
|
+ if (advance_hpp_check(&dummy_hpp, ret))
|
|
|
+ return ret;
|
|
|
+ }
|
|
|
+
|
|
|
+ perf_hpp__for_each_format(fmt) {
|
|
|
+ if (perf_hpp__should_skip(fmt))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ /* We need to add the length of the columns header. */
|
|
|
+ perf_hpp__reset_width(fmt, hists);
|
|
|
+
|
|
|
+ ret = fmt->header(fmt, &dummy_hpp, hists_to_evsel(hists));
|
|
|
+ if (advance_hpp_check(&dummy_hpp, ret))
|
|
|
+ break;
|
|
|
+
|
|
|
+ ret = scnprintf(dummy_hpp.buf, dummy_hpp.size, " ");
|
|
|
+ if (advance_hpp_check(&dummy_hpp, ret))
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static void hist_browser__show_headers(struct hist_browser *browser)
|
|
|
+{
|
|
|
+ char headers[1024];
|
|
|
+
|
|
|
+ hists__scnprintf_headers(headers, sizeof(headers), browser->hists);
|
|
|
+ ui_browser__gotorc(&browser->b, 0, 0);
|
|
|
+ ui_browser__set_color(&browser->b, HE_COLORSET_ROOT);
|
|
|
+ slsmg_write_nstring(headers, browser->b.width + 1);
|
|
|
+}
|
|
|
+
|
|
|
static void ui_browser__hists_init_top(struct ui_browser *browser)
|
|
|
{
|
|
|
if (browser->top == NULL) {
|
|
@@ -800,9 +885,15 @@ static void ui_browser__hists_init_top(struct ui_browser *browser)
|
|
|
static unsigned int hist_browser__refresh(struct ui_browser *browser)
|
|
|
{
|
|
|
unsigned row = 0;
|
|
|
+ u16 header_offset = 0;
|
|
|
struct rb_node *nd;
|
|
|
struct hist_browser *hb = container_of(browser, struct hist_browser, b);
|
|
|
|
|
|
+ if (hb->show_headers) {
|
|
|
+ hist_browser__show_headers(hb);
|
|
|
+ header_offset = 1;
|
|
|
+ }
|
|
|
+
|
|
|
ui_browser__hists_init_top(browser);
|
|
|
|
|
|
for (nd = browser->top; nd; nd = rb_next(nd)) {
|
|
@@ -817,11 +908,11 @@ static unsigned int hist_browser__refresh(struct ui_browser *browser)
|
|
|
continue;
|
|
|
|
|
|
row += hist_browser__show_entry(hb, h, row);
|
|
|
- if (row == browser->height)
|
|
|
+ if (row == browser->rows)
|
|
|
break;
|
|
|
}
|
|
|
|
|
|
- return row;
|
|
|
+ return row + header_offset;
|
|
|
}
|
|
|
|
|
|
static struct rb_node *hists__filter_entries(struct rb_node *nd,
|
|
@@ -1190,8 +1281,10 @@ static struct hist_browser *hist_browser__new(struct hists *hists)
|
|
|
if (browser) {
|
|
|
browser->hists = hists;
|
|
|
browser->b.refresh = hist_browser__refresh;
|
|
|
+ browser->b.refresh_dimensions = hist_browser__refresh_dimensions;
|
|
|
browser->b.seek = ui_browser__hists_seek;
|
|
|
browser->b.use_navkeypressed = true;
|
|
|
+ browser->show_headers = symbol_conf.show_hist_headers;
|
|
|
}
|
|
|
|
|
|
return browser;
|
|
@@ -1421,6 +1514,7 @@ static int perf_evsel__hists_browse(struct perf_evsel *evsel, int nr_events,
|
|
|
"d Zoom into current DSO\n" \
|
|
|
"E Expand all callchains\n" \
|
|
|
"F Toggle percentage of filtered entries\n" \
|
|
|
+ "H Display column headers\n" \
|
|
|
|
|
|
/* help messages are sorted by lexical order of the hotkey */
|
|
|
const char report_help[] = HIST_BROWSER_HELP_COMMON
|