diff options
Diffstat (limited to 'drivers/pci')
-rw-r--r-- | drivers/pci/msi.c | 35 | ||||
-rw-r--r-- | drivers/pci/pci-driver.c | 2 |
2 files changed, 28 insertions, 9 deletions
diff --git a/drivers/pci/msi.c b/drivers/pci/msi.c index e3a05cc9a595..8c61304cbb37 100644 --- a/drivers/pci/msi.c +++ b/drivers/pci/msi.c | |||
@@ -571,10 +571,9 @@ int pci_enable_msi(struct pci_dev* dev) | |||
571 | } | 571 | } |
572 | EXPORT_SYMBOL(pci_enable_msi); | 572 | EXPORT_SYMBOL(pci_enable_msi); |
573 | 573 | ||
574 | void pci_disable_msi(struct pci_dev* dev) | 574 | void pci_msi_shutdown(struct pci_dev* dev) |
575 | { | 575 | { |
576 | struct msi_desc *entry; | 576 | struct msi_desc *entry; |
577 | int default_irq; | ||
578 | 577 | ||
579 | if (!pci_msi_enable || !dev || !dev->msi_enabled) | 578 | if (!pci_msi_enable || !dev || !dev->msi_enabled) |
580 | return; | 579 | return; |
@@ -590,15 +589,26 @@ void pci_disable_msi(struct pci_dev* dev) | |||
590 | u32 mask = entry->msi_attrib.maskbits_mask; | 589 | u32 mask = entry->msi_attrib.maskbits_mask; |
591 | msi_set_mask_bits(dev->irq, mask, ~mask); | 590 | msi_set_mask_bits(dev->irq, mask, ~mask); |
592 | } | 591 | } |
593 | if (!entry->dev || entry->msi_attrib.type != PCI_CAP_ID_MSI) { | 592 | if (!entry->dev || entry->msi_attrib.type != PCI_CAP_ID_MSI) |
594 | return; | 593 | return; |
595 | } | ||
596 | |||
597 | default_irq = entry->msi_attrib.default_irq; | ||
598 | msi_free_irqs(dev); | ||
599 | 594 | ||
600 | /* Restore dev->irq to its default pin-assertion irq */ | 595 | /* Restore dev->irq to its default pin-assertion irq */ |
601 | dev->irq = default_irq; | 596 | dev->irq = entry->msi_attrib.default_irq; |
597 | } | ||
598 | void pci_disable_msi(struct pci_dev* dev) | ||
599 | { | ||
600 | struct msi_desc *entry; | ||
601 | |||
602 | if (!pci_msi_enable || !dev || !dev->msi_enabled) | ||
603 | return; | ||
604 | |||
605 | pci_msi_shutdown(dev); | ||
606 | |||
607 | entry = list_entry(dev->msi_list.next, struct msi_desc, list); | ||
608 | if (!entry->dev || entry->msi_attrib.type != PCI_CAP_ID_MSI) | ||
609 | return; | ||
610 | |||
611 | msi_free_irqs(dev); | ||
602 | } | 612 | } |
603 | EXPORT_SYMBOL(pci_disable_msi); | 613 | EXPORT_SYMBOL(pci_disable_msi); |
604 | 614 | ||
@@ -691,7 +701,7 @@ static void msix_free_all_irqs(struct pci_dev *dev) | |||
691 | msi_free_irqs(dev); | 701 | msi_free_irqs(dev); |
692 | } | 702 | } |
693 | 703 | ||
694 | void pci_disable_msix(struct pci_dev* dev) | 704 | void pci_msix_shutdown(struct pci_dev* dev) |
695 | { | 705 | { |
696 | if (!pci_msi_enable || !dev || !dev->msix_enabled) | 706 | if (!pci_msi_enable || !dev || !dev->msix_enabled) |
697 | return; | 707 | return; |
@@ -699,6 +709,13 @@ void pci_disable_msix(struct pci_dev* dev) | |||
699 | msix_set_enable(dev, 0); | 709 | msix_set_enable(dev, 0); |
700 | pci_intx_for_msi(dev, 1); | 710 | pci_intx_for_msi(dev, 1); |
701 | dev->msix_enabled = 0; | 711 | dev->msix_enabled = 0; |
712 | } | ||
713 | void pci_disable_msix(struct pci_dev* dev) | ||
714 | { | ||
715 | if (!pci_msi_enable || !dev || !dev->msix_enabled) | ||
716 | return; | ||
717 | |||
718 | pci_msix_shutdown(dev); | ||
702 | 719 | ||
703 | msix_free_all_irqs(dev); | 720 | msix_free_all_irqs(dev); |
704 | } | 721 | } |
diff --git a/drivers/pci/pci-driver.c b/drivers/pci/pci-driver.c index e8d94fafc280..72cf61ed8f96 100644 --- a/drivers/pci/pci-driver.c +++ b/drivers/pci/pci-driver.c | |||
@@ -360,6 +360,8 @@ static void pci_device_shutdown(struct device *dev) | |||
360 | 360 | ||
361 | if (drv && drv->shutdown) | 361 | if (drv && drv->shutdown) |
362 | drv->shutdown(pci_dev); | 362 | drv->shutdown(pci_dev); |
363 | pci_msi_shutdown(pci_dev); | ||
364 | pci_msix_shutdown(pci_dev); | ||
363 | } | 365 | } |
364 | 366 | ||
365 | /** | 367 | /** |