linux 2.6.16.38 w/ vs2.0.3-rc1
[linux-2.6.git] / drivers / sbus / sbus.c
index 935952e..5d30a3e 100644 (file)
@@ -1,10 +1,12 @@
-/* sbus.c: SBus support routines.
+/* $Id: sbus.c,v 1.100 2002/01/24 15:36:24 davem Exp $
+ * sbus.c:  SBus support routines.
  *
- * Copyright (C) 1995, 2006 David S. Miller (davem@davemloft.net)
+ * Copyright (C) 1995 David S. Miller (davem@caip.rutgers.edu)
  */
 
 #include <linux/kernel.h>
 #include <linux/slab.h>
+#include <linux/config.h>
 #include <linux/init.h>
 #include <linux/pci.h>
 
 #include <asm/sbus.h>
 #include <asm/dma.h>
 #include <asm/oplib.h>
-#include <asm/prom.h>
-#include <asm/of_device.h>
 #include <asm/bpp.h>
 #include <asm/irq.h>
 
-struct sbus_bus *sbus_root;
+struct sbus_bus *sbus_root = NULL;
 
-static void __init fill_sbus_device(struct device_node *dp, struct sbus_dev *sdev)
-{
-       unsigned long base;
-       void *pval;
-       int len;
+static struct linux_prom_irqs irqs[PROMINTR_MAX] __initdata = { { 0 } };
+#ifdef CONFIG_SPARC32
+static int interrupts[PROMINTR_MAX] __initdata = { 0 };
+#endif
 
-       sdev->prom_node = dp->node;
-       strcpy(sdev->prom_name, dp->name);
+#ifdef CONFIG_PCI
+extern int pcic_present(void);
+#endif
 
-       pval = of_get_property(dp, "reg", &len);
-       sdev->num_registers = 0;
-       if (pval) {
-               memcpy(sdev->reg_addrs, pval, len);
+/* Perhaps when I figure out more about the iommu we'll put a
+ * device registration routine here that probe_sbus() calls to
+ * setup the iommu for each Sbus.
+ */
+
+/* We call this for each SBus device, and fill the structure based
+ * upon the prom device tree.  We return the start of memory after
+ * the things we have allocated.
+ */
 
-               sdev->num_registers =
-                       len / sizeof(struct linux_prom_registers);
+/* #define DEBUG_FILL */
 
-               base = (unsigned long) sdev->reg_addrs[0].phys_addr;
+static void __init fill_sbus_device(int prom_node, struct sbus_dev *sdev)
+{
+       unsigned long address, base;
+       int len;
 
-               /* Compute the slot number. */
-               if (base >= SUN_SBUS_BVADDR && sparc_cpu_model == sun4m)
-                       sdev->slot = sbus_dev_slot(base);
-               else
-                       sdev->slot = sdev->reg_addrs[0].which_io;
+       sdev->prom_node = prom_node;
+       prom_getstring(prom_node, "name",
+                      sdev->prom_name, sizeof(sdev->prom_name));
+       address = prom_getint(prom_node, "address");
+       len = prom_getproperty(prom_node, "reg",
+                              (char *) sdev->reg_addrs,
+                              sizeof(sdev->reg_addrs));
+       if (len == -1) {
+               sdev->num_registers = 0;
+               goto no_regs;
        }
 
-       pval = of_get_property(dp, "ranges", &len);
-       sdev->num_device_ranges = 0;
-       if (pval) {
-               memcpy(sdev->device_ranges, pval, len);
-               sdev->num_device_ranges =
-                       len / sizeof(struct linux_prom_ranges);
+       if (len % sizeof(struct linux_prom_registers)) {
+               prom_printf("fill_sbus_device: proplen for regs of %s "
+                           " was %d, need multiple of %d\n",
+                           sdev->prom_name, len,
+                           (int) sizeof(struct linux_prom_registers));
+               prom_halt();
+       }
+       if (len > (sizeof(struct linux_prom_registers) * PROMREG_MAX)) {
+               prom_printf("fill_sbus_device: Too many register properties "
+                           "for device %s, len=%d\n",
+                           sdev->prom_name, len);
+               prom_halt();
        }
+       sdev->num_registers = len / sizeof(struct linux_prom_registers);
+       sdev->ranges_applied = 0;
 
-       sbus_fill_device_irq(sdev);
+       base = (unsigned long) sdev->reg_addrs[0].phys_addr;
 
-       sdev->ofdev.node = dp;
-       if (sdev->parent)
-               sdev->ofdev.dev.parent = &sdev->parent->ofdev.dev;
-       else
-               sdev->ofdev.dev.parent = &sdev->bus->ofdev.dev;
-       sdev->ofdev.dev.bus = &sbus_bus_type;
-       strcpy(sdev->ofdev.dev.bus_id, dp->path_component_name);
+       /* Compute the slot number. */
+       if (base >= SUN_SBUS_BVADDR && sparc_cpu_model == sun4m) {
+               sdev->slot = sbus_dev_slot(base);
+       } else {
+               sdev->slot = sdev->reg_addrs[0].which_io;
+       }
 
-       if (of_device_register(&sdev->ofdev) != 0)
-               printk(KERN_DEBUG "sbus: device registration error for %s!\n",
-                      sdev->ofdev.dev.bus_id);
+no_regs:
+       len = prom_getproperty(prom_node, "ranges",
+                              (char *)sdev->device_ranges,
+                              sizeof(sdev->device_ranges));
+       if (len == -1) {
+               sdev->num_device_ranges = 0;
+               goto no_ranges;
+       }
+       if (len % sizeof(struct linux_prom_ranges)) {
+               prom_printf("fill_sbus_device: proplen for ranges of %s "
+                           " was %d, need multiple of %d\n",
+                           sdev->prom_name, len,
+                           (int) sizeof(struct linux_prom_ranges));
+               prom_halt();
+       }
+       if (len > (sizeof(struct linux_prom_ranges) * PROMREG_MAX)) {
+               prom_printf("fill_sbus_device: Too many range properties "
+                           "for device %s, len=%d\n",
+                           sdev->prom_name, len);
+               prom_halt();
+       }
+       sdev->num_device_ranges =
+               len / sizeof(struct linux_prom_ranges);
+
+no_ranges:
+       /* XXX Unfortunately, IRQ issues are very arch specific.
+        * XXX Pull this crud out into an arch specific area
+        * XXX at some point. -DaveM
+        */
+#ifdef CONFIG_SPARC64
+       len = prom_getproperty(prom_node, "interrupts",
+                              (char *) irqs, sizeof(irqs));
+       if (len == -1 || len == 0) {
+               sdev->irqs[0] = 0;
+               sdev->num_irqs = 0;
+       } else {
+               unsigned int pri = irqs[0].pri;
+
+               sdev->num_irqs = 1;
+               if (pri < 0x20)
+                       pri += sdev->slot * 8;
+
+               sdev->irqs[0] = sbus_build_irq(sdev->bus, pri);
+       }
+#endif /* CONFIG_SPARC64 */
+
+#ifdef CONFIG_SPARC32
+       len = prom_getproperty(prom_node, "intr",
+                              (char *)irqs, sizeof(irqs));
+       if (len != -1) {
+               sdev->num_irqs = len / 8;
+               if (sdev->num_irqs == 0) {
+                       sdev->irqs[0] = 0;
+               } else if (sparc_cpu_model == sun4d) {
+                       extern unsigned int sun4d_build_irq(struct sbus_dev *sdev, int irq);
+
+                       for (len = 0; len < sdev->num_irqs; len++)
+                               sdev->irqs[len] = sun4d_build_irq(sdev, irqs[len].pri);
+               } else {
+                       for (len = 0; len < sdev->num_irqs; len++)
+                               sdev->irqs[len] = irqs[len].pri;
+               }
+       } else {
+               /* No "intr" node found-- check for "interrupts" node.
+                * This node contains SBus interrupt levels, not IPLs
+                * as in "intr", and no vector values.  We convert 
+                * SBus interrupt levels to PILs (platform specific).
+                */
+               len = prom_getproperty(prom_node, "interrupts", 
+                                       (char *)interrupts, sizeof(interrupts));
+               if (len == -1) {
+                       sdev->irqs[0] = 0;
+                       sdev->num_irqs = 0;
+               } else {
+                       sdev->num_irqs = len / sizeof(int);
+                       for (len = 0; len < sdev->num_irqs; len++) {
+                               sdev->irqs[len] = sbint_to_irq(sdev, interrupts[len]);
+                       }
+               }
+       } 
+#endif /* CONFIG_SPARC32 */
 }
 
-static void __init sbus_bus_ranges_init(struct device_node *dp, struct sbus_bus *sbus)
+/* This routine gets called from whoever needs the sbus first, to scan
+ * the SBus device tree.  Currently it just prints out the devices
+ * found on the bus and builds trees of SBUS structs and attached
+ * devices.
+ */
+
+extern void iommu_init(int iommu_node, struct sbus_bus *sbus);
+extern void iounit_init(int sbi_node, int iounit_node, struct sbus_bus *sbus);
+void sun4_init(void);
+#ifdef CONFIG_SUN_AUXIO
+extern void auxio_probe(void);
+#endif
+
+static void __init sbus_do_child_siblings(int start_node,
+                                         struct sbus_dev *child,
+                                         struct sbus_dev *parent,
+                                         struct sbus_bus *sbus)
 {
-       void *pval;
-       int len;
+       struct sbus_dev *this_dev = child;
+       int this_node = start_node;
+
+       /* Child already filled in, just need to traverse siblings. */
+       child->child = NULL;
+       child->parent = parent;
+       while((this_node = prom_getsibling(this_node)) != 0) {
+               this_dev->next = kmalloc(sizeof(struct sbus_dev), GFP_ATOMIC);
+               this_dev = this_dev->next;
+               this_dev->next = NULL;
+               this_dev->parent = parent;
+
+               this_dev->bus = sbus;
+               fill_sbus_device(this_node, this_dev);
+
+               if(prom_getchild(this_node)) {
+                       this_dev->child = kmalloc(sizeof(struct sbus_dev),
+                                                 GFP_ATOMIC);
+                       this_dev->child->bus = sbus;
+                       this_dev->child->next = NULL;
+                       fill_sbus_device(prom_getchild(this_node), this_dev->child);
+                       sbus_do_child_siblings(prom_getchild(this_node),
+                                              this_dev->child, this_dev, sbus);
+               } else {
+                       this_dev->child = NULL;
+               }
+       }
+}
 
-       pval = of_get_property(dp, "ranges", &len);
-       sbus->num_sbus_ranges = 0;
-       if (pval) {
-               memcpy(sbus->sbus_ranges, pval, len);
-               sbus->num_sbus_ranges =
-                       len / sizeof(struct linux_prom_ranges);
+/*
+ * XXX This functions appears to be a distorted version of
+ * prom_sbus_ranges_init(), with all sun4d stuff cut away.
+ * Ask DaveM what is going on here, how is sun4d supposed to work... XXX
+ */
+/* added back sun4d patch from Thomas Bogendoerfer - should be OK (crn) */
+
+static void __init sbus_bus_ranges_init(int parent_node, struct sbus_bus *sbus)
+{
+       int len;
 
-               sbus_arch_bus_ranges_init(dp->parent, sbus);
+       len = prom_getproperty(sbus->prom_node, "ranges",
+                              (char *) sbus->sbus_ranges,
+                              sizeof(sbus->sbus_ranges));
+       if (len == -1 || len == 0) {
+               sbus->num_sbus_ranges = 0;
+               return;
        }
+       sbus->num_sbus_ranges = len / sizeof(struct linux_prom_ranges);
+#ifdef CONFIG_SPARC32
+       if (sparc_cpu_model == sun4d) {
+               struct linux_prom_ranges iounit_ranges[PROMREG_MAX];
+               int num_iounit_ranges;
+
+               len = prom_getproperty(parent_node, "ranges",
+                                      (char *) iounit_ranges,
+                                      sizeof (iounit_ranges));
+               if (len != -1) {
+                       num_iounit_ranges = (len/sizeof(struct linux_prom_ranges));
+                       prom_adjust_ranges (sbus->sbus_ranges, sbus->num_sbus_ranges, iounit_ranges, num_iounit_ranges);
+               }
+       }
+#endif
 }
 
 static void __init __apply_ranges_to_regs(struct linux_prom_ranges *ranges,
@@ -159,127 +322,241 @@ static void __init sbus_fixup_all_regs(struct sbus_dev *first_sdev)
        }
 }
 
-/* We preserve the "probe order" of these bus and device lists to give
- * the same ordering as the old code.
- */
-static void __init sbus_insert(struct sbus_bus *sbus, struct sbus_bus **root)
-{
-       while (*root)
-               root = &(*root)->next;
-       *root = sbus;
-       sbus->next = NULL;
-}
+extern void register_proc_sparc_ioport(void);
+extern void firetruck_init(void);
 
-static void __init sdev_insert(struct sbus_dev *sdev, struct sbus_dev **root)
-{
-       while (*root)
-               root = &(*root)->next;
-       *root = sdev;
-       sdev->next = NULL;
-}
+#ifdef CONFIG_SUN4
+extern void sun4_dvma_init(void);
+#endif
 
-static void __init walk_children(struct device_node *dp, struct sbus_dev *parent, struct sbus_bus *sbus)
+static int __init sbus_init(void)
 {
-       dp = dp->child;
-       while (dp) {
-               struct sbus_dev *sdev;
-
-               sdev = kzalloc(sizeof(struct sbus_dev), GFP_ATOMIC);
-               if (sdev) {
-                       sdev_insert(sdev, &parent->child);
-
-                       sdev->bus = sbus;
-                       sdev->parent = parent;
+       int nd, this_sbus, sbus_devs, topnd, iommund;
+       unsigned int sbus_clock;
+       struct sbus_bus *sbus;
+       struct sbus_dev *this_dev;
+       int num_sbus = 0;  /* How many did we find? */
 
-                       fill_sbus_device(dp, sdev);
+#ifdef CONFIG_SPARC32
+       register_proc_sparc_ioport();
+#endif
 
-                       walk_children(dp, sdev, sbus);
+#ifdef CONFIG_SUN4
+       sun4_dvma_init();
+       return 0;
+#endif
+
+       topnd = prom_getchild(prom_root_node);
+       
+       /* Finding the first sbus is a special case... */
+       iommund = 0;
+       if(sparc_cpu_model == sun4u) {
+               nd = prom_searchsiblings(topnd, "sbus");
+               if(nd == 0) {
+#ifdef CONFIG_PCI
+                       if (!pcic_present()) {
+                               prom_printf("Neither SBUS nor PCI found.\n");
+                               prom_halt();
+                       } else {
+#ifdef CONFIG_SPARC64
+                               firetruck_init();
+#endif
+                       }
+                       return 0;
+#else
+                       prom_printf("YEEE, UltraSparc sbus not found\n");
+                       prom_halt();
+#endif
+               }
+       } else if(sparc_cpu_model == sun4d) {
+               if((iommund = prom_searchsiblings(topnd, "io-unit")) == 0 ||
+                  (nd = prom_getchild(iommund)) == 0 ||
+                  (nd = prom_searchsiblings(nd, "sbi")) == 0) {
+                       panic("sbi not found");
+               }
+       } else if((nd = prom_searchsiblings(topnd, "sbus")) == 0) {
+               if((iommund = prom_searchsiblings(topnd, "iommu")) == 0 ||
+                  (nd = prom_getchild(iommund)) == 0 ||
+                  (nd = prom_searchsiblings(nd, "sbus")) == 0) {
+#ifdef CONFIG_PCI
+                        if (!pcic_present()) {
+                                prom_printf("Neither SBUS nor PCI found.\n");
+                                prom_halt();
+                        }
+                        return 0;
+#else
+                       /* No reason to run further - the data access trap will occur. */
+                       panic("sbus not found");
+#endif
                }
-               dp = dp->sibling;
        }
-}
-
-static void __init build_one_sbus(struct device_node *dp, int num_sbus)
-{
-       struct sbus_bus *sbus;
-       unsigned int sbus_clock;
-       struct device_node *dev_dp;
-
-       sbus = kzalloc(sizeof(struct sbus_bus), GFP_ATOMIC);
-       if (!sbus)
-               return;
-
-       sbus_insert(sbus, &sbus_root);
-       sbus->prom_node = dp->node;
-
-       sbus_setup_iommu(sbus, dp);
-
-       printk("sbus%d: ", num_sbus);
-
-       sbus_clock = of_getintprop_default(dp, "clock-frequency",
-                                          (25*1000*1000));
-       sbus->clock_freq = sbus_clock;
-
-       printk("Clock %d.%d MHz\n", (int) ((sbus_clock/1000)/1000),
-              (int) (((sbus_clock/1000)%1000 != 0) ? 
-                     (((sbus_clock/1000)%1000) + 1000) : 0));
-
-       strcpy(sbus->prom_name, dp->name);
-
-       sbus_setup_arch_props(sbus, dp);
-
-       sbus_bus_ranges_init(dp, sbus);
 
-       sbus->ofdev.node = dp;
-       sbus->ofdev.dev.parent = NULL;
-       sbus->ofdev.dev.bus = &sbus_bus_type;
-       sprintf(sbus->ofdev.dev.bus_id, "sbus%d", num_sbus);
-
-       if (of_device_register(&sbus->ofdev) != 0)
-               printk(KERN_DEBUG "sbus: device registration error for %s!\n",
-                      sbus->ofdev.dev.bus_id);
-
-       dev_dp = dp->child;
-       while (dev_dp) {
-               struct sbus_dev *sdev;
+       /* Ok, we've found the first one, allocate first SBus struct
+        * and place in chain.
+        */
+       sbus = sbus_root = kmalloc(sizeof(struct sbus_bus), GFP_ATOMIC);
+       sbus->next = NULL;
+       sbus->prom_node = nd;
+       this_sbus = nd;
 
-               sdev = kzalloc(sizeof(struct sbus_dev), GFP_ATOMIC);
-               if (sdev) {
-                       sdev_insert(sdev, &sbus->devices);
+       if(iommund && sparc_cpu_model != sun4u && sparc_cpu_model != sun4d)
+               iommu_init(iommund, sbus);
 
-                       sdev->bus = sbus;
-                       sdev->parent = NULL;
-                       fill_sbus_device(dev_dp, sdev);
+       /* Loop until we find no more SBUS's */
+       while(this_sbus) {
+#ifdef CONFIG_SPARC64
+               /* IOMMU hides inside SBUS/SYSIO prom node on Ultra. */
+               if(sparc_cpu_model == sun4u) {
+                       extern void sbus_iommu_init(int prom_node, struct sbus_bus *sbus);
 
-                       walk_children(dev_dp, sdev, sbus);
+                       sbus_iommu_init(this_sbus, sbus);
+               }
+#endif /* CONFIG_SPARC64 */
+
+#ifdef CONFIG_SPARC32
+               if (sparc_cpu_model == sun4d)
+                       iounit_init(this_sbus, iommund, sbus);
+#endif /* CONFIG_SPARC32 */
+               printk("sbus%d: ", num_sbus);
+               sbus_clock = prom_getint(this_sbus, "clock-frequency");
+               if(sbus_clock == -1)
+                       sbus_clock = (25*1000*1000);
+               printk("Clock %d.%d MHz\n", (int) ((sbus_clock/1000)/1000),
+                      (int) (((sbus_clock/1000)%1000 != 0) ? 
+                             (((sbus_clock/1000)%1000) + 1000) : 0));
+
+               prom_getstring(this_sbus, "name",
+                              sbus->prom_name, sizeof(sbus->prom_name));
+               sbus->clock_freq = sbus_clock;
+#ifdef CONFIG_SPARC32
+               if (sparc_cpu_model == sun4d) {
+                       sbus->devid = prom_getint(iommund, "device-id");
+                       sbus->board = prom_getint(iommund, "board#");
+               }
+#endif
+               
+               sbus_bus_ranges_init(iommund, sbus);
+
+               sbus_devs = prom_getchild(this_sbus);
+               if (!sbus_devs) {
+                       sbus->devices = NULL;
+                       goto next_bus;
                }
-               dev_dp = dev_dp->sibling;
-       }
-
-       sbus_fixup_all_regs(sbus->devices);
-
-       dvma_init(sbus);
-}
 
-static int __init sbus_init(void)
-{
-       struct device_node *dp;
-       const char *sbus_name = "sbus";
-       int num_sbus = 0;
+               sbus->devices = kmalloc(sizeof(struct sbus_dev), GFP_ATOMIC);
+
+               this_dev = sbus->devices;
+               this_dev->next = NULL;
+
+               this_dev->bus = sbus;
+               this_dev->parent = NULL;
+               fill_sbus_device(sbus_devs, this_dev);
+
+               /* Should we traverse for children? */
+               if(prom_getchild(sbus_devs)) {
+                       /* Allocate device node */
+                       this_dev->child = kmalloc(sizeof(struct sbus_dev),
+                                                 GFP_ATOMIC);
+                       /* Fill it */
+                       this_dev->child->bus = sbus;
+                       this_dev->child->next = NULL;
+                       fill_sbus_device(prom_getchild(sbus_devs),
+                                        this_dev->child);
+                       sbus_do_child_siblings(prom_getchild(sbus_devs),
+                                              this_dev->child,
+                                              this_dev,
+                                              sbus);
+               } else {
+                       this_dev->child = NULL;
+               }
 
-       if (sbus_arch_preinit())
-               return 0;
+               while((sbus_devs = prom_getsibling(sbus_devs)) != 0) {
+                       /* Allocate device node */
+                       this_dev->next = kmalloc(sizeof(struct sbus_dev),
+                                                GFP_ATOMIC);
+                       this_dev = this_dev->next;
+                       this_dev->next = NULL;
+
+                       /* Fill it */
+                       this_dev->bus = sbus;
+                       this_dev->parent = NULL;
+                       fill_sbus_device(sbus_devs, this_dev);
+
+                       /* Is there a child node hanging off of us? */
+                       if(prom_getchild(sbus_devs)) {
+                               /* Get new device struct */
+                               this_dev->child = kmalloc(sizeof(struct sbus_dev),
+                                                         GFP_ATOMIC);
+                               /* Fill it */
+                               this_dev->child->bus = sbus;
+                               this_dev->child->next = NULL;
+                               fill_sbus_device(prom_getchild(sbus_devs),
+                                                this_dev->child);
+                               sbus_do_child_siblings(prom_getchild(sbus_devs),
+                                                      this_dev->child,
+                                                      this_dev,
+                                                      sbus);
+                       } else {
+                               this_dev->child = NULL;
+                       }
+               }
 
-       if (sparc_cpu_model == sun4d)
-               sbus_name = "sbi";
+               /* Walk all devices and apply parent ranges. */
+               sbus_fixup_all_regs(sbus->devices);
 
-       for_each_node_by_name(dp, sbus_name) {
-               build_one_sbus(dp, num_sbus);
+               dvma_init(sbus);
+       next_bus:
                num_sbus++;
+               if(sparc_cpu_model == sun4u) {
+                       this_sbus = prom_getsibling(this_sbus);
+                       if(!this_sbus)
+                               break;
+                       this_sbus = prom_searchsiblings(this_sbus, "sbus");
+               } else if(sparc_cpu_model == sun4d) {
+                       iommund = prom_getsibling(iommund);
+                       if(!iommund)
+                               break;
+                       iommund = prom_searchsiblings(iommund, "io-unit");
+                       if(!iommund)
+                               break;
+                       this_sbus = prom_searchsiblings(prom_getchild(iommund), "sbi");
+               } else {
+                       this_sbus = prom_getsibling(this_sbus);
+                       if(!this_sbus)
+                               break;
+                       this_sbus = prom_searchsiblings(this_sbus, "sbus");
+               }
+               if(this_sbus) {
+                       sbus->next = kmalloc(sizeof(struct sbus_bus), GFP_ATOMIC);
+                       sbus = sbus->next;
+                       sbus->next = NULL;
+                       sbus->prom_node = this_sbus;
+               } else {
+                       break;
+               }
+       } /* while(this_sbus) */
 
+       if (sparc_cpu_model == sun4d) {
+               extern void sun4d_init_sbi_irq(void);
+               sun4d_init_sbi_irq();
        }
-
-       sbus_arch_postinit();
+       
+#ifdef CONFIG_SPARC64
+       if (sparc_cpu_model == sun4u) {
+               firetruck_init();
+       }
+#endif
+#ifdef CONFIG_SUN_AUXIO
+       if (sparc_cpu_model == sun4u)
+               auxio_probe ();
+#endif
+#ifdef CONFIG_SPARC64
+       if (sparc_cpu_model == sun4u) {
+               extern void clock_probe(void);
+
+               clock_probe();
+       }
+#endif
 
        return 0;
 }