Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

PCI: Assign resources before drivers claim devices (pci_scan_root_bus())

Previously, pci_scan_root_bus() created a root PCI bus, enumerated the
devices on it, and called pci_bus_add_devices(), which made the devices
available for drivers to claim them.

Most callers assigned resources to devices after pci_scan_root_bus()
returns, which may be after drivers have claimed the devices. This is
incorrect; the PCI core should not change device resources while a driver
is managing the device.

Remove pci_bus_add_devices() from pci_scan_root_bus() and do it after any
resource assignment in the callers.

Note that ARM's pci_common_init_dev() already called pci_bus_add_devices()
after pci_scan_root_bus(), so we only need to remove the first call:

pci_common_init_dev
pcibios_init_hw
pci_scan_root_bus
pci_bus_add_devices # first call
pci_bus_assign_resources
pci_bus_add_devices # second call

[bhelgaas: changelog, drop "root_bus" var in alpha common_init_pci(),
return failure earlier in mn10300, add "return" in x86 pcibios_scan_root(),
return early if xtensa platform_pcibios_fixup() fails]
Signed-off-by: Yijing Wang <wangyijing@huawei.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
CC: Richard Henderson <rth@twiddle.net>
CC: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
CC: Matt Turner <mattst88@gmail.com>
CC: David Howells <dhowells@redhat.com>
CC: Tony Luck <tony.luck@intel.com>
CC: Michal Simek <monstr@monstr.eu>
CC: Ralf Baechle <ralf@linux-mips.org>
CC: Koichi Yasutake <yasutake.koichi@jp.panasonic.com>
CC: Sebastian Ott <sebott@linux.vnet.ibm.com>
CC: "David S. Miller" <davem@davemloft.net>
CC: Chris Metcalf <cmetcalf@ezchip.com>
CC: Chris Zankel <chris@zankel.net>
CC: Max Filippov <jcmvbkbc@gmail.com>
CC: Thomas Gleixner <tglx@linutronix.de>

authored by

Yijing Wang and committed by
Bjorn Helgaas
b97ea289 c90570d9

