ARM: SMP: provide individual IPI interrupt statistics
authorRussell King <rmk+kernel@arm.linux.org.uk>
Mon, 15 Nov 2010 14:40:29 +0000 (14:40 +0000)
committerRussell King <rmk+kernel@arm.linux.org.uk>
Mon, 20 Dec 2010 15:08:01 +0000 (15:08 +0000)
This separates out the individual IPI interrupt counts from the
total IPI count, which allows better visibility of what IPIs are
being used for.

Reviewed-by: Catalin Marinas <catalin.marinas@arm.com>
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
arch/arm/include/asm/hardirq.h
arch/arm/kernel/irq.c
arch/arm/kernel/smp.c

index 824b08aa3f7a3e6cb52b00c88d77a0f9de28ac60..c52e2507c9e02b9e9d8f9300428361ef634304f9 100644 (file)
@@ -5,13 +5,15 @@
 #include <linux/threads.h>
 #include <asm/irq.h>
 
+#define NR_IPI 5
+
 typedef struct {
        unsigned int __softirq_pending;
 #ifdef CONFIG_LOCAL_TIMERS
        unsigned int local_timer_irqs;
 #endif
 #ifdef CONFIG_SMP
-       unsigned int ipi_irqs;
+       unsigned int ipi_irqs[NR_IPI];
 #endif
 } ____cacheline_aligned irq_cpustat_t;
 
index 4e7a7d27221257c2a9916292655af26e32cb6e0c..6276f01df9e4e66d932ecc538322adb0fabc5376 100644 (file)
@@ -62,6 +62,11 @@ int show_interrupts(struct seq_file *p, void *v)
        for (prec = 3, n = 1000; prec < 10 && n <= nr_irqs; prec++)
                n *= 10;
 
+#ifdef CONFIG_SMP
+       if (prec < 4)
+               prec = 4;
+#endif
+
        if (i == 0) {
                char cpuname[12];
 
index 269237ed76a01b05bbfb32d762ab6fe97b3925a0..fa0c5f6e15873eecefc4d0dc568d44b069a8c76f 100644 (file)
@@ -382,16 +382,28 @@ void arch_send_call_function_single_ipi(int cpu)
        smp_cross_call(cpumask_of(cpu), IPI_CALL_FUNC_SINGLE);
 }
 
+static const char *ipi_types[NR_IPI] = {
+#define S(x,s) [x - IPI_TIMER] = s
+       S(IPI_TIMER, "Timer broadcast interrupts"),
+       S(IPI_RESCHEDULE, "Rescheduling interrupts"),
+       S(IPI_CALL_FUNC, "Function call interrupts"),
+       S(IPI_CALL_FUNC_SINGLE, "Single function call interrupts"),
+       S(IPI_CPU_STOP, "CPU stop interrupts"),
+};
+
 void show_ipi_list(struct seq_file *p, int prec)
 {
-       unsigned int cpu;
+       unsigned int cpu, i;
 
-       seq_printf(p, "%*s: ", prec, "IPI");
+       for (i = 0; i < NR_IPI; i++) {
+               seq_printf(p, "%*s%u: ", prec - 1, "IPI", i);
 
-       for_each_present_cpu(cpu)
-               seq_printf(p, "%10u ", __get_irq_stat(cpu, ipi_irqs));
+               for_each_present_cpu(cpu)
+                       seq_printf(p, "%10u ",
+                                  __get_irq_stat(cpu, ipi_irqs[i]));
 
-       seq_printf(p, " Inter-processor interrupts\n");
+               seq_printf(p, " %s\n", ipi_types[i]);
+       }
 }
 
 /*
@@ -506,7 +518,8 @@ asmlinkage void __exception do_IPI(int ipinr, struct pt_regs *regs)
        unsigned int cpu = smp_processor_id();
        struct pt_regs *old_regs = set_irq_regs(regs);
 
-       __inc_irq_stat(cpu, ipi_irqs);
+       if (ipinr >= IPI_TIMER && ipinr < IPI_TIMER + NR_IPI)
+               __inc_irq_stat(cpu, ipi_irqs[ipinr - IPI_TIMER]);
 
        switch (ipinr) {
        case IPI_TIMER: