RISC-V: implement low-level interrupt handling
authorChristoph Hellwig <hch@lst.de>
Sat, 4 Aug 2018 08:23:16 +0000 (10:23 +0200)
committerPalmer Dabbelt <palmer@sifive.com>
Mon, 13 Aug 2018 15:31:31 +0000 (08:31 -0700)
Add support for a routine that dispatches exceptions with the interrupt
flags set to either the IPI or irqdomain code (and the clock source in the
future).

Loosely based on the irq-riscv-int.c irqchip driver from the RISC-V tree.

Signed-off-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Palmer Dabbelt <palmer@sifive.com>
arch/riscv/kernel/entry.S
arch/riscv/kernel/irq.c

index 9aaf6c98677192ddacbcb2745dd55c8f630b3330..fa2c08e3c05e6ee74ea0258b62cdf6580f98a385 100644 (file)
@@ -168,8 +168,8 @@ ENTRY(handle_exception)
 
        /* Handle interrupts */
        move a0, sp /* pt_regs */
-       REG_L a1, handle_arch_irq
-       jr a1
+       move a1, s4 /* scause */
+       tail do_IRQ
 1:
        /* Exceptions run with interrupts enabled */
        csrs sstatus, SR_SIE
index 7bcdaed15703be6d8f141a8582abd024dc966fec..ab5f3e22c7cc19c3813c7f7452a9af5643361574 100644 (file)
@@ -1,21 +1,55 @@
+// SPDX-License-Identifier: GPL-2.0
 /*
  * Copyright (C) 2012 Regents of the University of California
  * Copyright (C) 2017 SiFive
- *
- *   This program is free software; you can redistribute it and/or
- *   modify it under the terms of the GNU General Public License
- *   as published by the Free Software Foundation, version 2.
- *
- *   This program is distributed in the hope that it will be useful,
- *   but WITHOUT ANY WARRANTY; without even the implied warranty of
- *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
- *   GNU General Public License for more details.
+ * Copyright (C) 2018 Christoph Hellwig
  */
 
 #include <linux/interrupt.h>
 #include <linux/irqchip.h>
 #include <linux/irqdomain.h>
 
+/*
+ * Possible interrupt causes:
+ */
+#define INTERRUPT_CAUSE_SOFTWARE    1
+#define INTERRUPT_CAUSE_TIMER       5
+#define INTERRUPT_CAUSE_EXTERNAL    9
+
+/*
+ * The high order bit of the trap cause register is always set for
+ * interrupts, which allows us to differentiate them from exceptions
+ * quickly.  The INTERRUPT_CAUSE_* macros don't contain that bit, so we
+ * need to mask it off.
+ */
+#define INTERRUPT_CAUSE_FLAG   (1UL << (__riscv_xlen - 1))
+
+asmlinkage void __irq_entry do_IRQ(struct pt_regs *regs, unsigned long cause)
+{
+       struct pt_regs *old_regs = set_irq_regs(regs);
+
+       irq_enter();
+       switch (cause & ~INTERRUPT_CAUSE_FLAG) {
+#ifdef CONFIG_SMP
+       case INTERRUPT_CAUSE_SOFTWARE:
+               /*
+                * We only use software interrupts to pass IPIs, so if a non-SMP
+                * system gets one, then we don't know what to do.
+                */
+               riscv_software_interrupt();
+               break;
+#endif
+       case INTERRUPT_CAUSE_EXTERNAL:
+               handle_arch_irq(regs);
+               break;
+       default:
+               panic("unexpected interrupt cause");
+       }
+       irq_exit();
+
+       set_irq_regs(old_regs);
+}
+
 void __init init_IRQ(void)
 {
        irqchip_init();