perf_counter: abstract wakeup flag setting in core to fix powerpc build
authorPaul Mackerras <paulus@samba.org>
Mon, 16 Mar 2009 10:00:00 +0000 (21:00 +1100)
committerIngo Molnar <mingo@elte.hu>
Mon, 6 Apr 2009 07:30:14 +0000 (09:30 +0200)
Impact: build fix for powerpc

Commit bd753921015e7905 ("perf_counter: software counter event
infrastructure") introduced a use of TIF_PERF_COUNTERS into the core
perfcounter code.  This breaks the build on powerpc because we use
a flag in a per-cpu area to signal wakeups on powerpc rather than
a thread_info flag, because the thread_info flags have to be
manipulated with atomic operations and are thus slower than per-cpu
flags.

This fixes the by changing the core to use an abstracted
set_perf_counter_pending() function, which is defined on x86 to set
the TIF_PERF_COUNTERS flag and on powerpc to set the per-cpu flag
(paca->perf_counter_pending).  It changes the previous powerpc
definition of set_perf_counter_pending to not take an argument and
adds a clear_perf_counter_pending, so as to simplify the definition
on x86.

On x86, set_perf_counter_pending() is defined as a macro.  Defining
it as a static inline in arch/x86/include/asm/perf_counters.h causes
compile failures because <asm/perf_counters.h> gets included early in
<linux/sched.h>, and the definitions of set_tsk_thread_flag etc. are
therefore not available in <asm/perf_counters.h>.  (On powerpc this
problem is avoided by defining set_perf_counter_pending etc. in
<asm/hw_irq.h>.)

Signed-off-by: Paul Mackerras <paulus@samba.org>
arch/powerpc/include/asm/hw_irq.h
arch/powerpc/kernel/irq.c
arch/powerpc/kernel/perf_counter.c
arch/x86/include/asm/perf_counter.h
kernel/perf_counter.c

index b43076ff92c9b82baa7a3bfbe35a7e9cbd5c21c1..cb32d571c9c7f2ac557477343890f73ae881c2d5 100644 (file)
@@ -142,10 +142,17 @@ static inline unsigned long get_perf_counter_pending(void)
        return x;
 }
 
-static inline void set_perf_counter_pending(int x)
+static inline void set_perf_counter_pending(void)
 {
        asm volatile("stb %0,%1(13)" : :
-               "r" (x),
+               "r" (1),
+               "i" (offsetof(struct paca_struct, perf_counter_pending)));
+}
+
+static inline void clear_perf_counter_pending(void)
+{
+       asm volatile("stb %0,%1(13)" : :
+               "r" (0),
                "i" (offsetof(struct paca_struct, perf_counter_pending)));
 }
 
@@ -158,7 +165,8 @@ static inline unsigned long get_perf_counter_pending(void)
        return 0;
 }
 
-static inline void set_perf_counter_pending(int x) {}
+static inline void set_perf_counter_pending(void) {}
+static inline void clear_perf_counter_pending(void) {}
 static inline void perf_counter_do_pending(void) {}
 #endif /* CONFIG_PERF_COUNTERS */
 
index 0d2e37c57738f8398e22985c393231e0dc9c0bbf..469e9635ff04b505812805f308190d50e158c59c 100644 (file)
@@ -104,13 +104,6 @@ static inline notrace void set_soft_enabled(unsigned long enable)
        : : "r" (enable), "i" (offsetof(struct paca_struct, soft_enabled)));
 }
 
-#ifdef CONFIG_PERF_COUNTERS
-notrace void __weak perf_counter_do_pending(void)
-{
-       set_perf_counter_pending(0);
-}
-#endif
-
 notrace void raw_local_irq_restore(unsigned long en)
 {
        /*
@@ -142,8 +135,10 @@ notrace void raw_local_irq_restore(unsigned long en)
                        iseries_handle_interrupts();
        }
 
-       if (get_perf_counter_pending())
+       if (get_perf_counter_pending()) {
+               clear_perf_counter_pending();
                perf_counter_do_pending();
+       }
 
        /*
         * if (get_paca()->hard_enabled) return;
index 0e33d27cd464124dbb0c540f49e393b7934846ec..5008762e8bf4c430106e2fb7f1a677050b2a0491 100644 (file)
@@ -653,7 +653,6 @@ void perf_counter_do_pending(void)
        struct cpu_hw_counters *cpuhw = &__get_cpu_var(cpu_hw_counters);
        struct perf_counter *counter;
 
-       set_perf_counter_pending(0);
        for (i = 0; i < cpuhw->n_counters; ++i) {
                counter = cpuhw->counter[i];
                if (counter && counter->wakeup_pending) {
@@ -811,7 +810,7 @@ static void perf_counter_interrupt(struct pt_regs *regs)
                        perf_counter_do_pending();
                        irq_exit();
                } else {
-                       set_perf_counter_pending(1);
+                       set_perf_counter_pending();
                }
        }
 }
index 2e08ed736647eb055613fa244d1ad62257211cff..1662043b340fb527d296d8bd049590c96a04a6db 100644 (file)
@@ -84,6 +84,9 @@ union cpuid10_edx {
 #define MSR_ARCH_PERFMON_FIXED_CTR2                    0x30b
 #define X86_PMC_IDX_FIXED_BUS_CYCLES                   (X86_PMC_IDX_FIXED + 2)
 
+#define set_perf_counter_pending()     \
+               set_tsk_thread_flag(current, TIF_PERF_COUNTERS);
+
 #ifdef CONFIG_PERF_COUNTERS
 extern void init_hw_perf_counters(void);
 extern void perf_counters_lapic_init(int nmi);
index 0018c5e81249e85b423614c35d57cc97cc95307d..b39456ad74a1450b7f2c66bccbb18c8e5e1a60ca 100644 (file)
@@ -1433,7 +1433,7 @@ static void perf_swcounter_interrupt(struct perf_counter *counter,
 
        if (nmi) {
                counter->wakeup_pending = 1;
-               set_tsk_thread_flag(current, TIF_PERF_COUNTERS);
+               set_perf_counter_pending();
        } else
                wake_up(&counter->waitq);
 }