Commit 10ecc818 authored by Bjorn Helgaas's avatar Bjorn Helgaas Committed by Bjorn Helgaas

PCI/ASPM: Use LTR if already enabled by platform

RussianNeuroMancer reported that the Intel 7265 wifi on a Dell Venue 11 Pro
7140 table stopped working after wakeup from suspend and bisected the
problem to 9ab105de ("PCI/ASPM: Disable ASPM L1.2 Substate if we don't
have LTR").  David Ward reported the same problem on a Dell Latitude 7350.

After af8bb9f8 ("PCI/ACPI: Request LTR control from platform before
using it"), we don't enable LTR unless the platform has granted LTR control
to us.  In addition, we don't notice if the platform had already enabled
LTR itself.

After 9ab105de ("PCI/ASPM: Disable ASPM L1.2 Substate if we don't have
LTR"), we avoid using LTR if we don't think the path to the device has LTR
enabled.

The combination means that if the platform itself enables LTR but declines
to give the OS control over LTR, we unnecessarily avoided using ASPM L1.2.

Link: https://bugzilla.kernel.org/show_bug.cgi?id=201469
Fixes: 9ab105de ("PCI/ASPM: Disable ASPM L1.2 Substate if we don't have LTR")
Fixes: af8bb9f8 ("PCI/ACPI: Request LTR control from platform before using it")
Reported-by: default avatarRussianNeuroMancer <russianneuromancer@ya.ru>
Reported-by: default avatarDavid Ward <david.ward@ll.mit.edu>
Signed-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
CC: stable@vger.kernel.org	# v4.18+
parent bfeffd15
...@@ -2071,11 +2071,8 @@ static void pci_configure_ltr(struct pci_dev *dev) ...@@ -2071,11 +2071,8 @@ static void pci_configure_ltr(struct pci_dev *dev)
{ {
#ifdef CONFIG_PCIEASPM #ifdef CONFIG_PCIEASPM
struct pci_host_bridge *host = pci_find_host_bridge(dev->bus); struct pci_host_bridge *host = pci_find_host_bridge(dev->bus);
u32 cap;
struct pci_dev *bridge; struct pci_dev *bridge;
u32 cap, ctl;
if (!host->native_ltr)
return;
if (!pci_is_pcie(dev)) if (!pci_is_pcie(dev))
return; return;
...@@ -2084,22 +2081,35 @@ static void pci_configure_ltr(struct pci_dev *dev) ...@@ -2084,22 +2081,35 @@ static void pci_configure_ltr(struct pci_dev *dev)
if (!(cap & PCI_EXP_DEVCAP2_LTR)) if (!(cap & PCI_EXP_DEVCAP2_LTR))
return; return;
/* pcie_capability_read_dword(dev, PCI_EXP_DEVCTL2, &ctl);
* Software must not enable LTR in an Endpoint unless the Root if (ctl & PCI_EXP_DEVCTL2_LTR_EN) {
* Complex and all intermediate Switches indicate support for LTR. if (pci_pcie_type(dev) == PCI_EXP_TYPE_ROOT_PORT) {
* PCIe r3.1, sec 6.18. dev->ltr_path = 1;
*/ return;
if (pci_pcie_type(dev) == PCI_EXP_TYPE_ROOT_PORT) }
dev->ltr_path = 1;
else {
bridge = pci_upstream_bridge(dev); bridge = pci_upstream_bridge(dev);
if (bridge && bridge->ltr_path) if (bridge && bridge->ltr_path)
dev->ltr_path = 1; dev->ltr_path = 1;
return;
} }
if (dev->ltr_path) if (!host->native_ltr)
return;
/*
* Software must not enable LTR in an Endpoint unless the Root
* Complex and all intermediate Switches indicate support for LTR.
* PCIe r4.0, sec 6.18.
*/
if (pci_pcie_type(dev) == PCI_EXP_TYPE_ROOT_PORT ||
((bridge = pci_upstream_bridge(dev)) &&
bridge->ltr_path)) {
pcie_capability_set_word(dev, PCI_EXP_DEVCTL2, pcie_capability_set_word(dev, PCI_EXP_DEVCTL2,
PCI_EXP_DEVCTL2_LTR_EN); PCI_EXP_DEVCTL2_LTR_EN);
dev->ltr_path = 1;
}
#endif #endif
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment