aboutsummaryrefslogtreecommitdiffstats
path: root/arch/powerpc/kernel
diff options
context:
space:
mode:
authorPaul Mackerras <paulus@samba.org>2009-03-21 00:31:47 -0400
committerIngo Molnar <mingo@elte.hu>2009-04-06 03:30:18 -0400
commit9aaa131a279834dff75c290c91f0058f62d72d46 (patch)
treed996052d5cdc48b38e300d2b3d87f64cf97ede15 /arch/powerpc/kernel
parentdb4fb5acf20295063d1d5105e67724eb51440207 (diff)
perf_counter: fix type/event_id layout on big-endian systems
Impact: build fix for powerpc Commit db3a944aca35ae61 ("perf_counter: revamp syscall input ABI") expanded the hw_event.type field into a union of structs containing bitfields. In particular it introduced a type field and a raw_type field, with the intention that the 1-bit raw_type field should overlay the most-significant bit of the 8-bit type field, and in fact perf_counter_alloc() now assumes that (or at least, assumes that raw_type doesn't overlay any of the bits that are 1 in the values of PERF_TYPE_{HARDWARE,SOFTWARE,TRACEPOINT}). Unfortunately this is not true on big-endian systems such as PowerPC, where bitfields are laid out from left to right, i.e. from most significant bit to least significant. This means that setting hw_event.type = PERF_TYPE_SOFTWARE will set hw_event.raw_type to 1. This fixes it by making the layout depend on whether or not __BIG_ENDIAN_BITFIELD is defined. It's a bit ugly, but that's what we get for using bitfields in a user/kernel ABI. Also, that commit didn't fix up some places in arch/powerpc/kernel/ perf_counter.c where hw_event.raw and hw_event.event_id were used. This fixes them too. Signed-off-by: Paul Mackerras <paulus@samba.org>
Diffstat (limited to 'arch/powerpc/kernel')
-rw-r--r--arch/powerpc/kernel/perf_counter.c9
1 files changed, 5 insertions, 4 deletions
diff --git a/arch/powerpc/kernel/perf_counter.c b/arch/powerpc/kernel/perf_counter.c
index 830ca9c4494c..6413d9c0313b 100644
--- a/arch/powerpc/kernel/perf_counter.c
+++ b/arch/powerpc/kernel/perf_counter.c
@@ -602,12 +602,13 @@ hw_perf_counter_init(struct perf_counter *counter)
602 return NULL; 602 return NULL;
603 if ((s64)counter->hw_event.irq_period < 0) 603 if ((s64)counter->hw_event.irq_period < 0)
604 return NULL; 604 return NULL;
605 ev = counter->hw_event.event_id; 605 if (!counter->hw_event.raw_type) {
606 if (!counter->hw_event.raw) { 606 ev = counter->hw_event.event_id;
607 if (ev >= ppmu->n_generic || 607 if (ev >= ppmu->n_generic || ppmu->generic_events[ev] == 0)
608 ppmu->generic_events[ev] == 0)
609 return NULL; 608 return NULL;
610 ev = ppmu->generic_events[ev]; 609 ev = ppmu->generic_events[ev];
610 } else {
611 ev = counter->hw_event.raw_event_id;
611 } 612 }
612 counter->hw.config_base = ev; 613 counter->hw.config_base = ev;
613 counter->hw.idx = 0; 614 counter->hw.idx = 0;