f50a7f3d92
Based on 1 normalized pattern(s): licensed under gplv2 extracted by the scancode license scanner the SPDX license identifier GPL-2.0-only has been chosen to replace the boilerplate/reference in 99 file(s). Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Reviewed-by: Alexios Zavras <alexios.zavras@intel.com> Reviewed-by: Richard Fontana <rfontana@redhat.com> Reviewed-by: Allison Randal <allison@lohutok.net> Reviewed-by: Steve Winslow <swinslow@gmail.com> Cc: linux-spdx@vger.kernel.org Link: https://lkml.kernel.org/r/20190528170027.163048684@linutronix.de Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
115 lines
2.6 KiB
C
115 lines
2.6 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright 2014, Michael Ellerman, IBM Corp.
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include <elf.h>
|
|
#include <limits.h>
|
|
#include <stdio.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
#include <sys/prctl.h>
|
|
|
|
#include "event.h"
|
|
#include "lib.h"
|
|
#include "utils.h"
|
|
|
|
/*
|
|
* Test that per-event excludes work.
|
|
*/
|
|
|
|
static int per_event_excludes(void)
|
|
{
|
|
struct event *e, events[4];
|
|
char *platform;
|
|
int i;
|
|
|
|
platform = (char *)get_auxv_entry(AT_BASE_PLATFORM);
|
|
FAIL_IF(!platform);
|
|
SKIP_IF(strcmp(platform, "power8") != 0);
|
|
|
|
/*
|
|
* We need to create the events disabled, otherwise the running/enabled
|
|
* counts don't match up.
|
|
*/
|
|
e = &events[0];
|
|
event_init_opts(e, PERF_COUNT_HW_INSTRUCTIONS,
|
|
PERF_TYPE_HARDWARE, "instructions");
|
|
e->attr.disabled = 1;
|
|
|
|
e = &events[1];
|
|
event_init_opts(e, PERF_COUNT_HW_INSTRUCTIONS,
|
|
PERF_TYPE_HARDWARE, "instructions(k)");
|
|
e->attr.disabled = 1;
|
|
e->attr.exclude_user = 1;
|
|
e->attr.exclude_hv = 1;
|
|
|
|
e = &events[2];
|
|
event_init_opts(e, PERF_COUNT_HW_INSTRUCTIONS,
|
|
PERF_TYPE_HARDWARE, "instructions(h)");
|
|
e->attr.disabled = 1;
|
|
e->attr.exclude_user = 1;
|
|
e->attr.exclude_kernel = 1;
|
|
|
|
e = &events[3];
|
|
event_init_opts(e, PERF_COUNT_HW_INSTRUCTIONS,
|
|
PERF_TYPE_HARDWARE, "instructions(u)");
|
|
e->attr.disabled = 1;
|
|
e->attr.exclude_hv = 1;
|
|
e->attr.exclude_kernel = 1;
|
|
|
|
FAIL_IF(event_open(&events[0]));
|
|
|
|
/*
|
|
* The open here will fail if we don't have per event exclude support,
|
|
* because the second event has an incompatible set of exclude settings
|
|
* and we're asking for the events to be in a group.
|
|
*/
|
|
for (i = 1; i < 4; i++)
|
|
FAIL_IF(event_open_with_group(&events[i], events[0].fd));
|
|
|
|
/*
|
|
* Even though the above will fail without per-event excludes we keep
|
|
* testing in order to be thorough.
|
|
*/
|
|
prctl(PR_TASK_PERF_EVENTS_ENABLE);
|
|
|
|
/* Spin for a while */
|
|
for (i = 0; i < INT_MAX; i++)
|
|
asm volatile("" : : : "memory");
|
|
|
|
prctl(PR_TASK_PERF_EVENTS_DISABLE);
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
FAIL_IF(event_read(&events[i]));
|
|
event_report(&events[i]);
|
|
}
|
|
|
|
/*
|
|
* We should see that all events have enabled == running. That
|
|
* shows that they were all on the PMU at once.
|
|
*/
|
|
for (i = 0; i < 4; i++)
|
|
FAIL_IF(events[i].result.running != events[i].result.enabled);
|
|
|
|
/*
|
|
* We can also check that the result for instructions is >= all the
|
|
* other counts. That's because it is counting all instructions while
|
|
* the others are counting a subset.
|
|
*/
|
|
for (i = 1; i < 4; i++)
|
|
FAIL_IF(events[0].result.value < events[i].result.value);
|
|
|
|
for (i = 0; i < 4; i++)
|
|
event_close(&events[i]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
return test_harness(per_event_excludes, "per_event_excludes");
|
|
}
|