#include "util/target.h"
 #include "util/session.h"
 #include "util/tool.h"
+#include "util/stat.h"
 #include "util/symbol.h"
 #include "util/record.h"
 #include "util/cpumap.h"
                pr_warning("WARNING: --timestamp-filename option is not available in pipe mode.\n");
        }
 
-       evlist__uniquify_name(rec->evlist);
+       /*
+        * Use global stat_config that is zero meaning aggr_mode is AGGR_NONE
+        * and hybrid_merge is false.
+        */
+       evlist__uniquify_evsel_names(rec->evlist, &stat_config);
 
        evlist__config(rec->evlist, opts, &callchain_param);
 
 
 #include "util/mmap.h"
 #include "util/session.h"
 #include "util/thread.h"
+#include "util/stat.h"
 #include "util/symbol.h"
 #include "util/synthetic-events.h"
 #include "util/top.h"
                }
        }
 
-       evlist__uniquify_name(top->evlist);
+       /*
+        * Use global stat_config that is zero meaning aggr_mode is AGGR_NONE
+        * and hybrid_merge is false.
+        */
+       evlist__uniquify_evsel_names(top->evlist, &stat_config);
        ret = perf_top__start_counters(top);
        if (ret)
                return ret;
 
        perf_cpu_map__put(user_requested_cpus);
 }
 
-void evlist__uniquify_name(struct evlist *evlist)
+/* Should uniquify be disabled for the evlist? */
+static bool evlist__disable_uniquify(const struct evlist *evlist)
 {
-       char *new_name, empty_attributes[2] = ":", *attributes;
-       struct evsel *pos;
+       struct evsel *counter;
+       struct perf_pmu *last_pmu = NULL;
+       bool first = true;
 
-       if (perf_pmus__num_core_pmus() == 1)
-               return;
+       evlist__for_each_entry(evlist, counter) {
+               /* If PMUs vary then uniquify can be useful. */
+               if (!first && counter->pmu != last_pmu)
+                       return false;
+               first = false;
+               if (counter->pmu) {
+                       /* Allow uniquify for uncore PMUs. */
+                       if (!counter->pmu->is_core)
+                               return false;
+                       /* Keep hybrid event names uniquified for clarity. */
+                       if (perf_pmus__num_core_pmus() > 1)
+                               return false;
+               }
+               last_pmu = counter->pmu;
+       }
+       return true;
+}
 
-       evlist__for_each_entry(evlist, pos) {
-               if (!evsel__is_hybrid(pos))
-                       continue;
+static bool evlist__set_needs_uniquify(struct evlist *evlist, const struct perf_stat_config *config)
+{
+       struct evsel *counter;
+       bool needs_uniquify = false;
 
-               if (strchr(pos->name, '/'))
-                       continue;
+       if (evlist__disable_uniquify(evlist)) {
+               evlist__for_each_entry(evlist, counter)
+                       counter->uniquified_name = true;
+               return false;
+       }
+
+       evlist__for_each_entry(evlist, counter) {
+               if (evsel__set_needs_uniquify(counter, config))
+                       needs_uniquify = true;
+       }
+       return needs_uniquify;
+}
 
-               attributes = strchr(pos->name, ':');
-               if (attributes)
-                       *attributes = '\0';
-               else
-                       attributes = empty_attributes;
+void evlist__uniquify_evsel_names(struct evlist *evlist, const struct perf_stat_config *config)
+{
+       if (evlist__set_needs_uniquify(evlist, config)) {
+               struct evsel *pos;
 
-               if (asprintf(&new_name, "%s/%s/%s", pos->pmu ? pos->pmu->name : "",
-                            pos->name, attributes + 1)) {
-                       free(pos->name);
-                       pos->name = new_name;
-               } else {
-                       *attributes = ':';
-               }
+               evlist__for_each_entry(evlist, pos)
+                       evsel__uniquify_counter(pos);
        }
 }
 
 
 struct pollfd;
 struct thread_map;
 struct perf_cpu_map;
+struct perf_stat_config;
 struct record_opts;
 struct strbuf;
 struct target;
 void evlist__format_evsels(struct evlist *evlist, struct strbuf *sb, size_t max_length);
 void evlist__check_mem_load_aux(struct evlist *evlist);
 void evlist__warn_user_requested_cpus(struct evlist *evlist, const char *cpu_list);
-void evlist__uniquify_name(struct evlist *evlist);
+void evlist__uniquify_evsel_names(struct evlist *evlist, const struct perf_stat_config *config);
 bool evlist__has_bpf_output(struct evlist *evlist);
 bool evlist__needs_bpf_sb_event(struct evlist *evlist);
 
 
                leader->core.nr_members--;
        }
 }
