Add explicit license headers to all files in src/device.
[coreboot.git] / src / devices / device.c
index 87e627d52cce6540dbf87229047d8f94f40b4b2a..67066e02653a69fdaae621af4b99a1581bbdbbd0 100644 (file)
@@ -1,7 +1,21 @@
+/*
+ * This file is part of the LinuxBIOS project.
+ *
+ * It was originally based on the Linux kernel (arch/i386/kernel/pci-pc.c).
+ *
+ * Modifications are:
+ * Copyright (C) 2003 Eric Biederman <ebiederm@xmission.com>
+ * Copyright (C) 2003-2004 Linux Networx
+ * (Written by Eric Biederman <ebiederman@lnxi.com> for Linux Networx)
+ * Copyright (C) 2003 Ronald G. Minnich <rminnich@gmail.com>
+ * Copyright (C) 2004-2005 Li-Ta Lo <ollie@lanl.gov>
+ * Copyright (C) 2005-2006 Tyan
+ * (Written by Yinghai Lu <yhlu@tyan.com> for Tyan)
+ * Copyright (C) 2005-2006 Stefan Reinauer <stepan@openbios.org>
+ */
+
 /*
  *      (c) 1999--2000 Martin Mares <mj@suse.cz>
- *      (c) 2003 Eric Biederman <ebiederm@xmission.com>
- *     (c) 2003 Linux Networx
  */
 /* lots of mods by ron minnich (rminnich@lanl.gov), with 
  * the final architecture guidance from Tom Merritt (tjm@codegen.com)
@@ -56,10 +70,12 @@ device_t alloc_dev(struct bus *parent, struct device_path *path)
        int link;
 
        spin_lock(&dev_lock);   
+
        /* Find the last child of our parent */
        for(child = parent->children; child && child->sibling; ) {
                child = child->sibling;
        }
+
        dev = malloc(sizeof(*dev));
        if (dev == 0) {
                die("DEV: out of memory.\n");
@@ -67,13 +83,12 @@ device_t alloc_dev(struct bus *parent, struct device_path *path)
        memset(dev, 0, sizeof(*dev));
        memcpy(&dev->path, path, sizeof(*path));
 
-
        /* Initialize the back pointers in the link fields */
        for(link = 0; link < MAX_LINKS; link++) {
                dev->link[link].dev  = dev;
                dev->link[link].link = link;
        }
-       
+
        /* By default devices are enabled */
        dev->enabled = 1;
 
@@ -153,8 +168,7 @@ static void read_resources(struct bus *bus)
                        }
                        if (!(links & (1 << link))) {
                                links |= (1 << link);
-                               read_resources(&curdev->link[resource->index]);
-                               
+                               read_resources(&curdev->link[link]);
                        }
                }
        }
@@ -180,6 +194,7 @@ static void pick_largest_resource(void *gp,
                state->seen_last = 1;
                return;
        }
