[PATCH 12/14] perf c2c: add function view browser UI
From: Jiebin Sun
Date: Fri Jun 26 2026 - 03:04:49 EST
Replace the stub perf_c2c__browse_function_view() with the full
interactive browser implementation:
- c2c_function_browser__new(): create browser with hierarchy support,
disabling callchains for clean function-level display
- c2c_function_browser__browse_cacheline_detail(): drill into
cacheline detail view via 'd' key
- c2c_function_browser__title(): display entry count in title bar
- c2c_function_browser__delete(): cleanup on exit
The browser saves/restores symbol_conf.use_callchain to avoid
corrupting the callchain display in the cacheline view after
returning from the function view.
The browser entry point perf_c2c__browse_function_view() is wired up by
the next patch, which adds the TAB key in the cacheline view to switch
into the function view.
Signed-off-by: Jiebin Sun <jiebin.sun@xxxxxxxxx>
Cc: Adrian Hunter <adrian.hunter@xxxxxxxxx>
Cc: Alexander Shishkin <alexander.shishkin@xxxxxxxxxxxxxxx>
Cc: Arnaldo Carvalho de Melo <acme@xxxxxxxxxx>
Cc: Dapeng Mi <dapeng1.mi@xxxxxxxxxxxxxxx>
Cc: Ian Rogers <irogers@xxxxxxxxxx>
Cc: Ingo Molnar <mingo@xxxxxxxxxx>
Cc: James Clark <james.clark@xxxxxxxxxx>
Cc: Jiri Olsa <jolsa@xxxxxxxxxx>
Cc: Mark Rutland <mark.rutland@xxxxxxx>
Cc: Namhyung Kim <namhyung@xxxxxxxxxx>
Cc: Peter Zijlstra <peterz@xxxxxxxxxxxxx>
Cc: Thomas Falcon <thomas.falcon@xxxxxxxxx>
Reviewed-by: Tianyou Li <tianyou.li@xxxxxxxxx>
Reviewed-by: Wangyang Guo <wangyang.guo@xxxxxxxxx>
---
tools/perf/ui/browsers/c2c-function.c | 159 +++++++++++++++++++++++++-
1 file changed, 153 insertions(+), 6 deletions(-)
diff --git a/tools/perf/ui/browsers/c2c-function.c b/tools/perf/ui/browsers/c2c-function.c
index 452d64007604..11366fd5e2c1 100644
--- a/tools/perf/ui/browsers/c2c-function.c
+++ b/tools/perf/ui/browsers/c2c-function.c
@@ -2,7 +2,7 @@
/*
* C2C Function Browser - function-level cacheline sharing analysis
*
- * Planned UI: 3-level hierarchy showing which functions share cachelines (not implemented yet):
+ * Displays a 3-level hierarchy showing which functions share cachelines:
* Level 1: Primary functions sorted by Cycles % (estimated load cycles)
* Level 2: Other functions sharing cachelines with the level-1 function
* Level 3: Specific shared cachelines between each pair of functions
@@ -44,7 +44,7 @@ struct perf_c2c_ext {
u64 total_cycles;
};
-static struct perf_c2c_ext c2c_ext __maybe_unused;
+static struct perf_c2c_ext c2c_ext;
struct c2c_function_browser {
struct hist_browser hb;
@@ -1319,7 +1319,7 @@ static void c2c_function__finalize(void)
* L2: sharing functions referenced from each L1 function
* L3: cachelines that pair L1 with L2
*/
-static __maybe_unused int build_function_view_hierarchy(void)
+static int build_function_view_hierarchy(void)
{
static const char output_fields[] =
"cycles_percent,total_stores,iaddr_symbol,symbol_view,cacheline_symbol";
@@ -1393,8 +1393,155 @@ static __maybe_unused int build_function_view_hierarchy(void)
return ret;
}
-int perf_c2c__browse_function_view(struct hists *hists __maybe_unused)
+static int c2c_function_browser__title(struct hist_browser *browser,
+ char *bf, size_t size)
{
- ui__warning("C2C function view is not implemented yet.\n");
- return -ENOSYS;
+ scnprintf(bf, size,
+ "Shared Data Functions Table (%" PRIu64 " entries, sorted on Cycles %%)",
+ browser->nr_non_filtered_entries);
+ return 0;
+}
+
+static struct c2c_function_browser *c2c_function_browser__new(struct hists *hists)
+{
+ struct c2c_function_browser *browser;
+
+ if (!hists)
+ return NULL;
+
+ browser = zalloc(sizeof(*browser));
+ if (!browser)
+ return NULL;
+
+ hist_browser__init(&browser->hb, hists);
+
+ browser->hb.title = c2c_function_browser__title;
+ browser->hb.c2c_filter = true;
+ browser->hb.show_headers = true;
+ /* Keep title line count consistent with forcing headers on. */
+ browser->hb.b.extra_title_lines = hists->hpp_list->nr_header_lines;
+ browser->hb.min_pcnt = 0.0;
+
+ /*
+ * Note: symbol_conf.report_hierarchy is deliberately left unset.
+ * The generic browser still descends into hroot_out children via
+ * rb_hierarchy_next()/can_goto_child(), which key off he->unfolded,
+ * so 'e'/'+' expands L1 -> L2 -> L3 correctly. Setting the flag would
+ * additionally make hist_entry__delete() recurse hroot_out and free
+ * each child, but our children borrow thread/ms (see
+ * c2c_child_entry__alloc()), so that would underflow their refcounts.
+ * Teardown is handled by c2c_he__free_hierarchy() instead.
+ */
+ return browser;
+}
+
+/*
+ * c2c_function_browser__delete - Free function browser
+ */
+static void c2c_function_browser__delete(struct c2c_function_browser *browser)
+{
+ free(browser);
+}
+
+static int c2c_function_browser__browse_cacheline_detail(struct hist_entry *he_selection,
+ struct hists *hists)
+{
+ struct rb_node *nd;
+ u64 cl_addr;
+
+ if (!he_selection || !he_selection->parent_he ||
+ !he_selection->parent_he->parent_he || !he_selection->mem_info)
+ return -1;
+
+ cl_addr = cl_address(mem_info__daddr(he_selection->mem_info)->addr, chk_double_cl);
+
+ for (nd = rb_first_cached(&hists->entries); nd; nd = rb_next(nd)) {
+ struct hist_entry *he_cl = rb_entry(nd, struct hist_entry, rb_node);
+ u64 this_cl;
+
+ if (!he_cl->mem_info)
+ continue;
+
+ this_cl = cl_address(mem_info__daddr(he_cl->mem_info)->addr, chk_double_cl);
+ if (this_cl == cl_addr)
+ return perf_c2c__browse_cacheline(he_cl);
+ }
+
+ return -1;
+}
+
+/*
+ * perf_c2c__browse_function_view - Browse function view with TAB key support
+ * @hists: Main cacheline histograms
+ *
+ * Returns: 0 on success, negative error code on failure
+ */
+int perf_c2c__browse_function_view(struct hists *hists)
+{
+ struct c2c_function_browser *sym_browser;
+ bool saved_use_callchain = symbol_conf.use_callchain;
+ int key, ret;
+ static const char help[] =
+ " d Display cacheline details for the selected entry\n"
+ " e/+ Expand/collapse the selected entry\n"
+ " TAB/ESC/q Return to the cacheline view\n";
+
+ if (!hists)
+ return -EINVAL;
+
+ /* Disable callchain before building so no callchain structs are allocated. */
+ symbol_conf.use_callchain = false;
+
+ ret = build_function_view_hierarchy();
+ if (ret) {
+ ui__error("Failed to build function view hierarchy (ret=%d)\n", ret);
+ goto out;
+ }
+
+ sym_browser = c2c_function_browser__new(&c2c_ext.function_hists.hists);
+ if (!sym_browser) {
+ ret = -ENOMEM;
+ goto out;
+ }
+
+ /* Reset abort key so we can receive Ctrl-C as a key. */
+ SLang_reset_tty();
+ SLang_init_tty(0, 0, 0);
+ SLtty_set_suspend_state(true);
+
+ sym_browser->hb.nr_non_filtered_entries =
+ c2c_ext.function_hists.hists.nr_non_filtered_entries;
+
+ while (1) {
+ key = hist_browser__run(&sym_browser->hb, "? - help", true, 0);
+
+ switch (key) {
+ case 'q':
+ case K_TAB:
+ case K_ESC:
+ goto browser_done;
+ case 'd':
+ /* Cacheline detail honors the user's callchain setting. */
+ symbol_conf.use_callchain = saved_use_callchain;
+ c2c_function_browser__browse_cacheline_detail(sym_browser->hb.he_selection,
+ hists);
+ /* Preserve any toggle made in the detail view, then
+ * re-disable callchain for the function view.
+ */
+ saved_use_callchain = symbol_conf.use_callchain;
+ symbol_conf.use_callchain = false;
+ break;
+ case '?':
+ ui_browser__help_window(&sym_browser->hb.b, help);
+ break;
+ default:
+ break;
+ }
+ }
+
+browser_done:
+ c2c_function_browser__delete(sym_browser);
+out:
+ symbol_conf.use_callchain = saved_use_callchain;
+ return ret;
}
--
2.52.0