diff options
Diffstat (limited to 'arch/sparc64')
-rw-r--r-- | arch/sparc64/kernel/ebus.c | 65 | ||||
-rw-r--r-- | arch/sparc64/kernel/isa.c | 68 |
2 files changed, 41 insertions, 92 deletions
diff --git a/arch/sparc64/kernel/ebus.c b/arch/sparc64/kernel/ebus.c index 919a91d9e5de..33d79e10c653 100644 --- a/arch/sparc64/kernel/ebus.c +++ b/arch/sparc64/kernel/ebus.c | |||
@@ -283,60 +283,32 @@ static inline void *ebus_alloc(size_t size) | |||
283 | return mem; | 283 | return mem; |
284 | } | 284 | } |
285 | 285 | ||
286 | static void __init ebus_ranges_init(struct linux_ebus *ebus) | 286 | int __init ebus_intmap_match(struct linux_ebus *ebus, |
287 | { | 287 | struct linux_prom_registers *reg, |
288 | struct linux_prom_ebus_ranges *rngs; | 288 | int *interrupt) |
289 | int len; | ||
290 | |||
291 | ebus->num_ebus_ranges = 0; | ||
292 | rngs = of_get_property(ebus->prom_node, "ranges", &len); | ||
293 | if (rngs) { | ||
294 | memcpy(ebus->ebus_ranges, rngs, len); | ||
295 | ebus->num_ebus_ranges = | ||
296 | (len / sizeof(struct linux_prom_ebus_ranges)); | ||
297 | } | ||
298 | } | ||
299 | |||
300 | static void __init ebus_intmap_init(struct linux_ebus *ebus) | ||
301 | { | 289 | { |
302 | struct linux_prom_ebus_intmap *imap; | 290 | struct linux_prom_ebus_intmap *imap; |
303 | struct linux_prom_ebus_intmask *imask; | 291 | struct linux_prom_ebus_intmask *imask; |
304 | int len; | 292 | unsigned int hi, lo, irq; |
293 | int i, len, n_imap; | ||
305 | 294 | ||
306 | ebus->num_ebus_intmap = 0; | ||
307 | imap = of_get_property(ebus->prom_node, "interrupt-map", &len); | 295 | imap = of_get_property(ebus->prom_node, "interrupt-map", &len); |
308 | if (!imap) | 296 | if (!imap) |
309 | return; | 297 | return 0; |
310 | 298 | n_imap = len / sizeof(imap[0]); | |
311 | memcpy(ebus->ebus_intmap, imap, len); | ||
312 | ebus->num_ebus_intmap = (len / sizeof(struct linux_prom_ebus_intmap)); | ||
313 | |||
314 | imask = of_get_property(ebus->prom_node, "interrupt-map-mask", &len); | ||
315 | if (!imask) { | ||
316 | prom_printf("EBUS: can't get interrupt-map-mask\n"); | ||
317 | prom_halt(); | ||
318 | } | ||
319 | memcpy(&ebus->ebus_intmask, imask, sizeof(ebus->ebus_intmask)); | ||
320 | } | ||
321 | |||
322 | int __init ebus_intmap_match(struct linux_ebus *ebus, | ||
323 | struct linux_prom_registers *reg, | ||
324 | int *interrupt) | ||
325 | { | ||
326 | unsigned int hi, lo, irq; | ||
327 | int i; | ||
328 | 299 | ||
329 | if (!ebus->num_ebus_intmap) | 300 | imask = of_get_property(ebus->prom_node, "interrupt-map-mask", NULL); |
301 | if (!imask) | ||
330 | return 0; | 302 | return 0; |
331 | 303 | ||
332 | hi = reg->which_io & ebus->ebus_intmask.phys_hi; | 304 | hi = reg->which_io & imask->phys_hi; |
333 | lo = reg->phys_addr & ebus->ebus_intmask.phys_lo; | 305 | lo = reg->phys_addr & imask->phys_lo; |
334 | irq = *interrupt & ebus->ebus_intmask.interrupt; | 306 | irq = *interrupt & imask->interrupt; |
335 | for (i = 0; i < ebus->num_ebus_intmap; i++) { | 307 | for (i = 0; i < n_imap; i++) { |
336 | if ((ebus->ebus_intmap[i].phys_hi == hi) && | 308 | if ((imap[i].phys_hi == hi) && |
337 | (ebus->ebus_intmap[i].phys_lo == lo) && | 309 | (imap[i].phys_lo == lo) && |
338 | (ebus->ebus_intmap[i].interrupt == irq)) { | 310 | (imap[i].interrupt == irq)) { |
339 | *interrupt = ebus->ebus_intmap[i].cinterrupt; | 311 | *interrupt = imap[i].cinterrupt; |
340 | return 0; | 312 | return 0; |
341 | } | 313 | } |
342 | } | 314 | } |
@@ -598,9 +570,6 @@ void __init ebus_init(void) | |||
598 | ebus->self = pdev; | 570 | ebus->self = pdev; |
599 | ebus->parent = pbm = cookie->pbm; | 571 | ebus->parent = pbm = cookie->pbm; |
600 | 572 | ||
601 | ebus_ranges_init(ebus); | ||
602 | ebus_intmap_init(ebus); | ||
603 | |||
604 | child = dp->child; | 573 | child = dp->child; |
605 | if (!child) | 574 | if (!child) |
606 | goto next_ebus; | 575 | goto next_ebus; |
diff --git a/arch/sparc64/kernel/isa.c b/arch/sparc64/kernel/isa.c index 8c8c5a491ad6..2693f2de549c 100644 --- a/arch/sparc64/kernel/isa.c +++ b/arch/sparc64/kernel/isa.c | |||
@@ -72,19 +72,30 @@ static struct { | |||
72 | static int __init isa_dev_get_irq_using_imap(struct sparc_isa_device *isa_dev, | 72 | static int __init isa_dev_get_irq_using_imap(struct sparc_isa_device *isa_dev, |
73 | struct sparc_isa_bridge *isa_br, | 73 | struct sparc_isa_bridge *isa_br, |
74 | int *interrupt, | 74 | int *interrupt, |
75 | struct linux_prom_registers *pregs) | 75 | struct linux_prom_registers *reg) |
76 | { | 76 | { |
77 | struct linux_prom_ebus_intmap *imap; | ||
78 | struct linux_prom_ebus_intmap *imask; | ||
77 | unsigned int hi, lo, irq; | 79 | unsigned int hi, lo, irq; |
78 | int i; | 80 | int i, len, n_imap; |
79 | 81 | ||
80 | hi = pregs->which_io & isa_br->isa_intmask.phys_hi; | 82 | imap = of_get_property(isa_br->prom_node, "interrupt-map", &len); |
81 | lo = pregs->phys_addr & isa_br->isa_intmask.phys_lo; | 83 | if (!imap) |
82 | irq = *interrupt & isa_br->isa_intmask.interrupt; | 84 | return 0; |
83 | for (i = 0; i < isa_br->num_isa_intmap; i++) { | 85 | n_imap = len / sizeof(imap[0]); |
84 | if ((isa_br->isa_intmap[i].phys_hi == hi) && | 86 | |
85 | (isa_br->isa_intmap[i].phys_lo == lo) && | 87 | imask = of_get_property(isa_br->prom_node, "interrupt-map-mask", NULL); |
86 | (isa_br->isa_intmap[i].interrupt == irq)) { | 88 | if (!imask) |
87 | *interrupt = isa_br->isa_intmap[i].cinterrupt; | 89 | return 0; |
90 | |||
91 | hi = reg->which_io & imask->phys_hi; | ||
92 | lo = reg->phys_addr & imask->phys_lo; | ||
93 | irq = *interrupt & imask->interrupt; | ||
94 | for (i = 0; i < n_imap; i++) { | ||
95 | if ((imap[i].phys_hi == hi) && | ||
96 | (imap[i].phys_lo == lo) && | ||
97 | (imap[i].interrupt == irq)) { | ||
98 | *interrupt = imap[i].cinterrupt; | ||
88 | return 0; | 99 | return 0; |
89 | } | 100 | } |
90 | } | 101 | } |
@@ -105,7 +116,8 @@ static void __init isa_dev_get_irq(struct sparc_isa_device *isa_dev, | |||
105 | struct pci_pbm_info *pbm; | 116 | struct pci_pbm_info *pbm; |
106 | int i; | 117 | int i; |
107 | 118 | ||
108 | if (isa_dev->bus->num_isa_intmap) { | 119 | if (of_find_property(isa_dev->bus->prom_node, |
120 | "interrupt-map", NULL)) { | ||
109 | if (!isa_dev_get_irq_using_imap(isa_dev, | 121 | if (!isa_dev_get_irq_using_imap(isa_dev, |
110 | isa_dev->bus, | 122 | isa_dev->bus, |
111 | &irq_prop, | 123 | &irq_prop, |
@@ -218,36 +230,6 @@ static void __init isa_fill_devices(struct sparc_isa_bridge *isa_br) | |||
218 | } | 230 | } |
219 | } | 231 | } |
220 | 232 | ||
221 | static void __init get_bridge_props(struct sparc_isa_bridge *isa_br) | ||
222 | { | ||
223 | struct device_node *dp = isa_br->prom_node; | ||
224 | void *pval; | ||
225 | int len; | ||
226 | |||
227 | pval = of_get_property(dp, "ranges", &len); | ||
228 | if (pval) { | ||
229 | memcpy(isa_br->isa_ranges, pval, len); | ||
230 | isa_br->num_isa_ranges = | ||
231 | len / sizeof(struct linux_prom_isa_ranges); | ||
232 | } else { | ||
233 | isa_br->num_isa_ranges = 0; | ||
234 | } | ||
235 | |||
236 | pval = of_get_property(dp, "interrupt-map", &len); | ||
237 | if (pval) { | ||
238 | memcpy(isa_br->isa_intmap, pval, len); | ||
239 | isa_br->num_isa_intmap = | ||
240 | (len / sizeof(struct linux_prom_isa_intmap)); | ||
241 | } else { | ||
242 | isa_br->num_isa_intmap = 0; | ||
243 | } | ||
244 | |||
245 | pval = of_get_property(dp, "interrupt-map-mask", &len); | ||
246 | if (pval) | ||
247 | memcpy(&isa_br->isa_intmask, pval, | ||
248 | sizeof(isa_br->isa_intmask)); | ||
249 | } | ||
250 | |||
251 | void __init isa_init(void) | 233 | void __init isa_init(void) |
252 | { | 234 | { |
253 | struct pci_dev *pdev; | 235 | struct pci_dev *pdev; |
@@ -288,8 +270,6 @@ void __init isa_init(void) | |||
288 | isa_br->index = index++; | 270 | isa_br->index = index++; |
289 | isa_br->prom_node = pdev_cookie->prom_node; | 271 | isa_br->prom_node = pdev_cookie->prom_node; |
290 | 272 | ||
291 | get_bridge_props(isa_br); | ||
292 | |||
293 | printk("isa%d:", isa_br->index); | 273 | printk("isa%d:", isa_br->index); |
294 | 274 | ||
295 | isa_fill_devices(isa_br); | 275 | isa_fill_devices(isa_br); |