Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 166911
b: refs/heads/master
c: d175138
h: refs/heads/master
i:
  166909: 8cb536d
  166907: 5ebcc19
  166903: a9d949c
  166895: 668d9a8
  166879: 933e21e
  166847: 4c45688
  166783: ce741aa
  166655: 2adcf33
  166399: 6aa0b1a
  165887: 0599574
v: v3
  • Loading branch information
David S. Miller committed Sep 30, 2009
1 parent 2078f0a commit ef70879
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 18 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 6e804251d119bbd5522d76bdb0f48f5c9a7abf51
refs/heads/master: d17513889a8b754c5872b6b46e6f7822338a0b79
59 changes: 42 additions & 17 deletions trunk/arch/sparc/kernel/perf_event.c
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,8 @@ struct cpu_hw_events {
struct perf_event *events[MAX_HWEVENTS];
unsigned long used_mask[BITS_TO_LONGS(MAX_HWEVENTS)];
unsigned long active_mask[BITS_TO_LONGS(MAX_HWEVENTS)];
int enabled;
u64 pcr;
int enabled;
};
DEFINE_PER_CPU(struct cpu_hw_events, cpu_hw_events) = { .enabled = 1, };

Expand Down Expand Up @@ -464,21 +465,30 @@ static u64 nop_for_index(int idx)
sparc_pmu->lower_nop, idx);
}

static inline void sparc_pmu_enable_event(struct hw_perf_event *hwc, int idx)
static inline void sparc_pmu_enable_event(struct cpu_hw_events *cpuc, struct hw_perf_event *hwc, int idx)
{
u64 val, mask = mask_for_index(idx);

val = pcr_ops->read();
pcr_ops->write((val & ~mask) | hwc->config);
val = cpuc->pcr;
val &= ~mask;
val |= hwc->config;
cpuc->pcr = val;

pcr_ops->write(cpuc->pcr);
}

static inline void sparc_pmu_disable_event(struct hw_perf_event *hwc, int idx)
static inline void sparc_pmu_disable_event(struct cpu_hw_events *cpuc, struct hw_perf_event *hwc, int idx)
{
u64 mask = mask_for_index(idx);
u64 nop = nop_for_index(idx);
u64 val = pcr_ops->read();
u64 val;

pcr_ops->write((val & ~mask) | nop);
val = cpuc->pcr;
val &= ~mask;
val |= nop;
cpuc->pcr = val;

pcr_ops->write(cpuc->pcr);
}

void hw_perf_enable(void)
Expand All @@ -493,7 +503,7 @@ void hw_perf_enable(void)
cpuc->enabled = 1;
barrier();

val = pcr_ops->read();
val = cpuc->pcr;

for (i = 0; i < MAX_HWEVENTS; i++) {
struct perf_event *cp = cpuc->events[i];
Expand All @@ -505,7 +515,9 @@ void hw_perf_enable(void)
val |= hwc->config_base;
}

pcr_ops->write(val);
cpuc->pcr = val;

pcr_ops->write(cpuc->pcr);
}

void hw_perf_disable(void)
Expand All @@ -518,10 +530,12 @@ void hw_perf_disable(void)

cpuc->enabled = 0;

val = pcr_ops->read();
val = cpuc->pcr;
val &= ~(PCR_UTRACE | PCR_STRACE |
sparc_pmu->hv_bit | sparc_pmu->irq_bit);
pcr_ops->write(val);
cpuc->pcr = val;

pcr_ops->write(cpuc->pcr);
}

static u32 read_pmc(int idx)
Expand Down Expand Up @@ -593,13 +607,13 @@ static int sparc_pmu_enable(struct perf_event *event)
if (test_and_set_bit(idx, cpuc->used_mask))
return -EAGAIN;

sparc_pmu_disable_event(hwc, idx);
sparc_pmu_disable_event(cpuc, hwc, idx);

cpuc->events[idx] = event;
set_bit(idx, cpuc->active_mask);

sparc_perf_event_set_period(event, hwc, idx);
sparc_pmu_enable_event(hwc, idx);
sparc_pmu_enable_event(cpuc, hwc, idx);
perf_event_update_userpage(event);
return 0;
}
Expand Down Expand Up @@ -635,7 +649,7 @@ static void sparc_pmu_disable(struct perf_event *event)
int idx = hwc->idx;

clear_bit(idx, cpuc->active_mask);
sparc_pmu_disable_event(hwc, idx);
sparc_pmu_disable_event(cpuc, hwc, idx);

barrier();

Expand All @@ -649,18 +663,29 @@ static void sparc_pmu_disable(struct perf_event *event)
static void sparc_pmu_read(struct perf_event *event)
{
struct hw_perf_event *hwc = &event->hw;

sparc_perf_event_update(event, hwc, hwc->idx);
}

static void sparc_pmu_unthrottle(struct perf_event *event)
{
struct cpu_hw_events *cpuc = &__get_cpu_var(cpu_hw_events);
struct hw_perf_event *hwc = &event->hw;
sparc_pmu_enable_event(hwc, hwc->idx);

sparc_pmu_enable_event(cpuc, hwc, hwc->idx);
}

static atomic_t active_events = ATOMIC_INIT(0);
static DEFINE_MUTEX(pmc_grab_mutex);

static void perf_stop_nmi_watchdog(void *unused)
{
struct cpu_hw_events *cpuc = &__get_cpu_var(cpu_hw_events);

stop_nmi_watchdog(NULL);
cpuc->pcr = pcr_ops->read();
}

void perf_event_grab_pmc(void)
{
if (atomic_inc_not_zero(&active_events))
Expand All @@ -669,7 +694,7 @@ void perf_event_grab_pmc(void)
mutex_lock(&pmc_grab_mutex);
if (atomic_read(&active_events) == 0) {
if (atomic_read(&nmi_active) > 0) {
on_each_cpu(stop_nmi_watchdog, NULL, 1);
on_each_cpu(perf_stop_nmi_watchdog, NULL, 1);
BUG_ON(atomic_read(&nmi_active) != 0);
}
atomic_inc(&active_events);
Expand Down Expand Up @@ -978,7 +1003,7 @@ static int __kprobes perf_event_nmi_handler(struct notifier_block *self,
continue;

if (perf_event_overflow(event, 1, &data, regs))
sparc_pmu_disable_event(hwc, idx);
sparc_pmu_disable_event(cpuc, hwc, idx);
}

return NOTIFY_STOP;
Expand Down

0 comments on commit ef70879

Please sign in to comment.