diff options
Diffstat (limited to 'drivers/pci')
-rw-r--r-- | drivers/pci/hotplug/cpqphp_pci.c | 85 |
1 files changed, 39 insertions, 46 deletions
diff --git a/drivers/pci/hotplug/cpqphp_pci.c b/drivers/pci/hotplug/cpqphp_pci.c index 2e96bae3c82a..1f1c90dd791d 100644 --- a/drivers/pci/hotplug/cpqphp_pci.c +++ b/drivers/pci/hotplug/cpqphp_pci.c | |||
@@ -494,7 +494,7 @@ int cpqhp_save_slot_config (struct controller *ctrl, struct pci_func * new_slot) | |||
494 | u8 secondary_bus; | 494 | u8 secondary_bus; |
495 | int sub_bus; | 495 | int sub_bus; |
496 | int max_functions; | 496 | int max_functions; |
497 | int function; | 497 | int function = 0; |
498 | int cloop = 0; | 498 | int cloop = 0; |
499 | int stop_it; | 499 | int stop_it; |
500 | 500 | ||
@@ -503,65 +503,58 @@ int cpqhp_save_slot_config (struct controller *ctrl, struct pci_func * new_slot) | |||
503 | ctrl->pci_bus->number = new_slot->bus; | 503 | ctrl->pci_bus->number = new_slot->bus; |
504 | pci_bus_read_config_dword (ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), PCI_VENDOR_ID, &ID); | 504 | pci_bus_read_config_dword (ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), PCI_VENDOR_ID, &ID); |
505 | 505 | ||
506 | if (ID != 0xFFFFFFFF) { /* device in slot */ | 506 | if (ID == 0xFFFFFFFF) |
507 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), 0x0B, &class_code); | 507 | return 2; |
508 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), PCI_HEADER_TYPE, &header_type); | ||
509 | |||
510 | if (header_type & 0x80) /* Multi-function device */ | ||
511 | max_functions = 8; | ||
512 | else | ||
513 | max_functions = 1; | ||
514 | |||
515 | function = 0; | ||
516 | 508 | ||
517 | do { | 509 | pci_bus_read_config_byte(ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), 0x0B, &class_code); |
518 | if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { | 510 | pci_bus_read_config_byte(ctrl->pci_bus, PCI_DEVFN(new_slot->device, 0), PCI_HEADER_TYPE, &header_type); |
519 | /* Recurse the subordinate bus */ | ||
520 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_SECONDARY_BUS, &secondary_bus); | ||
521 | 511 | ||
522 | sub_bus = (int) secondary_bus; | 512 | if (header_type & 0x80) /* Multi-function device */ |
513 | max_functions = 8; | ||
514 | else | ||
515 | max_functions = 1; | ||
523 | 516 | ||
524 | /* Save the config headers for the secondary | 517 | while (function < max_functions) { |
525 | * bus. | 518 | if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { |
526 | */ | 519 | /* Recurse the subordinate bus */ |
527 | rc = cpqhp_save_config(ctrl, sub_bus, 0); | 520 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_SECONDARY_BUS, &secondary_bus); |
528 | if (rc) | ||
529 | return(rc); | ||
530 | ctrl->pci_bus->number = new_slot->bus; | ||
531 | 521 | ||
532 | } /* End of IF */ | 522 | sub_bus = (int) secondary_bus; |
533 | 523 | ||
534 | new_slot->status = 0; | 524 | /* Save the config headers for the secondary |
525 | * bus. | ||
526 | */ | ||
527 | rc = cpqhp_save_config(ctrl, sub_bus, 0); | ||
528 | if (rc) | ||
529 | return(rc); | ||
530 | ctrl->pci_bus->number = new_slot->bus; | ||
535 | 531 | ||
536 | for (cloop = 0; cloop < 0x20; cloop++) { | 532 | } |
537 | pci_bus_read_config_dword (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), cloop << 2, (u32 *) & (new_slot-> config_space [cloop])); | ||
538 | } | ||
539 | 533 | ||
540 | function++; | 534 | new_slot->status = 0; |
541 | 535 | ||
542 | stop_it = 0; | 536 | for (cloop = 0; cloop < 0x20; cloop++) |
537 | pci_bus_read_config_dword(ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), cloop << 2, (u32 *) & (new_slot-> config_space [cloop])); | ||
543 | 538 | ||
544 | /* this loop skips to the next present function | 539 | function++; |
545 | * reading in the Class Code and the Header type. | ||
546 | */ | ||
547 | while ((function < max_functions) && (!stop_it)) { | ||
548 | pci_bus_read_config_dword (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_VENDOR_ID, &ID); | ||
549 | 540 | ||
550 | if (ID == 0xFFFFFFFF) { /* nothing there. */ | 541 | stop_it = 0; |
551 | function++; | ||
552 | } else { /* Something there */ | ||
553 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), 0x0B, &class_code); | ||
554 | 542 | ||
555 | pci_bus_read_config_byte (ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_HEADER_TYPE, &header_type); | 543 | /* this loop skips to the next present function |
544 | * reading in the Class Code and the Header type. | ||
545 | */ | ||
546 | while ((function < max_functions) && (!stop_it)) { | ||
547 | pci_bus_read_config_dword(ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_VENDOR_ID, &ID); | ||
556 | 548 | ||
557 | stop_it++; | 549 | if (ID == 0xFFFFFFFF) |
558 | } | 550 | function++; |
551 | else { | ||
552 | pci_bus_read_config_byte(ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), 0x0B, &class_code); | ||
553 | pci_bus_read_config_byte(ctrl->pci_bus, PCI_DEVFN(new_slot->device, function), PCI_HEADER_TYPE, &header_type); | ||
554 | stop_it++; | ||
559 | } | 555 | } |
556 | } | ||
560 | 557 | ||
561 | } while (function < max_functions); | ||
562 | } /* End of IF (device in slot?) */ | ||
563 | else { | ||
564 | return 2; | ||
565 | } | 558 | } |
566 | 559 | ||
567 | return 0; | 560 | return 0; |