libpfm4 is a library that takes a CPU vendor documentation
compatible string and fills out perf_event_attr. Typical
use case: user wants to look at events other than the
ones supported by perf using symbolic names.

Signed-off-by: Arun Sharma <asha...@fb.com>
Cc: Ingo Molnar <mi...@elte.hu>
Cc: Frederic Weisbecker <fweis...@gmail.com>
Cc: Mike Galbraith <efa...@gmx.de>
Cc: Paul Mackerras <pau...@samba.org>
Cc: Peter Zijlstra <a.p.zijls...@chello.nl>
Cc: Stephane Eranian <eran...@google.com>
Cc: Thomas Gleixner <t...@linutronix.de>
Cc: Tom Zanussi <tzanu...@gmail.com>
Cc: linux-ker...@vger.kernel.org
Cc: linux-perf-u...@vger.kernel.org
Cc: perfmon2-devel@lists.sourceforge.net
---
 tools/perf/Makefile            |   13 +++++
 tools/perf/feature-tests.mak   |   11 ++++
 tools/perf/perf.c              |    6 ++
 tools/perf/util/parse-events.c |  118 ++++++++++++++++++++++++++++++++++++++++
 tools/perf/util/parse-events.h |    3 +
 5 files changed, 151 insertions(+), 0 deletions(-)

diff --git a/tools/perf/Makefile b/tools/perf/Makefile
index 9b84218..617db4a 100644
--- a/tools/perf/Makefile
+++ b/tools/perf/Makefile
@@ -451,6 +451,19 @@ else
        endif
 endif
 
+ifdef NO_LIBPFM4
+       BASIC_CFLAGS += -DNO_LIBPFM4_SUPPORT
+else
+       FLAGS_LIBPFM4=$(ALL_CFLAGS) $(ALL_LDFLAGS) $(EXTLIBS) -lpfm
+       ifneq ($(call try-cc,$(SOURCE_LIBPFM4),$(FLAGS_LIBPFM4)),y)
+               msg := $(warning libpfm4 not found, events restricted to 
generic ones. Please install libpfm4-devel or libpfm4-dev);
+               BASIC_CFLAGS += -DNO_LIBPFM4_SUPPORT
+       else
+               BASIC_CFLAGS += -DLIBPFM4
+               EXTLIBS += -lpfm
+       endif
+endif
+
 ifdef NO_LIBPERL
        BASIC_CFLAGS += -DNO_LIBPERL
 else
diff --git a/tools/perf/feature-tests.mak b/tools/perf/feature-tests.mak
index b041ca6..ec6b27b 100644
--- a/tools/perf/feature-tests.mak
+++ b/tools/perf/feature-tests.mak
@@ -121,6 +121,17 @@ int main(void)
 }
 endef
 
+ifndef NO_LIBPFM4
+define SOURCE_LIBPFM4
+#include <perfmon/pfmlib_perf_event.h>
+
+int main(void)
+{
+       return pfm_initialize();
+}
+endef
+endif
+
 # try-cc
 # Usage: option = $(call try-cc, source-to-build, cc-options)
 try-cc = $(shell sh -c                                           \
diff --git a/tools/perf/perf.c b/tools/perf/perf.c
index 595d0f4..a847afb 100644
--- a/tools/perf/perf.c
+++ b/tools/perf/perf.c
@@ -472,6 +472,12 @@ int main(int argc, const char **argv)
        }
        cmd = argv[0];
 
