[PATCH 1/6] MIPS: Octeon: Enable per-CPU IRQs on all CPUs.
From: David Daney
Date: Thu Jan 06 2011 - 21:36:16 EST
We cannot use on_each_cpu() from low-level irq code, as it ends up
being run with interrupts disabled (a no-no). Instead use some direct
IPI message bits to enable and disable the MIPS CPU interrupts.
Also, enable the irq on all CPUs for MIPS CPU interrupts.
Signed-off-by: David Daney <ddaney@xxxxxxxxxxxxxxxxxx>
Cc: Thomas Gleixner <tglx@xxxxxxxxxxxxx>
---
arch/mips/cavium-octeon/octeon-irq.c | 30 +++++++++++++++++++++++++++---
arch/mips/cavium-octeon/smp.c | 10 ++++++++++
2 files changed, 37 insertions(+), 3 deletions(-)
diff --git a/arch/mips/cavium-octeon/octeon-irq.c b/arch/mips/cavium-octeon/octeon-irq.c
index ce7500c..023cf04 100644
--- a/arch/mips/cavium-octeon/octeon-irq.c
+++ b/arch/mips/cavium-octeon/octeon-irq.c
@@ -56,7 +56,7 @@ static void octeon_irq_core_eoi(unsigned int irq)
set_c0_status(0x100 << bit);
}
-static void octeon_irq_core_enable(unsigned int irq)
+static void octeon_irq_core_enable_local(unsigned int irq)
{
unsigned long flags;
unsigned int bit = irq - OCTEON_IRQ_SW0;
@@ -83,16 +83,40 @@ static void octeon_irq_core_disable_local(unsigned int irq)
local_irq_restore(flags);
}
+extern void octeon_send_ipi_single(int cpu, unsigned int action);
+
static void octeon_irq_core_disable(unsigned int irq)
{
#ifdef CONFIG_SMP
- on_each_cpu((void (*)(void *)) octeon_irq_core_disable_local,
- (void *) (long) irq, 1);
+ unsigned int bit = irq - OCTEON_IRQ_SW0;
+ int cpu;
+ for_each_online_cpu(cpu) {
+ if (cpu == smp_processor_id())
+ octeon_irq_core_disable_local(irq);
+ else
+ octeon_send_ipi_single(cpu, 0x100 << bit);
+ }
#else
octeon_irq_core_disable_local(irq);
#endif
}
+static void octeon_irq_core_enable(unsigned int irq)
+{
+#ifdef CONFIG_SMP
+ unsigned int bit = irq - OCTEON_IRQ_SW0;
+ int cpu;
+ for_each_online_cpu(cpu) {
+ if (cpu == smp_processor_id())
+ octeon_irq_core_enable_local(irq);
+ else
+ octeon_send_ipi_single(cpu, 0x10000 << bit);
+ }
+#else
+ octeon_irq_core_enable_local(irq);
+#endif
+}
+
static struct irq_chip octeon_irq_chip_core = {
.name = "Core",
.enable = octeon_irq_core_enable,
diff --git a/arch/mips/cavium-octeon/smp.c b/arch/mips/cavium-octeon/smp.c
index 391cefe..92d819b 100644
--- a/arch/mips/cavium-octeon/smp.c
+++ b/arch/mips/cavium-octeon/smp.c
@@ -48,6 +48,16 @@ static irqreturn_t mailbox_interrupt(int irq, void *dev_id)
/* Check if we've been told to flush the icache */
if (action & SMP_ICACHE_FLUSH)
asm volatile ("synci 0($0)\n");
+ if (action & 0xff00) {
+ /* Disable MIPS CPU irq*/
+ unsigned int mask = action & 0xff00;
+ clear_c0_status(mask);
+ }
+ if (action & 0xff0000) {
+ /* Enable MIPS CPU irq*/
+ unsigned int mask = (action >> 8) & 0xff00;
+ set_c0_status(mask);
+ }
return IRQ_HANDLED;
}
--
1.7.2.3
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/