4  * From setup-res.c, by:
 
   5  *      Dave Rusling (david.rusling@reo.mts.dec.com)
 
   6  *      David Mosberger (davidm@cs.arizona.edu)
 
   7  *      David Miller (davem@redhat.com)
 
   8  *      Ivan Kokshaysky (ink@jurassic.park.msu.ru)
 
  10 #include <linux/module.h>
 
  11 #include <linux/kernel.h>
 
  12 #include <linux/pci.h>
 
  13 #include <linux/errno.h>
 
  14 #include <linux/ioport.h>
 
  15 #include <linux/proc_fs.h>
 
  16 #include <linux/init.h>
 
  21  * pci_bus_alloc_resource - allocate a resource from a parent bus
 
  23  * @res: resource to allocate
 
  24  * @size: size of resource to allocate
 
  25  * @align: alignment of resource to allocate
 
  26  * @min: minimum /proc/iomem address to allocate
 
  27  * @type_mask: IORESOURCE_* type flags
 
  28  * @alignf: resource alignment function
 
  29  * @alignf_data: data argument for resource alignment function
 
  31  * Given the PCI bus a device resides on, the size, minimum address,
 
  32  * alignment and type, try to find an acceptable resource allocation
 
  33  * for a specific device resource.
 
  36 pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res,
 
  37                 resource_size_t size, resource_size_t align,
 
  38                 resource_size_t min, unsigned int type_mask,
 
  39                 void (*alignf)(void *, struct resource *, resource_size_t,
 
  45         type_mask |= IORESOURCE_IO | IORESOURCE_MEM;
 
  47         for (i = 0; i < PCI_BUS_NUM_RESOURCES; i++) {
 
  48                 struct resource *r = bus->resource[i];
 
  52                 /* type_mask must match */
 
  53                 if ((res->flags ^ r->flags) & type_mask)
 
  56                 /* We cannot allocate a non-prefetching resource
 
  57                    from a pre-fetching area */
 
  58                 if ((r->flags & IORESOURCE_PREFETCH) &&
 
  59                     !(res->flags & IORESOURCE_PREFETCH))
 
  62                 /* Ok, try it out.. */
 
  63                 ret = allocate_resource(r, res, size,
 
  77  * This adds a single pci device to the global
 
  78  * device list and adds sysfs and procfs entries
 
  80 int pci_bus_add_device(struct pci_dev *dev)
 
  83         retval = device_add(&dev->dev);
 
  87         down_write(&pci_bus_sem);
 
  88         list_add_tail(&dev->global_list, &pci_devices);
 
  89         up_write(&pci_bus_sem);
 
  91         pci_proc_attach_device(dev);
 
  92         pci_create_sysfs_dev_files(dev);
 
  97  * pci_bus_add_devices - insert newly discovered PCI devices
 
  98  * @bus: bus to check for new devices
 
 100  * Add newly discovered PCI devices (which are on the bus->devices
 
 101  * list) to the global PCI device list, add the sysfs and procfs
 
 102  * entries.  Where a bridge is found, add the discovered bus to
 
 103  * the parents list of child buses, and recurse (breadth-first
 
 104  * to be compatible with 2.4)
 
 106  * Call hotplug for each new devices.
 
 108 void pci_bus_add_devices(struct pci_bus *bus)
 
 111         struct pci_bus *child_bus;
 
 114         list_for_each_entry(dev, &bus->devices, bus_list) {
 
 116                  * Skip already-present devices (which are on the
 
 117                  * global device list.)
 
 119                 if (!list_empty(&dev->global_list))
 
 121                 retval = pci_bus_add_device(dev);
 
 123                         dev_err(&dev->dev, "Error adding device, continuing\n");
 
 126         list_for_each_entry(dev, &bus->devices, bus_list) {
 
 128                 BUG_ON(list_empty(&dev->global_list));
 
 131                  * If there is an unattached subordinate bus, attach
 
 132                  * it and then scan for unattached PCI devices.
 
 134                 if (dev->subordinate) {
 
 135                        if (list_empty(&dev->subordinate->node)) {
 
 136                                down_write(&pci_bus_sem);
 
 137                                list_add_tail(&dev->subordinate->node,
 
 138                                                &dev->bus->children);
 
 139                                up_write(&pci_bus_sem);
 
 141                         pci_bus_add_devices(dev->subordinate);
 
 143                         /* register the bus with sysfs as the parent is now
 
 144                          * properly registered. */
 
 145                         child_bus = dev->subordinate;
 
 146                         if (child_bus->is_added)
 
 148                         child_bus->dev.parent = child_bus->bridge;
 
 149                         retval = device_register(&child_bus->dev);
 
 151                                 dev_err(&dev->dev, "Error registering pci_bus,"
 
 154                                 child_bus->is_added = 1;
 
 155                                 retval = device_create_file(&child_bus->dev,
 
 156                                                         &dev_attr_cpuaffinity);
 
 159                                 dev_err(&dev->dev, "Error creating cpuaffinity"
 
 160                                         " file, continuing...\n");
 
 165 void pci_enable_bridges(struct pci_bus *bus)
 
 170         list_for_each_entry(dev, &bus->devices, bus_list) {
 
 171                 if (dev->subordinate) {
 
 172                         retval = pci_enable_device(dev);
 
 174                         pci_enable_bridges(dev->subordinate);
 
 179 /** pci_walk_bus - walk devices on/under bus, calling callback.
 
 180  *  @top      bus whose devices should be walked
 
 181  *  @cb       callback to be called for each device found
 
 182  *  @userdata arbitrary pointer to be passed to callback.
 
 184  *  Walk the given bus, including any bridged devices
 
 185  *  on buses under this bus.  Call the provided callback
 
 186  *  on each device found.
 
 188 void pci_walk_bus(struct pci_bus *top, void (*cb)(struct pci_dev *, void *),
 
 193         struct list_head *next;
 
 196         down_read(&pci_bus_sem);
 
 197         next = top->devices.next;
 
 199                 if (next == &bus->devices) {
 
 200                         /* end of this bus, go up or finish */
 
 203                         next = bus->self->bus_list.next;
 
 204                         bus = bus->self->bus;
 
 207                 dev = list_entry(next, struct pci_dev, bus_list);
 
 208                 if (dev->subordinate) {
 
 209                         /* this is a pci-pci bridge, do its devices next */
 
 210                         next = dev->subordinate->devices.next;
 
 211                         bus = dev->subordinate;
 
 213                         next = dev->bus_list.next;
 
 215                 /* Run device routines with the device locked */
 
 220         up_read(&pci_bus_sem);
 
 223 EXPORT_SYMBOL(pci_bus_alloc_resource);
 
 224 EXPORT_SYMBOL_GPL(pci_bus_add_device);
 
 225 EXPORT_SYMBOL(pci_bus_add_devices);
 
 226 EXPORT_SYMBOL(pci_enable_bridges);