diff options
| author | Mikulas Patocka <mpatocka@redhat.com> | 2018-11-26 11:37:13 -0500 |
|---|---|---|
| committer | Bjorn Helgaas <bhelgaas@google.com> | 2018-12-01 00:42:03 -0500 |
| commit | f1f90e254e46e0a14220e4090041f68256fbe297 (patch) | |
| tree | b0951795411236292d08404af7652f104731ae99 | |
| parent | 0d76bcc960e6057750fcf556b65da13f8bbdfd2b (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.c | 24 |
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 | } |
