aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVineet Gupta <vgupta@synopsys.com>2014-05-07 05:55:10 -0400
committerVineet Gupta <vgupta@synopsys.com>2014-07-23 01:46:45 -0400
commit2b75c0f93e395aa6130c20a08016b143e6ec8c53 (patch)
treedf879f7748f993f2e11150208e0c9ab01ef78112
parent4c834452aad01531db949414f94f817a86348d59 (diff)
ARC: [SMP] unify cpu private IRQ requests (TIMER/IPI)
The current cpu-private IRQ registration is ugly as it requires need to expose arch_unmask_irq() outside of intc code. So switch to percpu IRQ APIs: -request_percpu_irq [boot core] -enable_percpu_irq [all cores] Encapsulated in helper arc_request_percpu_irq() Signed-off-by: Vineet Gupta <vgupta@synopsys.com>
-rw-r--r--arch/arc/include/asm/irq.h4
-rw-r--r--arch/arc/kernel/irq.c26
-rw-r--r--arch/arc/kernel/smp.c15
-rw-r--r--arch/arc/kernel/time.c18
4 files changed, 37 insertions, 26 deletions
diff --git a/arch/arc/include/asm/irq.h b/arch/arc/include/asm/irq.h
index fb4efb648971..f38652fb2ed7 100644
--- a/arch/arc/include/asm/irq.h
+++ b/arch/arc/include/asm/irq.h
@@ -16,9 +16,13 @@
16#define TIMER0_IRQ 3 16#define TIMER0_IRQ 3
17#define TIMER1_IRQ 4 17#define TIMER1_IRQ 4
18 18
19#include <linux/interrupt.h>
19#include <asm-generic/irq.h> 20#include <asm-generic/irq.h>
20 21
21extern void arc_init_IRQ(void); 22extern void arc_init_IRQ(void);
22void arc_local_timer_setup(void); 23void arc_local_timer_setup(void);
24void arc_request_percpu_irq(int irq, int cpu,
25 irqreturn_t (*isr)(int irq, void *dev),
26 const char *irq_nm, void *percpu_dev);
23 27
24#endif 28#endif
diff --git a/arch/arc/kernel/irq.c b/arch/arc/kernel/irq.c
index 7d653c0d0773..835fa5e71b62 100644
--- a/arch/arc/kernel/irq.c
+++ b/arch/arc/kernel/irq.c
@@ -150,6 +150,32 @@ void arch_do_IRQ(unsigned int irq, struct pt_regs *regs)
150 set_irq_regs(old_regs); 150 set_irq_regs(old_regs);
151} 151}
152 152
153void arc_request_percpu_irq(int irq, int cpu,
154 irqreturn_t (*isr)(int irq, void *dev),
155 const char *irq_nm,
156 void *percpu_dev)
157{
158 /* Boot cpu calls request, all call enable */
159 if (!cpu) {
160 int rc;
161
162 /*
163 * These 2 calls are essential to making percpu IRQ APIs work
164 * Ideally these details could be hidden in irq chip map function
165 * but the issue is IPIs IRQs being static (non-DT) and platform
166 * specific, so we can't identify them there.
167 */
168 irq_set_percpu_devid(irq);
169 irq_modify_status(irq, IRQ_NOAUTOEN, 0); /* @irq, @clr, @set */
170
171 rc = request_percpu_irq(irq, isr, irq_nm, percpu_dev);
172 if (rc)
173 panic("Percpu IRQ request failed for %d\n", irq);
174 }
175
176 enable_percpu_irq(irq, 0);
177}
178
153/* 179/*
154 * arch_local_irq_enable - Enable interrupts. 180 * arch_local_irq_enable - Enable interrupts.
155 * 181 *
diff --git a/arch/arc/kernel/smp.c b/arch/arc/kernel/smp.c
index c802bb500602..1790c21760a1 100644
--- a/arch/arc/kernel/smp.c
+++ b/arch/arc/kernel/smp.c
@@ -136,7 +136,7 @@ void start_kernel_secondary(void)
136 pr_info("## CPU%u LIVE ##: Executing Code...\n", cpu); 136 pr_info("## CPU%u LIVE ##: Executing Code...\n", cpu);
137 137
138 if (machine_desc->init_smp) 138 if (machine_desc->init_smp)
139 machine_desc->init_smp(smp_processor_id()); 139 machine_desc->init_smp(cpu);
140 140
141 arc_local_timer_setup(); 141 arc_local_timer_setup();
142 142
@@ -338,18 +338,11 @@ irqreturn_t do_IPI(int irq, void *dev_id)
338 */ 338 */
339static DEFINE_PER_CPU(int, ipi_dev); 339static DEFINE_PER_CPU(int, ipi_dev);
340 340
341static struct irqaction arc_ipi_irq = {
342 .name = "IPI Interrupt",
343 .flags = IRQF_PERCPU,
344 .handler = do_IPI,
345};
346
347int smp_ipi_irq_setup(int cpu, int irq) 341int smp_ipi_irq_setup(int cpu, int irq)
348{ 342{
349 if (!cpu) 343 int *dev = per_cpu_ptr(&ipi_dev, cpu);
350 return setup_irq(irq, &arc_ipi_irq); 344
351 else 345 arc_request_percpu_irq(irq, cpu, do_IPI, "IPI Interrupt", dev);
352 arch_unmask_irq(irq);
353 346
354 return 0; 347 return 0;
355} 348}
diff --git a/arch/arc/kernel/time.c b/arch/arc/kernel/time.c
index 36c2aa99436f..f92cfb14ecd3 100644
--- a/arch/arc/kernel/time.c
+++ b/arch/arc/kernel/time.c
@@ -210,12 +210,6 @@ static irqreturn_t timer_irq_handler(int irq, void *dev_id)
210 return IRQ_HANDLED; 210 return IRQ_HANDLED;
211} 211}
212 212
213static struct irqaction arc_timer_irq = {
214 .name = "Timer0 (clock-evt-dev)",
215 .flags = IRQF_TIMER | IRQF_PERCPU,
216 .handler = timer_irq_handler,
217};
218
219/* 213/*
220 * Setup the local event timer for @cpu 214 * Setup the local event timer for @cpu
221 */ 215 */
@@ -228,15 +222,9 @@ void arc_local_timer_setup()
228 clockevents_config_and_register(evt, arc_get_core_freq(), 222 clockevents_config_and_register(evt, arc_get_core_freq(),
229 0, ARC_TIMER_MAX); 223 0, ARC_TIMER_MAX);
230 224
231 /* 225 /* setup the per-cpu timer IRQ handler - for all cpus */
232 * setup the per-cpu timer IRQ handler - for all cpus 226 arc_request_percpu_irq(TIMER0_IRQ, cpu, timer_irq_handler,
233 * For non boot CPU explicitly unmask at intc 227 "Timer0 (per-cpu-tick)", evt);
234 * setup_irq() -> .. -> irq_startup() already does this on boot-cpu
235 */
236 if (!cpu)
237 setup_irq(TIMER0_IRQ, &arc_timer_irq);
238 else
239 arch_unmask_irq(TIMER0_IRQ);
240} 228}
241 229
242/* 230/*