diff options
Diffstat (limited to 'arch/mips/kernel/irq-gic.c')
| -rw-r--r-- | arch/mips/kernel/irq-gic.c | 114 |
1 files changed, 43 insertions, 71 deletions
diff --git a/arch/mips/kernel/irq-gic.c b/arch/mips/kernel/irq-gic.c index d2072cd38592..b181f2f0ea8e 100644 --- a/arch/mips/kernel/irq-gic.c +++ b/arch/mips/kernel/irq-gic.c | |||
| @@ -14,38 +14,23 @@ | |||
| 14 | 14 | ||
| 15 | 15 | ||
| 16 | static unsigned long _gic_base; | 16 | static unsigned long _gic_base; |
| 17 | static unsigned int _irqbase, _mapsize, numvpes, numintrs; | 17 | static unsigned int _irqbase; |
| 18 | static struct gic_intr_map *_intrmap; | 18 | static unsigned int gic_irq_flags[GIC_NUM_INTRS]; |
| 19 | #define GIC_IRQ_FLAG_EDGE 0x0001 | ||
| 19 | 20 | ||
| 20 | static struct gic_pcpu_mask pcpu_masks[NR_CPUS]; | 21 | struct gic_pcpu_mask pcpu_masks[NR_CPUS]; |
| 21 | static struct gic_pending_regs pending_regs[NR_CPUS]; | 22 | static struct gic_pending_regs pending_regs[NR_CPUS]; |
| 22 | static struct gic_intrmask_regs intrmask_regs[NR_CPUS]; | 23 | static struct gic_intrmask_regs intrmask_regs[NR_CPUS]; |
| 23 | 24 | ||
| 24 | #define gic_wedgeb2bok 0 /* | ||
| 25 | * Can GIC handle b2b writes to wedge register? | ||
| 26 | */ | ||
| 27 | #if gic_wedgeb2bok == 0 | ||
| 28 | static DEFINE_SPINLOCK(gic_wedgeb2b_lock); | ||
| 29 | #endif | ||
| 30 | |||
| 31 | void gic_send_ipi(unsigned int intr) | 25 | void gic_send_ipi(unsigned int intr) |
| 32 | { | 26 | { |
| 33 | #if gic_wedgeb2bok == 0 | ||
| 34 | unsigned long flags; | ||
| 35 | #endif | ||
| 36 | pr_debug("CPU%d: %s status %08x\n", smp_processor_id(), __func__, | 27 | pr_debug("CPU%d: %s status %08x\n", smp_processor_id(), __func__, |
| 37 | read_c0_status()); | 28 | read_c0_status()); |
| 38 | if (!gic_wedgeb2bok) | ||
| 39 | spin_lock_irqsave(&gic_wedgeb2b_lock, flags); | ||
| 40 | GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), 0x80000000 | intr); | 29 | GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), 0x80000000 | intr); |
| 41 | if (!gic_wedgeb2bok) { | ||
| 42 | (void) GIC_REG(SHARED, GIC_SH_CONFIG); | ||
| 43 | spin_unlock_irqrestore(&gic_wedgeb2b_lock, flags); | ||
| 44 | } | ||
| 45 | } | 30 | } |
| 46 | 31 | ||
| 47 | /* This is Malta specific and needs to be exported */ | 32 | /* This is Malta specific and needs to be exported */ |
| 48 | static void vpe_local_setup(unsigned int numvpes) | 33 | static void __init vpe_local_setup(unsigned int numvpes) |
| 49 | { | 34 | { |
| 50 | int i; | 35 | int i; |
| 51 | unsigned long timer_interrupt = 5, perf_interrupt = 5; | 36 | unsigned long timer_interrupt = 5, perf_interrupt = 5; |
| @@ -105,44 +90,34 @@ unsigned int gic_get_int(void) | |||
| 105 | 90 | ||
| 106 | static unsigned int gic_irq_startup(unsigned int irq) | 91 | static unsigned int gic_irq_startup(unsigned int irq) |
| 107 | { | 92 | { |
| 108 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); | ||
| 109 | irq -= _irqbase; | 93 | irq -= _irqbase; |
| 110 | GIC_SET_INTR_MASK(irq, 1); | 94 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); |
| 95 | GIC_SET_INTR_MASK(irq); | ||
| 111 | return 0; | 96 | return 0; |
| 112 | } | 97 | } |
| 113 | 98 | ||
| 114 | static void gic_irq_ack(unsigned int irq) | 99 | static void gic_irq_ack(unsigned int irq) |
| 115 | { | 100 | { |
| 116 | #if gic_wedgeb2bok == 0 | ||
| 117 | unsigned long flags; | ||
| 118 | #endif | ||
| 119 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); | ||
| 120 | irq -= _irqbase; | 101 | irq -= _irqbase; |
| 121 | GIC_CLR_INTR_MASK(irq, 1); | 102 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); |
| 103 | GIC_CLR_INTR_MASK(irq); | ||
| 122 | 104 | ||
| 123 | if (_intrmap[irq].trigtype == GIC_TRIG_EDGE) { | 105 | if (gic_irq_flags[irq] & GIC_IRQ_FLAG_EDGE) |
| 124 | if (!gic_wedgeb2bok) | ||
| 125 | spin_lock_irqsave(&gic_wedgeb2b_lock, flags); | ||
| 126 | GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq); | 106 | GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq); |
| 127 | if (!gic_wedgeb2bok) { | ||
| 128 | (void) GIC_REG(SHARED, GIC_SH_CONFIG); | ||
| 129 | spin_unlock_irqrestore(&gic_wedgeb2b_lock, flags); | ||
| 130 | } | ||
| 131 | } | ||
| 132 | } | 107 | } |
| 133 | 108 | ||
| 134 | static void gic_mask_irq(unsigned int irq) | 109 | static void gic_mask_irq(unsigned int irq) |
| 135 | { | 110 | { |
| 136 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); | ||
| 137 | irq -= _irqbase; | 111 | irq -= _irqbase; |
| 138 | GIC_CLR_INTR_MASK(irq, 1); | 112 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); |
| 113 | GIC_CLR_INTR_MASK(irq); | ||
| 139 | } | 114 | } |
| 140 | 115 | ||
| 141 | static void gic_unmask_irq(unsigned int irq) | 116 | static void gic_unmask_irq(unsigned int irq) |
| 142 | { | 117 | { |
| 143 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); | ||
| 144 | irq -= _irqbase; | 118 | irq -= _irqbase; |
| 145 | GIC_SET_INTR_MASK(irq, 1); | 119 | pr_debug("CPU%d: %s: irq%d\n", smp_processor_id(), __func__, irq); |
| 120 | GIC_SET_INTR_MASK(irq); | ||
| 146 | } | 121 | } |
| 147 | 122 | ||
| 148 | #ifdef CONFIG_SMP | 123 | #ifdef CONFIG_SMP |
| @@ -155,9 +130,8 @@ static int gic_set_affinity(unsigned int irq, const struct cpumask *cpumask) | |||
| 155 | unsigned long flags; | 130 | unsigned long flags; |
| 156 | int i; | 131 | int i; |
| 157 | 132 | ||
| 158 | pr_debug(KERN_DEBUG "%s called\n", __func__); | ||
| 159 | irq -= _irqbase; | 133 | irq -= _irqbase; |
| 160 | 134 | pr_debug(KERN_DEBUG "%s(%d) called\n", __func__, irq); | |
| 161 | cpumask_and(&tmp, cpumask, cpu_online_mask); | 135 | cpumask_and(&tmp, cpumask, cpu_online_mask); |
| 162 | if (cpus_empty(tmp)) | 136 | if (cpus_empty(tmp)) |
| 163 | return -1; | 137 | return -1; |
| @@ -168,13 +142,6 @@ static int gic_set_affinity(unsigned int irq, const struct cpumask *cpumask) | |||
| 168 | /* Re-route this IRQ */ | 142 | /* Re-route this IRQ */ |
| 169 | GIC_SH_MAP_TO_VPE_SMASK(irq, first_cpu(tmp)); | 143 | GIC_SH_MAP_TO_VPE_SMASK(irq, first_cpu(tmp)); |
| 170 | 144 | ||
| 171 | /* | ||
| 172 | * FIXME: assumption that _intrmap is ordered and has no holes | ||
| 173 | */ | ||
| 174 | |||
| 175 | /* Update the intr_map */ | ||
| 176 | _intrmap[irq].cpunum = first_cpu(tmp); | ||
| 177 | |||
| 178 | /* Update the pcpu_masks */ | 145 | /* Update the pcpu_masks */ |
| 179 | for (i = 0; i < NR_CPUS; i++) | 146 | for (i = 0; i < NR_CPUS; i++) |
| 180 | clear_bit(irq, pcpu_masks[i].pcpu_mask); | 147 | clear_bit(irq, pcpu_masks[i].pcpu_mask); |
| @@ -201,8 +168,9 @@ static struct irq_chip gic_irq_controller = { | |||
| 201 | #endif | 168 | #endif |
| 202 | }; | 169 | }; |
| 203 | 170 | ||
| 204 | static void __init setup_intr(unsigned int intr, unsigned int cpu, | 171 | static void __init gic_setup_intr(unsigned int intr, unsigned int cpu, |
| 205 | unsigned int pin, unsigned int polarity, unsigned int trigtype) | 172 | unsigned int pin, unsigned int polarity, unsigned int trigtype, |
| 173 | unsigned int flags) | ||
| 206 | { | 174 | { |
| 207 | /* Setup Intr to Pin mapping */ | 175 | /* Setup Intr to Pin mapping */ |
| 208 | if (pin & GIC_MAP_TO_NMI_MSK) { | 176 | if (pin & GIC_MAP_TO_NMI_MSK) { |
| @@ -227,38 +195,43 @@ static void __init setup_intr(unsigned int intr, unsigned int cpu, | |||
| 227 | GIC_SET_TRIGGER(intr, trigtype); | 195 | GIC_SET_TRIGGER(intr, trigtype); |
| 228 | 196 | ||
| 229 | /* Init Intr Masks */ | 197 | /* Init Intr Masks */ |
| 230 | GIC_SET_INTR_MASK(intr, 0); | 198 | GIC_CLR_INTR_MASK(intr); |
| 199 | /* Initialise per-cpu Interrupt software masks */ | ||
| 200 | if (flags & GIC_FLAG_IPI) | ||
| 201 | set_bit(intr, pcpu_masks[cpu].pcpu_mask); | ||
| 202 | if (flags & GIC_FLAG_TRANSPARENT) | ||
| 203 | GIC_SET_INTR_MASK(intr); | ||
| 204 | if (trigtype == GIC_TRIG_EDGE) | ||
| 205 | gic_irq_flags[intr] |= GIC_IRQ_FLAG_EDGE; | ||
| 231 | } | 206 | } |
| 232 | 207 | ||
| 233 | static void __init gic_basic_init(void) | 208 | static void __init gic_basic_init(int numintrs, int numvpes, |
| 209 | struct gic_intr_map *intrmap, int mapsize) | ||
| 234 | { | 210 | { |
| 235 | unsigned int i, cpu; | 211 | unsigned int i, cpu; |
| 236 | 212 | ||
| 237 | /* Setup defaults */ | 213 | /* Setup defaults */ |
| 238 | for (i = 0; i < GIC_NUM_INTRS; i++) { | 214 | for (i = 0; i < numintrs; i++) { |
| 239 | GIC_SET_POLARITY(i, GIC_POL_POS); | 215 | GIC_SET_POLARITY(i, GIC_POL_POS); |
| 240 | GIC_SET_TRIGGER(i, GIC_TRIG_LEVEL); | 216 | GIC_SET_TRIGGER(i, GIC_TRIG_LEVEL); |
| 241 | GIC_SET_INTR_MASK(i, 0); | 217 | GIC_CLR_INTR_MASK(i); |
| 218 | if (i < GIC_NUM_INTRS) | ||
| 219 | gic_irq_flags[i] = 0; | ||
| 242 | } | 220 | } |
| 243 | 221 | ||
| 244 | /* Setup specifics */ | 222 | /* Setup specifics */ |
| 245 | for (i = 0; i < _mapsize; i++) { | 223 | for (i = 0; i < mapsize; i++) { |
| 246 | cpu = _intrmap[i].cpunum; | 224 | cpu = intrmap[i].cpunum; |
| 247 | if (cpu == X) | 225 | if (cpu == X) |
| 248 | continue; | 226 | continue; |
| 249 | 227 | if (cpu == 0 && i != 0 && intrmap[i].flags == 0) | |
| 250 | if (cpu == 0 && i != 0 && _intrmap[i].intrnum == 0 && | ||
| 251 | _intrmap[i].ipiflag == 0) | ||
| 252 | continue; | 228 | continue; |
| 253 | 229 | gic_setup_intr(i, | |
| 254 | setup_intr(_intrmap[i].intrnum, | 230 | intrmap[i].cpunum, |
| 255 | _intrmap[i].cpunum, | 231 | intrmap[i].pin, |
| 256 | _intrmap[i].pin, | 232 | intrmap[i].polarity, |
| 257 | _intrmap[i].polarity, | 233 | intrmap[i].trigtype, |
| 258 | _intrmap[i].trigtype); | 234 | intrmap[i].flags); |
| 259 | /* Initialise per-cpu Interrupt software masks */ | ||
| 260 | if (_intrmap[i].ipiflag) | ||
| 261 | set_bit(_intrmap[i].intrnum, pcpu_masks[cpu].pcpu_mask); | ||
| 262 | } | 235 | } |
| 263 | 236 | ||
| 264 | vpe_local_setup(numvpes); | 237 | vpe_local_setup(numvpes); |
| @@ -273,12 +246,11 @@ void __init gic_init(unsigned long gic_base_addr, | |||
| 273 | unsigned int irqbase) | 246 | unsigned int irqbase) |
| 274 | { | 247 | { |
| 275 | unsigned int gicconfig; | 248 | unsigned int gicconfig; |
| 249 | int numvpes, numintrs; | ||
| 276 | 250 | ||
| 277 | _gic_base = (unsigned long) ioremap_nocache(gic_base_addr, | 251 | _gic_base = (unsigned long) ioremap_nocache(gic_base_addr, |
| 278 | gic_addrspace_size); | 252 | gic_addrspace_size); |
| 279 | _irqbase = irqbase; | 253 | _irqbase = irqbase; |
| 280 | _intrmap = intr_map; | ||
| 281 | _mapsize = intr_map_size; | ||
| 282 | 254 | ||
| 283 | GICREAD(GIC_REG(SHARED, GIC_SH_CONFIG), gicconfig); | 255 | GICREAD(GIC_REG(SHARED, GIC_SH_CONFIG), gicconfig); |
| 284 | numintrs = (gicconfig & GIC_SH_CONFIG_NUMINTRS_MSK) >> | 256 | numintrs = (gicconfig & GIC_SH_CONFIG_NUMINTRS_MSK) >> |
| @@ -290,5 +262,5 @@ void __init gic_init(unsigned long gic_base_addr, | |||
| 290 | 262 | ||
| 291 | pr_debug("%s called\n", __func__); | 263 | pr_debug("%s called\n", __func__); |
| 292 | 264 | ||
| 293 | gic_basic_init(); | 265 | gic_basic_init(numintrs, numvpes, intr_map, intr_map_size); |
| 294 | } | 266 | } |
