diff options
author | Vineet Gupta <vgupta@synopsys.com> | 2014-05-07 05:55:10 -0400 |
---|---|---|
committer | Vineet Gupta <vgupta@synopsys.com> | 2014-07-23 01:46:45 -0400 |
commit | 2b75c0f93e395aa6130c20a08016b143e6ec8c53 (patch) | |
tree | df879f7748f993f2e11150208e0c9ab01ef78112 /arch/arc | |
parent | 4c834452aad01531db949414f94f817a86348d59 (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>
Diffstat (limited to 'arch/arc')
-rw-r--r-- | arch/arc/include/asm/irq.h | 4 | ||||
-rw-r--r-- | arch/arc/kernel/irq.c | 26 | ||||
-rw-r--r-- | arch/arc/kernel/smp.c | 15 | ||||
-rw-r--r-- | arch/arc/kernel/time.c | 18 |
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 | ||
21 | extern void arc_init_IRQ(void); | 22 | extern void arc_init_IRQ(void); |
22 | void arc_local_timer_setup(void); | 23 | void arc_local_timer_setup(void); |
24 | void 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 | ||
153 | void 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 | */ |
339 | static DEFINE_PER_CPU(int, ipi_dev); | 339 | static DEFINE_PER_CPU(int, ipi_dev); |
340 | 340 | ||
341 | static struct irqaction arc_ipi_irq = { | ||
342 | .name = "IPI Interrupt", | ||
343 | .flags = IRQF_PERCPU, | ||
344 | .handler = do_IPI, | ||
345 | }; | ||
346 | |||
347 | int smp_ipi_irq_setup(int cpu, int irq) | 341 | int 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 | ||
213 | static 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 | /* |