forked from mirrors/linux
		
	PCI: Put pci_dev in device tree as early as possible
We want to put pci_dev structs in the device tree as soon as possible so for_each_pci_dev() iteration will not miss them, but driver attachment needs to be delayed until after pci_assign_unassigned_resources() to make sure all devices have resources assigned first. This patch moves device registering from pci_bus_add_devices() to pci_device_add(), which happens earlier, leaving driver attachment in pci_bus_add_devices(). It also removes unattached child bus handling in pci_bus_add_devices(). That's not needed because child bus via pci_add_new_bus() is already in parent bus children list. [bhelgaas: changelog] Signed-off-by: Yinghai Lu <yinghai@kernel.org> Signed-off-by: Bjorn Helgaas <bhelgaas@google.com> Acked-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
This commit is contained in:
		
							parent
							
								
									58d9a38f6f
								
							
						
					
					
						commit
						4f535093cf
					
				
					 4 changed files with 41 additions and 82 deletions
				
			
		|  | @ -161,73 +161,35 @@ pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res, | ||||||
| void __weak pcibios_resource_survey_bus(struct pci_bus *bus) { } | void __weak pcibios_resource_survey_bus(struct pci_bus *bus) { } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * pci_bus_add_device - add a single device |  * pci_bus_add_device - start driver for a single device | ||||||
|  * @dev: device to add |  * @dev: device to add | ||||||
|  * |  * | ||||||
|  * This adds a single pci device to the global |  * This adds add sysfs entries and start device drivers | ||||||
|  * device list and adds sysfs and procfs entries |  | ||||||
|  */ |  */ | ||||||
| int pci_bus_add_device(struct pci_dev *dev) | int pci_bus_add_device(struct pci_dev *dev) | ||||||
| { | { | ||||||
| 	int retval; | 	int retval; | ||||||
| 
 | 
 | ||||||
| 	pci_fixup_device(pci_fixup_final, dev); | 	/*
 | ||||||
| 
 | 	 * Can not put in pci_device_add yet because resources | ||||||
| 	retval = pcibios_add_device(dev); | 	 * are not assigned yet for some devices. | ||||||
| 	if (retval) | 	 */ | ||||||
| 		return retval; | 	pci_create_sysfs_dev_files(dev); | ||||||
| 
 |  | ||||||
| 	dev->match_driver = false; |  | ||||||
| 	retval = device_add(&dev->dev); |  | ||||||
| 	if (retval) |  | ||||||
| 		return retval; |  | ||||||
| 
 | 
 | ||||||
| 	dev->match_driver = true; | 	dev->match_driver = true; | ||||||
| 	retval = device_attach(&dev->dev); | 	retval = device_attach(&dev->dev); | ||||||
| 	WARN_ON(retval < 0); | 	WARN_ON(retval < 0); | ||||||
| 
 | 
 | ||||||
| 	dev->is_added = 1; | 	dev->is_added = 1; | ||||||
| 	pci_proc_attach_device(dev); | 
 | ||||||
| 	pci_create_sysfs_dev_files(dev); |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * pci_bus_add_child - add a child bus |  * pci_bus_add_devices - start driver for PCI devices | ||||||
|  * @bus: bus to add |  | ||||||
|  * |  | ||||||
|  * This adds sysfs entries for a single bus |  | ||||||
|  */ |  | ||||||
| int pci_bus_add_child(struct pci_bus *bus) |  | ||||||
| { |  | ||||||
| 	int retval; |  | ||||||
| 
 |  | ||||||
| 	if (bus->bridge) |  | ||||||
| 		bus->dev.parent = bus->bridge; |  | ||||||
| 
 |  | ||||||
| 	retval = device_register(&bus->dev); |  | ||||||
| 	if (retval) |  | ||||||
| 		return retval; |  | ||||||
| 
 |  | ||||||
| 	bus->is_added = 1; |  | ||||||
| 
 |  | ||||||
| 	/* Create legacy_io and legacy_mem files for this bus */ |  | ||||||
| 	pci_create_legacy_files(bus); |  | ||||||
| 
 |  | ||||||
| 	return retval; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /**
 |  | ||||||
|  * pci_bus_add_devices - insert newly discovered PCI devices |  | ||||||
|  * @bus: bus to check for new devices |  * @bus: bus to check for new devices | ||||||
|  * |  * | ||||||
|  * Add newly discovered PCI devices (which are on the bus->devices |  * Start driver for PCI devices and add some sysfs entries. | ||||||
|  * list) to the global PCI device list, add the sysfs and procfs |  | ||||||
|  * entries.  Where a bridge is found, add the discovered bus to |  | ||||||
|  * the parents list of child buses, and recurse (breadth-first |  | ||||||
|  * to be compatible with 2.4) |  | ||||||
|  * |  | ||||||
|  * Call hotplug for each new devices. |  | ||||||
|  */ |  */ | ||||||
| void pci_bus_add_devices(const struct pci_bus *bus) | void pci_bus_add_devices(const struct pci_bus *bus) | ||||||
| { | { | ||||||
|  | @ -240,36 +202,20 @@ void pci_bus_add_devices(const struct pci_bus *bus) | ||||||
| 		if (dev->is_added) | 		if (dev->is_added) | ||||||
| 			continue; | 			continue; | ||||||
| 		retval = pci_bus_add_device(dev); | 		retval = pci_bus_add_device(dev); | ||||||
| 		if (retval) |  | ||||||
| 			dev_err(&dev->dev, "Error adding device, continuing\n"); |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	list_for_each_entry(dev, &bus->devices, bus_list) { | 	list_for_each_entry(dev, &bus->devices, bus_list) { | ||||||
| 		BUG_ON(!dev->is_added); | 		BUG_ON(!dev->is_added); | ||||||
| 
 | 
 | ||||||
| 		child = dev->subordinate; | 		child = dev->subordinate; | ||||||
| 		/*
 | 
 | ||||||
| 		 * If there is an unattached subordinate bus, attach |  | ||||||
| 		 * it and then scan for unattached PCI devices. |  | ||||||
| 		 */ |  | ||||||
| 		if (!child) | 		if (!child) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (list_empty(&child->node)) { |  | ||||||
| 			down_write(&pci_bus_sem); |  | ||||||
| 			list_add_tail(&child->node, &dev->bus->children); |  | ||||||
| 			up_write(&pci_bus_sem); |  | ||||||
| 		} |  | ||||||
| 		pci_bus_add_devices(child); | 		pci_bus_add_devices(child); | ||||||
| 
 | 
 | ||||||
| 		/*
 |  | ||||||
| 		 * register the bus with sysfs as the parent is now |  | ||||||
| 		 * properly registered. |  | ||||||
| 		 */ |  | ||||||
| 		if (child->is_added) | 		if (child->is_added) | ||||||
| 			continue; | 			continue; | ||||||
| 		retval = pci_bus_add_child(child); | 		child->is_added = 1; | ||||||
| 		if (retval) |  | ||||||
| 			dev_err(&dev->dev, "Error adding bus, continuing\n"); |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -48,12 +48,7 @@ static struct pci_bus *virtfn_add_bus(struct pci_bus *bus, int busnr) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	pci_bus_insert_busn_res(child, busnr, busnr); | 	pci_bus_insert_busn_res(child, busnr, busnr); | ||||||
| 	child->dev.parent = bus->bridge; | 	bus->is_added = 1; | ||||||
| 	rc = pci_bus_add_child(child); |  | ||||||
| 	if (rc) { |  | ||||||
| 		pci_remove_bus(child); |  | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	return child; | 	return child; | ||||||
| } | } | ||||||
|  | @ -123,8 +118,6 @@ static int virtfn_add(struct pci_dev *dev, int id, int reset) | ||||||
| 	virtfn->is_virtfn = 1; | 	virtfn->is_virtfn = 1; | ||||||
| 
 | 
 | ||||||
| 	rc = pci_bus_add_device(virtfn); | 	rc = pci_bus_add_device(virtfn); | ||||||
| 	if (rc) |  | ||||||
| 		goto failed1; |  | ||||||
| 	sprintf(buf, "virtfn%u", id); | 	sprintf(buf, "virtfn%u", id); | ||||||
| 	rc = sysfs_create_link(&dev->dev.kobj, &virtfn->dev.kobj, buf); | 	rc = sysfs_create_link(&dev->dev.kobj, &virtfn->dev.kobj, buf); | ||||||
| 	if (rc) | 	if (rc) | ||||||
|  |  | ||||||
|  | @ -203,7 +203,6 @@ extern int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type, | ||||||
| 				struct resource *res, unsigned int reg); | 				struct resource *res, unsigned int reg); | ||||||
| extern int pci_resource_bar(struct pci_dev *dev, int resno, | extern int pci_resource_bar(struct pci_dev *dev, int resno, | ||||||
| 			    enum pci_bar_type *type); | 			    enum pci_bar_type *type); | ||||||
| extern int pci_bus_add_child(struct pci_bus *bus); |  | ||||||
| extern void pci_enable_ari(struct pci_dev *dev); | extern void pci_enable_ari(struct pci_dev *dev); | ||||||
| /**
 | /**
 | ||||||
|  * pci_ari_enabled - query ARI forwarding status |  * pci_ari_enabled - query ARI forwarding status | ||||||
|  |  | ||||||
|  | @ -623,6 +623,7 @@ static struct pci_bus *pci_alloc_child_bus(struct pci_bus *parent, | ||||||
| { | { | ||||||
| 	struct pci_bus *child; | 	struct pci_bus *child; | ||||||
| 	int i; | 	int i; | ||||||
|  | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Allocate a new bus, and inherit stuff from the parent.. | 	 * Allocate a new bus, and inherit stuff from the parent.. | ||||||
|  | @ -637,8 +638,7 @@ static struct pci_bus *pci_alloc_child_bus(struct pci_bus *parent, | ||||||
| 	child->bus_flags = parent->bus_flags; | 	child->bus_flags = parent->bus_flags; | ||||||
| 
 | 
 | ||||||
| 	/* initialize some portions of the bus device, but don't register it
 | 	/* initialize some portions of the bus device, but don't register it
 | ||||||
| 	 * now as the parent is not properly set up yet.  This device will get | 	 * now as the parent is not properly set up yet. | ||||||
| 	 * registered later in pci_bus_add_devices() |  | ||||||
| 	 */ | 	 */ | ||||||
| 	child->dev.class = &pcibus_class; | 	child->dev.class = &pcibus_class; | ||||||
| 	dev_set_name(&child->dev, "%04x:%02x", pci_domain_nr(child), busnr); | 	dev_set_name(&child->dev, "%04x:%02x", pci_domain_nr(child), busnr); | ||||||
|  | @ -651,11 +651,14 @@ static struct pci_bus *pci_alloc_child_bus(struct pci_bus *parent, | ||||||
| 	child->primary = parent->busn_res.start; | 	child->primary = parent->busn_res.start; | ||||||
| 	child->busn_res.end = 0xff; | 	child->busn_res.end = 0xff; | ||||||
| 
 | 
 | ||||||
| 	if (!bridge) | 	if (!bridge) { | ||||||
| 		return child; | 		child->dev.parent = parent->bridge; | ||||||
|  | 		goto add_dev; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	child->self = bridge; | 	child->self = bridge; | ||||||
| 	child->bridge = get_device(&bridge->dev); | 	child->bridge = get_device(&bridge->dev); | ||||||
|  | 	child->dev.parent = child->bridge; | ||||||
| 	pci_set_bus_of_node(child); | 	pci_set_bus_of_node(child); | ||||||
| 	pci_set_bus_speed(child); | 	pci_set_bus_speed(child); | ||||||
| 
 | 
 | ||||||
|  | @ -666,6 +669,13 @@ static struct pci_bus *pci_alloc_child_bus(struct pci_bus *parent, | ||||||
| 	} | 	} | ||||||
| 	bridge->subordinate = child; | 	bridge->subordinate = child; | ||||||
| 
 | 
 | ||||||
|  | add_dev: | ||||||
|  | 	ret = device_register(&child->dev); | ||||||
|  | 	WARN_ON(ret < 0); | ||||||
|  | 
 | ||||||
|  | 	/* Create legacy_io and legacy_mem files for this bus */ | ||||||
|  | 	pci_create_legacy_files(child); | ||||||
|  | 
 | ||||||
| 	return child; | 	return child; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -1296,6 +1306,8 @@ static void pci_init_capabilities(struct pci_dev *dev) | ||||||
| 
 | 
 | ||||||
| void pci_device_add(struct pci_dev *dev, struct pci_bus *bus) | void pci_device_add(struct pci_dev *dev, struct pci_bus *bus) | ||||||
| { | { | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
| 	device_initialize(&dev->dev); | 	device_initialize(&dev->dev); | ||||||
| 	dev->dev.release = pci_release_dev; | 	dev->dev.release = pci_release_dev; | ||||||
| 
 | 
 | ||||||
|  | @ -1326,6 +1338,17 @@ void pci_device_add(struct pci_dev *dev, struct pci_bus *bus) | ||||||
| 	down_write(&pci_bus_sem); | 	down_write(&pci_bus_sem); | ||||||
| 	list_add_tail(&dev->bus_list, &bus->devices); | 	list_add_tail(&dev->bus_list, &bus->devices); | ||||||
| 	up_write(&pci_bus_sem); | 	up_write(&pci_bus_sem); | ||||||
|  | 
 | ||||||
|  | 	pci_fixup_device(pci_fixup_final, dev); | ||||||
|  | 	ret = pcibios_add_device(dev); | ||||||
|  | 	WARN_ON(ret < 0); | ||||||
|  | 
 | ||||||
|  | 	/* Notifier could use PCI capabilities */ | ||||||
|  | 	dev->match_driver = false; | ||||||
|  | 	ret = device_add(&dev->dev); | ||||||
|  | 	WARN_ON(ret < 0); | ||||||
|  | 
 | ||||||
|  | 	pci_proc_attach_device(dev); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct pci_dev *__ref pci_scan_single_device(struct pci_bus *bus, int devfn) | struct pci_dev *__ref pci_scan_single_device(struct pci_bus *bus, int devfn) | ||||||
|  | @ -1644,13 +1667,13 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | ||||||
| 	char bus_addr[64]; | 	char bus_addr[64]; | ||||||
| 	char *fmt; | 	char *fmt; | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 	b = pci_alloc_bus(); | 	b = pci_alloc_bus(); | ||||||
| 	if (!b) | 	if (!b) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	b->sysdata = sysdata; | 	b->sysdata = sysdata; | ||||||
| 	b->ops = ops; | 	b->ops = ops; | ||||||
|  | 	b->number = b->busn_res.start = bus; | ||||||
| 	b2 = pci_find_bus(pci_domain_nr(b), bus); | 	b2 = pci_find_bus(pci_domain_nr(b), bus); | ||||||
| 	if (b2) { | 	if (b2) { | ||||||
| 		/* If we already got to this bus through a different bridge, ignore it */ | 		/* If we already got to this bus through a different bridge, ignore it */ | ||||||
|  | @ -1685,8 +1708,6 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | ||||||
| 	/* Create legacy_io and legacy_mem files for this bus */ | 	/* Create legacy_io and legacy_mem files for this bus */ | ||||||
| 	pci_create_legacy_files(b); | 	pci_create_legacy_files(b); | ||||||
| 
 | 
 | ||||||
| 	b->number = b->busn_res.start = bus; |  | ||||||
| 
 |  | ||||||
| 	if (parent) | 	if (parent) | ||||||
| 		dev_info(parent, "PCI host bridge to bus %s\n", dev_name(&b->dev)); | 		dev_info(parent, "PCI host bridge to bus %s\n", dev_name(&b->dev)); | ||||||
| 	else | 	else | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Yinghai Lu
						Yinghai Lu