Commit b0b2072d authored by Stephane Eranian's avatar Stephane Eranian Committed by Ingo Molnar

perf_events: Fix BTS interrupt handling to avoid being dazed by NMI (v2)

Fix a bug introduced with commit de725dec and the change in the
meaning of the return value of intel_pmu_handle_irq(). With the
current code, when you are using the BTS, you get 'dazed by NMI'
each time the BTS buffer fills up.

BTS does interrupt on the PMU vector, thus NMI. You need to take
this into account in the return value of the function.

This version fixes initial patch which was missing changes to
perf_event_intel_ds.c.
Signed-off-by: default avatarStephane Eranian <eranian@google.com>
Acked-by: default avatarDon Zickus <dzickus@redhat.com>
Cc: peterz@infradead.org
Cc: paulus@samba.org
Cc: davem@davemloft.net
Cc: fweisbec@gmail.com
Cc: perfmon2-devel@lists.sf.net
Cc: eranian@gmail.com
Cc: robert.richter@amd.com
LKML-Reference: <4c8a1686.aae9d80a.5aa4.5e35@mx.google.com>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent e5f4d339
...@@ -713,18 +713,18 @@ static int intel_pmu_handle_irq(struct pt_regs *regs) ...@@ -713,18 +713,18 @@ static int intel_pmu_handle_irq(struct pt_regs *regs)
struct cpu_hw_events *cpuc; struct cpu_hw_events *cpuc;
int bit, loops; int bit, loops;
u64 status; u64 status;
int handled = 0; int handled;
perf_sample_data_init(&data, 0); perf_sample_data_init(&data, 0);
cpuc = &__get_cpu_var(cpu_hw_events); cpuc = &__get_cpu_var(cpu_hw_events);
intel_pmu_disable_all(); intel_pmu_disable_all();
intel_pmu_drain_bts_buffer(); handled = intel_pmu_drain_bts_buffer();
status = intel_pmu_get_status(); status = intel_pmu_get_status();
if (!status) { if (!status) {
intel_pmu_enable_all(0); intel_pmu_enable_all(0);
return 0; return handled;
} }
loops = 0; loops = 0;
......
...@@ -214,7 +214,7 @@ static void intel_pmu_disable_bts(void) ...@@ -214,7 +214,7 @@ static void intel_pmu_disable_bts(void)
update_debugctlmsr(debugctlmsr); update_debugctlmsr(debugctlmsr);
} }
static void intel_pmu_drain_bts_buffer(void) static int intel_pmu_drain_bts_buffer(void)
{ {
struct cpu_hw_events *cpuc = &__get_cpu_var(cpu_hw_events); struct cpu_hw_events *cpuc = &__get_cpu_var(cpu_hw_events);
struct debug_store *ds = cpuc->ds; struct debug_store *ds = cpuc->ds;
...@@ -231,16 +231,16 @@ static void intel_pmu_drain_bts_buffer(void) ...@@ -231,16 +231,16 @@ static void intel_pmu_drain_bts_buffer(void)
struct pt_regs regs; struct pt_regs regs;
if (!event) if (!event)
return; return 0;
if (!ds) if (!ds)
return; return 0;
at = (struct bts_record *)(unsigned long)ds->bts_buffer_base; at = (struct bts_record *)(unsigned long)ds->bts_buffer_base;
top = (struct bts_record *)(unsigned long)ds->bts_index; top = (struct bts_record *)(unsigned long)ds->bts_index;
if (top <= at) if (top <= at)
return; return 0;
ds->bts_index = ds->bts_buffer_base; ds->bts_index = ds->bts_buffer_base;
...@@ -256,7 +256,7 @@ static void intel_pmu_drain_bts_buffer(void) ...@@ -256,7 +256,7 @@ static void intel_pmu_drain_bts_buffer(void)
perf_prepare_sample(&header, &data, event, &regs); perf_prepare_sample(&header, &data, event, &regs);
if (perf_output_begin(&handle, event, header.size * (top - at), 1, 1)) if (perf_output_begin(&handle, event, header.size * (top - at), 1, 1))
return; return 1;
for (; at < top; at++) { for (; at < top; at++) {
data.ip = at->from; data.ip = at->from;
...@@ -270,6 +270,7 @@ static void intel_pmu_drain_bts_buffer(void) ...@@ -270,6 +270,7 @@ static void intel_pmu_drain_bts_buffer(void)
/* There's new data available. */ /* There's new data available. */
event->hw.interrupts++; event->hw.interrupts++;
event->pending_kill = POLL_IN; event->pending_kill = POLL_IN;
return 1;
} }
/* /*
......
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