+47 -6
+7
arch/alpha/kernel/pci.c
··· 338 338 339 339 bus = pci_scan_root_bus(NULL, next_busno, alpha_mv.pci_ops, 340 340 hose, &resources); 341 + if (!bus) 342 + continue; 341 343 hose->bus = bus; 342 344 hose->need_domain_info = need_domain_info; 343 345 next_busno = bus->busn_res.end + 1; ··· 355 353 356 354 pci_assign_unassigned_resources(); 357 355 pci_fixup_irqs(alpha_mv.pci_swizzle, alpha_mv.pci_map_irq); 356 + for (hose = hose_head; hose; hose = hose->next) { 357 + bus = hose->bus; 358 + if (bus) 359 + pci_bus_add_devices(bus); 360 + } 358 361 } 359 362 360 363
+5 -1
arch/frv/mb93090-mb00/pci-vdk.c
··· 316 316 317 317 int __init pcibios_init(void) 318 318 { 319 + struct pci_bus *bus; 319 320 struct pci_ops *dir = NULL; 320 321 LIST_HEAD(resources); 321 322 ··· 384 383 printk("PCI: Probing PCI hardware\n"); 385 384 pci_add_resource(&resources, &pci_ioport_resource); 386 385 pci_add_resource(&resources, &pci_iomem_resource); 387 - pci_scan_root_bus(NULL, 0, pci_root_ops, NULL, &resources); 386 + bus = pci_scan_root_bus(NULL, 0, pci_root_ops, NULL, &resources); 388 387 389 388 pcibios_irq_init(); 390 389 pcibios_fixup_irqs(); 391 390 pcibios_resource_survey(); 391 + if (!bus) 392 + return 0; 392 393 394 + pci_bus_add_devices(bus); 393 395 return 0; 394 396 } 395 397
+2
arch/ia64/sn/kernel/io_init.c
··· 271 271 if (bus == NULL) { 272 272 kfree(res); 273 273 kfree(controller); 274 + return; 274 275 } 276 + pci_bus_add_devices(bus); 275 277 } 276 278 277 279 /*
+4
arch/microblaze/pci/pci-common.c
··· 1382 1382 1383 1383 /* Call common code to handle resource allocation */ 1384 1384 pcibios_resource_survey(); 1385 + list_for_each_entry_safe(hose, tmp, &hose_list, list_node) { 1386 + if (hose->bus) 1387 + pci_bus_add_devices(hose->bus); 1388 + } 1385 1389 1386 1390 return 0; 1387 1391 }
+1
arch/mips/pci/pci.c
··· 114 114 pci_bus_size_bridges(bus); 115 115 pci_bus_assign_resources(bus); 116 116 } 117 + pci_bus_add_devices(bus); 117 118 } 118 119 } 119 120
+5 -1
arch/mn10300/unit-asb2305/pci.c
··· 342 342 { 343 343 resource_size_t io_offset, mem_offset; 344 344 LIST_HEAD(resources); 345 + struct pci_bus *bus; 345 346 346 347 ioport_resource.start = 0xA0000000; 347 348 ioport_resource.end = 0xDFFFFFFF; ··· 372 371 373 372 pci_add_resource_offset(&resources, &pci_ioport_resource, io_offset); 374 373 pci_add_resource_offset(&resources, &pci_iomem_resource, mem_offset); 375 - pci_scan_root_bus(NULL, 0, &pci_direct_ampci, NULL, &resources); 374 + bus = pci_scan_root_bus(NULL, 0, &pci_direct_ampci, NULL, &resources); 375 + if (!bus) 376 + return 0; 376 377 377 378 pcibios_irq_init(); 378 379 pcibios_fixup_irqs(); 379 380 pcibios_resource_survey(); 381 + pci_bus_add_devices(bus); 380 382 return 0; 381 383 } 382 384
+1 -1
arch/s390/pci/pci.c
··· 776 776 zpci_cleanup_bus_resources(zdev); 777 777 return -EIO; 778 778 } 779 - 780 779 zdev->bus->max_bus_speed = zdev->max_bus_speed; 780 + pci_bus_add_devices(zdev->bus); 781 781 return 0; 782 782 } 783 783
+1
arch/sh/drivers/pci/pci.c
··· 69 69 70 70 pci_bus_size_bridges(bus); 71 71 pci_bus_assign_resources(bus); 72 + pci_bus_add_devices(bus); 72 73 } else { 73 74 pci_free_resource_list(&resources); 74 75 }
+1
arch/sparc/kernel/leon_pci.c
··· 40 40 41 41 /* Assign devices with resources */ 42 42 pci_assign_unassigned_resources(); 43 + pci_bus_add_devices(root_bus); 43 44 } else { 44 45 pci_free_resource_list(&resources); 45 46 }
+2
arch/tile/kernel/pci.c
··· 339 339 struct pci_bus *next_bus; 340 340 struct pci_dev *dev; 341 341 342 + pci_bus_add_devices(root_bus); 343 + 342 344 list_for_each_entry(dev, &root_bus->devices, bus_list) { 343 345 /* 344 346 * Find the PCI host controller, ie. the 1st
+2
arch/tile/kernel/pci_gx.c
··· 1030 1030 alloc_mem_map_failed: 1031 1031 break; 1032 1032 } 1033 + 1034 + pci_bus_add_devices(root_bus); 1033 1035 } 1034 1036 1035 1037 return 0;
+2
arch/x86/pci/common.c
··· 490 490 if (!bus) { 491 491 pci_free_resource_list(&resources); 492 492 kfree(sd); 493 + return; 493 494 } 495 + pci_bus_add_devices(bus); 494 496 } 495 497 496 498 void __init pcibios_set_cache_line_size(void)
+13 -2
arch/xtensa/kernel/pci.c
··· 174 174 struct pci_controller *pci_ctrl; 175 175 struct list_head resources; 176 176 struct pci_bus *bus; 177 - int next_busno = 0; 177 + int next_busno = 0, ret; 178 178 179 179 printk("PCI: Probing PCI hardware\n"); 180 180 ··· 185 185 pci_controller_apertures(pci_ctrl, &resources); 186 186 bus = pci_scan_root_bus(NULL, pci_ctrl->first_busno, 187 187 pci_ctrl->ops, pci_ctrl, &resources); 188 + if (!bus) 189 + continue; 190 + 188 191 pci_ctrl->bus = bus; 189 192 pci_ctrl->last_busno = bus->busn_res.end; 190 193 if (next_busno <= pci_ctrl->last_busno) 191 194 next_busno = pci_ctrl->last_busno+1; 192 195 } 193 196 pci_bus_count = next_busno; 197 + ret = platform_pcibios_fixup(); 198 + if (ret) 199 + return ret; 194 200 195 - return platform_pcibios_fixup(); 201 + for (pci_ctrl = pci_ctrl_head; pci_ctrl; pci_ctrl = pci_ctrl->next) { 202 + if (pci_ctrl->bus) 203 + pci_bus_add_devices(pci_ctrl->bus); 204 + } 205 + 206 + return 0; 196 207 } 197 208 198 209 subsys_initcall(pcibios_init);
+1
drivers/pci/host/pci-versatile.c
··· 214 214 215 215 pci_fixup_irqs(pci_common_swizzle, of_irq_parse_and_map_pci); 216 216 pci_assign_unassigned_bus_resources(bus); 217 + pci_bus_add_devices(bus); 217 218 218 219 return 0; 219 220 }
-1
drivers/pci/probe.c
··· 2087 2087 if (!found) 2088 2088 pci_bus_update_busn_res_end(b, max); 2089 2089 2090 - pci_bus_add_devices(b); 2091 2090 return b; 2092 2091 } 2093 2092 EXPORT_SYMBOL(pci_scan_root_bus);