+#ifdef LIBPFM4
+       if (pfm_initialize() != PFM_SUCCESS) {
+               fprintf(stderr, "pfm_initialize failed\n");
+               return 1;
+       }
+#endif
        /*
         * We use PATH to find perf commands, but we prepend some higher
         * precedence paths: the "--exec-path" option, the PERF_EXEC_PATH
diff --git a/tools/perf/util/parse-events.c b/tools/perf/util/parse-events.c
index 54a7e26..91907a1 100644
--- a/tools/perf/util/parse-events.c
+++ b/tools/perf/util/parse-events.c
@@ -696,6 +696,48 @@ parse_numeric_event(const char **strp, struct 
perf_event_attr *attr)
        return EVT_FAILED;
 }
 
+#ifdef LIBPFM4
+static enum event_result
+parse_libpfm4_event(const char **strp, struct perf_event_attr *attr)
+{
+       int ret;
+       const char *str = *strp;
+       const char *comma_loc = NULL;
+       char *evt_name = NULL;
+       size_t len = 0;
+
+       comma_loc = strchr(str, ',');
+       if (comma_loc) {
+               /* take the event name up to the comma */
+               len = comma_loc - str;
+               evt_name = strndup(str, len+1);
+               if (evt_name == NULL) {
+                       pr_err("strndup returned NULL. Out of memory?");
+                       return EVT_FAILED;
+               }
+               evt_name[len] = '\0';
+       } else {
+               evt_name = (char *) *strp;
+       }
+
+       attr->size = sizeof(attr);
+       ret = pfm_get_perf_event_encoding(evt_name, PFM_PLM0|PFM_PLM3, attr,
+                                         NULL, NULL);
+       if (ret != PFM_SUCCESS) {
+               return EVT_FAILED;
+       }
+
+       if (comma_loc) {
+               *strp += len;
+               free(evt_name);
+       } else {
+               *strp += strlen(evt_name);
+       }
+
+       return EVT_HANDLED;
+}
+#endif
+
 static enum event_result
 parse_event_modifier(const char **strp, struct perf_event_attr *attr)
 {
@@ -762,6 +804,17 @@ parse_event_symbols(const struct option *opt, const char 
**str,
        if (ret != EVT_FAILED)
                goto modifier;
 
+#ifdef LIBPFM4
+       /*
+        * Handle libpfm4 before generic_hw events.
+        * Some events (eg: LLC_MISSES) fail otherwise.
+        */
+       ret = parse_libpfm4_event(str, attr);
+       if (ret != EVT_FAILED)
+               /* libpfm4 has its own modifier parsing code */
+               goto modifier;
+#endif
+
        ret = parse_generic_hw_event(str, attr);
        if (ret != EVT_FAILED)
                goto modifier;
@@ -987,6 +1040,24 @@ int print_hwcache_events(const char *event_glob)
        return printed;
 }
 
+#ifdef LIBPFM4
+static void print_umasks(pfm_event_info_t *info)
+{
+       int i, ret;
+       pfm_event_attr_info_t ainfo;
+
+       memset(&ainfo, 0, sizeof(ainfo));
+       ainfo.size = sizeof(ainfo);
+       pfm_for_each_event_attr(i, info) {
+               ret = pfm_get_event_attr_info(info->idx, i,
+                                             PFM_OS_PERF_EVENT_EXT, &ainfo);
+               if ((ret != PFM_SUCCESS) || (ainfo.type != PFM_ATTR_UMASK)) 
+                       continue;
+               printf("  \t:%-42s\n", ainfo.name);
+       }
+}
+#endif
+
 /*
  * Print the help text for the event symbols:
  */
@@ -1033,6 +1104,53 @@ void print_events(const char *event_glob)
        if (event_glob != NULL)
                return;
 
+#ifdef LIBPFM4
+       printf("\n");
+       pfm_for_all_pmus(i) {
+               int ret;
+               pfm_pmu_info_t pinfo;
+               int count;
+               int k;
+
+               memset(&pinfo, 0, sizeof(pinfo));
+               pinfo.size = sizeof(pinfo);
+               ret = pfm_get_pmu_info(i, &pinfo);
+               if (ret != PFM_SUCCESS)
+                       continue;
+               if (!pinfo.is_present)
+                       continue;
+               if (pinfo.pmu == PFM_PMU_PERF_EVENT)
+                       continue;
+
+               printf("\nDetected PMU:  %s -- %s Total events: %d\n",
+                       pinfo.name,
+                       pinfo.desc,
+                       pinfo.nevents);
+
+               count = 0;
+               for (k = pinfo.first_event; k != -1; k = pfm_get_event_next(k)) 
{
+                       pfm_event_info_t info;
+
+                       memset(&info, 0, sizeof(info));
+                       info.size = sizeof(info);
+                       ret = pfm_get_event_info(k, PFM_OS_PERF_EVENT_EXT, 
&info);
+                       if (ret != PFM_SUCCESS) {
+                               fprintf(stderr, "libpfm4: %s\n", 
pfm_strerror(ret));
+                       }
+                       if (info.pmu != pinfo.pmu)
+                               continue;
+
+                       count++;
+                       if (count > pinfo.nevents)
+                               break;
+                       printf("  %-42s [%s]\n",
+                               info.name,
+                               event_type_descriptors[PERF_TYPE_HARDWARE]);
+                       print_umasks(&info);
+               }
+       }
+#endif
+
        printf("\n");
        printf("  %-42s [%s]\n",
                "rNNN (see 'perf list --help' on how to encode it)",
diff --git a/tools/perf/util/parse-events.h b/tools/perf/util/parse-events.h
index 212f88e..89c153c 100644
--- a/tools/perf/util/parse-events.h
+++ b/tools/perf/util/parse-events.h
@@ -5,6 +5,9 @@
  */
 
 #include "../../../include/linux/perf_event.h"
+#ifdef LIBPFM4
+#include <perfmon/pfmlib_perf_event.h>
+#endif
 
 struct list_head;
 struct perf_evsel;
-- 
1.7.4


------------------------------------------------------------------------------
Colocation vs. Managed Hosting
A question and answer guide to determining the best fit
for your organization - today and in the future.
http://p.sf.net/sfu/internap-sfd2d
_______________________________________________
perfmon2-devel mailing list
perfmon2-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/perfmon2-devel

Reply via email to