diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-05-21 19:24:54 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-05-21 19:24:54 -0400 |
commit | 3bb07f1b73ea6313b843807063e183e168c9182a (patch) | |
tree | f0e2ab77b8bc993a843a0edede00668c589863cc | |
parent | 6326c71fd2fb3bef5fa33951479298b683da35fe (diff) | |
parent | 5420e46d4d79bcd5d5952df98d022c8412385d32 (diff) |
Merge tag 'pci-for-3.5' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci
Pull PCI changes from Bjorn Helgaas:
- Host bridge cleanups from Yinghai
- Disable Bus Master bit on PCI device shutdown (kexec-related)
- Stratus ftServer fix
- pci_dev_reset() locking fix
- IvyBridge graphics erratum workaround
* tag 'pci-for-3.5' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci: (21 commits)
microblaze/PCI: fix "io_offset undeclared" error
x86/PCI: only check for spinlock being held in SMP kernels
resources: add resource_overlaps()
PCI: fix uninitialized variable 'cap_mask'
MAINTAINERS: update PCI git tree and patchwork
PCI: disable Bus Master on PCI device shutdown
PCI: work around IvyBridge internal graphics FLR erratum
x86/PCI: fix unused variable warning in amd_bus.c
PCI: move mutex locking out of pci_dev_reset function
PCI: work around Stratus ftServer broken PCIe hierarchy
x86/PCI: merge pcibios_scan_root() and pci_scan_bus_on_node()
x86/PCI: dynamically allocate pci_root_info for native host bridge drivers
x86/PCI: embed pci_sysdata into pci_root_info on ACPI path
x86/PCI: embed name into pci_root_info struct
x86/PCI: add host bridge resource release for _CRS path
x86/PCI: refactor get_current_resources()
PCI: add host bridge release support
PCI: add generic device into pci_host_bridge struct
PCI: rename pci_host_bridge() to find_pci_root_bridge()
x86/PCI: fix memleak with get_current_resources()
...
-rw-r--r-- | Documentation/kernel-parameters.txt | 3 | ||||
-rw-r--r-- | MAINTAINERS | 10 | ||||
-rw-r--r-- | arch/microblaze/pci/pci-common.c | 1 | ||||
-rw-r--r-- | arch/x86/pci/acpi.c | 128 | ||||
-rw-r--r-- | arch/x86/pci/amd_bus.c | 91 | ||||
-rw-r--r-- | arch/x86/pci/broadcom_bus.c | 12 | ||||
-rw-r--r-- | arch/x86/pci/bus_numa.c | 69 | ||||
-rw-r--r-- | arch/x86/pci/bus_numa.h | 18 | ||||
-rw-r--r-- | arch/x86/pci/common.c | 43 | ||||
-rw-r--r-- | arch/x86/pci/i386.c | 2 | ||||
-rw-r--r-- | drivers/pci/Makefile | 2 | ||||
-rw-r--r-- | drivers/pci/host-bridge.c | 96 | ||||
-rw-r--r-- | drivers/pci/pci-driver.c | 6 | ||||
-rw-r--r-- | drivers/pci/pci.c | 30 | ||||
-rw-r--r-- | drivers/pci/pcie/portdrv_core.c | 2 | ||||
-rw-r--r-- | drivers/pci/probe.c | 154 | ||||
-rw-r--r-- | drivers/pci/quirks.c | 58 | ||||
-rw-r--r-- | include/asm-generic/pci-bridge.h | 6 | ||||
-rw-r--r-- | include/linux/ioport.h | 7 | ||||
-rw-r--r-- | include/linux/pci.h | 9 |
20 files changed, 442 insertions, 305 deletions
diff --git a/Documentation/kernel-parameters.txt b/Documentation/kernel-parameters.txt index c1601e5a8b71..f995195409fd 100644 --- a/Documentation/kernel-parameters.txt +++ b/Documentation/kernel-parameters.txt | |||
@@ -2161,6 +2161,9 @@ bytes respectively. Such letter suffixes can also be entirely omitted. | |||
2161 | on: Turn realloc on | 2161 | on: Turn realloc on |
2162 | realloc same as realloc=on | 2162 | realloc same as realloc=on |
2163 | noari do not use PCIe ARI. | 2163 | noari do not use PCIe ARI. |
2164 | pcie_scan_all Scan all possible PCIe devices. Otherwise we | ||
2165 | only look for one device below a PCIe downstream | ||
2166 | port. | ||
2164 | 2167 | ||
2165 | pcie_aspm= [PCIE] Forcibly enable or disable PCIe Active State Power | 2168 | pcie_aspm= [PCIE] Forcibly enable or disable PCIe Active State Power |
2166 | Management. | 2169 | Management. |
diff --git a/MAINTAINERS b/MAINTAINERS index 490dd6e640ac..7560921a4e15 100644 --- a/MAINTAINERS +++ b/MAINTAINERS | |||
@@ -5128,19 +5128,13 @@ F: Documentation/powerpc/eeh-pci-error-recovery.txt | |||
5128 | PCI SUBSYSTEM | 5128 | PCI SUBSYSTEM |
5129 | M: Bjorn Helgaas <bhelgaas@google.com> | 5129 | M: Bjorn Helgaas <bhelgaas@google.com> |
5130 | L: linux-pci@vger.kernel.org | 5130 | L: linux-pci@vger.kernel.org |
5131 | Q: http://patchwork.kernel.org/project/linux-pci/list/ | 5131 | Q: http://patchwork.ozlabs.org/project/linux-pci/list/ |
5132 | T: git git://git.kernel.org/pub/scm/linux/kernel/git/jbarnes/pci.git | 5132 | T: git git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/linux.git |
5133 | S: Supported | 5133 | S: Supported |
5134 | F: Documentation/PCI/ | 5134 | F: Documentation/PCI/ |
5135 | F: drivers/pci/ | 5135 | F: drivers/pci/ |
5136 | F: include/linux/pci* | 5136 | F: include/linux/pci* |
5137 | 5137 | ||
5138 | PCI HOTPLUG | ||
5139 | M: Bjorn Helgaas <bhelgaas@google.com> | ||
5140 | L: linux-pci@vger.kernel.org | ||
5141 | S: Supported | ||
5142 | F: drivers/pci/hotplug | ||
5143 | |||
5144 | PCMCIA SUBSYSTEM | 5138 | PCMCIA SUBSYSTEM |
5145 | P: Linux PCMCIA Team | 5139 | P: Linux PCMCIA Team |
5146 | L: linux-pcmcia@lists.infradead.org | 5140 | L: linux-pcmcia@lists.infradead.org |
diff --git a/arch/microblaze/pci/pci-common.c b/arch/microblaze/pci/pci-common.c index d10403dadd2b..ed22bfc5db14 100644 --- a/arch/microblaze/pci/pci-common.c +++ b/arch/microblaze/pci/pci-common.c | |||
@@ -1422,6 +1422,7 @@ int pcibios_enable_device(struct pci_dev *dev, int mask) | |||
1422 | 1422 | ||
1423 | static void __devinit pcibios_setup_phb_resources(struct pci_controller *hose, struct list_head *resources) | 1423 | static void __devinit pcibios_setup_phb_resources(struct pci_controller *hose, struct list_head *resources) |
1424 | { | 1424 | { |
1425 | unsigned long io_offset; | ||
1425 | struct resource *res; | 1426 | struct resource *res; |
1426 | int i; | 1427 | int i; |
1427 | 1428 | ||
diff --git a/arch/x86/pci/acpi.c b/arch/x86/pci/acpi.c index ed2835e148b5..fc09c2754e08 100644 --- a/arch/x86/pci/acpi.c +++ b/arch/x86/pci/acpi.c | |||
@@ -9,11 +9,11 @@ | |||
9 | 9 | ||
10 | struct pci_root_info { | 10 | struct pci_root_info { |
11 | struct acpi_device *bridge; | 11 | struct acpi_device *bridge; |
12 | char *name; | 12 | char name[16]; |
13 | unsigned int res_num; | 13 | unsigned int res_num; |
14 | struct resource *res; | 14 | struct resource *res; |
15 | struct list_head *resources; | ||
16 | int busnum; | 15 | int busnum; |
16 | struct pci_sysdata sd; | ||
17 | }; | 17 | }; |
18 | 18 | ||
19 | static bool pci_use_crs = true; | 19 | static bool pci_use_crs = true; |
@@ -245,13 +245,6 @@ setup_resource(struct acpi_resource *acpi_res, void *data) | |||
245 | return AE_OK; | 245 | return AE_OK; |
246 | } | 246 | } |
247 | 247 | ||
248 | static bool resource_contains(struct resource *res, resource_size_t point) | ||
249 | { | ||
250 | if (res->start <= point && point <= res->end) | ||
251 | return true; | ||
252 | return false; | ||
253 | } | ||
254 | |||
255 | static void coalesce_windows(struct pci_root_info *info, unsigned long type) | 248 | static void coalesce_windows(struct pci_root_info *info, unsigned long type) |
256 | { | 249 | { |
257 | int i, j; | 250 | int i, j; |
@@ -272,10 +265,7 @@ static void coalesce_windows(struct pci_root_info *info, unsigned long type) | |||
272 | * our resources no longer match the ACPI _CRS, but | 265 | * our resources no longer match the ACPI _CRS, but |
273 | * the kernel resource tree doesn't allow overlaps. | 266 | * the kernel resource tree doesn't allow overlaps. |
274 | */ | 267 | */ |
275 | if (resource_contains(res1, res2->start) || | 268 | if (resource_overlaps(res1, res2)) { |
276 | resource_contains(res1, res2->end) || | ||
277 | resource_contains(res2, res1->start) || | ||
278 | resource_contains(res2, res1->end)) { | ||
279 | res1->start = min(res1->start, res2->start); | 269 | res1->start = min(res1->start, res2->start); |
280 | res1->end = max(res1->end, res2->end); | 270 | res1->end = max(res1->end, res2->end); |
281 | dev_info(&info->bridge->dev, | 271 | dev_info(&info->bridge->dev, |
@@ -287,7 +277,8 @@ static void coalesce_windows(struct pci_root_info *info, unsigned long type) | |||
287 | } | 277 | } |
288 | } | 278 | } |
289 | 279 | ||
290 | static void add_resources(struct pci_root_info *info) | 280 | static void add_resources(struct pci_root_info *info, |
281 | struct list_head *resources) | ||
291 | { | 282 | { |
292 | int i; | 283 | int i; |
293 | struct resource *res, *root, *conflict; | 284 | struct resource *res, *root, *conflict; |
@@ -311,53 +302,74 @@ static void add_resources(struct pci_root_info *info) | |||
311 | "ignoring host bridge window %pR (conflicts with %s %pR)\n", | 302 | "ignoring host bridge window %pR (conflicts with %s %pR)\n", |
312 | res, conflict->name, conflict); | 303 | res, conflict->name, conflict); |
313 | else | 304 | else |
314 | pci_add_resource(info->resources, res); | 305 | pci_add_resource(resources, res); |
315 | } | 306 | } |
316 | } | 307 | } |
317 | 308 | ||
309 | static void free_pci_root_info_res(struct pci_root_info *info) | ||
310 | { | ||
311 | kfree(info->res); | ||
312 | info->res = NULL; | ||
313 | info->res_num = 0; | ||
314 | } | ||
315 | |||
316 | static void __release_pci_root_info(struct pci_root_info *info) | ||
317 | { | ||
318 | int i; | ||
319 | struct resource *res; | ||
320 | |||
321 | for (i = 0; i < info->res_num; i++) { | ||
322 | res = &info->res[i]; | ||
323 | |||
324 | if (!res->parent) | ||
325 | continue; | ||
326 | |||
327 | if (!(res->flags & (IORESOURCE_MEM | IORESOURCE_IO))) | ||
328 | continue; | ||
329 | |||
330 | release_resource(res); | ||
331 | } | ||
332 | |||
333 | free_pci_root_info_res(info); | ||
334 | |||
335 | kfree(info); | ||
336 | } | ||
337 | static void release_pci_root_info(struct pci_host_bridge *bridge) | ||
338 | { | ||
339 | struct pci_root_info *info = bridge->release_data; | ||
340 | |||
341 | __release_pci_root_info(info); | ||
342 | } | ||
343 | |||
318 | static void | 344 | static void |
319 | get_current_resources(struct acpi_device *device, int busnum, | 345 | probe_pci_root_info(struct pci_root_info *info, struct acpi_device *device, |
320 | int domain, struct list_head *resources) | 346 | int busnum, int domain) |
321 | { | 347 | { |
322 | struct pci_root_info info; | ||
323 | size_t size; | 348 | size_t size; |
324 | 349 | ||
325 | info.bridge = device; | 350 | info->bridge = device; |
326 | info.res_num = 0; | 351 | info->res_num = 0; |
327 | info.resources = resources; | ||
328 | acpi_walk_resources(device->handle, METHOD_NAME__CRS, count_resource, | 352 | acpi_walk_resources(device->handle, METHOD_NAME__CRS, count_resource, |
329 | &info); | 353 | info); |
330 | if (!info.res_num) | 354 | if (!info->res_num) |
331 | return; | 355 | return; |
332 | 356 | ||
333 | size = sizeof(*info.res) * info.res_num; | 357 | size = sizeof(*info->res) * info->res_num; |
334 | info.res = kmalloc(size, GFP_KERNEL); | 358 | info->res_num = 0; |
335 | if (!info.res) | 359 | info->res = kmalloc(size, GFP_KERNEL); |
360 | if (!info->res) | ||
336 | return; | 361 | return; |
337 | 362 | ||
338 | info.name = kasprintf(GFP_KERNEL, "PCI Bus %04x:%02x", domain, busnum); | 363 | sprintf(info->name, "PCI Bus %04x:%02x", domain, busnum); |
339 | if (!info.name) | ||
340 | goto name_alloc_fail; | ||
341 | 364 | ||
342 | info.res_num = 0; | ||
343 | acpi_walk_resources(device->handle, METHOD_NAME__CRS, setup_resource, | 365 | acpi_walk_resources(device->handle, METHOD_NAME__CRS, setup_resource, |
344 | &info); | 366 | info); |
345 | |||
346 | if (pci_use_crs) { | ||
347 | add_resources(&info); | ||
348 | |||
349 | return; | ||
350 | } | ||
351 | |||
352 | kfree(info.name); | ||
353 | |||
354 | name_alloc_fail: | ||
355 | kfree(info.res); | ||
356 | } | 367 | } |
357 | 368 | ||
358 | struct pci_bus * __devinit pci_acpi_scan_root(struct acpi_pci_root *root) | 369 | struct pci_bus * __devinit pci_acpi_scan_root(struct acpi_pci_root *root) |
359 | { | 370 | { |
360 | struct acpi_device *device = root->device; | 371 | struct acpi_device *device = root->device; |
372 | struct pci_root_info *info = NULL; | ||
361 | int domain = root->segment; | 373 | int domain = root->segment; |
362 | int busnum = root->secondary.start; | 374 | int busnum = root->secondary.start; |
363 | LIST_HEAD(resources); | 375 | LIST_HEAD(resources); |
@@ -389,17 +401,14 @@ struct pci_bus * __devinit pci_acpi_scan_root(struct acpi_pci_root *root) | |||
389 | if (node != -1 && !node_online(node)) | 401 | if (node != -1 && !node_online(node)) |
390 | node = -1; | 402 | node = -1; |
391 | 403 | ||
392 | /* Allocate per-root-bus (not per bus) arch-specific data. | 404 | info = kzalloc(sizeof(*info), GFP_KERNEL); |
393 | * TODO: leak; this memory is never freed. | 405 | if (!info) { |
394 | * It's arguable whether it's worth the trouble to care. | ||
395 | */ | ||
396 | sd = kzalloc(sizeof(*sd), GFP_KERNEL); | ||
397 | if (!sd) { | ||
398 | printk(KERN_WARNING "pci_bus %04x:%02x: " | 406 | printk(KERN_WARNING "pci_bus %04x:%02x: " |
399 | "ignored (out of memory)\n", domain, busnum); | 407 | "ignored (out of memory)\n", domain, busnum); |
400 | return NULL; | 408 | return NULL; |
401 | } | 409 | } |
402 | 410 | ||
411 | sd = &info->sd; | ||
403 | sd->domain = domain; | 412 | sd->domain = domain; |
404 | sd->node = node; | 413 | sd->node = node; |
405 | /* | 414 | /* |
@@ -413,22 +422,32 @@ struct pci_bus * __devinit pci_acpi_scan_root(struct acpi_pci_root *root) | |||
413 | * be replaced by sd. | 422 | * be replaced by sd. |
414 | */ | 423 | */ |
415 | memcpy(bus->sysdata, sd, sizeof(*sd)); | 424 | memcpy(bus->sysdata, sd, sizeof(*sd)); |
416 | kfree(sd); | 425 | kfree(info); |
417 | } else { | 426 | } else { |
418 | get_current_resources(device, busnum, domain, &resources); | 427 | probe_pci_root_info(info, device, busnum, domain); |
419 | 428 | ||
420 | /* | 429 | /* |
421 | * _CRS with no apertures is normal, so only fall back to | 430 | * _CRS with no apertures is normal, so only fall back to |
422 | * defaults or native bridge info if we're ignoring _CRS. | 431 | * defaults or native bridge info if we're ignoring _CRS. |
423 | */ | 432 | */ |
424 | if (!pci_use_crs) | 433 | if (pci_use_crs) |
434 | add_resources(info, &resources); | ||
435 | else { | ||
436 | free_pci_root_info_res(info); | ||
425 | x86_pci_root_bus_resources(busnum, &resources); | 437 | x86_pci_root_bus_resources(busnum, &resources); |
438 | } | ||
439 | |||
426 | bus = pci_create_root_bus(NULL, busnum, &pci_root_ops, sd, | 440 | bus = pci_create_root_bus(NULL, busnum, &pci_root_ops, sd, |
427 | &resources); | 441 | &resources); |
428 | if (bus) | 442 | if (bus) { |
429 | bus->subordinate = pci_scan_child_bus(bus); | 443 | bus->subordinate = pci_scan_child_bus(bus); |
430 | else | 444 | pci_set_host_bridge_release( |
445 | to_pci_host_bridge(bus->bridge), | ||
446 | release_pci_root_info, info); | ||
447 | } else { | ||
431 | pci_free_resource_list(&resources); | 448 | pci_free_resource_list(&resources); |
449 | __release_pci_root_info(info); | ||
450 | } | ||
432 | } | 451 | } |
433 | 452 | ||
434 | /* After the PCI-E bus has been walked and all devices discovered, | 453 | /* After the PCI-E bus has been walked and all devices discovered, |
@@ -445,9 +464,6 @@ struct pci_bus * __devinit pci_acpi_scan_root(struct acpi_pci_root *root) | |||
445 | } | 464 | } |
446 | } | 465 | } |
447 | 466 | ||
448 | if (!bus) | ||
449 | kfree(sd); | ||
450 | |||
451 | if (bus && node != -1) { | 467 | if (bus && node != -1) { |
452 | #ifdef CONFIG_ACPI_NUMA | 468 | #ifdef CONFIG_ACPI_NUMA |
453 | if (pxm >= 0) | 469 | if (pxm >= 0) |
diff --git a/arch/x86/pci/amd_bus.c b/arch/x86/pci/amd_bus.c index 0567df3890e1..5aed49bff058 100644 --- a/arch/x86/pci/amd_bus.c +++ b/arch/x86/pci/amd_bus.c | |||
@@ -32,6 +32,27 @@ static struct pci_hostbridge_probe pci_probes[] __initdata = { | |||
32 | 32 | ||
33 | #define RANGE_NUM 16 | 33 | #define RANGE_NUM 16 |
34 | 34 | ||
35 | static struct pci_root_info __init *find_pci_root_info(int node, int link) | ||
36 | { | ||
37 | struct pci_root_info *info; | ||
38 | |||
39 | /* find the position */ | ||
40 | list_for_each_entry(info, &pci_root_infos, list) | ||
41 | if (info->node == node && info->link == link) | ||
42 | return info; | ||
43 | |||
44 | return NULL; | ||
45 | } | ||
46 | |||
47 | static void __init set_mp_bus_range_to_node(int min_bus, int max_bus, int node) | ||
48 | { | ||
49 | #ifdef CONFIG_NUMA | ||
50 | int j; | ||
51 | |||
52 | for (j = min_bus; j <= max_bus; j++) | ||
53 | set_mp_bus_to_node(j, node); | ||
54 | #endif | ||
55 | } | ||
35 | /** | 56 | /** |
36 | * early_fill_mp_bus_to_node() | 57 | * early_fill_mp_bus_to_node() |
37 | * called before pcibios_scan_root and pci_scan_bus | 58 | * called before pcibios_scan_root and pci_scan_bus |
@@ -41,7 +62,6 @@ static struct pci_hostbridge_probe pci_probes[] __initdata = { | |||
41 | static int __init early_fill_mp_bus_info(void) | 62 | static int __init early_fill_mp_bus_info(void) |
42 | { | 63 | { |
43 | int i; | 64 | int i; |
44 | int j; | ||
45 | unsigned bus; | 65 | unsigned bus; |
46 | unsigned slot; | 66 | unsigned slot; |
47 | int node; | 67 | int node; |
@@ -50,7 +70,6 @@ static int __init early_fill_mp_bus_info(void) | |||
50 | int def_link; | 70 | int def_link; |
51 | struct pci_root_info *info; | 71 | struct pci_root_info *info; |
52 | u32 reg; | 72 | u32 reg; |
53 | struct resource *res; | ||
54 | u64 start; | 73 | u64 start; |
55 | u64 end; | 74 | u64 end; |
56 | struct range range[RANGE_NUM]; | 75 | struct range range[RANGE_NUM]; |
@@ -86,7 +105,6 @@ static int __init early_fill_mp_bus_info(void) | |||
86 | if (!found) | 105 | if (!found) |
87 | return 0; | 106 | return 0; |
88 | 107 | ||
89 | pci_root_num = 0; | ||
90 | for (i = 0; i < 4; i++) { | 108 | for (i = 0; i < 4; i++) { |
91 | int min_bus; | 109 | int min_bus; |
92 | int max_bus; | 110 | int max_bus; |
@@ -99,19 +117,11 @@ static int __init early_fill_mp_bus_info(void) | |||
99 | min_bus = (reg >> 16) & 0xff; | 117 | min_bus = (reg >> 16) & 0xff; |
100 | max_bus = (reg >> 24) & 0xff; | 118 | max_bus = (reg >> 24) & 0xff; |
101 | node = (reg >> 4) & 0x07; | 119 | node = (reg >> 4) & 0x07; |
102 | #ifdef CONFIG_NUMA | 120 | set_mp_bus_range_to_node(min_bus, max_bus, node); |
103 | for (j = min_bus; j <= max_bus; j++) | ||
104 | set_mp_bus_to_node(j, node); | ||
105 | #endif | ||
106 | link = (reg >> 8) & 0x03; | 121 | link = (reg >> 8) & 0x03; |
107 | 122 | ||
108 | info = &pci_root_info[pci_root_num]; | 123 | info = alloc_pci_root_info(min_bus, max_bus, node, link); |
109 | info->bus_min = min_bus; | ||
110 | info->bus_max = max_bus; | ||
111 | info->node = node; | ||
112 | info->link = link; | ||
113 | sprintf(info->name, "PCI Bus #%02x", min_bus); | 124 | sprintf(info->name, "PCI Bus #%02x", min_bus); |
114 | pci_root_num++; | ||
115 | } | 125 | } |
116 | 126 | ||
117 | /* get the default node and link for left over res */ | 127 | /* get the default node and link for left over res */ |
@@ -134,16 +144,10 @@ static int __init early_fill_mp_bus_info(void) | |||
134 | link = (reg >> 4) & 0x03; | 144 | link = (reg >> 4) & 0x03; |
135 | end = (reg & 0xfff000) | 0xfff; | 145 | end = (reg & 0xfff000) | 0xfff; |
136 | 146 | ||
137 | /* find the position */ | 147 | info = find_pci_root_info(node, link); |
138 | for (j = 0; j < pci_root_num; j++) { | 148 | if (!info) |
139 | info = &pci_root_info[j]; | ||
140 | if (info->node == node && info->link == link) | ||
141 | break; | ||
142 | } | ||
143 | if (j == pci_root_num) | ||
144 | continue; /* not found */ | 149 | continue; /* not found */ |
145 | 150 | ||
146 | info = &pci_root_info[j]; | ||
147 | printk(KERN_DEBUG "node %d link %d: io port [%llx, %llx]\n", | 151 | printk(KERN_DEBUG "node %d link %d: io port [%llx, %llx]\n", |
148 | node, link, start, end); | 152 | node, link, start, end); |
149 | 153 | ||
@@ -155,13 +159,8 @@ static int __init early_fill_mp_bus_info(void) | |||
155 | } | 159 | } |
156 | /* add left over io port range to def node/link, [0, 0xffff] */ | 160 | /* add left over io port range to def node/link, [0, 0xffff] */ |
157 | /* find the position */ | 161 | /* find the position */ |
158 | for (j = 0; j < pci_root_num; j++) { | 162 | info = find_pci_root_info(def_node, def_link); |
159 | info = &pci_root_info[j]; | 163 | if (info) { |
160 | if (info->node == def_node && info->link == def_link) | ||
161 | break; | ||
162 | } | ||
163 | if (j < pci_root_num) { | ||
164 | info = &pci_root_info[j]; | ||
165 | for (i = 0; i < RANGE_NUM; i++) { | 164 | for (i = 0; i < RANGE_NUM; i++) { |
166 | if (!range[i].end) | 165 | if (!range[i].end) |
167 | continue; | 166 | continue; |
@@ -214,16 +213,10 @@ static int __init early_fill_mp_bus_info(void) | |||
214 | end <<= 8; | 213 | end <<= 8; |
215 | end |= 0xffff; | 214 | end |= 0xffff; |
216 | 215 | ||
217 | /* find the position */ | 216 | info = find_pci_root_info(node, link); |
218 | for (j = 0; j < pci_root_num; j++) { | ||
219 | info = &pci_root_info[j]; | ||
220 | if (info->node == node && info->link == link) | ||
221 | break; | ||
222 | } | ||
223 | if (j == pci_root_num) | ||
224 | continue; /* not found */ | ||
225 | 217 | ||
226 | info = &pci_root_info[j]; | 218 | if (!info) |
219 | continue; | ||
227 | 220 | ||
228 | printk(KERN_DEBUG "node %d link %d: mmio [%llx, %llx]", | 221 | printk(KERN_DEBUG "node %d link %d: mmio [%llx, %llx]", |
229 | node, link, start, end); | 222 | node, link, start, end); |
@@ -291,14 +284,8 @@ static int __init early_fill_mp_bus_info(void) | |||
291 | * add left over mmio range to def node/link ? | 284 | * add left over mmio range to def node/link ? |
292 | * that is tricky, just record range in from start_min to 4G | 285 | * that is tricky, just record range in from start_min to 4G |
293 | */ | 286 | */ |
294 | for (j = 0; j < pci_root_num; j++) { | 287 | info = find_pci_root_info(def_node, def_link); |
295 | info = &pci_root_info[j]; | 288 | if (info) { |
296 | if (info->node == def_node && info->link == def_link) | ||
297 | break; | ||
298 | } | ||
299 | if (j < pci_root_num) { | ||
300 | info = &pci_root_info[j]; | ||
301 | |||
302 | for (i = 0; i < RANGE_NUM; i++) { | 289 | for (i = 0; i < RANGE_NUM; i++) { |
303 | if (!range[i].end) | 290 | if (!range[i].end) |
304 | continue; | 291 | continue; |
@@ -309,20 +296,16 @@ static int __init early_fill_mp_bus_info(void) | |||
309 | } | 296 | } |
310 | } | 297 | } |
311 | 298 | ||
312 | for (i = 0; i < pci_root_num; i++) { | 299 | list_for_each_entry(info, &pci_root_infos, list) { |
313 | int res_num; | ||
314 | int busnum; | 300 | int busnum; |
301 | struct pci_root_res *root_res; | ||
315 | 302 | ||
316 | info = &pci_root_info[i]; | ||
317 | res_num = info->res_num; | ||
318 | busnum = info->bus_min; | 303 | busnum = info->bus_min; |
319 | printk(KERN_DEBUG "bus: [%02x, %02x] on node %x link %x\n", | 304 | printk(KERN_DEBUG "bus: [%02x, %02x] on node %x link %x\n", |
320 | info->bus_min, info->bus_max, info->node, info->link); | 305 | info->bus_min, info->bus_max, info->node, info->link); |
321 | for (j = 0; j < res_num; j++) { | 306 | list_for_each_entry(root_res, &info->resources, list) |
322 | res = &info->res[j]; | 307 | printk(KERN_DEBUG "bus: %02x %pR\n", |
323 | printk(KERN_DEBUG "bus: %02x index %x %pR\n", | 308 | busnum, &root_res->res); |
324 | busnum, j, res); | ||
325 | } | ||
326 | } | 309 | } |
327 | 310 | ||
328 | return 0; | 311 | return 0; |
diff --git a/arch/x86/pci/broadcom_bus.c b/arch/x86/pci/broadcom_bus.c index f3a7c569a403..614392ced7d6 100644 --- a/arch/x86/pci/broadcom_bus.c +++ b/arch/x86/pci/broadcom_bus.c | |||
@@ -22,19 +22,15 @@ | |||
22 | static void __init cnb20le_res(u8 bus, u8 slot, u8 func) | 22 | static void __init cnb20le_res(u8 bus, u8 slot, u8 func) |
23 | { | 23 | { |
24 | struct pci_root_info *info; | 24 | struct pci_root_info *info; |
25 | struct pci_root_res *root_res; | ||
25 | struct resource res; | 26 | struct resource res; |
26 | u16 word1, word2; | 27 | u16 word1, word2; |
27 | u8 fbus, lbus; | 28 | u8 fbus, lbus; |
28 | int i; | ||
29 | |||
30 | info = &pci_root_info[pci_root_num]; | ||
31 | pci_root_num++; | ||
32 | 29 | ||
33 | /* read the PCI bus numbers */ | 30 | /* read the PCI bus numbers */ |
34 | fbus = read_pci_config_byte(bus, slot, func, 0x44); | 31 | fbus = read_pci_config_byte(bus, slot, func, 0x44); |
35 | lbus = read_pci_config_byte(bus, slot, func, 0x45); | 32 | lbus = read_pci_config_byte(bus, slot, func, 0x45); |
36 | info->bus_min = fbus; | 33 | info = alloc_pci_root_info(fbus, lbus, 0, 0); |
37 | info->bus_max = lbus; | ||
38 | 34 | ||
39 | /* | 35 | /* |
40 | * Add the legacy IDE ports on bus 0 | 36 | * Add the legacy IDE ports on bus 0 |
@@ -86,8 +82,8 @@ static void __init cnb20le_res(u8 bus, u8 slot, u8 func) | |||
86 | res.flags = IORESOURCE_BUS; | 82 | res.flags = IORESOURCE_BUS; |
87 | printk(KERN_INFO "CNB20LE PCI Host Bridge (domain 0000 %pR)\n", &res); | 83 | printk(KERN_INFO "CNB20LE PCI Host Bridge (domain 0000 %pR)\n", &res); |
88 | 84 | ||
89 | for (i = 0; i < info->res_num; i++) | 85 | list_for_each_entry(root_res, &info->resources, list) |
90 | printk(KERN_INFO "host bridge window %pR\n", &info->res[i]); | 86 | printk(KERN_INFO "host bridge window %pR\n", &root_res->res); |
91 | } | 87 | } |
92 | 88 | ||
93 | static int __init broadcom_postcore_init(void) | 89 | static int __init broadcom_postcore_init(void) |
diff --git a/arch/x86/pci/bus_numa.c b/arch/x86/pci/bus_numa.c index fd3f65510e9d..306579f7d0fd 100644 --- a/arch/x86/pci/bus_numa.c +++ b/arch/x86/pci/bus_numa.c | |||
@@ -4,35 +4,38 @@ | |||
4 | 4 | ||
5 | #include "bus_numa.h" | 5 | #include "bus_numa.h" |
6 | 6 | ||
7 | int pci_root_num; | 7 | LIST_HEAD(pci_root_infos); |
8 | struct pci_root_info pci_root_info[PCI_ROOT_NR]; | ||
9 | 8 | ||
10 | void x86_pci_root_bus_resources(int bus, struct list_head *resources) | 9 | static struct pci_root_info *x86_find_pci_root_info(int bus) |
11 | { | 10 | { |
12 | int i; | ||
13 | int j; | ||
14 | struct pci_root_info *info; | 11 | struct pci_root_info *info; |
15 | 12 | ||
16 | if (!pci_root_num) | 13 | if (list_empty(&pci_root_infos)) |
17 | goto default_resources; | 14 | return NULL; |
18 | 15 | ||
19 | for (i = 0; i < pci_root_num; i++) { | 16 | list_for_each_entry(info, &pci_root_infos, list) |
20 | if (pci_root_info[i].bus_min == bus) | 17 | if (info->bus_min == bus) |
21 | break; | 18 | return info; |
22 | } | 19 | |
20 | return NULL; | ||
21 | } | ||
23 | 22 | ||
24 | if (i == pci_root_num) | 23 | void x86_pci_root_bus_resources(int bus, struct list_head *resources) |
24 | { | ||
25 | struct pci_root_info *info = x86_find_pci_root_info(bus); | ||
26 | struct pci_root_res *root_res; | ||
27 | |||
28 | if (!info) | ||
25 | goto default_resources; | 29 | goto default_resources; |
26 | 30 | ||
27 | printk(KERN_DEBUG "PCI: root bus %02x: hardware-probed resources\n", | 31 | printk(KERN_DEBUG "PCI: root bus %02x: hardware-probed resources\n", |
28 | bus); | 32 | bus); |
29 | 33 | ||
30 | info = &pci_root_info[i]; | 34 | list_for_each_entry(root_res, &info->resources, list) { |
31 | for (j = 0; j < info->res_num; j++) { | ||
32 | struct resource *res; | 35 | struct resource *res; |
33 | struct resource *root; | 36 | struct resource *root; |
34 | 37 | ||
35 | res = &info->res[j]; | 38 | res = &root_res->res; |
36 | pci_add_resource(resources, res); | 39 | pci_add_resource(resources, res); |
37 | if (res->flags & IORESOURCE_IO) | 40 | if (res->flags & IORESOURCE_IO) |
38 | root = &ioport_resource; | 41 | root = &ioport_resource; |
@@ -53,11 +56,32 @@ default_resources: | |||
53 | pci_add_resource(resources, &iomem_resource); | 56 | pci_add_resource(resources, &iomem_resource); |
54 | } | 57 | } |
55 | 58 | ||
59 | struct pci_root_info __init *alloc_pci_root_info(int bus_min, int bus_max, | ||
60 | int node, int link) | ||
61 | { | ||
62 | struct pci_root_info *info; | ||
63 | |||
64 | info = kzalloc(sizeof(*info), GFP_KERNEL); | ||
65 | |||
66 | if (!info) | ||
67 | return info; | ||
68 | |||
69 | INIT_LIST_HEAD(&info->resources); | ||
70 | info->bus_min = bus_min; | ||
71 | info->bus_max = bus_max; | ||
72 | info->node = node; | ||
73 | info->link = link; | ||
74 | |||
75 | list_add_tail(&info->list, &pci_root_infos); | ||
76 | |||
77 | return info; | ||
78 | } | ||
79 | |||
56 | void __devinit update_res(struct pci_root_info *info, resource_size_t start, | 80 | void __devinit update_res(struct pci_root_info *info, resource_size_t start, |
57 | resource_size_t end, unsigned long flags, int merge) | 81 | resource_size_t end, unsigned long flags, int merge) |
58 | { | 82 | { |
59 | int i; | ||
60 | struct resource *res; | 83 | struct resource *res; |
84 | struct pci_root_res *root_res; | ||
61 | 85 | ||
62 | if (start > end) | 86 | if (start > end) |
63 | return; | 87 | return; |
@@ -69,11 +93,11 @@ void __devinit update_res(struct pci_root_info *info, resource_size_t start, | |||
69 | goto addit; | 93 | goto addit; |
70 | 94 | ||
71 | /* try to merge it with old one */ | 95 | /* try to merge it with old one */ |
72 | for (i = 0; i < info->res_num; i++) { | 96 | list_for_each_entry(root_res, &info->resources, list) { |
73 | resource_size_t final_start, final_end; | 97 | resource_size_t final_start, final_end; |
74 | resource_size_t common_start, common_end; | 98 | resource_size_t common_start, common_end; |
75 | 99 | ||
76 | res = &info->res[i]; | 100 | res = &root_res->res; |
77 | if (res->flags != flags) | 101 | if (res->flags != flags) |
78 | continue; | 102 | continue; |
79 | 103 | ||
@@ -93,14 +117,15 @@ void __devinit update_res(struct pci_root_info *info, resource_size_t start, | |||
93 | addit: | 117 | addit: |
94 | 118 | ||
95 | /* need to add that */ | 119 | /* need to add that */ |
96 | if (info->res_num >= RES_NUM) | 120 | root_res = kzalloc(sizeof(*root_res), GFP_KERNEL); |
121 | if (!root_res) | ||
97 | return; | 122 | return; |
98 | 123 | ||
99 | res = &info->res[info->res_num]; | 124 | res = &root_res->res; |
100 | res->name = info->name; | 125 | res->name = info->name; |
101 | res->flags = flags; | 126 | res->flags = flags; |
102 | res->start = start; | 127 | res->start = start; |
103 | res->end = end; | 128 | res->end = end; |
104 | res->child = NULL; | 129 | |
105 | info->res_num++; | 130 | list_add_tail(&root_res->list, &info->resources); |
106 | } | 131 | } |
diff --git a/arch/x86/pci/bus_numa.h b/arch/x86/pci/bus_numa.h index 804a4b40c31a..226a466b2b2b 100644 --- a/arch/x86/pci/bus_numa.h +++ b/arch/x86/pci/bus_numa.h | |||
@@ -4,22 +4,24 @@ | |||
4 | * sub bus (transparent) will use entres from 3 to store extra from | 4 | * sub bus (transparent) will use entres from 3 to store extra from |
5 | * root, so need to make sure we have enough slot there. | 5 | * root, so need to make sure we have enough slot there. |
6 | */ | 6 | */ |
7 | #define RES_NUM 16 | 7 | struct pci_root_res { |
8 | struct list_head list; | ||
9 | struct resource res; | ||
10 | }; | ||
11 | |||
8 | struct pci_root_info { | 12 | struct pci_root_info { |
13 | struct list_head list; | ||
9 | char name[12]; | 14 | char name[12]; |
10 | unsigned int res_num; | 15 | struct list_head resources; |
11 | struct resource res[RES_NUM]; | ||
12 | int bus_min; | 16 | int bus_min; |
13 | int bus_max; | 17 | int bus_max; |
14 | int node; | 18 | int node; |
15 | int link; | 19 | int link; |
16 | }; | 20 | }; |
17 | 21 | ||
18 | /* 4 at this time, it may become to 32 */ | 22 | extern struct list_head pci_root_infos; |
19 | #define PCI_ROOT_NR 4 | 23 | struct pci_root_info *alloc_pci_root_info(int bus_min, int bus_max, |
20 | extern int pci_root_num; | 24 | int node, int link); |
21 | extern struct pci_root_info pci_root_info[PCI_ROOT_NR]; | ||
22 | |||
23 | extern void update_res(struct pci_root_info *info, resource_size_t start, | 25 | extern void update_res(struct pci_root_info *info, resource_size_t start, |
24 | resource_size_t end, unsigned long flags, int merge); | 26 | resource_size_t end, unsigned long flags, int merge); |
25 | #endif | 27 | #endif |
diff --git a/arch/x86/pci/common.c b/arch/x86/pci/common.c index 323481e06ef8..0ad990a20d4a 100644 --- a/arch/x86/pci/common.c +++ b/arch/x86/pci/common.c | |||
@@ -11,6 +11,7 @@ | |||
11 | #include <linux/dmi.h> | 11 | #include <linux/dmi.h> |
12 | #include <linux/slab.h> | 12 | #include <linux/slab.h> |
13 | 13 | ||
14 | #include <asm-generic/pci-bridge.h> | ||
14 | #include <asm/acpi.h> | 15 | #include <asm/acpi.h> |
15 | #include <asm/segment.h> | 16 | #include <asm/segment.h> |
16 | #include <asm/io.h> | 17 | #include <asm/io.h> |
@@ -229,6 +230,14 @@ static int __devinit assign_all_busses(const struct dmi_system_id *d) | |||
229 | } | 230 | } |
230 | #endif | 231 | #endif |
231 | 232 | ||
233 | static int __devinit set_scan_all(const struct dmi_system_id *d) | ||
234 | { | ||
235 | printk(KERN_INFO "PCI: %s detected, enabling pci=pcie_scan_all\n", | ||
236 | d->ident); | ||
237 | pci_add_flags(PCI_SCAN_ALL_PCIE_DEVS); | ||
238 | return 0; | ||
239 | } | ||
240 | |||
232 | static const struct dmi_system_id __devinitconst pciprobe_dmi_table[] = { | 241 | static const struct dmi_system_id __devinitconst pciprobe_dmi_table[] = { |
233 | #ifdef __i386__ | 242 | #ifdef __i386__ |
234 | /* | 243 | /* |
@@ -420,6 +429,13 @@ static const struct dmi_system_id __devinitconst pciprobe_dmi_table[] = { | |||
420 | DMI_MATCH(DMI_PRODUCT_NAME, "ProLiant DL585 G2"), | 429 | DMI_MATCH(DMI_PRODUCT_NAME, "ProLiant DL585 G2"), |
421 | }, | 430 | }, |
422 | }, | 431 | }, |
432 | { | ||
433 | .callback = set_scan_all, | ||
434 | .ident = "Stratus/NEC ftServer", | ||
435 | .matches = { | ||
436 | DMI_MATCH(DMI_SYS_VENDOR, "ftServer"), | ||
437 | }, | ||
438 | }, | ||
423 | {} | 439 | {} |
424 | }; | 440 | }; |
425 | 441 | ||
@@ -430,9 +446,7 @@ void __init dmi_check_pciprobe(void) | |||
430 | 446 | ||
431 | struct pci_bus * __devinit pcibios_scan_root(int busnum) | 447 | struct pci_bus * __devinit pcibios_scan_root(int busnum) |
432 | { | 448 | { |
433 | LIST_HEAD(resources); | ||
434 | struct pci_bus *bus = NULL; | 449 | struct pci_bus *bus = NULL; |
435 | struct pci_sysdata *sd; | ||
436 | 450 | ||
437 | while ((bus = pci_find_next_bus(bus)) != NULL) { | 451 | while ((bus = pci_find_next_bus(bus)) != NULL) { |
438 | if (bus->number == busnum) { | 452 | if (bus->number == busnum) { |
@@ -441,28 +455,10 @@ struct pci_bus * __devinit pcibios_scan_root(int busnum) | |||
441 | } | 455 | } |
442 | } | 456 | } |
443 | 457 | ||
444 | /* Allocate per-root-bus (not per bus) arch-specific data. | 458 | return pci_scan_bus_on_node(busnum, &pci_root_ops, |
445 | * TODO: leak; this memory is never freed. | 459 | get_mp_bus_to_node(busnum)); |
446 | * It's arguable whether it's worth the trouble to care. | ||
447 | */ | ||
448 | sd = kzalloc(sizeof(*sd), GFP_KERNEL); | ||
449 | if (!sd) { | ||
450 | printk(KERN_ERR "PCI: OOM, not probing PCI bus %02x\n", busnum); | ||
451 | return NULL; | ||
452 | } | ||
453 | |||
454 | sd->node = get_mp_bus_to_node(busnum); | ||
455 | |||
456 | printk(KERN_DEBUG "PCI: Probing PCI hardware (bus %02x)\n", busnum); | ||
457 | x86_pci_root_bus_resources(busnum, &resources); | ||
458 | bus = pci_scan_root_bus(NULL, busnum, &pci_root_ops, sd, &resources); | ||
459 | if (!bus) { | ||
460 | pci_free_resource_list(&resources); | ||
461 | kfree(sd); | ||
462 | } | ||
463 | |||
464 | return bus; | ||
465 | } | 460 | } |
461 | |||
466 | void __init pcibios_set_cache_line_size(void) | 462 | void __init pcibios_set_cache_line_size(void) |
467 | { | 463 | { |
468 | struct cpuinfo_x86 *c = &boot_cpu_data; | 464 | struct cpuinfo_x86 *c = &boot_cpu_data; |
@@ -656,6 +652,7 @@ struct pci_bus * __devinit pci_scan_bus_on_node(int busno, struct pci_ops *ops, | |||
656 | } | 652 | } |
657 | sd->node = node; | 653 | sd->node = node; |
658 | x86_pci_root_bus_resources(busno, &resources); | 654 | x86_pci_root_bus_resources(busno, &resources); |
655 | printk(KERN_DEBUG "PCI: Probing PCI hardware (bus %02x)\n", busno); | ||
659 | bus = pci_scan_root_bus(NULL, busno, ops, sd, &resources); | 656 | bus = pci_scan_root_bus(NULL, busno, ops, sd, &resources); |
660 | if (!bus) { | 657 | if (!bus) { |
661 | pci_free_resource_list(&resources); | 658 | pci_free_resource_list(&resources); |
diff --git a/arch/x86/pci/i386.c b/arch/x86/pci/i386.c index 831971e731f7..dd8ca6f7223b 100644 --- a/arch/x86/pci/i386.c +++ b/arch/x86/pci/i386.c | |||
@@ -57,7 +57,7 @@ static struct pcibios_fwaddrmap *pcibios_fwaddrmap_lookup(struct pci_dev *dev) | |||
57 | { | 57 | { |
58 | struct pcibios_fwaddrmap *map; | 58 | struct pcibios_fwaddrmap *map; |
59 | 59 | ||
60 | WARN_ON(!spin_is_locked(&pcibios_fwaddrmap_lock)); | 60 | WARN_ON_SMP(!spin_is_locked(&pcibios_fwaddrmap_lock)); |
61 | 61 | ||
62 | list_for_each_entry(map, &pcibios_fwaddrmappings, list) | 62 | list_for_each_entry(map, &pcibios_fwaddrmappings, list) |
63 | if (map->dev == dev) | 63 | if (map->dev == dev) |
diff --git a/drivers/pci/Makefile b/drivers/pci/Makefile index 165274c064bc..01c001f3b766 100644 --- a/drivers/pci/Makefile +++ b/drivers/pci/Makefile | |||
@@ -2,7 +2,7 @@ | |||
2 | # Makefile for the PCI bus specific drivers. | 2 | # Makefile for the PCI bus specific drivers. |
3 | # | 3 | # |
4 | 4 | ||
5 | obj-y += access.o bus.o probe.o remove.o pci.o \ | 5 | obj-y += access.o bus.o probe.o host-bridge.o remove.o pci.o \ |
6 | pci-driver.o search.o pci-sysfs.o rom.o setup-res.o \ | 6 | pci-driver.o search.o pci-sysfs.o rom.o setup-res.o \ |
7 | irq.o vpd.o | 7 | irq.o vpd.o |
8 | obj-$(CONFIG_PROC_FS) += proc.o | 8 | obj-$(CONFIG_PROC_FS) += proc.o |
diff --git a/drivers/pci/host-bridge.c b/drivers/pci/host-bridge.c new file mode 100644 index 000000000000..a68dc613a5be --- /dev/null +++ b/drivers/pci/host-bridge.c | |||
@@ -0,0 +1,96 @@ | |||
1 | /* | ||
2 | * host bridge related code | ||
3 | */ | ||
4 | |||
5 | #include <linux/kernel.h> | ||
6 | #include <linux/init.h> | ||
7 | #include <linux/pci.h> | ||
8 | #include <linux/module.h> | ||
9 | |||
10 | #include "pci.h" | ||
11 | |||
12 | static struct pci_bus *find_pci_root_bus(struct pci_dev *dev) | ||
13 | { | ||
14 | struct pci_bus *bus; | ||
15 | |||
16 | bus = dev->bus; | ||
17 | while (bus->parent) | ||
18 | bus = bus->parent; | ||
19 | |||
20 | return bus; | ||
21 | } | ||
22 | |||
23 | static struct pci_host_bridge *find_pci_host_bridge(struct pci_dev *dev) | ||
24 | { | ||
25 | struct pci_bus *bus = find_pci_root_bus(dev); | ||
26 | |||
27 | return to_pci_host_bridge(bus->bridge); | ||
28 | } | ||
29 | |||
30 | void pci_set_host_bridge_release(struct pci_host_bridge *bridge, | ||
31 | void (*release_fn)(struct pci_host_bridge *), | ||
32 | void *release_data) | ||
33 | { | ||
34 | bridge->release_fn = release_fn; | ||
35 | bridge->release_data = release_data; | ||
36 | } | ||
37 | |||
38 | static bool resource_contains(struct resource *res1, struct resource *res2) | ||
39 | { | ||
40 | return res1->start <= res2->start && res1->end >= res2->end; | ||
41 | } | ||
42 | |||
43 | void pcibios_resource_to_bus(struct pci_dev *dev, struct pci_bus_region *region, | ||
44 | struct resource *res) | ||
45 | { | ||
46 | struct pci_host_bridge *bridge = find_pci_host_bridge(dev); | ||
47 | struct pci_host_bridge_window *window; | ||
48 | resource_size_t offset = 0; | ||
49 | |||
50 | list_for_each_entry(window, &bridge->windows, list) { | ||
51 | if (resource_type(res) != resource_type(window->res)) | ||
52 | continue; | ||
53 | |||
54 | if (resource_contains(window->res, res)) { | ||
55 | offset = window->offset; | ||
56 | break; | ||
57 | } | ||
58 | } | ||
59 | |||
60 | region->start = res->start - offset; | ||
61 | region->end = res->end - offset; | ||
62 | } | ||
63 | EXPORT_SYMBOL(pcibios_resource_to_bus); | ||
64 | |||
65 | static bool region_contains(struct pci_bus_region *region1, | ||
66 | struct pci_bus_region *region2) | ||
67 | { | ||
68 | return region1->start <= region2->start && region1->end >= region2->end; | ||
69 | } | ||
70 | |||
71 | void pcibios_bus_to_resource(struct pci_dev *dev, struct resource *res, | ||
72 | struct pci_bus_region *region) | ||
73 | { | ||
74 | struct pci_host_bridge *bridge = find_pci_host_bridge(dev); | ||
75 | struct pci_host_bridge_window *window; | ||
76 | resource_size_t offset = 0; | ||
77 | |||
78 | list_for_each_entry(window, &bridge->windows, list) { | ||
79 | struct pci_bus_region bus_region; | ||
80 | |||
81 | if (resource_type(res) != resource_type(window->res)) | ||
82 | continue; | ||
83 | |||
84 | bus_region.start = window->res->start - window->offset; | ||
85 | bus_region.end = window->res->end - window->offset; | ||
86 | |||
87 | if (region_contains(&bus_region, region)) { | ||
88 | offset = window->offset; | ||
89 | break; | ||
90 | } | ||
91 | } | ||
92 | |||
93 | res->start = region->start + offset; | ||
94 | res->end = region->end + offset; | ||
95 | } | ||
96 | EXPORT_SYMBOL(pcibios_bus_to_resource); | ||
diff --git a/drivers/pci/pci-driver.c b/drivers/pci/pci-driver.c index 6b54b23b990b..bf0cee629b60 100644 --- a/drivers/pci/pci-driver.c +++ b/drivers/pci/pci-driver.c | |||
@@ -421,6 +421,12 @@ static void pci_device_shutdown(struct device *dev) | |||
421 | pci_msix_shutdown(pci_dev); | 421 | pci_msix_shutdown(pci_dev); |
422 | 422 | ||
423 | /* | 423 | /* |
424 | * Turn off Bus Master bit on the device to tell it to not | ||
425 | * continue to do DMA | ||
426 | */ | ||
427 | pci_disable_device(pci_dev); | ||
428 | |||
429 | /* | ||
424 | * Devices may be enabled to wake up by runtime PM, but they need not | 430 | * Devices may be enabled to wake up by runtime PM, but they need not |
425 | * be supposed to wake up the system from its "power off" state (e.g. | 431 | * be supposed to wake up the system from its "power off" state (e.g. |
426 | * ACPI S5). Therefore disable wakeup for all devices that aren't | 432 | * ACPI S5). Therefore disable wakeup for all devices that aren't |
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 111569ccab43..8f169002dc7e 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -22,6 +22,7 @@ | |||
22 | #include <linux/interrupt.h> | 22 | #include <linux/interrupt.h> |
23 | #include <linux/device.h> | 23 | #include <linux/device.h> |
24 | #include <linux/pm_runtime.h> | 24 | #include <linux/pm_runtime.h> |
25 | #include <asm-generic/pci-bridge.h> | ||
25 | #include <asm/setup.h> | 26 | #include <asm/setup.h> |
26 | #include "pci.h" | 27 | #include "pci.h" |
27 | 28 | ||
@@ -3164,18 +3165,12 @@ static int pci_parent_bus_reset(struct pci_dev *dev, int probe) | |||
3164 | return 0; | 3165 | return 0; |
3165 | } | 3166 | } |
3166 | 3167 | ||
3167 | static int pci_dev_reset(struct pci_dev *dev, int probe) | 3168 | static int __pci_dev_reset(struct pci_dev *dev, int probe) |
3168 | { | 3169 | { |
3169 | int rc; | 3170 | int rc; |
3170 | 3171 | ||
3171 | might_sleep(); | 3172 | might_sleep(); |
3172 | 3173 | ||
3173 | if (!probe) { | ||
3174 | pci_cfg_access_lock(dev); | ||
3175 | /* block PM suspend, driver probe, etc. */ | ||
3176 | device_lock(&dev->dev); | ||
3177 | } | ||
3178 | |||
3179 | rc = pci_dev_specific_reset(dev, probe); | 3174 | rc = pci_dev_specific_reset(dev, probe); |
3180 | if (rc != -ENOTTY) | 3175 | if (rc != -ENOTTY) |
3181 | goto done; | 3176 | goto done; |
@@ -3194,14 +3189,27 @@ static int pci_dev_reset(struct pci_dev *dev, int probe) | |||
3194 | 3189 | ||
3195 | rc = pci_parent_bus_reset(dev, probe); | 3190 | rc = pci_parent_bus_reset(dev, probe); |
3196 | done: | 3191 | done: |
3192 | return rc; | ||
3193 | } | ||
3194 | |||
3195 | static int pci_dev_reset(struct pci_dev *dev, int probe) | ||
3196 | { | ||
3197 | int rc; | ||
3198 | |||
3199 | if (!probe) { | ||
3200 | pci_cfg_access_lock(dev); | ||
3201 | /* block PM suspend, driver probe, etc. */ | ||
3202 | device_lock(&dev->dev); | ||
3203 | } | ||
3204 | |||
3205 | rc = __pci_dev_reset(dev, probe); | ||
3206 | |||
3197 | if (!probe) { | 3207 | if (!probe) { |
3198 | device_unlock(&dev->dev); | 3208 | device_unlock(&dev->dev); |
3199 | pci_cfg_access_unlock(dev); | 3209 | pci_cfg_access_unlock(dev); |
3200 | } | 3210 | } |
3201 | |||
3202 | return rc; | 3211 | return rc; |
3203 | } | 3212 | } |
3204 | |||
3205 | /** | 3213 | /** |
3206 | * __pci_reset_function - reset a PCI device function | 3214 | * __pci_reset_function - reset a PCI device function |
3207 | * @dev: PCI device to reset | 3215 | * @dev: PCI device to reset |
@@ -3246,7 +3254,7 @@ EXPORT_SYMBOL_GPL(__pci_reset_function); | |||
3246 | */ | 3254 | */ |
3247 | int __pci_reset_function_locked(struct pci_dev *dev) | 3255 | int __pci_reset_function_locked(struct pci_dev *dev) |
3248 | { | 3256 | { |
3249 | return pci_dev_reset(dev, 1); | 3257 | return __pci_dev_reset(dev, 0); |
3250 | } | 3258 | } |
3251 | EXPORT_SYMBOL_GPL(__pci_reset_function_locked); | 3259 | EXPORT_SYMBOL_GPL(__pci_reset_function_locked); |
3252 | 3260 | ||
@@ -3893,6 +3901,8 @@ static int __init pci_setup(char *str) | |||
3893 | pcie_bus_config = PCIE_BUS_PERFORMANCE; | 3901 | pcie_bus_config = PCIE_BUS_PERFORMANCE; |
3894 | } else if (!strncmp(str, "pcie_bus_peer2peer", 18)) { | 3902 | } else if (!strncmp(str, "pcie_bus_peer2peer", 18)) { |
3895 | pcie_bus_config = PCIE_BUS_PEER2PEER; | 3903 | pcie_bus_config = PCIE_BUS_PEER2PEER; |
3904 | } else if (!strncmp(str, "pcie_scan_all", 13)) { | ||
3905 | pci_add_flags(PCI_SCAN_ALL_PCIE_DEVS); | ||
3896 | } else { | 3906 | } else { |
3897 | printk(KERN_ERR "PCI: Unknown option `%s'\n", | 3907 | printk(KERN_ERR "PCI: Unknown option `%s'\n", |
3898 | str); | 3908 | str); |
diff --git a/drivers/pci/pcie/portdrv_core.c b/drivers/pci/pcie/portdrv_core.c index 2f589a54f9bd..75915b30ad19 100644 --- a/drivers/pci/pcie/portdrv_core.c +++ b/drivers/pci/pcie/portdrv_core.c | |||
@@ -249,7 +249,7 @@ static int get_port_device_capability(struct pci_dev *dev) | |||
249 | int services = 0, pos; | 249 | int services = 0, pos; |
250 | u16 reg16; | 250 | u16 reg16; |
251 | u32 reg32; | 251 | u32 reg32; |
252 | int cap_mask; | 252 | int cap_mask = 0; |
253 | int err; | 253 | int err; |
254 | 254 | ||
255 | if (pcie_ports_disabled) | 255 | if (pcie_ports_disabled) |
diff --git a/drivers/pci/probe.c b/drivers/pci/probe.c index 5e1ca3c58a7d..658ac977cb56 100644 --- a/drivers/pci/probe.c +++ b/drivers/pci/probe.c | |||
@@ -10,18 +10,16 @@ | |||
10 | #include <linux/module.h> | 10 | #include <linux/module.h> |
11 | #include <linux/cpumask.h> | 11 | #include <linux/cpumask.h> |
12 | #include <linux/pci-aspm.h> | 12 | #include <linux/pci-aspm.h> |
13 | #include <asm-generic/pci-bridge.h> | ||
13 | #include "pci.h" | 14 | #include "pci.h" |
14 | 15 | ||
15 | #define CARDBUS_LATENCY_TIMER 176 /* secondary latency timer */ | 16 | #define CARDBUS_LATENCY_TIMER 176 /* secondary latency timer */ |
16 | #define CARDBUS_RESERVE_BUSNR 3 | 17 | #define CARDBUS_RESERVE_BUSNR 3 |
17 | 18 | ||
18 | static LIST_HEAD(pci_host_bridges); | ||
19 | |||
20 | /* Ugh. Need to stop exporting this to modules. */ | 19 | /* Ugh. Need to stop exporting this to modules. */ |
21 | LIST_HEAD(pci_root_buses); | 20 | LIST_HEAD(pci_root_buses); |
22 | EXPORT_SYMBOL(pci_root_buses); | 21 | EXPORT_SYMBOL(pci_root_buses); |
23 | 22 | ||
24 | |||
25 | static int find_anything(struct device *dev, void *data) | 23 | static int find_anything(struct device *dev, void *data) |
26 | { | 24 | { |
27 | return 1; | 25 | return 1; |
@@ -44,82 +42,6 @@ int no_pci_devices(void) | |||
44 | } | 42 | } |
45 | EXPORT_SYMBOL(no_pci_devices); | 43 | EXPORT_SYMBOL(no_pci_devices); |
46 | 44 | ||
47 | static struct pci_host_bridge *pci_host_bridge(struct pci_dev *dev) | ||
48 | { | ||
49 | struct pci_bus *bus; | ||
50 | struct pci_host_bridge *bridge; | ||
51 | |||
52 | bus = dev->bus; | ||
53 | while (bus->parent) | ||
54 | bus = bus->parent; | ||
55 | |||
56 | list_for_each_entry(bridge, &pci_host_bridges, list) { | ||
57 | if (bridge->bus == bus) | ||
58 | return bridge; | ||
59 | } | ||
60 | |||
61 | return NULL; | ||
62 | } | ||
63 | |||
64 | static bool resource_contains(struct resource *res1, struct resource *res2) | ||
65 | { | ||
66 | return res1->start <= res2->start && res1->end >= res2->end; | ||
67 | } | ||
68 | |||
69 | void pcibios_resource_to_bus(struct pci_dev *dev, struct pci_bus_region *region, | ||
70 | struct resource *res) | ||
71 | { | ||
72 | struct pci_host_bridge *bridge = pci_host_bridge(dev); | ||
73 | struct pci_host_bridge_window *window; | ||
74 | resource_size_t offset = 0; | ||
75 | |||
76 | list_for_each_entry(window, &bridge->windows, list) { | ||
77 | if (resource_type(res) != resource_type(window->res)) | ||
78 | continue; | ||
79 | |||
80 | if (resource_contains(window->res, res)) { | ||
81 | offset = window->offset; | ||
82 | break; | ||
83 | } | ||
84 | } | ||
85 | |||
86 | region->start = res->start - offset; | ||
87 | region->end = res->end - offset; | ||
88 | } | ||
89 | EXPORT_SYMBOL(pcibios_resource_to_bus); | ||
90 | |||
91 | static bool region_contains(struct pci_bus_region *region1, | ||
92 | struct pci_bus_region *region2) | ||
93 | { | ||
94 | return region1->start <= region2->start && region1->end >= region2->end; | ||
95 | } | ||
96 | |||
97 | void pcibios_bus_to_resource(struct pci_dev *dev, struct resource *res, | ||
98 | struct pci_bus_region *region) | ||
99 | { | ||
100 | struct pci_host_bridge *bridge = pci_host_bridge(dev); | ||
101 | struct pci_host_bridge_window *window; | ||
102 | struct pci_bus_region bus_region; | ||
103 | resource_size_t offset = 0; | ||
104 | |||
105 | list_for_each_entry(window, &bridge->windows, list) { | ||
106 | if (resource_type(res) != resource_type(window->res)) | ||
107 | continue; | ||
108 | |||
109 | bus_region.start = window->res->start - window->offset; | ||
110 | bus_region.end = window->res->end - window->offset; | ||
111 | |||
112 | if (region_contains(&bus_region, region)) { | ||
113 | offset = window->offset; | ||
114 | break; | ||
115 | } | ||
116 | } | ||
117 | |||
118 | res->start = region->start + offset; | ||
119 | res->end = region->end + offset; | ||
120 | } | ||
121 | EXPORT_SYMBOL(pcibios_bus_to_resource); | ||
122 | |||
123 | /* | 45 | /* |
124 | * PCI Bus Class | 46 | * PCI Bus Class |
125 | */ | 47 | */ |
@@ -501,6 +423,19 @@ static struct pci_bus * pci_alloc_bus(void) | |||
501 | return b; | 423 | return b; |
502 | } | 424 | } |
503 | 425 | ||
426 | static struct pci_host_bridge *pci_alloc_host_bridge(struct pci_bus *b) | ||
427 | { | ||
428 | struct pci_host_bridge *bridge; | ||
429 | |||
430 | bridge = kzalloc(sizeof(*bridge), GFP_KERNEL); | ||
431 | if (bridge) { | ||
432 | INIT_LIST_HEAD(&bridge->windows); | ||
433 | bridge->bus = b; | ||
434 | } | ||
435 | |||
436 | return bridge; | ||
437 | } | ||
438 | |||
504 | static unsigned char pcix_bus_speed[] = { | 439 | static unsigned char pcix_bus_speed[] = { |
505 | PCI_SPEED_UNKNOWN, /* 0 */ | 440 | PCI_SPEED_UNKNOWN, /* 0 */ |
506 | PCI_SPEED_66MHz_PCIX, /* 1 */ | 441 | PCI_SPEED_66MHz_PCIX, /* 1 */ |
@@ -1201,7 +1136,14 @@ int pci_cfg_space_size(struct pci_dev *dev) | |||
1201 | 1136 | ||
1202 | static void pci_release_bus_bridge_dev(struct device *dev) | 1137 | static void pci_release_bus_bridge_dev(struct device *dev) |
1203 | { | 1138 | { |
1204 | kfree(dev); | 1139 | struct pci_host_bridge *bridge = to_pci_host_bridge(dev); |
1140 | |||
1141 | if (bridge->release_fn) | ||
1142 | bridge->release_fn(bridge); | ||
1143 | |||
1144 | pci_free_resource_list(&bridge->windows); | ||
1145 | |||
1146 | kfree(bridge); | ||
1205 | } | 1147 | } |
1206 | 1148 | ||
1207 | struct pci_dev *alloc_pci_dev(void) | 1149 | struct pci_dev *alloc_pci_dev(void) |
@@ -1395,10 +1337,13 @@ static unsigned no_next_fn(struct pci_dev *dev, unsigned fn) | |||
1395 | static int only_one_child(struct pci_bus *bus) | 1337 | static int only_one_child(struct pci_bus *bus) |
1396 | { | 1338 | { |
1397 | struct pci_dev *parent = bus->self; | 1339 | struct pci_dev *parent = bus->self; |
1340 | |||
1398 | if (!parent || !pci_is_pcie(parent)) | 1341 | if (!parent || !pci_is_pcie(parent)) |
1399 | return 0; | 1342 | return 0; |
1400 | if (parent->pcie_type == PCI_EXP_TYPE_ROOT_PORT || | 1343 | if (parent->pcie_type == PCI_EXP_TYPE_ROOT_PORT) |
1401 | parent->pcie_type == PCI_EXP_TYPE_DOWNSTREAM) | 1344 | return 1; |
1345 | if (parent->pcie_type == PCI_EXP_TYPE_DOWNSTREAM && | ||
1346 | !pci_has_flag(PCI_SCAN_ALL_PCIE_DEVS)) | ||
1402 | return 1; | 1347 | return 1; |
1403 | return 0; | 1348 | return 0; |
1404 | } | 1349 | } |
@@ -1650,28 +1595,19 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | |||
1650 | int error; | 1595 | int error; |
1651 | struct pci_host_bridge *bridge; | 1596 | struct pci_host_bridge *bridge; |
1652 | struct pci_bus *b, *b2; | 1597 | struct pci_bus *b, *b2; |
1653 | struct device *dev; | ||
1654 | struct pci_host_bridge_window *window, *n; | 1598 | struct pci_host_bridge_window *window, *n; |
1655 | struct resource *res; | 1599 | struct resource *res; |
1656 | resource_size_t offset; | 1600 | resource_size_t offset; |
1657 | char bus_addr[64]; | 1601 | char bus_addr[64]; |
1658 | char *fmt; | 1602 | char *fmt; |
1659 | 1603 | ||
1660 | bridge = kzalloc(sizeof(*bridge), GFP_KERNEL); | ||
1661 | if (!bridge) | ||
1662 | return NULL; | ||
1663 | 1604 | ||
1664 | b = pci_alloc_bus(); | 1605 | b = pci_alloc_bus(); |
1665 | if (!b) | 1606 | if (!b) |
1666 | goto err_bus; | 1607 | return NULL; |
1667 | |||
1668 | dev = kzalloc(sizeof(*dev), GFP_KERNEL); | ||
1669 | if (!dev) | ||
1670 | goto err_dev; | ||
1671 | 1608 | ||
1672 | b->sysdata = sysdata; | 1609 | b->sysdata = sysdata; |
1673 | b->ops = ops; | 1610 | b->ops = ops; |
1674 | |||
1675 | b2 = pci_find_bus(pci_domain_nr(b), bus); | 1611 | b2 = pci_find_bus(pci_domain_nr(b), bus); |
1676 | if (b2) { | 1612 | if (b2) { |
1677 | /* If we already got to this bus through a different bridge, ignore it */ | 1613 | /* If we already got to this bus through a different bridge, ignore it */ |
@@ -1679,13 +1615,17 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | |||
1679 | goto err_out; | 1615 | goto err_out; |
1680 | } | 1616 | } |
1681 | 1617 | ||
1682 | dev->parent = parent; | 1618 | bridge = pci_alloc_host_bridge(b); |
1683 | dev->release = pci_release_bus_bridge_dev; | 1619 | if (!bridge) |
1684 | dev_set_name(dev, "pci%04x:%02x", pci_domain_nr(b), bus); | 1620 | goto err_out; |
1685 | error = device_register(dev); | 1621 | |
1622 | bridge->dev.parent = parent; | ||
1623 | bridge->dev.release = pci_release_bus_bridge_dev; | ||
1624 | dev_set_name(&bridge->dev, "pci%04x:%02x", pci_domain_nr(b), bus); | ||
1625 | error = device_register(&bridge->dev); | ||
1686 | if (error) | 1626 | if (error) |
1687 | goto dev_reg_err; | 1627 | goto bridge_dev_reg_err; |
1688 | b->bridge = get_device(dev); | 1628 | b->bridge = get_device(&bridge->dev); |
1689 | device_enable_async_suspend(b->bridge); | 1629 | device_enable_async_suspend(b->bridge); |
1690 | pci_set_bus_of_node(b); | 1630 | pci_set_bus_of_node(b); |
1691 | 1631 | ||
@@ -1704,9 +1644,6 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | |||
1704 | 1644 | ||
1705 | b->number = b->secondary = bus; | 1645 | b->number = b->secondary = bus; |
1706 | 1646 | ||
1707 | bridge->bus = b; | ||
1708 | INIT_LIST_HEAD(&bridge->windows); | ||
1709 | |||
1710 | if (parent) | 1647 | if (parent) |
1711 | dev_info(parent, "PCI host bridge to bus %s\n", dev_name(&b->dev)); | 1648 | dev_info(parent, "PCI host bridge to bus %s\n", dev_name(&b->dev)); |
1712 | else | 1649 | else |
@@ -1732,25 +1669,18 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus, | |||
1732 | } | 1669 | } |
1733 | 1670 | ||
1734 | down_write(&pci_bus_sem); | 1671 | down_write(&pci_bus_sem); |
1735 | list_add_tail(&bridge->list, &pci_host_bridges); | ||
1736 | list_add_tail(&b->node, &pci_root_buses); | 1672 | list_add_tail(&b->node, &pci_root_buses); |
1737 | up_write(&pci_bus_sem); | 1673 | up_write(&pci_bus_sem); |
1738 | 1674 | ||
1739 | return b; | 1675 | return b; |
1740 | 1676 | ||
1741 | class_dev_reg_err: | 1677 | class_dev_reg_err: |
1742 | device_unregister(dev); | 1678 | put_device(&bridge->dev); |
1743 | dev_reg_err: | 1679 | device_unregister(&bridge->dev); |
1744 | down_write(&pci_bus_sem); | 1680 | bridge_dev_reg_err: |
1745 | list_del(&bridge->list); | 1681 | kfree(bridge); |
1746 | list_del(&b->node); | ||
1747 | up_write(&pci_bus_sem); | ||
1748 | err_out: | 1682 | err_out: |
1749 | kfree(dev); | ||
1750 | err_dev: | ||
1751 | kfree(b); | 1683 | kfree(b); |
1752 | err_bus: | ||
1753 | kfree(bridge); | ||
1754 | return NULL; | 1684 | return NULL; |
1755 | } | 1685 | } |
1756 | 1686 | ||
diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c index 953ec3f08470..2a7521677541 100644 --- a/drivers/pci/quirks.c +++ b/drivers/pci/quirks.c | |||
@@ -3097,16 +3097,74 @@ static int reset_intel_82599_sfp_virtfn(struct pci_dev *dev, int probe) | |||
3097 | return 0; | 3097 | return 0; |
3098 | } | 3098 | } |
3099 | 3099 | ||
3100 | #include "../gpu/drm/i915/i915_reg.h" | ||
3101 | #define MSG_CTL 0x45010 | ||
3102 | #define NSDE_PWR_STATE 0xd0100 | ||
3103 | #define IGD_OPERATION_TIMEOUT 10000 /* set timeout 10 seconds */ | ||
3104 | |||
3105 | static int reset_ivb_igd(struct pci_dev *dev, int probe) | ||
3106 | { | ||
3107 | void __iomem *mmio_base; | ||
3108 | unsigned long timeout; | ||
3109 | u32 val; | ||
3110 | |||
3111 | if (probe) | ||
3112 | return 0; | ||
3113 | |||
3114 | mmio_base = pci_iomap(dev, 0, 0); | ||
3115 | if (!mmio_base) | ||
3116 | return -ENOMEM; | ||
3117 | |||
3118 | iowrite32(0x00000002, mmio_base + MSG_CTL); | ||
3119 | |||
3120 | /* | ||
3121 | * Clobbering SOUTH_CHICKEN2 register is fine only if the next | ||
3122 | * driver loaded sets the right bits. However, this's a reset and | ||
3123 | * the bits have been set by i915 previously, so we clobber | ||
3124 | * SOUTH_CHICKEN2 register directly here. | ||
3125 | */ | ||
3126 | iowrite32(0x00000005, mmio_base + SOUTH_CHICKEN2); | ||
3127 | |||
3128 | val = ioread32(mmio_base + PCH_PP_CONTROL) & 0xfffffffe; | ||
3129 | iowrite32(val, mmio_base + PCH_PP_CONTROL); | ||
3130 | |||
3131 | timeout = jiffies + msecs_to_jiffies(IGD_OPERATION_TIMEOUT); | ||
3132 | do { | ||
3133 | val = ioread32(mmio_base + PCH_PP_STATUS); | ||
3134 | if ((val & 0xb0000000) == 0) | ||
3135 | goto reset_complete; | ||
3136 | msleep(10); | ||
3137 | } while (time_before(jiffies, timeout)); | ||
3138 | dev_warn(&dev->dev, "timeout during reset\n"); | ||
3139 | |||
3140 | reset_complete: | ||
3141 | iowrite32(0x00000002, mmio_base + NSDE_PWR_STATE); | ||
3142 | |||
3143 | pci_iounmap(dev, mmio_base); | ||
3144 | return 0; | ||
3145 | } | ||
3146 | |||
3100 | #define PCI_DEVICE_ID_INTEL_82599_SFP_VF 0x10ed | 3147 | #define PCI_DEVICE_ID_INTEL_82599_SFP_VF 0x10ed |
3148 | #define PCI_DEVICE_ID_INTEL_IVB_M_VGA 0x0156 | ||
3149 | #define PCI_DEVICE_ID_INTEL_IVB_M2_VGA 0x0166 | ||
3101 | 3150 | ||
3102 | static const struct pci_dev_reset_methods pci_dev_reset_methods[] = { | 3151 | static const struct pci_dev_reset_methods pci_dev_reset_methods[] = { |
3103 | { PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82599_SFP_VF, | 3152 | { PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82599_SFP_VF, |
3104 | reset_intel_82599_sfp_virtfn }, | 3153 | reset_intel_82599_sfp_virtfn }, |
3154 | { PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_IVB_M_VGA, | ||
3155 | reset_ivb_igd }, | ||
3156 | { PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_IVB_M2_VGA, | ||
3157 | reset_ivb_igd }, | ||
3105 | { PCI_VENDOR_ID_INTEL, PCI_ANY_ID, | 3158 | { PCI_VENDOR_ID_INTEL, PCI_ANY_ID, |
3106 | reset_intel_generic_dev }, | 3159 | reset_intel_generic_dev }, |
3107 | { 0 } | 3160 | { 0 } |
3108 | }; | 3161 | }; |
3109 | 3162 | ||
3163 | /* | ||
3164 | * These device-specific reset methods are here rather than in a driver | ||
3165 | * because when a host assigns a device to a guest VM, the host may need | ||
3166 | * to reset the device but probably doesn't have a driver for it. | ||
3167 | */ | ||
3110 | int pci_dev_specific_reset(struct pci_dev *dev, int probe) | 3168 | int pci_dev_specific_reset(struct pci_dev *dev, int probe) |
3111 | { | 3169 | { |
3112 | const struct pci_dev_reset_methods *i; | 3170 | const struct pci_dev_reset_methods *i; |
diff --git a/include/asm-generic/pci-bridge.h b/include/asm-generic/pci-bridge.h index a5b5d5a89a4f..20db2e5a0a69 100644 --- a/include/asm-generic/pci-bridge.h +++ b/include/asm-generic/pci-bridge.h | |||
@@ -30,6 +30,12 @@ enum { | |||
30 | PCI_ENABLE_PROC_DOMAINS = 0x00000010, | 30 | PCI_ENABLE_PROC_DOMAINS = 0x00000010, |
31 | /* ... except for domain 0 */ | 31 | /* ... except for domain 0 */ |
32 | PCI_COMPAT_DOMAIN_0 = 0x00000020, | 32 | PCI_COMPAT_DOMAIN_0 = 0x00000020, |
33 | |||
34 | /* PCIe downstream ports are bridges that normally lead to only a | ||
35 | * device 0, but if this is set, we scan all possible devices, not | ||
36 | * just device 0. | ||
37 | */ | ||
38 | PCI_SCAN_ALL_PCIE_DEVS = 0x00000040, | ||
33 | }; | 39 | }; |
34 | 40 | ||
35 | #ifdef CONFIG_PCI | 41 | #ifdef CONFIG_PCI |
diff --git a/include/linux/ioport.h b/include/linux/ioport.h index e885ba23de70..589e0e75efae 100644 --- a/include/linux/ioport.h +++ b/include/linux/ioport.h | |||
@@ -223,5 +223,12 @@ extern int | |||
223 | walk_system_ram_range(unsigned long start_pfn, unsigned long nr_pages, | 223 | walk_system_ram_range(unsigned long start_pfn, unsigned long nr_pages, |
224 | void *arg, int (*func)(unsigned long, unsigned long, void *)); | 224 | void *arg, int (*func)(unsigned long, unsigned long, void *)); |
225 | 225 | ||
226 | /* True if any part of r1 overlaps r2 */ | ||
227 | static inline bool resource_overlaps(struct resource *r1, struct resource *r2) | ||
228 | { | ||
229 | return (r1->start <= r2->end && r1->end >= r2->start); | ||
230 | } | ||
231 | |||
232 | |||
226 | #endif /* __ASSEMBLY__ */ | 233 | #endif /* __ASSEMBLY__ */ |
227 | #endif /* _LINUX_IOPORT_H */ | 234 | #endif /* _LINUX_IOPORT_H */ |
diff --git a/include/linux/pci.h b/include/linux/pci.h index e444f5b49118..17b7b5b01b4a 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h | |||
@@ -375,11 +375,18 @@ struct pci_host_bridge_window { | |||
375 | }; | 375 | }; |
376 | 376 | ||
377 | struct pci_host_bridge { | 377 | struct pci_host_bridge { |
378 | struct list_head list; | 378 | struct device dev; |
379 | struct pci_bus *bus; /* root bus */ | 379 | struct pci_bus *bus; /* root bus */ |
380 | struct list_head windows; /* pci_host_bridge_windows */ | 380 | struct list_head windows; /* pci_host_bridge_windows */ |
381 | void (*release_fn)(struct pci_host_bridge *); | ||
382 | void *release_data; | ||
381 | }; | 383 | }; |
382 | 384 | ||
385 | #define to_pci_host_bridge(n) container_of(n, struct pci_host_bridge, dev) | ||
386 | void pci_set_host_bridge_release(struct pci_host_bridge *bridge, | ||
387 | void (*release_fn)(struct pci_host_bridge *), | ||
388 | void *release_data); | ||
389 | |||
383 | /* | 390 | /* |
384 | * The first PCI_BRIDGE_RESOURCE_NUM PCI bus resources (those that correspond | 391 | * The first PCI_BRIDGE_RESOURCE_NUM PCI bus resources (those that correspond |
385 | * to P2P or CardBus bridge windows) go in a table. Additional ones (for | 392 | * to P2P or CardBus bridge windows) go in a table. Additional ones (for |