Commit 21acec08 authored by David S. Miller's avatar David S. Miller

[SPARC64]: In ISA support, is interrupt-map exists use it.

parent 3de143ef
...@@ -20,22 +20,24 @@ static void __init report_dev(struct sparc_isa_device *isa_dev, int child) ...@@ -20,22 +20,24 @@ static void __init report_dev(struct sparc_isa_device *isa_dev, int child)
printk(" [%s", isa_dev->prom_name); printk(" [%s", isa_dev->prom_name);
} }
static void __init isa_dev_get_resource(struct sparc_isa_device *isa_dev) static void __init isa_dev_get_resource(struct sparc_isa_device *isa_dev,
struct linux_prom_registers *pregs,
int pregs_size)
{ {
struct linux_prom_registers regs[PROMREG_MAX]; struct linux_prom_registers regs[PROMREG_MAX];
unsigned long base, len; unsigned long base, len;
int prop_len; int prop_len;
prop_len = prom_getproperty(isa_dev->prom_node, "reg", prop_len = prom_getproperty(isa_dev->prom_node, "reg",
(char *) regs, sizeof(regs)); (char *) pregs, pregs_size);
if (prop_len <= 0) if (prop_len <= 0)
return; return;
/* Only the first one is interesting. */ /* Only the first one is interesting. */
len = regs[0].reg_size; len = pregs[0].reg_size;
base = (((unsigned long)regs[0].which_io << 32) | base = (((unsigned long)pregs[0].which_io << 32) |
(unsigned long)regs[0].phys_addr); (unsigned long)pregs[0].phys_addr);
base += isa_dev->bus->parent->io_space.start; base += isa_dev->bus->parent->io_space.start;
isa_dev->resource.start = base; isa_dev->resource.start = base;
...@@ -53,6 +55,9 @@ static void __init isa_dev_get_resource(struct sparc_isa_device *isa_dev) ...@@ -53,6 +55,9 @@ static void __init isa_dev_get_resource(struct sparc_isa_device *isa_dev)
* *
* The P1275 standard for ISA devices seems to also have been * The P1275 standard for ISA devices seems to also have been
* totally ignored. * totally ignored.
*
* On later systems, an interrupt-map and interrupt-map-mask scheme
* akin to EBUS is used.
*/ */
static struct { static struct {
int obp_irq; int obp_irq;
...@@ -67,33 +72,72 @@ static struct { ...@@ -67,33 +72,72 @@ static struct {
{ 0, 0x00 } /* end of table */ { 0, 0x00 } /* end of table */
}; };
static void __init isa_dev_get_irq(struct sparc_isa_device *isa_dev) static int __init isa_dev_get_irq_using_imap(struct sparc_isa_device *isa_dev,
struct sparc_isa_bridge *isa_br,
int *interrupt,
struct linux_prom_registers *pregs)
{
unsigned int hi, lo, irq;
int i;
hi = pregs->which_io & isa_br->isa_intmask.phys_hi;
lo = pregs->phys_addr & isa_br->isa_intmask.phys_lo;
irq = *interrupt & isa_br->isa_intmask.interrupt;
for (i = 0; i < isa_br->num_isa_intmap; i++) {
if ((isa_br->isa_intmap[i].phys_hi == hi) &&
(isa_br->isa_intmap[i].phys_lo == lo) &&
(isa_br->isa_intmap[i].interrupt == irq)) {
*interrupt = isa_br->isa_intmap[i].cinterrupt;
return 0;
}
}
return -1;
}
static void __init isa_dev_get_irq(struct sparc_isa_device *isa_dev,
struct linux_prom_registers *pregs)
{ {
int irq_prop; int irq_prop;
irq_prop = prom_getintdefault(isa_dev->prom_node, irq_prop = prom_getintdefault(isa_dev->prom_node,
"interrupts", -1); "interrupts", -1);
if (irq_prop <= 0) { if (irq_prop <= 0) {
isa_dev->irq = PCI_IRQ_NONE; goto no_irq;
} else { } else {
struct pci_controller_info *pcic;
struct pci_pbm_info *pbm;
int i; int i;
if (isa_dev->bus->num_isa_intmap) {
if (!isa_dev_get_irq_using_imap(isa_dev,
isa_dev->bus,
&irq_prop,
pregs))
goto route_irq;
}
for (i = 0; grover_irq_table[i].obp_irq != 0; i++) { for (i = 0; grover_irq_table[i].obp_irq != 0; i++) {
if (grover_irq_table[i].obp_irq == irq_prop) { if (grover_irq_table[i].obp_irq == irq_prop) {
struct pci_controller_info *pcic;
struct pci_pbm_info *pbm;
int ino = grover_irq_table[i].pci_ino; int ino = grover_irq_table[i].pci_ino;
if (ino == 0) { if (ino == 0)
isa_dev->irq = PCI_IRQ_NONE; goto no_irq;
} else {
pbm = isa_dev->bus->parent; irq_prop = ino;
pcic = pbm->parent; goto route_irq;
isa_dev->irq = pcic->irq_build(pbm, NULL, ino);
}
} }
} }
goto no_irq;
route_irq:
pbm = isa_dev->bus->parent;
pcic = pbm->parent;
isa_dev->irq = pcic->irq_build(pbm, NULL, irq_prop);
return;
} }
no_irq:
isa_dev->irq = PCI_IRQ_NONE;
} }
static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev) static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev)
...@@ -105,6 +149,7 @@ static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev) ...@@ -105,6 +149,7 @@ static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev)
printk(" ->"); printk(" ->");
while (node != 0) { while (node != 0) {
struct linux_prom_registers regs[PROMREG_MAX];
struct sparc_isa_device *isa_dev; struct sparc_isa_device *isa_dev;
int prop_len; int prop_len;
...@@ -138,8 +183,8 @@ static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev) ...@@ -138,8 +183,8 @@ static void __init isa_fill_children(struct sparc_isa_device *parent_isa_dev)
if (prop_len <= 0) if (prop_len <= 0)
isa_dev->compatible[0] = '\0'; isa_dev->compatible[0] = '\0';
isa_dev_get_resource(isa_dev); isa_dev_get_resource(isa_dev, regs, sizeof(regs));
isa_dev_get_irq(isa_dev); isa_dev_get_irq(isa_dev, regs);
report_dev(isa_dev, 1); report_dev(isa_dev, 1);
...@@ -152,6 +197,7 @@ static void __init isa_fill_devices(struct sparc_isa_bridge *isa_br) ...@@ -152,6 +197,7 @@ static void __init isa_fill_devices(struct sparc_isa_bridge *isa_br)
int node = prom_getchild(isa_br->prom_node); int node = prom_getchild(isa_br->prom_node);
while (node != 0) { while (node != 0) {
struct linux_prom_registers regs[PROMREG_MAX];
struct sparc_isa_device *isa_dev; struct sparc_isa_device *isa_dev;
int prop_len; int prop_len;
...@@ -194,8 +240,8 @@ static void __init isa_fill_devices(struct sparc_isa_bridge *isa_br) ...@@ -194,8 +240,8 @@ static void __init isa_fill_devices(struct sparc_isa_bridge *isa_br)
if (prop_len <= 0) if (prop_len <= 0)
isa_dev->compatible[0] = '\0'; isa_dev->compatible[0] = '\0';
isa_dev_get_resource(isa_dev); isa_dev_get_resource(isa_dev, regs, sizeof(regs));
isa_dev_get_irq(isa_dev); isa_dev_get_irq(isa_dev, regs);
report_dev(isa_dev, 0); report_dev(isa_dev, 0);
...@@ -260,6 +306,21 @@ void __init isa_init(void) ...@@ -260,6 +306,21 @@ void __init isa_init(void)
isa_br->num_isa_ranges = isa_br->num_isa_ranges =
(prop_len / sizeof(struct linux_prom_isa_ranges)); (prop_len / sizeof(struct linux_prom_isa_ranges));
prop_len = prom_getproperty(isa_br->prom_node,
"interrupt-map",
(char *) isa_br->isa_intmap,
sizeof(isa_br->isa_intmap));
if (prop_len <= 0)
isa_br->num_isa_intmap = 0;
else
isa_br->num_isa_intmap =
(prop_len / sizeof(struct linux_prom_isa_intmap));
prop_len = prom_getproperty(isa_br->prom_node,
"interrupt-map-mask",
(char *) &(isa_br->isa_intmask),
sizeof(isa_br->isa_intmask));
printk("isa%d:", isa_br->index); printk("isa%d:", isa_br->index);
isa_fill_devices(isa_br); isa_fill_devices(isa_br);
......
...@@ -34,6 +34,11 @@ struct sparc_isa_bridge { ...@@ -34,6 +34,11 @@ struct sparc_isa_bridge {
#define linux_prom_isa_ranges linux_prom_ebus_ranges #define linux_prom_isa_ranges linux_prom_ebus_ranges
struct linux_prom_isa_ranges isa_ranges[PROMREG_MAX]; struct linux_prom_isa_ranges isa_ranges[PROMREG_MAX];
int num_isa_ranges; int num_isa_ranges;
#define linux_prom_isa_intmap linux_prom_ebus_intmap
struct linux_prom_isa_intmap isa_intmap[PROMREG_MAX];
int num_isa_intmap;
#define linux_prom_isa_intmask linux_prom_ebus_intmask
struct linux_prom_isa_intmap isa_intmask;
}; };
extern struct sparc_isa_bridge *isa_chain; extern struct sparc_isa_bridge *isa_chain;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment