| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Benchmark scanning sysfs files for PMU information.
- *
- * Copyright 2023 Google LLC.
- */
- #include <stdio.h>
- #include "bench.h"
- #include "util/debug.h"
- #include "util/pmu.h"
- #include "util/pmus.h"
- #include "util/stat.h"
- #include <linux/atomic.h>
- #include <linux/err.h>
- #include <linux/time64.h>
- #include <subcmd/parse-options.h>
- static unsigned int iterations = 100;
- struct pmu_scan_result {
- char *name;
- int nr_aliases;
- int nr_formats;
- int nr_caps;
- bool is_core;
- };
- static const struct option options[] = {
- OPT_UINTEGER('i', "iterations", &iterations,
- "Number of iterations used to compute average"),
- OPT_END()
- };
- static const char *const bench_usage[] = {
- "perf bench internals pmu-scan <options>",
- NULL
- };
- static int nr_pmus;
- static struct pmu_scan_result *results;
- static int save_result(void)
- {
- struct perf_pmu *pmu = NULL;
- struct list_head *list;
- struct pmu_scan_result *r;
- while ((pmu = perf_pmus__scan(pmu)) != NULL) {
- r = realloc(results, (nr_pmus + 1) * sizeof(*r));
- if (r == NULL)
- return -ENOMEM;
- results = r;
- r = results + nr_pmus;
- r->name = strdup(pmu->name);
- r->is_core = pmu->is_core;
- r->nr_caps = pmu->nr_caps;
- r->nr_aliases = perf_pmu__num_events(pmu);
- r->nr_formats = 0;
- list_for_each(list, &pmu->format)
- r->nr_formats++;
- pr_debug("pmu[%d] name=%s, nr_caps=%d, nr_aliases=%d, nr_formats=%d\n",
- nr_pmus, r->name, r->nr_caps, r->nr_aliases, r->nr_formats);
- nr_pmus++;
- }
- perf_pmus__destroy();
- return 0;
- }
- static int check_result(bool core_only)
- {
- struct pmu_scan_result *r;
- struct perf_pmu *pmu;
- struct list_head *list;
- int nr;
- for (int i = 0; i < nr_pmus; i++) {
- r = &results[i];
- if (core_only && !r->is_core)
- continue;
- pmu = perf_pmus__find(r->name);
- if (pmu == NULL) {
- pr_err("Cannot find PMU %s\n", r->name);
- return -1;
- }
- if (pmu->nr_caps != (u32)r->nr_caps) {
- pr_err("Unmatched number of event caps in %s: expect %d vs got %d\n",
- pmu->name, r->nr_caps, pmu->nr_caps);
- return -1;
- }
- nr = perf_pmu__num_events(pmu);
- if (nr != r->nr_aliases) {
- pr_err("Unmatched number of event aliases in %s: expect %d vs got %d\n",
- pmu->name, r->nr_aliases, nr);
- return -1;
- }
- nr = 0;
- list_for_each(list, &pmu->format)
- nr++;
- if (nr != r->nr_formats) {
- pr_err("Unmatched number of event formats in %s: expect %d vs got %d\n",
- pmu->name, r->nr_formats, nr);
- return -1;
- }
- }
- return 0;
- }
- static void delete_result(void)
- {
- for (int i = 0; i < nr_pmus; i++)
- free(results[i].name);
- free(results);
- results = NULL;
- nr_pmus = 0;
- }
- static int run_pmu_scan(void)
- {
- struct stats stats;
- struct timeval start, end, diff;
- double time_average, time_stddev;
- u64 runtime_us;
- int ret;
- init_stats(&stats);
- pr_info("Computing performance of sysfs PMU event scan for %u times\n",
- iterations);
- if (save_result() < 0) {
- pr_err("Failed to initialize PMU scan result\n");
- return -1;
- }
- for (int j = 0; j < 2; j++) {
- bool core_only = (j == 0);
- for (unsigned int i = 0; i < iterations; i++) {
- gettimeofday(&start, NULL);
- if (core_only)
- perf_pmus__scan_core(NULL);
- else
- perf_pmus__scan(NULL);
- gettimeofday(&end, NULL);
- timersub(&end, &start, &diff);
- runtime_us = diff.tv_sec * USEC_PER_SEC + diff.tv_usec;
- update_stats(&stats, runtime_us);
- ret = check_result(core_only);
- perf_pmus__destroy();
- if (ret < 0)
- break;
- }
- time_average = avg_stats(&stats);
- time_stddev = stddev_stats(&stats);
- pr_info(" Average%s PMU scanning took: %.3f usec (+- %.3f usec)\n",
- core_only ? " core" : "", time_average, time_stddev);
- }
- delete_result();
- return 0;
- }
- int bench_pmu_scan(int argc, const char **argv)
- {
- int err = 0;
- argc = parse_options(argc, argv, options, bench_usage, 0);
- if (argc) {
- usage_with_options(bench_usage, options);
- exit(EXIT_FAILURE);
- }
- err = run_pmu_scan();
- return err;
- }
|