diff options
Diffstat (limited to 'src/plugins/perfmon/intel/core.c')
-rw-r--r-- | src/plugins/perfmon/intel/core.c | 78 |
1 files changed, 77 insertions, 1 deletions
diff --git a/src/plugins/perfmon/intel/core.c b/src/plugins/perfmon/intel/core.c index cef6f32d7e6..d6a16b2125e 100644 --- a/src/plugins/perfmon/intel/core.c +++ b/src/plugins/perfmon/intel/core.c @@ -16,6 +16,7 @@ #include <vnet/vnet.h> #include <perfmon/perfmon.h> #include <perfmon/intel/core.h> +#include <perfmon/intel/dispatch_wrapper.h> #include <linux/perf_event.h> static perfmon_event_t events[] = { @@ -25,9 +26,12 @@ static perfmon_event_t events[] = { event, umask, edge, any, inv, cmask), \ .name = #n "." #suffix, \ .description = desc, \ + .implemented = 1, \ .exclude_kernel = 1 }, - foreach_perf_intel_core_event + foreach_perf_intel_core_event foreach_perf_intel_peusdo_event + foreach_perf_intel_tremont_event + #undef _ }; @@ -55,6 +59,10 @@ format_intel_core_config (u8 *s, va_list *args) if ((v = (config >> 24) & 0xff)) s = format (s, ", cmask=0x%02x", v); + /* show the raw config, for convenience sake */ + if (!((config >> 16) & 0xffff)) + s = format (s, ", raw=r%x", config & 0xffff); + return s; } @@ -71,11 +79,79 @@ intel_core_init (vlib_main_t *vm, perfmon_source_t *src) return 0; } +perfmon_event_type_t +intel_core_get_event_type (u32 event) +{ + u64 config = events[event].config; + u8 eventcode = (config & 0xFF); + u8 umask = ((config >> 8) & 0xFF); + + if (!eventcode) /* is fixed or pseudo */ + { + if (umask >= 0x80) /* is pseudo */ + return PERFMON_EVENT_TYPE_PSEUDO; + else /* is fixed */ + return PERFMON_EVENT_TYPE_FIXED; + } + else + return PERFMON_EVENT_TYPE_GENERAL; +} + +static u8 +is_enough_counters (perfmon_bundle_t *b) +{ + u8 bl[PERFMON_EVENT_TYPE_MAX]; + u8 cpu[PERFMON_EVENT_TYPE_MAX]; + + clib_memset (&bl, 0, sizeof (bl)); + clib_memset (&cpu, 0, sizeof (cpu)); + + /* how many does this uarch support */ + if (!clib_get_pmu_counter_count (&cpu[PERFMON_EVENT_TYPE_FIXED], + &cpu[PERFMON_EVENT_TYPE_GENERAL])) + return 0; + + /* how many does the bundle require */ + for (u16 i = 0; i < b->n_events; i++) + { + /* if source allows us to identify events, otherwise assume general */ + if (b->src->get_event_type) + bl[b->src->get_event_type (b->events[i])]++; + else + bl[PERFMON_EVENT_TYPE_GENERAL]++; + } + + /* consciously ignoring pseudo events here */ + return cpu[PERFMON_EVENT_TYPE_GENERAL] >= bl[PERFMON_EVENT_TYPE_GENERAL] && + cpu[PERFMON_EVENT_TYPE_FIXED] >= bl[PERFMON_EVENT_TYPE_FIXED]; +} + +u8 +intel_bundle_supported (perfmon_bundle_t *b) +{ + perfmon_cpu_supports_t *supports = b->cpu_supports; + + if (!is_enough_counters (b)) + return 0; + + if (!b->cpu_supports) + return 1; + + for (int i = 0; i < b->n_cpu_supports; ++i) + if (supports[i].cpu_supports ()) + return 1; + + return 0; +} + PERFMON_REGISTER_SOURCE (intel_core) = { .name = "intel-core", .description = "intel arch core events", .events = events, .n_events = ARRAY_LEN (events), .init_fn = intel_core_init, + .get_event_type = intel_core_get_event_type, .format_config = format_intel_core_config, + .bundle_support = intel_bundle_supported, + .config_dispatch_wrapper = intel_config_dispatch_wrapper, }; |