Commit 6ec2159c authored by Pete Zaitcev's avatar Pete Zaitcev Committed by David S. Miller

[SPARC]: Streamlined probing for Zilog.

parent 1ca5b0e0
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* This is based on the old drivers/sbus/char/zs.c code. A lot * This is based on the old drivers/sbus/char/zs.c code. A lot
* of code has been simply moved over directly from there but * of code has been simply moved over directly from there but
* much has been rewritten. Credits therefore go out to Eddie * much has been rewritten. Credits therefore go out to Eddie
* C. Dost, Peter Zaitcev, Ted Ts'o and Alex Buell for their * C. Dost, Pete Zaitcev, Ted Ts'o and Alex Buell for their
* work there. * work there.
* *
* Copyright (C) 2002 David S. Miller (davem@redhat.com) * Copyright (C) 2002 David S. Miller (davem@redhat.com)
...@@ -992,7 +992,6 @@ static struct uart_ops sunzilog_pops = { ...@@ -992,7 +992,6 @@ static struct uart_ops sunzilog_pops = {
static struct uart_sunzilog_port *sunzilog_port_table; static struct uart_sunzilog_port *sunzilog_port_table;
static struct zilog_layout **sunzilog_chip_regs; static struct zilog_layout **sunzilog_chip_regs;
static int *sunzilog_nodes;
static struct uart_sunzilog_port *sunzilog_irq_chain; static struct uart_sunzilog_port *sunzilog_irq_chain;
static int zilog_irq = -1; static int zilog_irq = -1;
...@@ -1025,12 +1024,8 @@ static void __init sunzilog_alloc_tables(void) ...@@ -1025,12 +1024,8 @@ static void __init sunzilog_alloc_tables(void)
alloc_one_table(NUM_CHANNELS * sizeof(struct uart_sunzilog_port)); alloc_one_table(NUM_CHANNELS * sizeof(struct uart_sunzilog_port));
sunzilog_chip_regs = (struct zilog_layout **) sunzilog_chip_regs = (struct zilog_layout **)
alloc_one_table(NUM_SUNZILOG * sizeof(struct zilog_layout *)); alloc_one_table(NUM_SUNZILOG * sizeof(struct zilog_layout *));
sunzilog_nodes = (int *)
alloc_one_table(NUM_SUNZILOG * sizeof(int));
if (sunzilog_port_table == NULL || if (sunzilog_port_table == NULL || sunzilog_chip_regs == NULL) {
sunzilog_chip_regs == NULL ||
sunzilog_nodes == NULL) {
prom_printf("sunzilog_init: Cannot alloc SunZilog tables.\n"); prom_printf("sunzilog_init: Cannot alloc SunZilog tables.\n");
prom_halt(); prom_halt();
} }
...@@ -1041,24 +1036,13 @@ static void __init sunzilog_alloc_tables(void) ...@@ -1041,24 +1036,13 @@ static void __init sunzilog_alloc_tables(void)
/* We used to attempt to use the address property of the Zilog device node /* We used to attempt to use the address property of the Zilog device node
* but that totally is not necessary on sparc64. * but that totally is not necessary on sparc64.
*/ */
static struct zilog_layout * __init get_zs_sun4u(int chip) static struct zilog_layout * __init get_zs_sun4u(int chip, int node)
{ {
unsigned long mapped_addr = 0xdeadbeefUL; unsigned long mapped_addr = 0xdeadbeefUL;
unsigned int sun4u_ino; unsigned int sun4u_ino;
int busnode, zsnode, seen; int zsnode, seen;
if (central_bus) zsnode = node;
busnode = central_bus->child->prom_node;
else
busnode = prom_searchsiblings(prom_getchild(prom_root_node), "sbus");
if (busnode == 0 || busnode == -1) {
prom_printf("SunZilog: Cannot find bus of Zilog %d in get_zs_sun4u.\n",
chip);
prom_halt();
}
zsnode = prom_getchild(busnode);
seen = 0; seen = 0;
while (zsnode) { while (zsnode) {
int slave; int slave;
...@@ -1111,7 +1095,6 @@ static struct zilog_layout * __init get_zs_sun4u(int chip) ...@@ -1111,7 +1095,6 @@ static struct zilog_layout * __init get_zs_sun4u(int chip)
((u64)zsregs[0].phys_addr); ((u64)zsregs[0].phys_addr);
} }
sunzilog_nodes[chip] = zsnode;
if (zilog_irq == -1) { if (zilog_irq == -1) {
if (central_bus) { if (central_bus) {
unsigned long iclr, imap; unsigned long iclr, imap;
...@@ -1142,31 +1125,35 @@ static struct zilog_layout * __init get_zs_sun4u(int chip) ...@@ -1142,31 +1125,35 @@ static struct zilog_layout * __init get_zs_sun4u(int chip)
return (struct zilog_layout *) mapped_addr; return (struct zilog_layout *) mapped_addr;
} }
#else /* CONFIG_SPARC64 */ #else /* CONFIG_SPARC64 */
static struct zilog_layout * __init get_zs_sun4cmd(int chip)
/*
* XXX The sun4d case is utterly screwed: it tries to re-walk the tree
* (for the 3rd time) in order to find bootbus and cpu. Streamline it.
*/
static struct zilog_layout * __init get_zs_sun4cmd(int chip, int node)
{ {
struct linux_prom_irqs irq_info[2]; struct linux_prom_irqs irq_info[2];
unsigned long mapped_addr = 0; unsigned long mapped_addr = 0;
int zsnode, chipid, cpunode, bbnode; int zsnode, cpunode, bbnode;
struct linux_prom_registers zsreg[4];
struct resource res;
if (sparc_cpu_model == sun4d) { if (sparc_cpu_model == sun4d) {
int walk, no; int walk;
zsnode = 0; zsnode = 0;
bbnode = 0; bbnode = 0;
no = 0; cpunode = 0;
for (walk = prom_getchild(prom_root_node); for (walk = prom_getchild(prom_root_node);
(walk = prom_searchsiblings(walk, "cpu-unit")) != 0; (walk = prom_searchsiblings(walk, "cpu-unit")) != 0;
walk = prom_getsibling(walk)) { walk = prom_getsibling(walk)) {
bbnode = prom_getchild(walk); bbnode = prom_getchild(walk);
if (bbnode && if (bbnode &&
(bbnode = prom_searchsiblings(bbnode, "bootbus"))) { (bbnode = prom_searchsiblings(bbnode, "bootbus"))) {
if (no == (chip / 2)) { if ((zsnode = prom_getchild(bbnode)) == node) {
cpunode = walk; cpunode = walk;
zsnode = prom_getchild(bbnode);
chipid = (chip & 1);
break; break;
} }
no++;
} }
} }
if (!walk) { if (!walk) {
...@@ -1174,89 +1161,70 @@ static struct zilog_layout * __init get_zs_sun4cmd(int chip) ...@@ -1174,89 +1161,70 @@ static struct zilog_layout * __init get_zs_sun4cmd(int chip)
(chip / 2)); (chip / 2));
prom_halt(); prom_halt();
} }
} else {
int tmp; if (prom_getproperty(zsnode, "reg",
(char *) zsreg, sizeof(zsreg)) == -1) {
zsnode = prom_getchild(prom_root_node); prom_printf("SunZilog: Cannot map Zilog %d\n", chip);
if ((tmp = prom_searchsiblings(zsnode, "obio"))) {
zsnode = prom_getchild(tmp);
if (!zsnode) {
prom_printf("SunZilog: Child of obio node does "
"not exist.\n");
prom_halt(); prom_halt();
} }
} /* XXX Looks like an off by one? */
chipid = 0; prom_apply_generic_ranges(bbnode, cpunode, zsreg, 1);
} res.start = zsreg[0].phys_addr;
res.end = res.start + (8 - 1);
while (zsnode) { res.flags = zsreg[0].which_io | IORESOURCE_IO;
struct linux_prom_registers zsreg[4]; mapped_addr = sbus_ioremap(&res, 0, 8, "Zilog Serial");
struct resource res;
zsnode = prom_searchsiblings(zsnode, "zs"); } else {
if (zsnode == 0 || zsnode == -1) zsnode = node;
break;
#if 0 /* XXX When was this used? */
if (prom_getintdefault(zsnode, "slave", -1) != chipid) { if (prom_getintdefault(zsnode, "slave", -1) != chipid) {
zsnode = prom_getsibling(zsnode); zsnode = prom_getsibling(zsnode);
continue; continue;
} }
#endif
if (sparc_cpu_model == sun4d) { /*
/* Sun4d Zilog nodes lack the address property, so just * "address" is only present on ports that OBP opened
* map it like a normal device. * (from Mitch Bradley's "Hitchhiker's Guide to OBP").
* We do not use it.
*/ */
if (prom_getproperty(zsnode, "reg", if (prom_getproperty(zsnode, "reg",
(char *) zsreg, sizeof(zsreg)) == -1) { (char *) zsreg, sizeof(zsreg)) == -1) {
prom_printf("SunZilog: Cannot map Zilog %d " prom_printf("SunZilog: Cannot map Zilog %d\n", chip);
"regs on sun4c.\n", chip);
prom_halt(); prom_halt();
} }
prom_apply_generic_ranges(bbnode, cpunode, zsreg, 1); if (sparc_cpu_model == sun4m) /* Crude. Pass parent. XXX */
prom_apply_obio_ranges(zsreg, 1);
res.start = zsreg[0].phys_addr; res.start = zsreg[0].phys_addr;
res.end = res.start + (8 - 1); res.end = res.start + (8 - 1);
res.flags = zsreg[0].which_io | IORESOURCE_IO; res.flags = zsreg[0].which_io | IORESOURCE_IO;
mapped_addr = sbus_ioremap(&res, 0, 8, "Zilog Serial"); mapped_addr = sbus_ioremap(&res, 0, 8, "Zilog Serial");
} else {
unsigned int vaddr[2];
if (prom_getproperty(zsnode, "address",
(void *) vaddr, sizeof(vaddr))
% sizeof(unsigned int)) {
prom_printf("SunZilog: Cannot get address property for "
"Zilog %d.\n", chip);
prom_halt();
}
mapped_addr = (unsigned long) vaddr[0];
} }
sunzilog_nodes[chip] = zsnode;
if (prom_getproperty(zsnode, "intr", if (prom_getproperty(zsnode, "intr",
(char *) irq_info, sizeof(irq_info)) (char *) irq_info, sizeof(irq_info))
% sizeof(struct linux_prom_irqs)) { % sizeof(struct linux_prom_irqs)) {
prom_printf("SunZilog: Cannot get IRQ property for " prom_printf("SunZilog: Cannot get IRQ property for Zilog %d.\n",
"Zilog %d.\n", chip); chip);
prom_halt(); prom_halt();
} }
if (zilog_irq == -1) { if (zilog_irq == -1) {
zilog_irq = irq_info[0].pri; zilog_irq = irq_info[0].pri;
} else if (zilog_irq != irq_info[0].pri) { } else if (zilog_irq != irq_info[0].pri) {
/* XXX. Dumb. Should handle per-chip IRQ, for add-ons. */
prom_printf("SunZilog: Inconsistent IRQ layout for Zilog %d.\n", prom_printf("SunZilog: Inconsistent IRQ layout for Zilog %d.\n",
chip); chip);
prom_halt(); prom_halt();
} }
break;
}
if (!zsnode) {
prom_printf("SunZilog: OBP node for Zilog %d not found.\n", chip);
prom_halt();
}
return (struct zilog_layout *) mapped_addr; return (struct zilog_layout *) mapped_addr;
} }
#endif /* !(CONFIG_SPARC64) */ #endif /* !(CONFIG_SPARC64) */
/* Get the address of the registers for SunZilog instance CHIP. */ /* Get the address of the registers for SunZilog instance CHIP. */
static struct zilog_layout * __init get_zs(int chip) static struct zilog_layout * __init get_zs(int chip, int node)
{ {
if (chip < 0 || chip >= NUM_SUNZILOG) { if (chip < 0 || chip >= NUM_SUNZILOG) {
prom_printf("SunZilog: Illegal chip number %d in get_zs.\n", chip); prom_printf("SunZilog: Illegal chip number %d in get_zs.\n", chip);
...@@ -1264,7 +1232,7 @@ static struct zilog_layout * __init get_zs(int chip) ...@@ -1264,7 +1232,7 @@ static struct zilog_layout * __init get_zs(int chip)
} }
#ifdef CONFIG_SPARC64 #ifdef CONFIG_SPARC64
return get_zs_sun4u(chip); return get_zs_sun4u(chip, node);
#else #else
if (sparc_cpu_model == sun4) { if (sparc_cpu_model == sun4) {
...@@ -1279,14 +1247,13 @@ static struct zilog_layout * __init get_zs(int chip) ...@@ -1279,14 +1247,13 @@ static struct zilog_layout * __init get_zs(int chip)
res.start = 0xf0000000; res.start = 0xf0000000;
break; break;
}; };
sunzilog_nodes[chip] = 0;
zilog_irq = 12; zilog_irq = 12;
res.end = (res.start + (8 - 1)); res.end = (res.start + (8 - 1));
res.flags = IORESOURCE_IO; res.flags = IORESOURCE_IO;
return (struct zilog_layout *) sbus_ioremap(&res, 0, 8, "SunZilog"); return (struct zilog_layout *) sbus_ioremap(&res, 0, 8, "SunZilog");
} }
return get_zs_sun4cmd(chip); return get_zs_sun4cmd(chip, node);
#endif #endif
} }
...@@ -1459,6 +1426,52 @@ static int __init sunzilog_console_init(void) ...@@ -1459,6 +1426,52 @@ static int __init sunzilog_console_init(void)
return 0; return 0;
} }
/*
* We scan the PROM tree recursively. This is the most reliable way
* to find Zilog nodes on various platforms. However, we face an extreme
* shortage of kernel stack, so we must be very careful. To that end,
* we scan only to a certain depth, and we use a common property buffer
* in the scan structure.
*/
#define ZS_PROPSIZE 128
#define ZS_SCAN_DEPTH 5
struct zs_probe_scan {
int depth;
void (*scanner)(struct zs_probe_scan *t, int node);
int devices;
char prop[ZS_PROPSIZE];
};
static int __inline__ sunzilog_node_ok(int node, const char *name, int len)
{
if (strncmp(name, "zs", len) == 0)
return 1;
/* Don't fold this procedure just yet. Compare to su_node_ok(). */
return 0;
}
static void __init sunzilog_scan(struct zs_probe_scan *t, int node)
{
int len;
for (; node != 0; node = prom_getsibling(node)) {
len = prom_getproperty(node, "name", t->prop, ZS_PROPSIZE);
if (len <= 1)
continue; /* Broken PROM node */
if (sunzilog_node_ok(node, t->prop, len)) {
(*t->scanner)(t, node);
} else {
if (t->depth < ZS_SCAN_DEPTH) {
t->depth++;
sunzilog_scan(t, prom_getchild(node));
--t->depth;
}
}
}
}
static void __init sunzilog_prepare(void) static void __init sunzilog_prepare(void)
{ {
struct uart_sunzilog_port *up; struct uart_sunzilog_port *up;
...@@ -1471,12 +1484,9 @@ static void __init sunzilog_prepare(void) ...@@ -1471,12 +1484,9 @@ static void __init sunzilog_prepare(void)
up[channel].next = NULL; up[channel].next = NULL;
for (chip = 0; chip < NUM_SUNZILOG; chip++) { for (chip = 0; chip < NUM_SUNZILOG; chip++) {
if (!sunzilog_chip_regs[chip]) { rp = sunzilog_chip_regs[chip];
sunzilog_chip_regs[chip] = rp = get_zs(chip);
up[(chip * 2) + 0].port.membase = (char *) &rp->channelA; up[(chip * 2) + 0].port.membase = (char *) &rp->channelA;
up[(chip * 2) + 1].port.membase = (char *) &rp->channelB; up[(chip * 2) + 1].port.membase = (char *) &rp->channelB;
}
/* Channel A */ /* Channel A */
up[(chip * 2) + 0].port.iotype = SERIAL_IO_MEM; up[(chip * 2) + 0].port.iotype = SERIAL_IO_MEM;
...@@ -1593,11 +1603,25 @@ static void __init sunzilog_init_hw(void) ...@@ -1593,11 +1603,25 @@ static void __init sunzilog_init_hw(void)
} }
} }
static struct zilog_layout * __init get_zs(int chip, int node);
static void __init sunzilog_scan_probe(struct zs_probe_scan *t, int node)
{
sunzilog_chip_regs[t->devices] = get_zs(t->devices, node);
t->devices++;
}
static int __init sunzilog_ports_init(void) static int __init sunzilog_ports_init(void)
{ {
struct zs_probe_scan scan;
int ret; int ret;
printk(KERN_INFO "Serial: Sun Zilog driver.\n"); printk(KERN_INFO "Serial: Sun Zilog driver (%d chips).\n", NUM_SUNZILOG);
scan.scanner = sunzilog_scan_probe;
scan.depth = 0;
scan.devices = 0;
sunzilog_scan(&scan, prom_getchild(prom_root_node));
sunzilog_prepare(); sunzilog_prepare();
...@@ -1635,56 +1659,34 @@ static int __init sunzilog_ports_init(void) ...@@ -1635,56 +1659,34 @@ static int __init sunzilog_ports_init(void)
return ret; return ret;
} }
static int __init sunzilog_init(void) static void __init sunzilog_scan_count(struct zs_probe_scan *t, int node)
{ {
int node; t->devices++;
}
static int __init sunzilog_ports_count(void)
{
struct zs_probe_scan scan;
/* Sun4 Zilog setup is hard coded, no probing to do. */ /* Sun4 Zilog setup is hard coded, no probing to do. */
if (sparc_cpu_model == sun4) { if (sparc_cpu_model == sun4)
NUM_SUNZILOG = 2; return 2;
} else if (sparc_cpu_model == sun4d) {
int bbnode;
node = prom_getchild(prom_root_node);
NUM_SUNZILOG = 0;
while (node &&
(node = prom_searchsiblings(node, "cpu-unit"))) {
bbnode = prom_getchild(node);
if (bbnode && prom_searchsiblings(bbnode, "bootbus"))
NUM_SUNZILOG += 2;
node = prom_getsibling(node);
}
} else if (sparc_cpu_model == sun4u) {
int central_node;
/* Central bus zilogs must be checked for first,
* since Enterprise boxes might have SBUSes as well.
*/
central_node = prom_finddevice("/central");
if (central_node != 0 && central_node != -1)
node = prom_searchsiblings(prom_getchild(central_node), "fhc");
else
node = prom_searchsiblings(prom_getchild(prom_root_node), "sbus");
if (node != 0 && node != -1)
node = prom_getchild(node);
if (node == 0 || node == -1)
return -ENODEV;
node = prom_searchsiblings(node, "zs"); scan.scanner = sunzilog_scan_count;
if (!node) scan.depth = 0;
return -ENODEV; scan.devices = 0;
NUM_SUNZILOG = 2; sunzilog_scan(&scan, prom_getchild(prom_root_node));
} else {
node = prom_getchild(prom_root_node);
node = prom_searchsiblings(node, "obio");
if (node)
node = prom_getchild(node);
if (!node)
return -ENODEV;
NUM_SUNZILOG = 2; return scan.devices;
} }
static int __init sunzilog_init(void)
{
NUM_SUNZILOG = sunzilog_ports_count();
if (NUM_SUNZILOG == 0)
return -ENODEV;
sunzilog_alloc_tables(); sunzilog_alloc_tables();
......
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