const char *pmu_name = &big_c_string[table_pmu->pmu_name.offset];
                 int ret;
 
-                if (!perf_pmu__name_wildcard_match(pmu, pmu_name))
+                if (pmu && !perf_pmu__name_wildcard_match(pmu, pmu_name))
                         continue;
 
                 ret = pmu_events_table__find_event_pmu(table, table_pmu, name, fn, data);
         return 0;
 }
 
+static int pmu_metrics_table__find_metric_pmu(const struct pmu_metrics_table *table,
+                                            const struct pmu_table_entry *pmu,
+                                            const char *metric,
+                                            pmu_metric_iter_fn fn,
+                                            void *data)
+{
+        struct pmu_metric pm = {
+                .pmu = &big_c_string[pmu->pmu_name.offset],
+        };
+        int low = 0, high = pmu->num_entries - 1;
+
+        while (low <= high) {
+                int cmp, mid = (low + high) / 2;
+
+                decompress_metric(pmu->entries[mid].offset, &pm);
+
+                if (!pm.metric_name && !metric)
+                        goto do_call;
+
+                if (!pm.metric_name && metric) {
+                        low = mid + 1;
+                        continue;
+                }
+                if (pm.metric_name && !metric) {
+                        high = mid - 1;
+                        continue;
+                }
+
+                cmp = strcmp(pm.metric_name, metric);
+                if (cmp < 0) {
+                        low = mid + 1;
+                        continue;
+                }
+                if (cmp > 0) {
+                        high = mid - 1;
+                        continue;
+                }
+  do_call:
+                return fn ? fn(&pm, table, data) : 0;
+        }
+        return PMU_METRICS__NOT_FOUND;
+}
+
 int pmu_metrics_table__for_each_metric(const struct pmu_metrics_table *table,
                                      pmu_metric_iter_fn fn,
                                      void *data)
         return 0;
 }
 
+int pmu_metrics_table__find_metric(const struct pmu_metrics_table *table,
+                                 struct perf_pmu *pmu,
+                                 const char *metric,
+                                 pmu_metric_iter_fn fn,
+                                 void *data)
+{
+        for (size_t i = 0; i < table->num_pmus; i++) {
+                const struct pmu_table_entry *table_pmu = &table->pmus[i];
+                const char *pmu_name = &big_c_string[table_pmu->pmu_name.offset];
+                int ret;
+
+                if (pmu && !perf_pmu__name_wildcard_match(pmu, pmu_name))
+                        continue;
+
+                ret = pmu_metrics_table__find_metric_pmu(table, table_pmu, metric, fn, data);
+                if (ret != PMU_METRICS__NOT_FOUND)
+                        return ret;
+        }
+        return PMU_METRICS__NOT_FOUND;
+}
+
 static const struct pmu_events_map *map_for_cpu(struct perf_cpu cpu)
 {
         static struct {
 
                 const char *pmu_name = &big_c_string[table_pmu->pmu_name.offset];
                 int ret;
 
-                if (!perf_pmu__name_wildcard_match(pmu, pmu_name))
+                if (pmu && !perf_pmu__name_wildcard_match(pmu, pmu_name))
                         continue;
 
                 ret = pmu_events_table__find_event_pmu(table, table_pmu, name, fn, data);
         return 0;
 }
 
+static int pmu_metrics_table__find_metric_pmu(const struct pmu_metrics_table *table,
+                                            const struct pmu_table_entry *pmu,
+                                            const char *metric,
+                                            pmu_metric_iter_fn fn,
+                                            void *data)
+{
+        struct pmu_metric pm = {
+                .pmu = &big_c_string[pmu->pmu_name.offset],
+        };
+        int low = 0, high = pmu->num_entries - 1;
+
+        while (low <= high) {
+                int cmp, mid = (low + high) / 2;
+
+                decompress_metric(pmu->entries[mid].offset, &pm);
+
+                if (!pm.metric_name && !metric)
+                        goto do_call;
+
+                if (!pm.metric_name && metric) {
+                        low = mid + 1;
+                        continue;
+                }
+                if (pm.metric_name && !metric) {
+                        high = mid - 1;
+                        continue;
+                }
+
+                cmp = strcmp(pm.metric_name, metric);
+                if (cmp < 0) {
+                        low = mid + 1;
+                        continue;
+                }
+                if (cmp > 0) {
+                        high = mid - 1;
+                        continue;
+                }
+  do_call:
+                return fn ? fn(&pm, table, data) : 0;
+        }
+        return PMU_METRICS__NOT_FOUND;
+}
+
 int pmu_metrics_table__for_each_metric(const struct pmu_metrics_table *table,
                                      pmu_metric_iter_fn fn,
                                      void *data)
         return 0;
 }
 
