const char *metric_unit;
 };
 
-static bool record_evsel(int *ind, struct evsel **start,
-                        int idnum,
-                        struct evsel **metric_events,
-                        struct evsel *ev)
-{
-       metric_events[*ind] = ev;
-       if (*ind == 0)
-               *start = ev;
-       if (++*ind == idnum) {
-               metric_events[*ind] = NULL;
-               return true;
-       }
-       return false;
-}
-
 static struct evsel *find_evsel_group(struct evlist *perf_evlist,
                                      const char **ids,
                                      int idnum,
                                      struct evsel **metric_events)
 {
-       struct evsel *ev, *start = NULL;
-       int ind = 0;
+       struct evsel *ev;
+       int i = 0;
+       bool leader_found;
 
        evlist__for_each_entry (perf_evlist, ev) {
-               if (ev->collect_stat)
-                       continue;
-               if (!strcmp(ev->name, ids[ind])) {
-                       if (record_evsel(&ind, &start, idnum,
-                                        metric_events, ev))
-                               return start;
+               if (!strcmp(ev->name, ids[i])) {
+                       if (!metric_events[i])
+                               metric_events[i] = ev;
                } else {
-                       /*
-                        * We saw some other event that is not
-                        * in our list of events. Discard
-                        * the whole match and start again.
-                        */
-                       ind = 0;
-                       start = NULL;
-                       if (!strcmp(ev->name, ids[ind])) {
-                               if (record_evsel(&ind, &start, idnum,
-                                                metric_events, ev))
-                                       return start;
+                       if (++i == idnum) {
+                               /* Discard the whole match and start again */
+                               i = 0;
+                               memset(metric_events, 0,
+                                      sizeof(struct evsel *) * idnum);
+                               continue;
+                       }
+
+                       if (!strcmp(ev->name, ids[i]))
+                               metric_events[i] = ev;
+                       else {
+                               /* Discard the whole match and start again */
+                               i = 0;
+                               memset(metric_events, 0,
+                                      sizeof(struct evsel *) * idnum);
+                               continue;
                        }
                }
        }
-       /*
-        * This can happen when an alias expands to multiple
-        * events, like for uncore events.
-        * We don't support this case for now.
-        */
-       return NULL;
+
+       if (i != idnum - 1) {
+               /* Not whole match */
+               return NULL;
+       }
+
+       metric_events[idnum] = NULL;
+
+       for (i = 0; i < idnum; i++) {
+               leader_found = false;
+               evlist__for_each_entry(perf_evlist, ev) {
+                       if (!leader_found && (ev == metric_events[i]))
+                               leader_found = true;
+
+                       if (leader_found &&
+                           !strcmp(ev->name, metric_events[i]->name)) {
+                               ev->metric_leader = metric_events[i];
+                       }
+               }
+       }
+
+       return metric_events[0];
 }
 
 static int metricgroup__setup_events(struct list_head *groups,
 
        int cpu;
        struct runtime_stat *stat;
        struct stats stats;
+       u64 metric_total;
+       int metric_other;
 };
 
 static int saved_value_cmp(struct rb_node *rb_node, const void *entry)
 {
        int ctx = evsel_context(counter);
        u64 count_ns = count;
+       struct saved_value *v;
 
        count *= counter->scale;
 
                update_runtime_stat(st, STAT_APERF, ctx, cpu, count);
 
        if (counter->collect_stat) {
-               struct saved_value *v = saved_value_lookup(counter, cpu, true,
-                                                          STAT_NONE, 0, st);
+               v = saved_value_lookup(counter, cpu, true, STAT_NONE, 0, st);
                update_stats(&v->stats, count);
+               if (counter->metric_leader)
+                       v->metric_total += count;
+       } else if (counter->metric_leader) {
+               v = saved_value_lookup(counter->metric_leader,
+                                      cpu, true, STAT_NONE, 0, st);
+               v->metric_total += count;
+               v->metric_other++;
        }
 }
 
        char *n, *pn;
 
        expr__ctx_init(&pctx);
-       expr__add_id(&pctx, name, avg);
        for (i = 0; metric_events[i]; i++) {
                struct saved_value *v;
                struct stats *stats;
+               u64 metric_total = 0;
 
                if (!strcmp(metric_events[i]->name, "duration_time")) {
                        stats = &walltime_nsecs_stats;
                                break;
                        stats = &v->stats;
                        scale = 1.0;
+
+                       if (v->metric_other)
+                               metric_total = v->metric_total;
                }
 
                n = strdup(metric_events[i]->name);
                pn = strchr(n, ' ');
                if (pn)
                        *pn = 0;
-               expr__add_id(&pctx, n, avg_stats(stats)*scale);
+
+               if (metric_total)
+                       expr__add_id(&pctx, n, metric_total);
+               else
+                       expr__add_id(&pctx, n, avg_stats(stats)*scale);
        }
+
+       expr__add_id(&pctx, name, avg);
+
        if (!metric_events[i]) {
                const char *p = metric_expr;