diff options
author | Chris Metcalf <cmetcalf@tilera.com> | 2010-06-25 16:41:11 -0400 |
---|---|---|
committer | Chris Metcalf <cmetcalf@tilera.com> | 2010-07-06 13:34:01 -0400 |
commit | fb702b942bf638baa6cbbbda9f76794db62921ef (patch) | |
tree | c065b0ab61cbb80b6209c725836a6864624b3c46 /arch/tile/kernel/smp.c | |
parent | de5d9bf6541736dc7ad264d2b5cc99bc1b2ad958 (diff) |
arch/tile: Enable more sophisticated IRQ model for 32-bit chips.
This model is based on the on-chip interrupt model used by the
TILE-Gx next-generation hardware, and interacts much more cleanly
with the Linux generic IRQ layer.
The change includes modifications to the Tilera hypervisor, which
are reflected in the hypervisor headers in arch/tile/include/arch/.
Signed-off-by: Chris Metcalf <cmetcalf@tilera.com>
Acked-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'arch/tile/kernel/smp.c')
-rw-r--r-- | arch/tile/kernel/smp.c | 72 |
1 files changed, 63 insertions, 9 deletions
diff --git a/arch/tile/kernel/smp.c b/arch/tile/kernel/smp.c index 782c1bfa6dfe..1cb5ec79de04 100644 --- a/arch/tile/kernel/smp.c +++ b/arch/tile/kernel/smp.c | |||
@@ -15,10 +15,18 @@ | |||
15 | */ | 15 | */ |
16 | 16 | ||
17 | #include <linux/smp.h> | 17 | #include <linux/smp.h> |
18 | #include <linux/interrupt.h> | ||
19 | #include <linux/io.h> | ||
18 | #include <linux/irq.h> | 20 | #include <linux/irq.h> |
21 | #include <linux/module.h> | ||
19 | #include <asm/cacheflush.h> | 22 | #include <asm/cacheflush.h> |
20 | 23 | ||
21 | HV_Topology smp_topology __write_once; | 24 | HV_Topology smp_topology __write_once; |
25 | EXPORT_SYMBOL(smp_topology); | ||
26 | |||
27 | #if CHIP_HAS_IPI() | ||
28 | static unsigned long __iomem *ipi_mappings[NR_CPUS]; | ||
29 | #endif | ||
22 | 30 | ||
23 | 31 | ||
24 | /* | 32 | /* |
@@ -100,7 +108,6 @@ void on_each_cpu_mask(const struct cpumask *mask, void (*func)(void *), | |||
100 | /* Handler to start the current cpu. */ | 108 | /* Handler to start the current cpu. */ |
101 | static void smp_start_cpu_interrupt(void) | 109 | static void smp_start_cpu_interrupt(void) |
102 | { | 110 | { |
103 | extern unsigned long start_cpu_function_addr; | ||
104 | get_irq_regs()->pc = start_cpu_function_addr; | 111 | get_irq_regs()->pc = start_cpu_function_addr; |
105 | } | 112 | } |
106 | 113 | ||
@@ -174,12 +181,8 @@ void flush_icache_range(unsigned long start, unsigned long end) | |||
174 | } | 181 | } |
175 | 182 | ||
176 | 183 | ||
177 | /* | 184 | /* Called when smp_send_reschedule() triggers IRQ_RESCHEDULE. */ |
178 | * The smp_send_reschedule() path does not use the hv_message_intr() | 185 | static irqreturn_t handle_reschedule_ipi(int irq, void *token) |
179 | * path but instead the faster tile_dev_intr() path for interrupts. | ||
180 | */ | ||
181 | |||
182 | irqreturn_t handle_reschedule_ipi(int irq, void *token) | ||
183 | { | 186 | { |
184 | /* | 187 | /* |
185 | * Nothing to do here; when we return from interrupt, the | 188 | * Nothing to do here; when we return from interrupt, the |
@@ -191,12 +194,63 @@ irqreturn_t handle_reschedule_ipi(int irq, void *token) | |||
191 | return IRQ_HANDLED; | 194 | return IRQ_HANDLED; |
192 | } | 195 | } |
193 | 196 | ||
197 | static struct irqaction resched_action = { | ||
198 | .handler = handle_reschedule_ipi, | ||
199 | .name = "resched", | ||
200 | .dev_id = handle_reschedule_ipi /* unique token */, | ||
201 | }; | ||
202 | |||
203 | void __init ipi_init(void) | ||
204 | { | ||
205 | #if CHIP_HAS_IPI() | ||
206 | int cpu; | ||
207 | /* Map IPI trigger MMIO addresses. */ | ||
208 | for_each_possible_cpu(cpu) { | ||
209 | HV_Coord tile; | ||
210 | HV_PTE pte; | ||
211 | unsigned long offset; | ||
212 | |||
213 | tile.x = cpu_x(cpu); | ||
214 | tile.y = cpu_y(cpu); | ||
215 | if (hv_get_ipi_pte(tile, 1, &pte) != 0) | ||
216 | panic("Failed to initialize IPI for cpu %d\n", cpu); | ||
217 | |||
218 | offset = hv_pte_get_pfn(pte) << PAGE_SHIFT; | ||
219 | ipi_mappings[cpu] = ioremap_prot(offset, PAGE_SIZE, pte); | ||
220 | } | ||
221 | #endif | ||
222 | |||
223 | /* Bind handle_reschedule_ipi() to IRQ_RESCHEDULE. */ | ||
224 | tile_irq_activate(IRQ_RESCHEDULE, TILE_IRQ_PERCPU); | ||
225 | BUG_ON(setup_irq(IRQ_RESCHEDULE, &resched_action)); | ||
226 | } | ||
227 | |||
228 | #if CHIP_HAS_IPI() | ||
229 | |||
230 | void smp_send_reschedule(int cpu) | ||
231 | { | ||
232 | WARN_ON(cpu_is_offline(cpu)); | ||
233 | |||
234 | /* | ||
235 | * We just want to do an MMIO store. The traditional writeq() | ||
236 | * functions aren't really correct here, since they're always | ||
237 | * directed at the PCI shim. For now, just do a raw store, | ||
238 | * casting away the __iomem attribute. | ||
239 | */ | ||
240 | ((unsigned long __force *)ipi_mappings[cpu])[IRQ_RESCHEDULE] = 0; | ||
241 | } | ||
242 | |||
243 | #else | ||
244 | |||
194 | void smp_send_reschedule(int cpu) | 245 | void smp_send_reschedule(int cpu) |
195 | { | 246 | { |
196 | HV_Coord coord; | 247 | HV_Coord coord; |
197 | 248 | ||
198 | WARN_ON(cpu_is_offline(cpu)); | 249 | WARN_ON(cpu_is_offline(cpu)); |
199 | coord.y = cpu / smp_width; | 250 | |
200 | coord.x = cpu % smp_width; | 251 | coord.y = cpu_y(cpu); |
252 | coord.x = cpu_x(cpu); | ||
201 | hv_trigger_ipi(coord, IRQ_RESCHEDULE); | 253 | hv_trigger_ipi(coord, IRQ_RESCHEDULE); |
202 | } | 254 | } |
255 | |||
256 | #endif /* CHIP_HAS_IPI() */ | ||