Skip to content

Commit ca2b497

Browse files
committed
arm64: perf: Reject stand-alone CHAIN events for PMUv3
It doesn't make sense for a perf event to be configured as a CHAIN event in isolation, so extend the arm_pmu structure with a ->filter_match() function to allow the backend PMU implementation to reject CHAIN events early. Cc: <stable@vger.kernel.org> Reviewed-by: Suzuki K Poulose <suzuki.poulose@arm.com> Signed-off-by: Will Deacon <will.deacon@arm.com>
1 parent d91680e commit ca2b497

File tree

3 files changed

+15
-1
lines changed

3 files changed

+15
-1
lines changed

arch/arm64/kernel/perf_event.c

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -966,6 +966,12 @@ static int armv8pmu_set_event_filter(struct hw_perf_event *event,
966966
return 0;
967967
}
968968

969+
static int armv8pmu_filter_match(struct perf_event *event)
970+
{
971+
unsigned long evtype = event->hw.config_base & ARMV8_PMU_EVTYPE_EVENT;
972+
return evtype != ARMV8_PMUV3_PERFCTR_CHAIN;
973+
}
974+
969975
static void armv8pmu_reset(void *info)
970976
{
971977
struct arm_pmu *cpu_pmu = (struct arm_pmu *)info;
@@ -1114,6 +1120,7 @@ static int armv8_pmu_init(struct arm_pmu *cpu_pmu)
11141120
cpu_pmu->stop = armv8pmu_stop,
11151121
cpu_pmu->reset = armv8pmu_reset,
11161122
cpu_pmu->set_event_filter = armv8pmu_set_event_filter;
1123+
cpu_pmu->filter_match = armv8pmu_filter_match;
11171124

11181125
return 0;
11191126
}

drivers/perf/arm_pmu.c

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -485,7 +485,13 @@ static int armpmu_filter_match(struct perf_event *event)
485485
{
486486
struct arm_pmu *armpmu = to_arm_pmu(event->pmu);
487487
unsigned int cpu = smp_processor_id();
488-
return cpumask_test_cpu(cpu, &armpmu->supported_cpus);
488+
int ret;
489+
490+
ret = cpumask_test_cpu(cpu, &armpmu->supported_cpus);
491+
if (ret && armpmu->filter_match)
492+
return armpmu->filter_match(event);
493+
494+
return ret;
489495
}
490496

491497
static ssize_t armpmu_cpumask_show(struct device *dev,

include/linux/perf/arm_pmu.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,7 @@ struct arm_pmu {
9999
void (*stop)(struct arm_pmu *);
100100
void (*reset)(void *);
101101
int (*map_event)(struct perf_event *event);
102+
int (*filter_match)(struct perf_event *event);
102103
int num_events;
103104
bool secure_access; /* 32-bit ARM only */
104105
#define ARMV8_PMUV3_MAX_COMMON_EVENTS 0x40

0 commit comments

Comments
 (0)