Commit 85bbba1c authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'perf_urgent_for_v5.13_rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip

Pull x86 perf fix from Borislav Petkov:
 "Handle power-gating of AMD IOMMU perf counters properly when they are
  used"

* tag 'perf_urgent_for_v5.13_rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip:
  x86/events/amd/iommu: Fix invalid Perf result due to IOMMU PMC power-gating
parents dd3e4012 e10de314
...@@ -19,8 +19,6 @@ ...@@ -19,8 +19,6 @@
#include "../perf_event.h" #include "../perf_event.h"
#include "iommu.h" #include "iommu.h"
#define COUNTER_SHIFT 16
/* iommu pmu conf masks */ /* iommu pmu conf masks */
#define GET_CSOURCE(x) ((x)->conf & 0xFFULL) #define GET_CSOURCE(x) ((x)->conf & 0xFFULL)
#define GET_DEVID(x) (((x)->conf >> 8) & 0xFFFFULL) #define GET_DEVID(x) (((x)->conf >> 8) & 0xFFFFULL)
...@@ -286,22 +284,31 @@ static void perf_iommu_start(struct perf_event *event, int flags) ...@@ -286,22 +284,31 @@ static void perf_iommu_start(struct perf_event *event, int flags)
WARN_ON_ONCE(!(hwc->state & PERF_HES_UPTODATE)); WARN_ON_ONCE(!(hwc->state & PERF_HES_UPTODATE));
hwc->state = 0; hwc->state = 0;
/*
* To account for power-gating, which prevents write to
* the counter, we need to enable the counter
* before setting up counter register.
*/
perf_iommu_enable_event(event);
if (flags & PERF_EF_RELOAD) { if (flags & PERF_EF_RELOAD) {
u64 prev_raw_count = local64_read(&hwc->prev_count); u64 count = 0;
struct amd_iommu *iommu = perf_event_2_iommu(event); struct amd_iommu *iommu = perf_event_2_iommu(event);
/*
* Since the IOMMU PMU only support counting mode,
* the counter always start with value zero.
*/
amd_iommu_pc_set_reg(iommu, hwc->iommu_bank, hwc->iommu_cntr, amd_iommu_pc_set_reg(iommu, hwc->iommu_bank, hwc->iommu_cntr,
IOMMU_PC_COUNTER_REG, &prev_raw_count); IOMMU_PC_COUNTER_REG, &count);
} }
perf_iommu_enable_event(event);
perf_event_update_userpage(event); perf_event_update_userpage(event);
} }
static void perf_iommu_read(struct perf_event *event) static void perf_iommu_read(struct perf_event *event)
{ {
u64 count, prev, delta; u64 count;
struct hw_perf_event *hwc = &event->hw; struct hw_perf_event *hwc = &event->hw;
struct amd_iommu *iommu = perf_event_2_iommu(event); struct amd_iommu *iommu = perf_event_2_iommu(event);
...@@ -312,14 +319,11 @@ static void perf_iommu_read(struct perf_event *event) ...@@ -312,14 +319,11 @@ static void perf_iommu_read(struct perf_event *event)
/* IOMMU pc counter register is only 48 bits */ /* IOMMU pc counter register is only 48 bits */
count &= GENMASK_ULL(47, 0); count &= GENMASK_ULL(47, 0);
prev = local64_read(&hwc->prev_count); /*
if (local64_cmpxchg(&hwc->prev_count, prev, count) != prev) * Since the counter always start with value zero,
return; * simply just accumulate the count for the event.
*/
/* Handle 48-bit counter overflow */ local64_add(count, &event->count);
delta = (count << COUNTER_SHIFT) - (prev << COUNTER_SHIFT);
delta >>= COUNTER_SHIFT;
local64_add(delta, &event->count);
} }
static void perf_iommu_stop(struct perf_event *event, int flags) static void perf_iommu_stop(struct perf_event *event, int flags)
...@@ -329,15 +333,16 @@ static void perf_iommu_stop(struct perf_event *event, int flags) ...@@ -329,15 +333,16 @@ static void perf_iommu_stop(struct perf_event *event, int flags)
if (hwc->state & PERF_HES_UPTODATE) if (hwc->state & PERF_HES_UPTODATE)
return; return;
/*
* To account for power-gating, in which reading the counter would
* return zero, we need to read the register before disabling.
*/
perf_iommu_read(event);
hwc->state |= PERF_HES_UPTODATE;
perf_iommu_disable_event(event); perf_iommu_disable_event(event);
WARN_ON_ONCE(hwc->state & PERF_HES_STOPPED); WARN_ON_ONCE(hwc->state & PERF_HES_STOPPED);
hwc->state |= PERF_HES_STOPPED; hwc->state |= PERF_HES_STOPPED;
if (hwc->state & PERF_HES_UPTODATE)
return;
perf_iommu_read(event);
hwc->state |= PERF_HES_UPTODATE;
} }
static int perf_iommu_add(struct perf_event *event, int flags) static int perf_iommu_add(struct perf_event *event, int flags)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment