Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 | // SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2006,2007 Felix Fietkau <nbd@openwrt.org> * Copyright (C) 2006,2007 Eugene Konev <ejka@openwrt.org> */ #include <linux/interrupt.h> #include <linux/io.h> #include <linux/irq.h> #include <asm/irq_cpu.h> #include <asm/mipsregs.h> #include <asm/mach-ar7/ar7.h> #define EXCEPT_OFFSET 0x80 #define PACE_OFFSET 0xA0 #define CHNLS_OFFSET 0x200 #define REG_OFFSET(irq, reg) ((irq) / 32 * 0x4 + reg * 0x10) #define SEC_REG_OFFSET(reg) (EXCEPT_OFFSET + reg * 0x8) #define SEC_SR_OFFSET (SEC_REG_OFFSET(0)) /* 0x80 */ #define CR_OFFSET(irq) (REG_OFFSET(irq, 1)) /* 0x10 */ #define SEC_CR_OFFSET (SEC_REG_OFFSET(1)) /* 0x88 */ #define ESR_OFFSET(irq) (REG_OFFSET(irq, 2)) /* 0x20 */ #define SEC_ESR_OFFSET (SEC_REG_OFFSET(2)) /* 0x90 */ #define ECR_OFFSET(irq) (REG_OFFSET(irq, 3)) /* 0x30 */ #define SEC_ECR_OFFSET (SEC_REG_OFFSET(3)) /* 0x98 */ #define PIR_OFFSET (0x40) #define MSR_OFFSET (0x44) #define PM_OFFSET(irq) (REG_OFFSET(irq, 5)) /* 0x50 */ #define TM_OFFSET(irq) (REG_OFFSET(irq, 6)) /* 0x60 */ #define REG(addr) ((u32 *)(KSEG1ADDR(AR7_REGS_IRQ) + addr)) #define CHNL_OFFSET(chnl) (CHNLS_OFFSET + (chnl * 4)) static int ar7_irq_base; static void ar7_unmask_irq(struct irq_data *d) { writel(1 << ((d->irq - ar7_irq_base) % 32), REG(ESR_OFFSET(d->irq - ar7_irq_base))); } static void ar7_mask_irq(struct irq_data *d) { writel(1 << ((d->irq - ar7_irq_base) % 32), REG(ECR_OFFSET(d->irq - ar7_irq_base))); } static void ar7_ack_irq(struct irq_data *d) { writel(1 << ((d->irq - ar7_irq_base) % 32), REG(CR_OFFSET(d->irq - ar7_irq_base))); } static void ar7_unmask_sec_irq(struct irq_data *d) { writel(1 << (d->irq - ar7_irq_base - 40), REG(SEC_ESR_OFFSET)); } static void ar7_mask_sec_irq(struct irq_data *d) { writel(1 << (d->irq - ar7_irq_base - 40), REG(SEC_ECR_OFFSET)); } static void ar7_ack_sec_irq(struct irq_data *d) { writel(1 << (d->irq - ar7_irq_base - 40), REG(SEC_CR_OFFSET)); } static struct irq_chip ar7_irq_type = { .name = "AR7", .irq_unmask = ar7_unmask_irq, .irq_mask = ar7_mask_irq, .irq_ack = ar7_ack_irq }; static struct irq_chip ar7_sec_irq_type = { .name = "AR7", .irq_unmask = ar7_unmask_sec_irq, .irq_mask = ar7_mask_sec_irq, .irq_ack = ar7_ack_sec_irq, }; static struct irqaction ar7_cascade_action = { .handler = no_action, .name = "AR7 cascade interrupt", .flags = IRQF_NO_THREAD, }; static void __init ar7_irq_init(int base) { int i; /* * Disable interrupts and clear pending */ writel(0xffffffff, REG(ECR_OFFSET(0))); writel(0xff, REG(ECR_OFFSET(32))); writel(0xffffffff, REG(SEC_ECR_OFFSET)); writel(0xffffffff, REG(CR_OFFSET(0))); writel(0xff, REG(CR_OFFSET(32))); writel(0xffffffff, REG(SEC_CR_OFFSET)); ar7_irq_base = base; for (i = 0; i < 40; i++) { writel(i, REG(CHNL_OFFSET(i))); /* Primary IRQ's */ irq_set_chip_and_handler(base + i, &ar7_irq_type, handle_level_irq); /* Secondary IRQ's */ if (i < 32) irq_set_chip_and_handler(base + i + 40, &ar7_sec_irq_type, handle_level_irq); } setup_irq(2, &ar7_cascade_action); setup_irq(ar7_irq_base, &ar7_cascade_action); set_c0_status(IE_IRQ0); } void __init arch_init_irq(void) { mips_cpu_irq_init(); ar7_irq_init(8); } static void ar7_cascade(void) { u32 status; int i, irq; /* Primary IRQ's */ irq = readl(REG(PIR_OFFSET)) & 0x3f; if (irq) { do_IRQ(ar7_irq_base + irq); return; } /* Secondary IRQ's are cascaded through primary '0' */ writel(1, REG(CR_OFFSET(irq))); status = readl(REG(SEC_SR_OFFSET)); for (i = 0; i < 32; i++) { if (status & 1) { do_IRQ(ar7_irq_base + i + 40); return; } status >>= 1; } spurious_interrupt(); } asmlinkage void plat_irq_dispatch(void) { unsigned int pending = read_c0_status() & read_c0_cause() & ST0_IM; if (pending & STATUSF_IP7) /* cpu timer */ do_IRQ(7); else if (pending & STATUSF_IP2) /* int0 hardware line */ ar7_cascade(); else spurious_interrupt(); } |