+int pmu_metrics_table__find_metric(const struct pmu_metrics_table *table,
+                                 struct perf_pmu *pmu,
+                                 const char *metric,
+                                 pmu_metric_iter_fn fn,
+                                 void *data)
+{
+        for (size_t i = 0; i < table->num_pmus; i++) {
+                const struct pmu_table_entry *table_pmu = &table->pmus[i];
+                const char *pmu_name = &big_c_string[table_pmu->pmu_name.offset];
+                int ret;
+
+                if (pmu && !perf_pmu__name_wildcard_match(pmu, pmu_name))
+                        continue;
+
+                ret = pmu_metrics_table__find_metric_pmu(table, table_pmu, metric, fn, data);
+                if (ret != PMU_METRICS__NOT_FOUND)
+                        return ret;
+        }
+        return PMU_METRICS__NOT_FOUND;
+}
+
 static const struct pmu_events_map *map_for_cpu(struct perf_cpu cpu)
 {
         static struct {
 
 struct pmu_metrics_table;
 
 #define PMU_EVENTS__NOT_FOUND -1000
+#define PMU_METRICS__NOT_FOUND -1000
 
 typedef int (*pmu_event_iter_fn)(const struct pmu_event *pe,
                                 const struct pmu_events_table *table,
                                    pmu_event_iter_fn fn,
                                    void *data);
 /*
- * Search for table and entry matching with pmu__name_match. Each matching event
- * has fn called on it. 0 implies to success/continue the search while non-zero
- * means to terminate. The special value PMU_EVENTS__NOT_FOUND is used to
- * indicate no event was found in one of the tables which doesn't terminate the
- * search of all tables.
+ * Search for a table and entry matching with pmu__name_wildcard_match or any
+ * tables if pmu is NULL. Each matching event has fn called on it. 0 implies to
+ * success/continue the search while non-zero means to terminate. The special
+ * value PMU_EVENTS__NOT_FOUND is used to indicate no event was found in one of
+ * the tables which doesn't terminate the search of all tables.
  */
 int pmu_events_table__find_event(const struct pmu_events_table *table,
                                  struct perf_pmu *pmu,
 
 int pmu_metrics_table__for_each_metric(const struct pmu_metrics_table *table, pmu_metric_iter_fn fn,
                                     void *data);
+/*
+ * Search for a table and entry matching with pmu__name_wildcard_match or any
+ * tables if pmu is NULL. Each matching metric has fn called on it. 0 implies to
+ * success/continue the search while non-zero means to terminate. The special
+ * value PMU_METRICS__NOT_FOUND is used to indicate no metric was found in one
+ * of the tables which doesn't terminate the search of all tables.
+ */
+int pmu_metrics_table__find_metric(const struct pmu_metrics_table *table,
+                                  struct perf_pmu *pmu,
+                                  const char *metric,
+                                  pmu_metric_iter_fn fn,
+                                  void *data);
 
 const struct pmu_events_table *perf_pmu__find_events_table(struct perf_pmu *pmu);
 const struct pmu_metrics_table *pmu_metrics_table__find(void);
 
        return 0;
 }
 
-static bool match_metric(const char *metric_or_groups, const char *sought)
+static bool match_metric_or_groups(const char *metric_or_groups, const char *sought)
 {
        int len;
        char *m;
            (metric_or_groups[len] == 0 || metric_or_groups[len] == ';'))
                return true;
        m = strchr(metric_or_groups, ';');
-       return m && match_metric(m + 1, sought);
+       return m && match_metric_or_groups(m + 1, sought);
 }
 
-static bool match_pm_metric(const struct pmu_metric *pm, const char *pmu, const char *metric)
+static bool match_pm_metric_or_groups(const struct pmu_metric *pm, const char *pmu,
+                                     const char *metric_or_groups)
 {
        const char *pm_pmu = pm->pmu ?: "cpu";
 
        if (strcmp(pmu, "all") && strcmp(pm_pmu, pmu))
                return false;
 
-       return match_metric(pm->metric_group, metric) ||
-              match_metric(pm->metric_name, metric);
+       return match_metric_or_groups(pm->metric_group, metric_or_groups) ||
+              match_metric_or_groups(pm->metric_name, metric_or_groups);
 }
 
 /** struct mep - RB-tree node for building printing information. */
        const struct pmu_metrics_table *table;
 };
 
-static bool metricgroup__find_metric(const char *pmu,
-                                    const char *metric,
-                                    const struct pmu_metrics_table *table,
-                                    struct pmu_metric *pm);
-
 static int add_metric(struct list_head *metric_list,
                      const struct pmu_metric *pm,
                      const char *modifier,
                      const struct visited_metric *visited,
                      const struct pmu_metrics_table *table);
 
