diff options
| author | Bjorn Helgaas <bhelgaas@google.com> | 2012-06-16 16:40:22 -0400 |
|---|---|---|
| committer | Bjorn Helgaas <bhelgaas@google.com> | 2012-06-16 16:40:22 -0400 |
| commit | fbebb9fd22581b6422d60669c4ff86ce99d6cdba (patch) | |
| tree | f7063ec22814ee7782d1ccdb6f8653404d6714c8 | |
| parent | cfaf025112d3856637ff34a767ef785ef5cf2ca9 (diff) | |
PCI: add infrastructure for devices with broken INTx masking
pci_intx_mask_supported() assumes INTx masking is supported if the
PCI_COMMAND_INTX_DISABLE bit is writable. But when that bit is set,
some devices don't actually mask INTx or update PCI_STATUS_INTERRUPT
as we expect.
This patch adds a way for quirks to identify these broken devices.
[bhelgaas: split out from Chelsio quirk addition]
Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
| -rw-r--r-- | drivers/pci/pci.c | 3 | ||||
| -rw-r--r-- | drivers/pci/quirks.c | 10 | ||||
| -rw-r--r-- | include/linux/pci.h | 1 |
3 files changed, 14 insertions, 0 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 447e83472c01..9ae517a68360 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
| @@ -2876,6 +2876,9 @@ bool pci_intx_mask_supported(struct pci_dev *dev) | |||
| 2876 | bool mask_supported = false; | 2876 | bool mask_supported = false; |
| 2877 | u16 orig, new; | 2877 | u16 orig, new; |
| 2878 | 2878 | ||
| 2879 | if (dev->broken_intx_masking) | ||
| 2880 | return false; | ||
| 2881 | |||
| 2879 | pci_cfg_access_lock(dev); | 2882 | pci_cfg_access_lock(dev); |
| 2880 | 2883 | ||
| 2881 | pci_read_config_word(dev, PCI_COMMAND, &orig); | 2884 | pci_read_config_word(dev, PCI_COMMAND, &orig); |
diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c index 2a7521677541..cc13415416d7 100644 --- a/drivers/pci/quirks.c +++ b/drivers/pci/quirks.c | |||
| @@ -2929,6 +2929,16 @@ static void __devinit disable_igfx_irq(struct pci_dev *dev) | |||
| 2929 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x0102, disable_igfx_irq); | 2929 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x0102, disable_igfx_irq); |
| 2930 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x010a, disable_igfx_irq); | 2930 | DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x010a, disable_igfx_irq); |
| 2931 | 2931 | ||
| 2932 | /* | ||
| 2933 | * Some devices may pass our check in pci_intx_mask_supported if | ||
| 2934 | * PCI_COMMAND_INTX_DISABLE works though they actually do not properly | ||
| 2935 | * support this feature. | ||
| 2936 | */ | ||
| 2937 | static void __devinit quirk_broken_intx_masking(struct pci_dev *dev) | ||
| 2938 | { | ||
| 2939 | dev->broken_intx_masking = 1; | ||
| 2940 | } | ||
| 2941 | |||
| 2932 | static void pci_do_fixups(struct pci_dev *dev, struct pci_fixup *f, | 2942 | static void pci_do_fixups(struct pci_dev *dev, struct pci_fixup *f, |
| 2933 | struct pci_fixup *end) | 2943 | struct pci_fixup *end) |
| 2934 | { | 2944 | { |
diff --git a/include/linux/pci.h b/include/linux/pci.h index d8c379dba6ad..40a039f1dffb 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h | |||
| @@ -324,6 +324,7 @@ struct pci_dev { | |||
| 324 | unsigned int is_hotplug_bridge:1; | 324 | unsigned int is_hotplug_bridge:1; |
| 325 | unsigned int __aer_firmware_first_valid:1; | 325 | unsigned int __aer_firmware_first_valid:1; |
| 326 | unsigned int __aer_firmware_first:1; | 326 | unsigned int __aer_firmware_first:1; |
| 327 | unsigned int broken_intx_masking:1; | ||
| 327 | pci_dev_flags_t dev_flags; | 328 | pci_dev_flags_t dev_flags; |
| 328 | atomic_t enable_cnt; /* pci_enable_device has been called */ | 329 | atomic_t enable_cnt; /* pci_enable_device has been called */ |
| 329 | 330 | ||
