perf tests: Add tool PMU test
authorIan Rogers <irogers@google.com>
Wed, 2 Oct 2024 03:20:13 +0000 (20:20 -0700)
committerNamhyung Kim <namhyung@kernel.org>
Fri, 11 Oct 2024 06:41:13 +0000 (23:41 -0700)
Ensure parsing with and without PMU creates events with the expected
config values. This ensures the tool.json doesn't get out of sync with
tool_pmu_event enum.

Signed-off-by: Ian Rogers <irogers@google.com>
Acked-by: Namhyung Kim <namhyung@kernel.org>
Link: https://lore.kernel.org/r/20241002032016.333748-11-irogers@google.com
Signed-off-by: Namhyung Kim <namhyung@kernel.org>
tools/perf/tests/Build
tools/perf/tests/builtin-test.c
tools/perf/tests/tests.h
tools/perf/tests/tool_pmu.c [new file with mode: 0644]

index 5671ee53001959ed8e2ec4a0a8f479620e139d40..a771e49282471240c4d6988a0d66c013ab73756a 100644 (file)
@@ -67,6 +67,7 @@ perf-test-y += sigtrap.o
 perf-test-y += event_groups.o
 perf-test-y += symbols.o
 perf-test-y += util.o
+perf-test-y += tool_pmu.o
 
 ifeq ($(SRCARCH),$(filter $(SRCARCH),x86 arm arm64 powerpc))
 perf-test-$(CONFIG_DWARF_UNWIND) += dwarf-unwind.o
index 470a9709427ddaad73d98ae52a6a7c6f626eecb3..3b30f258c39592fc4b9e323c03125168a7682b08 100644 (file)
@@ -73,6 +73,7 @@ static struct test_suite *generic_tests[] = {
        &suite__PERF_RECORD,
        &suite__pmu,
        &suite__pmu_events,
+       &suite__tool_pmu,
        &suite__dso_data,
        &suite__perf_evsel__roundtrip_name_test,
 #ifdef HAVE_LIBTRACEEVENT
index 6ea2be86b7bf8bb4e4a3757d27233be3082fbb51..1ed76d4156b6af55af04a380eeb7abd45ea1f666 100644 (file)
@@ -83,6 +83,7 @@ DECLARE_SUITE(perf_evsel__tp_sched_test);
 DECLARE_SUITE(syscall_openat_tp_fields);
 DECLARE_SUITE(pmu);
 DECLARE_SUITE(pmu_events);
+DECLARE_SUITE(tool_pmu);
 DECLARE_SUITE(attr);
 DECLARE_SUITE(dso_data);
 DECLARE_SUITE(dso_data_cache);
diff --git a/tools/perf/tests/tool_pmu.c b/tools/perf/tests/tool_pmu.c
new file mode 100644 (file)
index 0000000..94d0dd8
--- /dev/null
@@ -0,0 +1,111 @@
+// SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
+#include "debug.h"
+#include "evlist.h"
+#include "parse-events.h"
+#include "tests.h"
+#include "tool_pmu.h"
+
+static int do_test(enum tool_pmu_event ev, bool with_pmu)
+{
+       struct evlist *evlist = evlist__new();
+       struct evsel *evsel;
+       struct parse_events_error err;
+       int ret;
+       char str[128];
+       bool found = false;
+
+       if (!evlist) {
+               pr_err("evlist allocation failed\n");
+               return TEST_FAIL;
+       }
+
+       if (with_pmu)
+               snprintf(str, sizeof(str), "tool/%s/", tool_pmu__event_to_str(ev));
+       else
+               strncpy(str, tool_pmu__event_to_str(ev), sizeof(str));
+
+       parse_events_error__init(&err);
+       ret = parse_events(evlist, str, &err);
+       if (ret) {
+               evlist__delete(evlist);
+               if (tool_pmu__skip_event(tool_pmu__event_to_str(ev))) {
+                       ret = TEST_OK;
+                       goto out;
+               }
+
+               pr_debug("FAILED %s:%d failed to parse event '%s', err %d\n",
+                        __FILE__, __LINE__, str, ret);
+               parse_events_error__print(&err, str);
+               ret = TEST_FAIL;
+               goto out;
+       }
+
+       ret = TEST_OK;
+       if (with_pmu ? (evlist->core.nr_entries != 1) : (evlist->core.nr_entries < 1)) {
+               pr_debug("FAILED %s:%d Unexpected number of events for '%s' of %d\n",
+                        __FILE__, __LINE__, str, evlist->core.nr_entries);
+               ret = TEST_FAIL;
+               goto out;
+       }
+
+       evlist__for_each_entry(evlist, evsel) {
+               if (perf_pmu__is_tool(evsel->pmu)) {
+                       if (evsel->core.attr.config != ev) {
+                               pr_debug("FAILED %s:%d Unexpected config for '%s', %lld != %d\n",
+                                       __FILE__, __LINE__, str, evsel->core.attr.config, ev);
+                               ret = TEST_FAIL;
+                               goto out;
+                       }
+                       found = true;
+               }
+       }
+
+       if (!found && !tool_pmu__skip_event(tool_pmu__event_to_str(ev))) {
+               pr_debug("FAILED %s:%d Didn't find tool event '%s' in parsed evsels\n",
+                        __FILE__, __LINE__, str);
+               ret = TEST_FAIL;
+       }
+
+out:
+       evlist__delete(evlist);
+       return ret;
+}
+
+static int test__tool_pmu_without_pmu(struct test_suite *test __maybe_unused,
+                                     int subtest __maybe_unused)
+{
+       int i;
+
+       tool_pmu__for_each_event(i) {
+               int ret = do_test(i, /*with_pmu=*/false);
+
+               if (ret != TEST_OK)
+                       return ret;
+       }
+       return TEST_OK;
+}
+
+static int test__tool_pmu_with_pmu(struct test_suite *test __maybe_unused,
+                                  int subtest __maybe_unused)
+{
+       int i;
+
+       tool_pmu__for_each_event(i) {
+               int ret = do_test(i, /*with_pmu=*/true);
+
+               if (ret != TEST_OK)
+                       return ret;
+       }
+       return TEST_OK;
+}
+
+static struct test_case tests__tool_pmu[] = {
+       TEST_CASE("Parsing without PMU name", tool_pmu_without_pmu),
+       TEST_CASE("Parsing with PMU name", tool_pmu_with_pmu),
+       {       .name = NULL, }
+};
+
+struct test_suite suite__tool_pmu = {
+       .desc = "Tool PMU",
+       .test_cases = tests__tool_pmu,
+};