diff options
-rw-r--r-- | drivers/pci/quirks.c | 62 |
1 files changed, 62 insertions, 0 deletions
diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c index f0b3b7166b46..bbad4a9f264f 100644 --- a/drivers/pci/quirks.c +++ b/drivers/pci/quirks.c | |||
@@ -1778,6 +1778,68 @@ static void __devinit quirk_nvidia_ck804_msi_ht_cap(struct pci_dev *dev) | |||
1778 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_CK804_PCIE, | 1778 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_CK804_PCIE, |
1779 | quirk_nvidia_ck804_msi_ht_cap); | 1779 | quirk_nvidia_ck804_msi_ht_cap); |
1780 | 1780 | ||
1781 | /* | ||
1782 | * Force enable MSI mapping capability on HT bridges */ | ||
1783 | static inline void ht_enable_msi_mapping(struct pci_dev *dev) | ||
1784 | { | ||
1785 | int pos, ttl = 48; | ||
1786 | |||
1787 | pos = pci_find_ht_capability(dev, HT_CAPTYPE_MSI_MAPPING); | ||
1788 | while (pos && ttl--) { | ||
1789 | u8 flags; | ||
1790 | |||
1791 | if (pci_read_config_byte(dev, pos + HT_MSI_FLAGS, | ||
1792 | &flags) == 0) { | ||
1793 | dev_info(&dev->dev, "Enabling HT MSI Mapping\n"); | ||
1794 | |||
1795 | pci_write_config_byte(dev, pos + HT_MSI_FLAGS, | ||
1796 | flags | HT_MSI_FLAGS_ENABLE); | ||
1797 | } | ||
1798 | pos = pci_find_next_ht_capability(dev, pos, | ||
1799 | HT_CAPTYPE_MSI_MAPPING); | ||
1800 | } | ||
1801 | } | ||
1802 | |||
1803 | static void __devinit nv_msi_ht_cap_quirk(struct pci_dev *dev) | ||
1804 | { | ||
1805 | struct pci_dev *host_bridge; | ||
1806 | int pos, ttl = 48; | ||
1807 | |||
1808 | /* | ||
1809 | * HT MSI mapping should be disabled on devices that are below | ||
1810 | * a non-Hypertransport host bridge. Locate the host bridge... | ||
1811 | */ | ||
1812 | host_bridge = pci_get_bus_and_slot(0, PCI_DEVFN(0, 0)); | ||
1813 | if (host_bridge == NULL) { | ||
1814 | dev_warn(&dev->dev, | ||
1815 | "nv_msi_ht_cap_quirk didn't locate host bridge\n"); | ||
1816 | return; | ||
1817 | } | ||
1818 | |||
1819 | pos = pci_find_ht_capability(host_bridge, HT_CAPTYPE_SLAVE); | ||
1820 | if (pos != 0) { | ||
1821 | /* Host bridge is to HT */ | ||
1822 | ht_enable_msi_mapping(dev); | ||
1823 | return; | ||
1824 | } | ||
1825 | |||
1826 | /* Host bridge is not to HT, disable HT MSI mapping on this device */ | ||
1827 | pos = pci_find_ht_capability(dev, HT_CAPTYPE_MSI_MAPPING); | ||
1828 | while (pos && ttl--) { | ||
1829 | u8 flags; | ||
1830 | |||
1831 | if (pci_read_config_byte(dev, pos + HT_MSI_FLAGS, | ||
1832 | &flags) == 0) { | ||
1833 | dev_info(&dev->dev, "Quirk disabling HT MSI mapping"); | ||
1834 | pci_write_config_byte(dev, pos + HT_MSI_FLAGS, | ||
1835 | flags & ~HT_MSI_FLAGS_ENABLE); | ||
1836 | } | ||
1837 | pos = pci_find_next_ht_capability(dev, pos, | ||
1838 | HT_CAPTYPE_MSI_MAPPING); | ||
1839 | } | ||
1840 | } | ||
1841 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_NVIDIA, PCI_ANY_ID, nv_msi_ht_cap_quirk); | ||
1842 | |||
1781 | static void __devinit quirk_msi_intx_disable_bug(struct pci_dev *dev) | 1843 | static void __devinit quirk_msi_intx_disable_bug(struct pci_dev *dev) |
1782 | { | 1844 | { |
1783 | dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG; | 1845 | dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG; |