// SPDX-License-Identifier: GPL-2.0 #include #include #include #include "evsel.h" #include "stat.h" #include "color.h" #include "debug.h" #include "pmu.h" #include "rblist.h" #include "evlist.h" #include "expr.h" #include "metricgroup.h" #include "cgroup.h" #include "units.h" #include #include "iostat.h" #include "util/hashmap.h" #include "tool_pmu.h" static bool tool_pmu__is_time_event(const struct perf_stat_config *config, const struct evsel *evsel, int *tool_aggr_idx) { enum tool_pmu_event event = evsel__tool_event(evsel); int aggr_idx; if (event != TOOL_PMU__EVENT_DURATION_TIME && event != TOOL_PMU__EVENT_USER_TIME && event != TOOL_PMU__EVENT_SYSTEM_TIME) return false; if (config) { cpu_aggr_map__for_each_idx(aggr_idx, config->aggr_map) { if (config->aggr_map->map[aggr_idx].cpu.cpu == 0) { *tool_aggr_idx = aggr_idx; return true; } } pr_debug("Unexpected CPU0 missing in aggregation for tool event.\n"); } *tool_aggr_idx = 0; /* Assume the first aggregation index works. */ return true; } static int prepare_metric(struct perf_stat_config *config, const struct metric_expr *mexp, const struct evsel *evsel, struct expr_parse_ctx *pctx, int aggr_idx) { struct evsel * const *metric_events = mexp->metric_events; struct metric_ref *metric_refs = mexp->metric_refs; int i; for (i = 0; metric_events[i]; i++) { int source_count = 0, tool_aggr_idx; bool is_tool_time = tool_pmu__is_time_event(config, metric_events[i], &tool_aggr_idx); struct perf_stat_evsel *ps = metric_events[i]->stats; struct perf_stat_aggr *aggr; char *n; double val; /* * If there are multiple uncore PMUs and we're not reading the * leader's stats, determine the stats for the appropriate * uncore PMU. */ if (evsel && evsel->metric_leader && evsel->pmu != evsel->metric_leader->pmu && mexp->metric_events[i]->pmu == evsel->metric_leader->pmu) { struct evsel *pos; evlist__for_each_entry(evsel->evlist, pos) { if (pos->pmu != evsel->pmu) continue; if (pos->metric_leader != mexp->metric_events[i]) continue; ps = pos->stats; source_count = 1; break; } } /* Time events are always on CPU0, the first aggregation index. */ aggr = &ps->aggr[is_tool_time ? tool_aggr_idx : aggr_idx]; if (!aggr || !metric_events[i]->supported) { /* * Not supported events will have a count of 0, which * can be confusing in a metric. Explicitly set the * value to NAN. Not counted events (enable time of 0) * are read as 0. */ val = NAN; source_count = 0; } else { val = aggr->counts.val; if (is_tool_time) val *= 1e-9; /* Convert time event nanoseconds to seconds. */ if (!source_count) source_count = evsel__source_count(metric_events[i]); } n = strdup(evsel__metric_id(metric_events[i])); if (!n) return -ENOMEM; expr__add_id_val_source_count(pctx, n, val, source_count); } for (int j = 0; metric_refs && metric_refs[j].metric_name; j++) { int ret = expr__add_ref(pctx, &metric_refs[j]); if (ret) return ret; } return i; } static void generic_metric(struct perf_stat_config *config, struct metric_expr *mexp, struct evsel *evsel, int aggr_idx, struct perf_stat_output_ctx *out) { print_metric_t print_metric = out->print_metric; const char *metric_name = mexp->metric_name; const char *metric_expr = mexp->metric_expr; const char *metric_threshold = mexp->metric_threshold; const char *metric_unit = mexp->metric_unit; struct evsel * const *metric_events = mexp->metric_events; int runtime = mexp->runtime; struct expr_parse_ctx *pctx; double ratio, scale, threshold; int i; void *ctxp = out->ctx; enum metric_threshold_classify thresh = METRIC_THRESHOLD_UNKNOWN; pctx = expr__ctx_new(); if (!pctx) return; if (config->user_requested_cpu_list) pctx->sctx.user_requested_cpu_list = strdup(config->user_requested_cpu_list); pctx->sctx.runtime = runtime; pctx->sctx.system_wide = config->system_wide; i = prepare_metric(config, mexp, evsel, pctx, aggr_idx); if (i < 0) { expr__ctx_free(pctx); return; } if (!metric_events[i]) { if (expr__parse(&ratio, pctx, metric_expr) == 0) { char *unit; char metric_bf[128]; if (metric_threshold && expr__parse(&threshold, pctx, metric_threshold) == 0 && !isnan(threshold)) { thresh = fpclassify(threshold) == FP_ZERO ? METRIC_THRESHOLD_GOOD : METRIC_THRESHOLD_BAD; } if (metric_unit && metric_name) { if (perf_pmu__convert_scale(metric_unit, &unit, &scale) >= 0) { ratio *= scale; } if (strstr(metric_expr, "?")) scnprintf(metric_bf, sizeof(metric_bf), "%s %s_%d", unit, metric_name, runtime); else scnprintf(metric_bf, sizeof(metric_bf), "%s %s", unit, metric_name); print_metric(config, ctxp, thresh, "%8.1f", metric_bf, ratio); } else { print_metric(config, ctxp, thresh, "%8.2f", metric_name ? metric_name : out->force_header ? evsel->name : "", ratio); } } else { print_metric(config, ctxp, thresh, /*fmt=*/NULL, out->force_header ? (metric_name ?: evsel->name) : "", 0); } } else { print_metric(config, ctxp, thresh, /*fmt=*/NULL, out->force_header ? (metric_name ?: evsel->name) : "", 0); } expr__ctx_free(pctx); } double test_generic_metric(struct metric_expr *mexp, int aggr_idx) { struct expr_parse_ctx *pctx; double ratio = 0.0; pctx = expr__ctx_new(); if (!pctx) return NAN; if (prepare_metric(/*config=*/NULL, mexp, /*evsel=*/NULL, pctx, aggr_idx) < 0) goto out; if (expr__parse(&ratio, pctx, mexp->metric_expr)) ratio = 0.0; out: expr__ctx_free(pctx); return ratio; } static void perf_stat__print_metricgroup_header(struct perf_stat_config *config, struct evsel *evsel, void *ctxp, const char *name, struct perf_stat_output_ctx *out) { bool need_full_name = perf_pmus__num_core_pmus() > 1; static const char *last_name; static const struct perf_pmu *last_pmu; char full_name[64]; /* * A metricgroup may have several metric events, * e.g.,TopdownL1 on e-core of ADL. * The name has been output by the first metric * event. Only align with other metics from * different metric events. */ if (last_name && !strcmp(last_name, name) && last_pmu == evsel->pmu) { out->print_metricgroup_header(config, ctxp, NULL); return; } if (need_full_name && evsel->pmu) scnprintf(full_name, sizeof(full_name), "%s (%s)", name, evsel->pmu->name); else scnprintf(full_name, sizeof(full_name), "%s", name); out->print_metricgroup_header(config, ctxp, full_name); last_name = name; last_pmu = evsel->pmu; } /** * perf_stat__print_shadow_stats_metricgroup - Print out metrics associated with the evsel * For the non-default, all metrics associated * with the evsel are printed. * For the default mode, only the metrics from * the same metricgroup and the name of the * metricgroup are printed. To print the metrics * from the next metricgroup (if available), * invoke the function with correspoinding * metric_expr. */ void *perf_stat__print_shadow_stats_metricgroup(struct perf_stat_config *config, struct evsel *evsel, int aggr_idx, int *num, void *from, struct perf_stat_output_ctx *out) { struct metric_event *me; struct metric_expr *mexp = from; void *ctxp = out->ctx; bool header_printed = false; const char *name = NULL; struct rblist *metric_events = &evsel->evlist->metric_events; me = metricgroup__lookup(metric_events, evsel, false); if (me == NULL) return NULL; if (!mexp) mexp = list_first_entry(&me->head, typeof(*mexp), nd); list_for_each_entry_from(mexp, &me->head, nd) { /* Print the display name of the Default metricgroup */ if (!config->metric_only && me->is_default) { if (!name) name = mexp->default_metricgroup_name; /* * Two or more metricgroup may share the same metric * event, e.g., TopdownL1 and TopdownL2 on SPR. * Return and print the prefix, e.g., noise, running * for the next metricgroup. */ if (strcmp(name, mexp->default_metricgroup_name)) return (void *)mexp; /* Only print the name of the metricgroup once */ if (!header_printed && !evsel->default_show_events) { header_printed = true; perf_stat__print_metricgroup_header(config, evsel, ctxp, name, out); } } if ((*num)++ > 0 && out->new_line) out->new_line(config, ctxp); generic_metric(config, mexp, evsel, aggr_idx, out); } return NULL; } void perf_stat__print_shadow_stats(struct perf_stat_config *config, struct evsel *evsel, int aggr_idx, struct perf_stat_output_ctx *out) { print_metric_t print_metric = out->print_metric; void *ctxp = out->ctx; int num = 0; if (config->iostat_run) iostat_print_metric(config, evsel, out); perf_stat__print_shadow_stats_metricgroup(config, evsel, aggr_idx, &num, NULL, out); if (num == 0) { print_metric(config, ctxp, METRIC_THRESHOLD_UNKNOWN, /*fmt=*/NULL, /*unit=*/NULL, 0); } } /** * perf_stat__skip_metric_event - Skip the evsel in the Default metricgroup, * if it's not running or not the metric event. */ bool perf_stat__skip_metric_event(struct evsel *evsel, u64 ena, u64 run) { if (!evsel->default_metricgroup) return false; if (!ena || !run) return true; return !metricgroup__lookup(&evsel->evlist->metric_events, evsel, false); }