+
+bool evsel__set_needs_uniquify(struct evsel *counter, const struct perf_stat_config *config)
+{
+       struct evsel *evsel;
+
+       if (counter->needs_uniquify) {
+               /* Already set. */
+               return true;
+       }
+
+       if (counter->merged_stat) {
+               /* Counter won't be shown. */
+               return false;
+       }
+
+       if (counter->use_config_name || counter->is_libpfm_event) {
+               /* Original name will be used. */
+               return false;
+       }
+
+       if (!config->hybrid_merge && evsel__is_hybrid(counter)) {
+               /* Unique hybrid counters necessary. */
+               counter->needs_uniquify = true;
+               return true;
+       }
+
+       if  (counter->core.attr.type < PERF_TYPE_MAX && counter->core.attr.type != PERF_TYPE_RAW) {
+               /* Legacy event, don't uniquify. */
+               return false;
+       }
+
+       if (counter->pmu && counter->pmu->is_core &&
+           counter->alternate_hw_config != PERF_COUNT_HW_MAX) {
+               /* A sysfs or json event replacing a legacy event, don't uniquify. */
+               return false;
+       }
+
+       if (config->aggr_mode == AGGR_NONE) {
+               /* Always unique with no aggregation. */
+               counter->needs_uniquify = true;
+               return true;
+       }
+
+       /*
+        * Do other non-merged events in the evlist have the same name? If so
+        * uniquify is necessary.
+        */
+       evlist__for_each_entry(counter->evlist, evsel) {
+               if (evsel == counter || evsel->merged_stat || evsel->pmu == counter->pmu)
+                       continue;
+
+               if (evsel__name_is(counter, evsel__name(evsel))) {
+                       counter->needs_uniquify = true;
+                       return true;
+               }
+       }
+       return false;
+}
+
+void evsel__uniquify_counter(struct evsel *counter)
+{
+       const char *name, *pmu_name;
+       char *new_name, *config;
+       int ret;
+
+       /* No uniquification necessary. */
+       if (!counter->needs_uniquify)
+               return;
+
+       /* The evsel was already uniquified. */
+       if (counter->uniquified_name)
+               return;
+
+       /* Avoid checking to uniquify twice. */
+       counter->uniquified_name = true;
+
+       name = evsel__name(counter);
+       pmu_name = counter->pmu->name;
+       /* Already prefixed by the PMU name. */
+       if (!strncmp(name, pmu_name, strlen(pmu_name)))
+               return;
+
+       config = strchr(name, '/');
+       if (config) {
+               int len = config - name;
+
+               if (config[1] == '/') {
+                       /* case: event// */
+                       ret = asprintf(&new_name, "%s/%.*s/%s", pmu_name, len, name, config + 2);
+               } else {
+                       /* case: event/.../ */
+                       ret = asprintf(&new_name, "%s/%.*s,%s", pmu_name, len, name, config + 1);
+               }
+       } else {
+               config = strchr(name, ':');
+               if (config) {
+                       /* case: event:.. */
+                       int len = config - name;
+
+                       ret = asprintf(&new_name, "%s/%.*s/%s", pmu_name, len, name, config + 1);
+               } else {
+                       /* case: event */
+                       ret = asprintf(&new_name, "%s/%s/", pmu_name, name);
+               }
+       }
+       if (ret > 0) {
+               free(counter->name);
+               counter->name = new_name;
+       } else {
+               /* ENOMEM from asprintf. */
+               counter->uniquified_name = false;
+       }
+}
 
 struct bpf_object;
 struct cgroup;
 struct perf_counts;
+struct perf_stat_config;
 struct perf_stat_evsel;
 union perf_event;
 struct bpf_counter_ops;
 
 bool arch_evsel__must_be_in_group(const struct evsel *evsel);
 