+static int metricgroup__find_metric_callback(const struct pmu_metric *pm,
+                                            const struct pmu_metrics_table *table  __maybe_unused,
+                                            void *vdata)
+{
+       struct pmu_metric *copied_pm = vdata;
+
+       memcpy(copied_pm, pm, sizeof(*pm));
+       return 0;
+}
+
 /**
  * resolve_metric - Locate metrics within the root metric and recursively add
  *                    references to them.
  *       architecture perf is running upon.
  */
 static int resolve_metric(struct list_head *metric_list,
-                         const char *pmu,
+                         struct perf_pmu *pmu,
                          const char *modifier,
                          bool metric_no_group,
                          bool metric_no_threshold,
        hashmap__for_each_entry(root_metric->pctx->ids, cur, bkt) {
                struct pmu_metric pm;
 
-               if (metricgroup__find_metric(pmu, cur->pkey, table, &pm)) {
+               if (pmu_metrics_table__find_metric(table, pmu, cur->pkey,
+                                                  metricgroup__find_metric_callback,
+                                                  &pm) != PMU_METRICS__NOT_FOUND) {
                        pending = realloc(pending,
                                        (pending_cnt + 1) * sizeof(struct to_resolve));
                        if (!pending)
        }
        if (!ret) {
                /* Resolve referenced metrics. */
-               const char *pmu = pm->pmu ?: "cpu";
+               struct perf_pmu *pmu;
+
+               if (pm->pmu && pm->pmu[0] != '\0')
+                       pmu = perf_pmus__find(pm->pmu);
+               else
+                       pmu = perf_pmus__scan_core(/*pmu=*/ NULL);
 
                ret = resolve_metric(metric_list, pmu, modifier, metric_no_group,
                                     metric_no_threshold, user_requested_cpu_list,
        return ret;
 }
 
-struct metricgroup__find_metric_data {
-       const char *pmu;
-       const char *metric;
-       struct pmu_metric *pm;
-};
-
-static int metricgroup__find_metric_callback(const struct pmu_metric *pm,
-                                            const struct pmu_metrics_table *table  __maybe_unused,
-                                            void *vdata)
-{
-       struct metricgroup__find_metric_data *data = vdata;
-       const char *pm_pmu = pm->pmu ?: "cpu";
-
-       if (strcmp(data->pmu, "all") && strcmp(pm_pmu, data->pmu))
-               return 0;
-
-       if (!match_metric(pm->metric_name, data->metric))
-               return 0;
-
-       memcpy(data->pm, pm, sizeof(*pm));
-       return 1;
-}
-
-static bool metricgroup__find_metric(const char *pmu,
-                                    const char *metric,
-                                    const struct pmu_metrics_table *table,
-                                    struct pmu_metric *pm)
-{
-       struct metricgroup__find_metric_data data = {
-               .pmu = pmu,
-               .metric = metric,
-               .pm = pm,
-       };
-
-       return pmu_metrics_table__for_each_metric(table, metricgroup__find_metric_callback, &data)
-               ? true : false;
-}
-
 static int add_metric(struct list_head *metric_list,
                      const struct pmu_metric *pm,
                      const char *modifier,
        struct metricgroup_add_iter_data *d = data;
        int ret;
 
-       if (!match_pm_metric(pm, d->pmu, d->metric_name))
+       if (!match_pm_metric_or_groups(pm, d->pmu, d->metric_name))
                return 0;
 
        ret = add_metric(d->metric_list, pm, d->modifier, d->metric_no_group,
        struct metricgroup__add_metric_data *data = vdata;
        int ret = 0;
 
-       if (pm->metric_expr && match_pm_metric(pm, data->pmu, data->metric_name)) {
+       if (pm->metric_expr && match_pm_metric_or_groups(pm, data->pmu, data->metric_name)) {
                bool metric_no_group = data->metric_no_group ||
-                       match_metric(pm->metricgroup_no_group, data->metric_name);
+                       match_metric_or_groups(pm->metricgroup_no_group, data->metric_name);
 
                data->has_match = true;
                ret = add_metric(data->list, pm, data->modifier, metric_no_group,
 
 struct metricgroup__has_metric_data {
        const char *pmu;
-       const char *metric;
+       const char *metric_or_groups;
 };
-static int metricgroup__has_metric_callback(const struct pmu_metric *pm,
-                                           const struct pmu_metrics_table *table __maybe_unused,
-                                           void *vdata)
+static int metricgroup__has_metric_or_groups_callback(const struct pmu_metric *pm,
+                                                     const struct pmu_metrics_table *table
+                                                       __maybe_unused,
+                                                     void *vdata)
 {
        struct metricgroup__has_metric_data *data = vdata;
 
-       return match_pm_metric(pm, data->pmu, data->metric) ? 1 : 0;
+       return match_pm_metric_or_groups(pm, data->pmu, data->metric_or_groups) ? 1 : 0;
 }
 
-bool metricgroup__has_metric(const char *pmu, const char *metric)
+bool metricgroup__has_metric_or_groups(const char *pmu, const char *metric_or_groups)
 {
        const struct pmu_metrics_table *table = pmu_metrics_table__find();
        struct metricgroup__has_metric_data data = {
                .pmu = pmu,
-               .metric = metric,
+               .metric_or_groups = metric_or_groups,
        };
 
        if (!table)
                return false;
 
-       return pmu_metrics_table__for_each_metric(table, metricgroup__has_metric_callback, &data)
+       return pmu_metrics_table__for_each_metric(table,
+                                                 metricgroup__has_metric_or_groups_callback,
+                                                 &data)
                ? true : false;
 }