diff options
author | Sasha Neftin <sasha.neftin@intel.com> | 2017-04-03 17:02:50 -0400 |
---|---|---|
committer | Bjorn Helgaas <bhelgaas@google.com> | 2017-04-03 17:02:50 -0400 |
commit | f65fd1aa4f9881d5540192d11f7b8ed2fec936db (patch) | |
tree | a6a61973770b1c570019cd9b6218983f3b020b65 | |
parent | c1ae3cfa0e89fa1a7ecc4c99031f5e9ae99d9201 (diff) |
PCI: Avoid FLR for Intel 82579 NICs
Per Intel Specification Update 335553-002 (see link below), some 82579
network adapters advertise a Function Level Reset (FLR) capability, but
they can hang when an FLR is triggered.
To reproduce the problem, attach the device to a VM, then detach and try to
attach again.
Add a quirk to prevent the use of FLR on these devices.
[bhelgaas: changelog, comments]
Link: http://www.intel.com/content/dam/www/public/us/en/documents/specification-updates/82579lm-82579v-gigabit-network-connection-spec-update.pdf
Signed-off-by: Sasha Neftin <sasha.neftin@intel.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
-rw-r--r-- | drivers/pci/pci.c | 6 | ||||
-rw-r--r-- | drivers/pci/quirks.c | 8 | ||||
-rw-r--r-- | include/linux/pci.h | 2 |
3 files changed, 16 insertions, 0 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 7904d02ffdb9..bef14777bb30 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -3781,6 +3781,9 @@ static int pcie_flr(struct pci_dev *dev, int probe) | |||
3781 | if (!(cap & PCI_EXP_DEVCAP_FLR)) | 3781 | if (!(cap & PCI_EXP_DEVCAP_FLR)) |
3782 | return -ENOTTY; | 3782 | return -ENOTTY; |
3783 | 3783 | ||
3784 | if (dev->dev_flags & PCI_DEV_FLAGS_NO_FLR_RESET) | ||
3785 | return -ENOTTY; | ||
3786 | |||
3784 | if (probe) | 3787 | if (probe) |
3785 | return 0; | 3788 | return 0; |
3786 | 3789 | ||
@@ -3801,6 +3804,9 @@ static int pci_af_flr(struct pci_dev *dev, int probe) | |||
3801 | if (!pos) | 3804 | if (!pos) |
3802 | return -ENOTTY; | 3805 | return -ENOTTY; |
3803 | 3806 | ||
3807 | if (dev->dev_flags & PCI_DEV_FLAGS_NO_FLR_RESET) | ||
3808 | return -ENOTTY; | ||
3809 | |||
3804 | pci_read_config_byte(dev, pos + PCI_AF_CAP, &cap); | 3810 | pci_read_config_byte(dev, pos + PCI_AF_CAP, &cap); |
3805 | if (!(cap & PCI_AF_CAP_TP) || !(cap & PCI_AF_CAP_FLR)) | 3811 | if (!(cap & PCI_AF_CAP_TP) || !(cap & PCI_AF_CAP_FLR)) |
3806 | return -ENOTTY; | 3812 | return -ENOTTY; |
diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c index f754453fe754..823271b13d12 100644 --- a/drivers/pci/quirks.c +++ b/drivers/pci/quirks.c | |||
@@ -4633,3 +4633,11 @@ DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2030, quirk_no_aersid); | |||
4633 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2031, quirk_no_aersid); | 4633 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2031, quirk_no_aersid); |
4634 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2032, quirk_no_aersid); | 4634 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2032, quirk_no_aersid); |
4635 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2033, quirk_no_aersid); | 4635 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x2033, quirk_no_aersid); |
4636 | |||
4637 | /* FLR may cause some 82579 devices to hang. */ | ||
4638 | static void quirk_intel_no_flr(struct pci_dev *dev) | ||
4639 | { | ||
4640 | dev->dev_flags |= PCI_DEV_FLAGS_NO_FLR_RESET; | ||
4641 | } | ||
4642 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x1502, quirk_intel_no_flr); | ||
4643 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x1503, quirk_intel_no_flr); | ||
diff --git a/include/linux/pci.h b/include/linux/pci.h index eb3da1a04e6c..22cad2c66d59 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h | |||
@@ -178,6 +178,8 @@ enum pci_dev_flags { | |||
178 | PCI_DEV_FLAGS_NO_PM_RESET = (__force pci_dev_flags_t) (1 << 7), | 178 | PCI_DEV_FLAGS_NO_PM_RESET = (__force pci_dev_flags_t) (1 << 7), |
179 | /* Get VPD from function 0 VPD */ | 179 | /* Get VPD from function 0 VPD */ |
180 | PCI_DEV_FLAGS_VPD_REF_F0 = (__force pci_dev_flags_t) (1 << 8), | 180 | PCI_DEV_FLAGS_VPD_REF_F0 = (__force pci_dev_flags_t) (1 << 8), |
181 | /* Do not use FLR even if device advertises PCI_AF_CAP */ | ||
182 | PCI_DEV_FLAGS_NO_FLR_RESET = (__force pci_dev_flags_t) (1 << 10), | ||
181 | }; | 183 | }; |
182 | 184 | ||
183 | enum pci_irq_reroute_variant { | 185 | enum pci_irq_reroute_variant { |