aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMikulas Patocka <mpatocka@redhat.com>2018-11-26 11:37:13 -0500
committerBjorn Helgaas <bhelgaas@google.com>2018-12-01 00:42:03 -0500
commitf1f90e254e46e0a14220e4090041f68256fbe297 (patch)
treeb0951795411236292d08404af7652f104731ae99
parent0d76bcc960e6057750fcf556b65da13f8bbdfd2b (diff)
PCI: Fix incorrect value returned from pcie_get_speed_cap()
The macros PCI_EXP_LNKCAP_SLS_*GB are values, not bit masks. We must mask the register and compare it against them. This fixes errors like this: amdgpu: [powerplay] failed to send message 261 ret is 0 when a PCIe-v3 card is plugged into a PCIe-v1 slot, because the slot is being incorrectly reported as PCIe-v3 capable. 6cf57be0f78e, which appeared in v4.17, added pcie_get_speed_cap() with the incorrect test of PCI_EXP_LNKCAP_SLS as a bitmask. 5d9a63304032, which appeared in v4.19, changed amdgpu to use pcie_get_speed_cap(), so the amdgpu bug reports below are regressions in v4.19. Fixes: 6cf57be0f78e ("PCI: Add pcie_get_speed_cap() to find max supported link speed") Fixes: 5d9a63304032 ("drm/amdgpu: use pcie functions for link width and speed") Link: https://bugs.freedesktop.org/show_bug.cgi?id=108704 Link: https://bugs.freedesktop.org/show_bug.cgi?id=108778 Signed-off-by: Mikulas Patocka <mpatocka@redhat.com> [bhelgaas: update comment, remove use of PCI_EXP_LNKCAP_SLS_8_0GB and PCI_EXP_LNKCAP_SLS_16_0GB since those should be covered by PCI_EXP_LNKCAP2, remove test of PCI_EXP_LNKCAP for zero, since that register is required] Signed-off-by: Bjorn Helgaas <bhelgaas@google.com> Acked-by: Alex Deucher <alexander.deucher@amd.com> Cc: stable@vger.kernel.org # v4.17+
-rw-r--r--drivers/pci/pci.c24
1 files changed, 11 insertions, 13 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c
index d068f11d08a7..c9d8e3c837de 100644
--- a/drivers/pci/pci.c
+++ b/drivers/pci/pci.c
@@ -5556,9 +5556,13 @@ enum pci_bus_speed pcie_get_speed_cap(struct pci_dev *dev)
5556 u32 lnkcap2, lnkcap; 5556 u32 lnkcap2, lnkcap;
5557 5557
5558 /* 5558 /*
5559 * PCIe r4.0 sec 7.5.3.18 recommends using the Supported Link 5559 * Link Capabilities 2 was added in PCIe r3.0, sec 7.8.18. The
5560 * Speeds Vector in Link Capabilities 2 when supported, falling 5560 * implementation note there recommends using the Supported Link
5561 * back to Max Link Speed in Link Capabilities otherwise. 5561 * Speeds Vector in Link Capabilities 2 when supported.
5562 *
5563 * Without Link Capabilities 2, i.e., prior to PCIe r3.0, software
5564 * should use the Supported Link Speeds field in Link Capabilities,
5565 * where only 2.5 GT/s and 5.0 GT/s speeds were defined.
5562 */ 5566 */
5563 pcie_capability_read_dword(dev, PCI_EXP_LNKCAP2, &lnkcap2); 5567 pcie_capability_read_dword(dev, PCI_EXP_LNKCAP2, &lnkcap2);
5564 if (lnkcap2) { /* PCIe r3.0-compliant */ 5568 if (lnkcap2) { /* PCIe r3.0-compliant */
@@ -5574,16 +5578,10 @@ enum pci_bus_speed pcie_get_speed_cap(struct pci_dev *dev)
5574 } 5578 }
5575 5579
5576 pcie_capability_read_dword(dev, PCI_EXP_LNKCAP, &lnkcap); 5580 pcie_capability_read_dword(dev, PCI_EXP_LNKCAP, &lnkcap);
5577 if (lnkcap) { 5581 if ((lnkcap & PCI_EXP_LNKCAP_SLS) == PCI_EXP_LNKCAP_SLS_5_0GB)
5578 if (lnkcap & PCI_EXP_LNKCAP_SLS_16_0GB) 5582 return PCIE_SPEED_5_0GT;
5579 return PCIE_SPEED_16_0GT; 5583 else if ((lnkcap & PCI_EXP_LNKCAP_SLS) == PCI_EXP_LNKCAP_SLS_2_5GB)
5580 else if (lnkcap & PCI_EXP_LNKCAP_SLS_8_0GB) 5584 return PCIE_SPEED_2_5GT;
5581 return PCIE_SPEED_8_0GT;
5582 else if (lnkcap & PCI_EXP_LNKCAP_SLS_5_0GB)
5583 return PCIE_SPEED_5_0GT;
5584 else if (lnkcap & PCI_EXP_LNKCAP_SLS_2_5GB)
5585 return PCIE_SPEED_2_5GT;
5586 }
5587 5585
5588 return PCI_SPEED_UNKNOWN; 5586 return PCI_SPEED_UNKNOWN;
5589} 5587}