blob: 03d85b2b3eeabae9462217f117867516b79e8341 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Steven J. Hill5792bf62014-01-01 16:35:32 +01002 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07006 * Carsten Langgaard, carstenl@mips.com
7 * Copyright (C) 2000, 2001, 2004 MIPS Technologies, Inc.
8 * Copyright (C) 2001 Ralf Baechle
Deng-Cheng Zhu13361132013-10-30 15:52:10 -05009 * Copyright (C) 2013 Imagination Technologies Ltd.
Linus Torvalds1da177e2005-04-16 15:20:36 -070010 *
Linus Torvalds1da177e2005-04-16 15:20:36 -070011 * Routines for generic manipulation of the interrupts found on the MIPS
Steven J. Hill5792bf62014-01-01 16:35:32 +010012 * Malta board. The interrupt controller is located in the South Bridge
13 * a PIIX4 device with two internal 82C95 interrupt controllers.
Linus Torvalds1da177e2005-04-16 15:20:36 -070014 */
15#include <linux/init.h>
16#include <linux/irq.h>
Paul Burton38ec82f2016-09-19 22:21:23 +010017#include <linux/irqchip.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070018#include <linux/sched.h>
Ralf Baechle631330f2009-06-19 14:05:26 +010019#include <linux/smp.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070020#include <linux/interrupt.h>
Dmitri Vorobiev54bf0382008-01-24 19:52:49 +030021#include <linux/io.h>
Paul Burton38ec82f2016-09-19 22:21:23 +010022#include <linux/of_irq.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070023#include <linux/kernel_stat.h>
Ahmed S. Darwish25b8ac32007-02-05 04:42:11 +020024#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070025#include <linux/random.h>
26
Ralf Baechle39b8d522008-04-28 17:14:26 +010027#include <asm/traps.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070028#include <asm/i8259.h>
Ralf Baechlee01402b2005-07-14 15:57:16 +000029#include <asm/irq_cpu.h>
Ralf Baechleba38cdf2006-10-15 09:17:43 +010030#include <asm/irq_regs.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070031#include <asm/mips-boards/malta.h>
32#include <asm/mips-boards/maltaint.h>
Paul Burton72eb2992017-08-12 21:36:34 -070033#include <asm/mips-cps.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070034#include <asm/gt64120.h>
35#include <asm/mips-boards/generic.h>
36#include <asm/mips-boards/msc01_pci.h>
Ralf Baechlee01402b2005-07-14 15:57:16 +000037#include <asm/msc01_ic.h>
David Howellsb81947c2012-03-28 18:30:02 +010038#include <asm/setup.h>
Deng-Cheng Zhu13361132013-10-30 15:52:10 -050039#include <asm/rtlx.h>
Ralf Baechle39b8d522008-04-28 17:14:26 +010040
Linus Torvalds1da177e2005-04-16 15:20:36 -070041static inline int mips_pcibios_iack(void)
42{
43 int irq;
Linus Torvalds1da177e2005-04-16 15:20:36 -070044
45 /*
46 * Determine highest priority pending interrupt by performing
47 * a PCI Interrupt Acknowledge cycle.
48 */
Chris Dearmanb72c0522007-04-27 15:58:41 +010049 switch (mips_revision_sconid) {
50 case MIPS_REVISION_SCON_SOCIT:
51 case MIPS_REVISION_SCON_ROCIT:
52 case MIPS_REVISION_SCON_SOCITSC:
53 case MIPS_REVISION_SCON_SOCITSCP:
Dmitri Vorobievaf825582008-01-24 19:52:45 +030054 MSC_READ(MSC01_PCI_IACK, irq);
Linus Torvalds1da177e2005-04-16 15:20:36 -070055 irq &= 0xff;
56 break;
Chris Dearmanb72c0522007-04-27 15:58:41 +010057 case MIPS_REVISION_SCON_GT64120:
Linus Torvalds1da177e2005-04-16 15:20:36 -070058 irq = GT_READ(GT_PCI0_IACK_OFS);
59 irq &= 0xff;
60 break;
Chris Dearmanb72c0522007-04-27 15:58:41 +010061 case MIPS_REVISION_SCON_BONITO:
Linus Torvalds1da177e2005-04-16 15:20:36 -070062 /* The following will generate a PCI IACK cycle on the
63 * Bonito controller. It's a little bit kludgy, but it
64 * was the easiest way to implement it in hardware at
65 * the given time.
66 */
67 BONITO_PCIMAP_CFG = 0x20000;
68
69 /* Flush Bonito register block */
Ralf Baechle6be63bb2011-03-29 11:48:22 +020070 (void) BONITO_PCIMAP_CFG;
Ralf Baechle70342282013-01-22 12:59:30 +010071 iob(); /* sync */
Linus Torvalds1da177e2005-04-16 15:20:36 -070072
Chris Dearmanaccfd352009-07-10 01:53:54 -070073 irq = __raw_readl((u32 *)_pcictrl_bonito_pcicfg);
Ralf Baechle70342282013-01-22 12:59:30 +010074 iob(); /* sync */
Linus Torvalds1da177e2005-04-16 15:20:36 -070075 irq &= 0xff;
76 BONITO_PCIMAP_CFG = 0;
77 break;
78 default:
Steven J. Hill5792bf62014-01-01 16:35:32 +010079 pr_emerg("Unknown system controller.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -070080 return -1;
81 }
82 return irq;
83}
84
Ralf Baechle937a8012006-10-07 19:44:33 +010085static void corehi_irqdispatch(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -070086{
Ralf Baechle937a8012006-10-07 19:44:33 +010087 unsigned int intedge, intsteer, pcicmd, pcibadaddr;
Dmitri Vorobievaf825582008-01-24 19:52:45 +030088 unsigned int pcimstat, intisr, inten, intpol;
Ralf Baechle21a151d2007-10-11 23:46:15 +010089 unsigned int intrcause, datalo, datahi;
Ralf Baechleba38cdf2006-10-15 09:17:43 +010090 struct pt_regs *regs = get_irq_regs();
Linus Torvalds1da177e2005-04-16 15:20:36 -070091
Steven J. Hill5792bf62014-01-01 16:35:32 +010092 pr_emerg("CoreHI interrupt, shouldn't happen, we die here!\n");
93 pr_emerg("epc : %08lx\nStatus: %08lx\n"
94 "Cause : %08lx\nbadVaddr : %08lx\n",
95 regs->cp0_epc, regs->cp0_status,
96 regs->cp0_cause, regs->cp0_badvaddr);
Ralf Baechlee01402b2005-07-14 15:57:16 +000097
98 /* Read all the registers and then print them as there is a
99 problem with interspersed printk's upsetting the Bonito controller.
100 Do it for the others too.
101 */
102
Chris Dearmanb72c0522007-04-27 15:58:41 +0100103 switch (mips_revision_sconid) {
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300104 case MIPS_REVISION_SCON_SOCIT:
Chris Dearmanb72c0522007-04-27 15:58:41 +0100105 case MIPS_REVISION_SCON_ROCIT:
106 case MIPS_REVISION_SCON_SOCITSC:
107 case MIPS_REVISION_SCON_SOCITSCP:
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300108 ll_msc_irq();
109 break;
110 case MIPS_REVISION_SCON_GT64120:
111 intrcause = GT_READ(GT_INTRCAUSE_OFS);
112 datalo = GT_READ(GT_CPUERR_ADDRLO_OFS);
113 datahi = GT_READ(GT_CPUERR_ADDRHI_OFS);
Steven J. Hill5792bf62014-01-01 16:35:32 +0100114 pr_emerg("GT_INTRCAUSE = %08x\n", intrcause);
115 pr_emerg("GT_CPUERR_ADDR = %02x%08x\n",
Dmitri Vorobiev8216d342008-01-24 19:52:42 +0300116 datahi, datalo);
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300117 break;
118 case MIPS_REVISION_SCON_BONITO:
119 pcibadaddr = BONITO_PCIBADADDR;
120 pcimstat = BONITO_PCIMSTAT;
121 intisr = BONITO_INTISR;
122 inten = BONITO_INTEN;
123 intpol = BONITO_INTPOL;
124 intedge = BONITO_INTEDGE;
125 intsteer = BONITO_INTSTEER;
126 pcicmd = BONITO_PCICMD;
Steven J. Hill5792bf62014-01-01 16:35:32 +0100127 pr_emerg("BONITO_INTISR = %08x\n", intisr);
128 pr_emerg("BONITO_INTEN = %08x\n", inten);
129 pr_emerg("BONITO_INTPOL = %08x\n", intpol);
130 pr_emerg("BONITO_INTEDGE = %08x\n", intedge);
131 pr_emerg("BONITO_INTSTEER = %08x\n", intsteer);
132 pr_emerg("BONITO_PCICMD = %08x\n", pcicmd);
133 pr_emerg("BONITO_PCIBADADDR = %08x\n", pcibadaddr);
134 pr_emerg("BONITO_PCIMSTAT = %08x\n", pcimstat);
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300135 break;
136 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700137
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300138 die("CoreHi interrupt", regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700139}
140
Andrew Bresticker18743d22014-09-18 14:47:24 -0700141static irqreturn_t corehi_handler(int irq, void *dev_id)
142{
143 corehi_irqdispatch();
144 return IRQ_HANDLED;
145}
146
Steven J. Hill5792bf62014-01-01 16:35:32 +0100147static msc_irqmap_t msc_irqmap[] __initdata = {
Ralf Baechlee01402b2005-07-14 15:57:16 +0000148 {MSC01C_INT_TMR, MSC01_IRQ_EDGE, 0},
149 {MSC01C_INT_PCI, MSC01_IRQ_LEVEL, 0},
150};
Steven J. Hill5792bf62014-01-01 16:35:32 +0100151static int msc_nr_irqs __initdata = ARRAY_SIZE(msc_irqmap);
Ralf Baechlee01402b2005-07-14 15:57:16 +0000152
Steven J. Hill5792bf62014-01-01 16:35:32 +0100153static msc_irqmap_t msc_eicirqmap[] __initdata = {
Ralf Baechlee01402b2005-07-14 15:57:16 +0000154 {MSC01E_INT_SW0, MSC01_IRQ_LEVEL, 0},
155 {MSC01E_INT_SW1, MSC01_IRQ_LEVEL, 0},
156 {MSC01E_INT_I8259A, MSC01_IRQ_LEVEL, 0},
157 {MSC01E_INT_SMI, MSC01_IRQ_LEVEL, 0},
158 {MSC01E_INT_COREHI, MSC01_IRQ_LEVEL, 0},
159 {MSC01E_INT_CORELO, MSC01_IRQ_LEVEL, 0},
160 {MSC01E_INT_TMR, MSC01_IRQ_EDGE, 0},
161 {MSC01E_INT_PCI, MSC01_IRQ_LEVEL, 0},
162 {MSC01E_INT_PERFCTR, MSC01_IRQ_LEVEL, 0},
163 {MSC01E_INT_CPUCTR, MSC01_IRQ_LEVEL, 0}
164};
Ralf Baechle39b8d522008-04-28 17:14:26 +0100165
Steven J. Hill5792bf62014-01-01 16:35:32 +0100166static int msc_nr_eicirqs __initdata = ARRAY_SIZE(msc_eicirqmap);
Ralf Baechlee01402b2005-07-14 15:57:16 +0000167
Linus Torvalds1da177e2005-04-16 15:20:36 -0700168void __init arch_init_irq(void)
169{
Paul Burton38ec82f2016-09-19 22:21:23 +0100170 int corehi_irq;
Andrew Bresticker18743d22014-09-18 14:47:24 -0700171
Matt Redfearn9eec1c02017-04-06 16:58:09 +0100172 /*
173 * Preallocate the i8259's expected virq's here. Since irqchip_init()
174 * will probe the irqchips in hierarchial order, i8259 is probed last.
175 * If anything allocates a virq before the i8259 is probed, it will
176 * be given one of the i8259's expected range and consequently setup
177 * of the i8259 will fail.
178 */
179 WARN(irq_alloc_descs(I8259A_IRQ_BASE, I8259A_IRQ_BASE,
180 16, numa_node_id()) < 0,
181 "Cannot reserve i8259 virqs at IRQ%d\n", I8259A_IRQ_BASE);
182
Paul Burton38ec82f2016-09-19 22:21:23 +0100183 i8259_set_poll(mips_pcibios_iack);
184 irqchip_init();
Ralf Baechle39b8d522008-04-28 17:14:26 +0100185
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300186 switch (mips_revision_sconid) {
187 case MIPS_REVISION_SCON_SOCIT:
188 case MIPS_REVISION_SCON_ROCIT:
Ralf Baechlee01402b2005-07-14 15:57:16 +0000189 if (cpu_has_veic)
Dmitri Vorobievf8071492008-01-24 19:52:47 +0300190 init_msc_irqs(MIPS_MSC01_IC_REG_BASE,
191 MSC01E_INT_BASE, msc_eicirqmap,
192 msc_nr_eicirqs);
Ralf Baechlee01402b2005-07-14 15:57:16 +0000193 else
Dmitri Vorobievf8071492008-01-24 19:52:47 +0300194 init_msc_irqs(MIPS_MSC01_IC_REG_BASE,
195 MSC01C_INT_BASE, msc_irqmap,
196 msc_nr_irqs);
Chris Dearmand725cf32007-05-08 14:05:39 +0100197 break;
198
Dmitri Vorobievaf825582008-01-24 19:52:45 +0300199 case MIPS_REVISION_SCON_SOCITSC:
200 case MIPS_REVISION_SCON_SOCITSCP:
Chris Dearmand725cf32007-05-08 14:05:39 +0100201 if (cpu_has_veic)
Dmitri Vorobievf8071492008-01-24 19:52:47 +0300202 init_msc_irqs(MIPS_SOCITSC_IC_REG_BASE,
203 MSC01E_INT_BASE, msc_eicirqmap,
204 msc_nr_eicirqs);
Chris Dearmand725cf32007-05-08 14:05:39 +0100205 else
Dmitri Vorobievf8071492008-01-24 19:52:47 +0300206 init_msc_irqs(MIPS_SOCITSC_IC_REG_BASE,
207 MSC01C_INT_BASE, msc_irqmap,
208 msc_nr_irqs);
Ralf Baechlee01402b2005-07-14 15:57:16 +0000209 }
210
Paul Burton72eb2992017-08-12 21:36:34 -0700211 if (mips_gic_present()) {
Andrew Bresticker18743d22014-09-18 14:47:24 -0700212 corehi_irq = MIPS_CPU_IRQ_BASE + MIPSCPU_INT_COREHI;
Paul Burton1eed4002017-03-30 12:06:12 -0700213 } else if (cpu_has_veic) {
214 set_vi_handler(MSC01E_INT_COREHI, corehi_irqdispatch);
215 corehi_irq = MSC01E_INT_BASE + MSC01E_INT_COREHI;
Ralf Baechle39b8d522008-04-28 17:14:26 +0100216 } else {
Paul Burton1eed4002017-03-30 12:06:12 -0700217 corehi_irq = MIPS_CPU_IRQ_BASE + MIPSCPU_INT_COREHI;
Chris Dearman7098f742009-07-10 01:54:09 -0700218 }
Andrew Bresticker18743d22014-09-18 14:47:24 -0700219
afzal mohammedac8fd122020-03-05 17:27:53 +0530220 if (request_irq(corehi_irq, corehi_handler, IRQF_NO_THREAD, "CoreHi",
221 NULL))
222 pr_err("Failed to request irq %d (CoreHi)\n", corehi_irq);
Ralf Baechle39b8d522008-04-28 17:14:26 +0100223}