Merge branch 'perf/urgent' into perf/core
[firefly-linux-kernel-4.4.55.git] / tools / perf / builtin-report.c
index f815de25d0fc5b76eef680411930589654e793e8..e93c69a8e72001d97f73a86895ec1a4dab6ac7fe 100644 (file)
@@ -14,7 +14,6 @@
 #include "util/cache.h"
 #include <linux/rbtree.h>
 #include "util/symbol.h"
-#include "util/string.h"
 #include "util/callchain.h"
 #include "util/strlist.h"
 #include "util/values.h"
@@ -81,15 +80,20 @@ static int perf_session__add_hist_entry(struct perf_session *self,
                                        struct addr_location *al,
                                        struct sample_data *data)
 {
-       struct symbol **syms = NULL, *parent = NULL;
+       struct map_symbol *syms = NULL;
+       struct symbol *parent = NULL;
        bool hit;
+       int err;
        struct hist_entry *he;
        struct event_stat_id *stats;
        struct perf_event_attr *attr;
 
-       if ((sort__has_parent || symbol_conf.use_callchain) && data->callchain)
+       if ((sort__has_parent || symbol_conf.use_callchain) && data->callchain) {
                syms = perf_session__resolve_callchain(self, al->thread,
                                                       data->callchain, &parent);
+               if (syms == NULL)
+                       return -ENOMEM;
+       }
 
        attr = perf_header__find_attr(data->id, &self->header);
        if (attr)
@@ -108,9 +112,12 @@ static int perf_session__add_hist_entry(struct perf_session *self,
 
        if (symbol_conf.use_callchain) {
                if (!hit)
-                       callchain_init(&he->callchain);
-               append_chain(&he->callchain, data->callchain, syms);
+                       callchain_init(he->callchain);
+               err = append_chain(he->callchain, data->callchain, syms);
                free(syms);
+
+               if (err)
+                       return err;
        }
 
        return 0;
@@ -267,6 +274,7 @@ static int __cmd_report(void)
        int ret = -EINVAL;
        struct perf_session *session;
        struct rb_node *next;
+       const char *help = "For a higher level overview, try: perf report --sort comm,dso";
 
        session = perf_session__new(input_name, O_RDONLY, force);
        if (session == NULL)
@@ -297,34 +305,44 @@ static int __cmd_report(void)
        next = rb_first(&session->stats_by_id);
        while (next) {
                struct event_stat_id *stats;
+               u64 nr_hists;
 
                stats = rb_entry(next, struct event_stat_id, rb_node);
                perf_session__collapse_resort(&stats->hists);
-               perf_session__output_resort(&stats->hists, stats->stats.total);
-               if (rb_first(&session->stats_by_id) ==
-                   rb_last(&session->stats_by_id))
-                       fprintf(stdout, "# Samples: %Ld\n#\n",
-                               stats->stats.total);
-               else
-                       fprintf(stdout, "# Samples: %Ld %s\n#\n",
-                               stats->stats.total,
-                               __event_name(stats->type, stats->config));
-
-               perf_session__fprintf_hists(&stats->hists, NULL, false, stdout,
+               nr_hists = perf_session__output_resort(&stats->hists,
+                                                      stats->stats.total);
+               if (use_browser)
+                       perf_session__browse_hists(&stats->hists, nr_hists,
+                                                  stats->stats.total, help,
+                                                  input_name);
+               else {
+                       if (rb_first(&session->stats_by_id) ==
+                           rb_last(&session->stats_by_id))
+                               fprintf(stdout, "# Samples: %Ld\n#\n",
+                                       stats->stats.total);
+                       else
+                               fprintf(stdout, "# Samples: %Ld %s\n#\n",
+                                       stats->stats.total,
+                                       __event_name(stats->type, stats->config));
+
+                       perf_session__fprintf_hists(&stats->hists, NULL, false, stdout,
                                            stats->stats.total);
-               fprintf(stdout, "\n\n");
+                       fprintf(stdout, "\n\n");
+               }
+
                next = rb_next(&stats->rb_node);
        }
 
-       if (sort_order == default_sort_order &&
-           parent_pattern == default_parent_pattern)
-               fprintf(stdout, "#\n# (For a higher level overview, try: perf report --sort comm,dso)\n#\n");
+       if (!use_browser && sort_order == default_sort_order &&
+           parent_pattern == default_parent_pattern) {
+               fprintf(stdout, "#\n# (%s)\n#\n", help);
 
-       if (show_threads) {
-               bool raw_printing_style = !strcmp(pretty_printing_style, "raw");
-               perf_read_values_display(stdout, &show_threads_values,
-                                        raw_printing_style);
-               perf_read_values_destroy(&show_threads_values);
+               if (show_threads) {
+                       bool style = !strcmp(pretty_printing_style, "raw");
+                       perf_read_values_display(stdout, &show_threads_values,
+                                                style);
+                       perf_read_values_destroy(&show_threads_values);
+               }
        }
 out_delete:
        perf_session__delete(session);
@@ -447,7 +465,7 @@ int cmd_report(int argc, const char **argv, const char *prefix __used)
 {
        argc = parse_options(argc, argv, options, report_usage, 0);
 
-       setup_pager();
+       setup_browser();
 
        if (symbol__init() < 0)
                return -1;
@@ -455,7 +473,8 @@ int cmd_report(int argc, const char **argv, const char *prefix __used)
        setup_sorting(report_usage, options);
 
        if (parent_pattern != default_parent_pattern) {
-               sort_dimension__add("parent");
+               if (sort_dimension__add("parent") < 0)
+                       return -1;
                sort_parent.elide = 1;
        } else
                symbol_conf.exclude_other = false;