+       if (resource->flags & IORESOURCE_FIXED ) return; //skip it 
        if (last && (
                    (last->align < resource->align) ||
                    ((last->align == resource->align) &&
@@ -192,7 +207,8 @@ static void pick_largest_resource(void *gp,
        if (!state->result || 
                (state->result->align < resource->align) ||
                ((state->result->align == resource->align) &&
-                       (state->result->size < resource->size))) {
+                       (state->result->size < resource->size)))
+       {
                state->result_dev = dev;
                state->result = resource;
        }    
@@ -262,7 +278,6 @@ void compute_allocate_resource(
                (bridge->flags & IORESOURCE_PREFETCH)? "prefmem" : "mem",
                base, bridge->size, bridge->align, bridge->gran);
 
-
        /* We want different minimum alignments for different kinds of
         * resources.  These minimums are not device type specific
         * but resource type specific.
@@ -290,7 +305,6 @@ void compute_allocate_resource(
                 * return them.   Some resources must be set even when they have
                 * no size.  PCI bridge resources are a good example of this.
                 */
-
                /* Propogate the resource alignment to the bridge register  */
                if (resource->align > bridge->align) {
                        bridge->align = resource->align;
@@ -302,9 +316,11 @@ void compute_allocate_resource(
                if (align < min_align) {
                        align = min_align;
                }
+
                if (resource->flags & IORESOURCE_FIXED) {
                        continue;
                }
+
                /* Propogate the resource limit to the bridge register */
                if (bridge->limit > resource->limit) {
                        bridge->limit = resource->limit;
@@ -366,47 +382,80 @@ void compute_allocate_resource(
 
 }
 
+#if CONFIG_CONSOLE_VGA == 1
+device_t vga_pri = 0;
 static void allocate_vga_resource(void)
 {
 #warning "FIXME modify allocate_vga_resource so it is less pci centric!"
 #warning "This function knows to much about PCI stuff, it should be just a ietrator/visitor."
 
        /* FIXME handle the VGA pallette snooping */
-       struct device *dev, *vga;
+       struct device *dev, *vga, *vga_onboard, *vga_first, *vga_last;
        struct bus *bus;
        bus = 0;
        vga = 0;
-       for (dev = all_devices; dev; dev = dev->next) {
+       vga_onboard = 0;
+       vga_first = 0;
+       vga_last = 0;
+       for(dev = all_devices; dev; dev = dev->next) {
+               if (!dev->enabled) continue;
                if (((dev->class >> 16) == PCI_BASE_CLASS_DISPLAY) &&
-                   ((dev->class >> 8) != PCI_CLASS_DISPLAY_OTHER)) {
-                       if (!vga) {
-                               printk_debug("Allocating VGA resource %s\n",
-                                            dev_path(dev));
-                               printk_debug("parent of the vga device %s\n",
-                                            dev_path(dev->bus->dev));
-                               vga = dev;
-                       }
-                       if (vga == dev) {
-                               /* All legacy VGA cards have MEM & I/O space registers */
-                               dev->command |= PCI_COMMAND_MEMORY | PCI_COMMAND_IO;
-                       } else {
-                               /* It isn't safe to enable other VGA cards */
-                               dev->command &= ~(PCI_COMMAND_MEMORY | PCI_COMMAND_IO);
-                       }
+                       ((dev->class >> 8) != PCI_CLASS_DISPLAY_OTHER)) 
+               {
+                        if (!vga_first) {
+                                if (dev->on_mainboard) {
+                                        vga_onboard = dev;
+                                } else {
+                                        vga_first = dev;
+                                }
+                        } else {
+                                if (dev->on_mainboard) {
+                                        vga_onboard = dev;
+                                } else {
+                                        vga_last = dev;
+                                }
+                        }
+
+                       /* It isn't safe to enable other VGA cards */
+                       dev->command &= ~(PCI_COMMAND_MEMORY | PCI_COMMAND_IO);
                }
        }
+       
+        vga = vga_last;
+
+        if(!vga) {
+                vga = vga_first;
+        }
+
+#if CONFIG_CONSOLE_VGA_ONBOARD_AT_FIRST == 1
+        if (vga_onboard) // will use on board vga as pri
+#else
+        if (!vga) // will use last add on adapter as pri
+#endif
+        {
+                vga = vga_onboard;
+        }
+
+       
        if (vga) {
+               /* vga is first add on card or the only onboard vga */
+               printk_debug("Allocating VGA resource %s\n", dev_path(vga));
+               /* All legacy VGA cards have MEM & I/O space registers */
+               vga->command |= (PCI_COMMAND_MEMORY | PCI_COMMAND_IO);
+               vga_pri = vga;
                bus = vga->bus;
        }
        /* Now walk up the bridges setting the VGA enable */
-       while (bus) {
-               printk_info("Enabling VGA forward on bus connect to %s\n",
-                           dev_path(bus->dev));
+       while(bus) {
+               printk_debug("Setting PCI_BRIDGE_CTL_VGA for bridge %s\n",
+                            dev_path(bus->dev));
                bus->bridge_ctrl |= PCI_BRIDGE_CTL_VGA;
                bus = (bus == bus->dev->bus)? 0 : bus->dev->bus;
        } 
 }
 
+#endif
+
 
 /**
  * @brief  Assign the computed resources to the devices on the bus.
@@ -428,7 +477,7 @@ void assign_resources(struct bus *bus)
        printk_spew("%s assign_resources, bus %d link: %d\n", 
                dev_path(bus->dev), bus->secondary, bus->link);
 
-       for (curdev = bus->children; curdev; curdev = curdev->sibling) {
+       for(curdev = bus->children; curdev; curdev = curdev->sibling) {
                if (!curdev->enabled || !curdev->resources) {
                        continue;
                }
@@ -473,6 +522,69 @@ void enable_resources(struct device *dev)
        dev->ops->enable_resources(dev);
 }
 
+/** 
+ * @brief Reset all of the devices a bus
+ *
+ * Reset all of the devices on a bus and clear the bus's reset_needed flag.
+ *
+ * @param bus pointer to the bus structure
+ *
+ * @return 1 if the bus was successfully reset, 0 otherwise.
+ *
+ */
+int reset_bus(struct bus *bus)
+{
+       if (bus && bus->dev && bus->dev->ops && bus->dev->ops->reset_bus)
+       {
+               bus->dev->ops->reset_bus(bus);
+               bus->reset_needed = 0;
+               return 1;
+       }
+       return 0;
+}
+
+/** 
+ * @brief Scan for devices on a bus.
+ *
+ * If there are bridges on the bus, recursively scan the buses behind the bridges.
+ * If the setting up and tuning of the bus causes a reset to be required, 
+ * reset the bus and scan it again.
+ *
+ * @param bus pointer to the bus device
+ * @param max current bus number
+ *
+ * @return The maximum bus number found, after scanning all subordinate busses
+ */
+unsigned int scan_bus(device_t bus, unsigned int max)
+{
+       unsigned int new_max;
+       int do_scan_bus;
+       if (    !bus ||
+               !bus->enabled ||
+               !bus->ops ||
+               !bus->ops->scan_bus)
+       {
+               return max;
+       }
+       do_scan_bus = 1;
+       while(do_scan_bus) {
+               int link;
+               new_max = bus->ops->scan_bus(bus, max);
+               do_scan_bus = 0;
+               for(link = 0; link < bus->links; link++) {
+                       if (bus->link[link].reset_needed) {
+                               if (reset_bus(&bus->link[link])) {
+                                       do_scan_bus = 1;
+                               } else {
+                                       bus->bus->reset_needed = 1;
+                               }
+                       }
+               }
+       }
+       return new_max;
+}
+
+
 /**
  * @brief Determine the existence of devices and extend the device tree.
  *
@@ -508,7 +620,7 @@ void dev_enumerate(void)
                printk_err("dev_root missing scan_bus operation");
                return;
        }
-       subordinate = root->ops->scan_bus(root, 0);
+       subordinate = scan_bus(root, 0);
        printk_info("done\n");
 }
 
@@ -545,7 +657,7 @@ void dev_configure(void)
 
        printk_info("Reading resources...\n");
        root->ops->read_resources(root);
-       printk_info("Done\n");
+       printk_info("Done reading resources.\n");
 
        /* Get the resources */
        io  = &root->resource[0];
@@ -561,19 +673,21 @@ void dev_configure(void)
        mem->flags |= IORESOURCE_ASSIGNED;
        mem->flags &= ~IORESOURCE_STORED;
 
+#if CONFIG_CONSOLE_VGA == 1
        /* Allocate the VGA I/O resource.. */
        allocate_vga_resource(); 
+#endif
 
        /* Store the computed resource allocations into device registers ... */
        printk_info("Setting resources...\n");
        root->ops->set_resources(root);
-       printk_info("Done\n");
+       printk_info("Done setting resources.\n");
 #if 0
        mem->flags |= IORESOURCE_STORED;
        report_resource_stored(root, mem, "");
 #endif
 
-       printk_info("done.\n");
+       printk_info("Done allocating resources.\n");
 }
 
 /**
@@ -584,7 +698,7 @@ void dev_configure(void)
  */
 void dev_enable(void)
 {
-       printk_info("Enabling resourcess...\n");
+       printk_info("Enabling resources...\n");
 
        /* now enable everything. */
        enable_resources(&dev_root);
@@ -604,10 +718,14 @@ void dev_initialize(void)
        struct device *dev;
 
        printk_info("Initializing devices...\n");
-       for (dev = all_devices; dev; dev = dev->next) {
+       for(dev = all_devices; dev; dev = dev->next) {
                if (dev->enabled && !dev->initialized && 
                        dev->ops && dev->ops->init) 
                {
+                       if (dev->path.type == DEVICE_PATH_I2C) {
+                               printk_debug("smbus: %s[%d]->",
+                                       dev_path(dev->bus->dev), dev->bus->link);
+                       }
                        printk_debug("%s init\n", dev_path(dev));
                        dev->initialized = 1;
                        dev->ops->init(dev);