Commit c5760066 authored by Anton Blanchard's avatar Anton Blanchard

ppc64: merge Paul's interrupt-map parsing code

parent a7872552
...@@ -1538,137 +1538,185 @@ finish_node(struct device_node *np, unsigned long mem_start, ...@@ -1538,137 +1538,185 @@ finish_node(struct device_node *np, unsigned long mem_start,
return mem_start; return mem_start;
} }
/* This routine walks the interrupt tree for a given device node and gather /*
* all necessary informations according to the draft interrupt mapping * Find the interrupt parent of a node.
* for CHRP. The current version was only tested on Apple "Core99" machines
* and may not handle cascaded controllers correctly.
*/ */
__init static struct device_node * __init
static unsigned long intr_parent(struct device_node *p)
finish_node_interrupts(struct device_node *np, unsigned long mem_start)
{ {
/* Finish this node */ phandle *parp;
unsigned int *isizep, *asizep, *interrupts, *map, *map_mask, *reg;
phandle *parent, map_parent;
struct device_node *node, *parent_node;
int l, isize, ipsize, asize, map_size, regpsize;
/* Currently, we don't look at all nodes with no "interrupts" property */
interrupts = (unsigned int *)get_property(np, "interrupts", &l);
if (interrupts == NULL)
return mem_start;
ipsize = l>>2;
reg = (unsigned int *)get_property(np, "reg", &l);
regpsize = l>>2;
/* We assume default interrupt cell size is 1 (bugus ?) */ parp = (phandle *) get_property(p, "interrupt-parent", NULL);
isize = 1; if (parp == NULL)
node = np; return p->parent;
return find_phandle(*parp);
}
do { /*
/* We adjust the cell size if the current parent contains an #interrupt-cells * Find out the size of each entry of the interrupts property
* property */ * for a node.
isizep = (unsigned int *)get_property(node, "#interrupt-cells", &l);
if (isizep)
isize = *isizep;
/* We don't do interrupt cascade (ISA) for now, we stop on the first
* controller found
*/ */
if (get_property(node, "interrupt-controller", &l)) { static int __init
int i,j; prom_n_intr_cells(struct device_node *np)
{
np->intrs = (struct interrupt_info *) mem_start; struct device_node *p;
np->n_intrs = ipsize / isize; unsigned int *icp;
mem_start += np->n_intrs * sizeof(struct interrupt_info);
for (i = 0; i < np->n_intrs; ++i) { for (p = np; (p = intr_parent(p)) != NULL; ) {
np->intrs[i].line = openpic_to_irq(virt_irq_create_mapping(*interrupts++)); icp = (unsigned int *)
np->intrs[i].sense = 1; get_property(p, "#interrupt-cells", NULL);
if (isize > 1) if (icp != NULL)
np->intrs[i].sense = *interrupts++; return *icp;
for (j=2; j<isize; j++) if (get_property(p, "interrupt-controller", NULL) != NULL
interrupts++; || get_property(p, "interrupt-map", NULL) != NULL) {
printk("oops, node %s doesn't have #interrupt-cells\n",
p->full_name);
return 1;
} }
return mem_start;
} }
/* We lookup for an interrupt-map. This code can only handle one interrupt #ifdef DEBUG_IRQ
* per device in the map. We also don't handle #address-cells in the parent printk("prom_n_intr_cells failed for %s\n", np->full_name);
* I skip the pci node itself here, may not be necessary but I don't like it's #endif
* reg property. return 1;
*/ }
if (np != node)
map = (unsigned int *)get_property(node, "interrupt-map", &l); /*
else * Map an interrupt from a device up to the platform interrupt
map = NULL; * descriptor.
if (map && l) {
int i, found, temp_isize, temp_asize;
map_size = l>>2;
map_mask = (unsigned int *)get_property(node, "interrupt-map-mask", &l);
asizep = (unsigned int *)get_property(node, "#address-cells", &l);
if (asizep && l == sizeof(unsigned int))
asize = *asizep;
else
asize = 0;
found = 0;
while (map_size>0 && !found) {
found = 1;
for (i=0; i<asize; i++) {
unsigned int mask = map_mask ? map_mask[i] : 0xffffffff;
if (!reg || (i>=regpsize) || ((mask & *map) != (mask & reg[i])))
found = 0;
map++;
map_size--;
}
for (i=0; i<isize; i++) {
unsigned int mask = map_mask ? map_mask[i+asize] : 0xffffffff;
if ((mask & *map) != (mask & interrupts[i]))
found = 0;
map++;
map_size--;
}
map_parent = *((phandle *)map);
map+=1; map_size-=1;
parent_node = find_phandle(map_parent);
temp_isize = isize;
temp_asize = 0;
if (parent_node) {
isizep = (unsigned int *)get_property(parent_node, "#interrupt-cells", &l);
if (isizep)
temp_isize = *isizep;
asizep = (unsigned int *)get_property(parent_node, "#address-cells", &l);
if (asizep && l == sizeof(unsigned int))
temp_asize = *asizep;
}
if (!found) {
map += temp_isize + temp_asize;
map_size -= temp_isize + temp_asize;
}
}
if (found) {
/* Mapped to a new parent. Use the reg and interrupts specified in
* the map as the new search parameters. Then search from the parent.
*/ */
node = parent_node; static int __init
reg = map; map_interrupt(unsigned int **irq, struct device_node **ictrler,
regpsize = temp_asize; struct device_node *np, unsigned int *ints, int nintrc)
interrupts = map + temp_asize; {
ipsize = temp_isize; struct device_node *p, *ipar;
unsigned int *imap, *imask, *ip;
int i, imaplen, match;
int newintrc, newaddrc;
unsigned int *reg;
int naddrc;
reg = (unsigned int *) get_property(np, "reg", NULL);
naddrc = prom_n_addr_cells(np);
p = intr_parent(np);
while (p != NULL) {
if (get_property(p, "interrupt-controller", NULL) != NULL)
/* this node is an interrupt controller, stop here */
break;
imap = (unsigned int *)
get_property(p, "interrupt-map", &imaplen);
if (imap == NULL) {
p = intr_parent(p);
continue; continue;
} }
imask = (unsigned int *)
get_property(p, "interrupt-map-mask", NULL);
if (imask == NULL) {
printk("oops, %s has interrupt-map but no mask\n",
p->full_name);
return 0;
}
imaplen /= sizeof(unsigned int);
match = 0;
ipar = NULL;
while (imaplen > 0 && !match) {
/* check the child-interrupt field */
match = 1;
for (i = 0; i < naddrc && match; ++i)
match = ((reg[i] ^ imap[i]) & imask[i]) == 0;
for (; i < naddrc + nintrc && match; ++i)
match = ((ints[i-naddrc] ^ imap[i]) & imask[i]) == 0;
imap += naddrc + nintrc;
imaplen -= naddrc + nintrc;
/* grab the interrupt parent */
ipar = find_phandle((phandle) *imap++);
--imaplen;
if (ipar == NULL) {
printk("oops, no int parent %x in map of %s\n",
imap[-1], p->full_name);
return 0;
}
/* find the parent's # addr and intr cells */
ip = (unsigned int *)
get_property(ipar, "#interrupt-cells", NULL);
if (ip == NULL) {
printk("oops, no #interrupt-cells on %s\n",
ipar->full_name);
return 0;
}
newintrc = *ip;
ip = (unsigned int *)
get_property(ipar, "#address-cells", NULL);
newaddrc = (ip == NULL)? 0: *ip;
imap += newaddrc + newintrc;
imaplen -= newaddrc + newintrc;
}
if (imaplen < 0) {
printk("oops, error decoding int-map on %s, len=%d\n",
p->full_name, imaplen);
return 0;
}
if (!match) {
#ifdef DEBUG_IRQ
printk("oops, no match in %s int-map for %s\n",
p->full_name, np->full_name);
#endif
return 0;
}
p = ipar;
naddrc = newaddrc;
nintrc = newintrc;
ints = imap - nintrc;
reg = ints - naddrc;
} }
/* We look for an explicit interrupt-parent. #ifdef DEBUG_IRQ
if (p == NULL)
printk("hmmm, int tree for %s doesn't have ctrler\n",
np->full_name);
#endif
*irq = ints;
*ictrler = p;
return nintrc;
}
/*
* New version of finish_node_interrupts.
*/ */
parent = (phandle *)get_property(node, "interrupt-parent", &l); static unsigned long __init
if (parent && (l == sizeof(phandle)) && finish_node_interrupts(struct device_node *np, unsigned long mem_start)
(parent_node = find_phandle(*parent))) { {
node = parent_node; unsigned int *ints;
int intlen, intrcells;
int i, j, n;
unsigned int *irq;
struct device_node *ic;
ints = (unsigned int *) get_property(np, "interrupts", &intlen);
if (ints == NULL)
return mem_start;
intrcells = prom_n_intr_cells(np);
intlen /= intrcells * sizeof(unsigned int);
np->n_intrs = intlen;
np->intrs = (struct interrupt_info *) mem_start;
mem_start += intlen * sizeof(struct interrupt_info);
for (i = 0; i < intlen; ++i) {
np->intrs[i].line = 0;
np->intrs[i].sense = 1;
n = map_interrupt(&irq, &ic, np, ints, intrcells);
if (n <= 0)
continue; continue;
np->intrs[i].line = openpic_to_irq(virt_irq_create_mapping(irq[0]));
if (n > 1)
np->intrs[i].sense = irq[1];
if (n > 2) {
printk("hmmm, got %d intr cells for %s:", n,
np->full_name);
for (j = 0; j < n; ++j)
printk(" %d", irq[j]);
printk("\n");
}
ints += intrcells;
} }
/* Default, get real parent */
node = node->parent;
} while (node);
return mem_start; return mem_start;
} }
......
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