+bool evsel__set_needs_uniquify(struct evsel *counter, const struct perf_stat_config *config);
+void evsel__uniquify_counter(struct evsel *counter);
+
 /*
  * Macro to swap the bit-field postition and size.
  * Used when,
 
                if (verbose > 0) {
                        struct strbuf sb = STRBUF_INIT;
 
-                       evlist__uniquify_name(evlist);
+                       evlist__uniquify_evsel_names(evlist, &stat_config);
                        evlist__format_evsels(evlist, &sb, 2048);
                        pr_debug("evlist after sorting/fixing: '%s'\n", sb.buf);
                        strbuf_release(&sb);
 
        }
 }
 
-static void evsel__uniquify_counter(struct evsel *counter)
-{
-       const char *name, *pmu_name;
-       char *new_name, *config;
-       int ret;
-
-       /* No uniquification necessary. */
-       if (!counter->needs_uniquify)
-               return;
-
-       /* The evsel was already uniquified. */
-       if (counter->uniquified_name)
-               return;
-
-       /* Avoid checking to uniquify twice. */
-       counter->uniquified_name = true;
-
-       name = evsel__name(counter);
-       pmu_name = counter->pmu->name;
-       /* Already prefixed by the PMU name. */
-       if (!strncmp(name, pmu_name, strlen(pmu_name)))
-               return;
-
-       config = strchr(name, '/');
-       if (config) {
-               int len = config - name;
-
-               if (config[1] == '/') {
-                       /* case: event// */
-                       ret = asprintf(&new_name, "%s/%.*s/%s", pmu_name, len, name, config + 2);
-               } else {
-                       /* case: event/.../ */
-                       ret = asprintf(&new_name, "%s/%.*s,%s", pmu_name, len, name, config + 1);
-               }
-       } else {
-               config = strchr(name, ':');
-               if (config) {
-                       /* case: event:.. */
-                       int len = config - name;
-
-                       ret = asprintf(&new_name, "%s/%.*s/%s", pmu_name, len, name, config + 1);
-               } else {
-                       /* case: event */
-                       ret = asprintf(&new_name, "%s/%s/", pmu_name, name);
-               }
-       }
-       if (ret > 0) {
-               free(counter->name);
-               counter->name = new_name;
-       } else {
-               /* ENOMEM from asprintf. */
-               counter->uniquified_name = false;
-       }
-}
-
 /**
  * should_skip_zero_count() - Check if the event should print 0 values.
  * @config: The perf stat configuration (including aggregation mode).
        if (counter->merged_stat)
                return;
 
-       evsel__uniquify_counter(counter);
-
        val = aggr->counts.val;
        ena = aggr->counts.ena;
        run = aggr->counts.run;
                print_metric_end(config, os);
 }
 
-/* Should uniquify be disabled for the evlist? */
-static bool evlist__disable_uniquify(const struct evlist *evlist)
-{
-       struct evsel *counter;
-       struct perf_pmu *last_pmu = NULL;
-       bool first = true;
-
-       evlist__for_each_entry(evlist, counter) {
-               /* If PMUs vary then uniquify can be useful. */
-               if (!first && counter->pmu != last_pmu)
-                       return false;
-               first = false;
-               if (counter->pmu) {
-                       /* Allow uniquify for uncore PMUs. */
-                       if (!counter->pmu->is_core)
-                               return false;
-                       /* Keep hybrid event names uniquified for clarity. */
-                       if (perf_pmus__num_core_pmus() > 1)
-                               return false;
-               }
-       }
-       return true;
-}
-
-static void evsel__set_needs_uniquify(struct evsel *counter, const struct perf_stat_config *config)
-{
-       struct evsel *evsel;
-
-       if (counter->merged_stat) {
-               /* Counter won't be shown. */
-               return;
-       }
-
-       if (counter->use_config_name || counter->is_libpfm_event) {
-               /* Original name will be used. */
-               return;
-       }
-
-       if (!config->hybrid_merge && evsel__is_hybrid(counter)) {
-               /* Unique hybrid counters necessary. */
-               counter->needs_uniquify = true;
-               return;
-       }
-
-       if  (counter->core.attr.type < PERF_TYPE_MAX && counter->core.attr.type != PERF_TYPE_RAW) {
-               /* Legacy event, don't uniquify. */
-               return;
-       }
-
-       if (counter->pmu && counter->pmu->is_core &&
-           counter->alternate_hw_config != PERF_COUNT_HW_MAX) {
-               /* A sysfs or json event replacing a legacy event, don't uniquify. */
-               return;
-       }
-
-       if (config->aggr_mode == AGGR_NONE) {
-               /* Always unique with no aggregation. */
-               counter->needs_uniquify = true;
-               return;
-       }
-
-       /*
-        * Do other non-merged events in the evlist have the same name? If so
-        * uniquify is necessary.
-        */
-       evlist__for_each_entry(counter->evlist, evsel) {
-               if (evsel == counter || evsel->merged_stat)
-                       continue;
-
-               if (evsel__name_is(counter, evsel__name(evsel))) {
-                       counter->needs_uniquify = true;
-                       return;
-               }
-       }
-}
-
-static void evlist__set_needs_uniquify(struct evlist *evlist, const struct perf_stat_config *config)
-{
-       struct evsel *counter;
-
-       if (evlist__disable_uniquify(evlist)) {
-               evlist__for_each_entry(evlist, counter)
-                       counter->uniquified_name = true;
-               return;
-       }
-
-       evlist__for_each_entry(evlist, counter)
-               evsel__set_needs_uniquify(counter, config);
-}
-
 void evlist__print_counters(struct evlist *evlist, struct perf_stat_config *config,
                            struct target *_target, struct timespec *ts,
                            int argc, const char **argv)
                .first = true,
        };
 
-       evlist__set_needs_uniquify(evlist, config);
+       evlist__uniquify_evsel_names(evlist, config);
 
        if (config->iostat_run)
                evlist->selected = evlist__first(evlist);