|
@@ -1724,11 +1724,15 @@ int pci_setup_device(struct pci_dev *dev)
|
|
|
static void pci_configure_mps(struct pci_dev *dev)
|
|
|
{
|
|
|
struct pci_dev *bridge = pci_upstream_bridge(dev);
|
|
|
- int mps, p_mps, rc;
|
|
|
+ int mps, mpss, p_mps, rc;
|
|
|
|
|
|
if (!pci_is_pcie(dev) || !bridge || !pci_is_pcie(bridge))
|
|
|
return;
|
|
|
|
|
|
+ /* MPS and MRRS fields are of type 'RsvdP' for VFs, short-circuit out */
|
|
|
+ if (dev->is_virtfn)
|
|
|
+ return;
|
|
|
+
|
|
|
mps = pcie_get_mps(dev);
|
|
|
p_mps = pcie_get_mps(bridge);
|
|
|
|
|
@@ -1748,6 +1752,14 @@ static void pci_configure_mps(struct pci_dev *dev)
|
|
|
if (pcie_bus_config != PCIE_BUS_DEFAULT)
|
|
|
return;
|
|
|
|
|
|
+ mpss = 128 << dev->pcie_mpss;
|
|
|
+ if (mpss < p_mps && pci_pcie_type(bridge) == PCI_EXP_TYPE_ROOT_PORT) {
|
|
|
+ pcie_set_mps(bridge, mpss);
|
|
|
+ pci_info(dev, "Upstream bridge's Max Payload Size set to %d (was %d, max %d)\n",
|
|
|
+ mpss, p_mps, 128 << bridge->pcie_mpss);
|
|
|
+ p_mps = pcie_get_mps(bridge);
|
|
|
+ }
|
|
|
+
|
|
|
rc = pcie_set_mps(dev, p_mps);
|
|
|
if (rc) {
|
|
|
pci_warn(dev, "can't set Max Payload Size to %d; if necessary, use \"pci=pcie_bus_safe\" and report a bug\n",
|
|
@@ -1756,7 +1768,7 @@ static void pci_configure_mps(struct pci_dev *dev)
|
|
|
}
|
|
|
|
|
|
pci_info(dev, "Max Payload Size set to %d (was %d, max %d)\n",
|
|
|
- p_mps, mps, 128 << dev->pcie_mpss);
|
|
|
+ p_mps, mps, mpss);
|
|
|
}
|
|
|
|
|
|
static struct hpp_type0 pci_default_type0 = {
|
|
@@ -2156,8 +2168,8 @@ static bool pci_bus_wait_crs(struct pci_bus *bus, int devfn, u32 *l,
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
-bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *l,
|
|
|
- int timeout)
|
|
|
+bool pci_bus_generic_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *l,
|
|
|
+ int timeout)
|
|
|
{
|
|
|
if (pci_bus_read_config_dword(bus, devfn, PCI_VENDOR_ID, l))
|
|
|
return false;
|
|
@@ -2172,6 +2184,24 @@ bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *l,
|
|
|
|
|
|
return true;
|
|
|
}
|
|
|
+
|
|
|
+bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *l,
|
|
|
+ int timeout)
|
|
|
+{
|
|
|
+#ifdef CONFIG_PCI_QUIRKS
|
|
|
+ struct pci_dev *bridge = bus->self;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Certain IDT switches have an issue where they improperly trigger
|
|
|
+ * ACS Source Validation errors on completions for config reads.
|
|
|
+ */
|
|
|
+ if (bridge && bridge->vendor == PCI_VENDOR_ID_IDT &&
|
|
|
+ bridge->device == 0x80b5)
|
|
|
+ return pci_idt_bus_quirk(bus, devfn, l, timeout);
|
|
|
+#endif
|
|
|
+
|
|
|
+ return pci_bus_generic_read_dev_vendor_id(bus, devfn, l, timeout);
|
|
|
+}
|
|
|
EXPORT_SYMBOL(pci_bus_read_dev_vendor_id);
|
|
|
|
|
|
/*
|
|
@@ -2205,6 +2235,25 @@ static struct pci_dev *pci_scan_device(struct pci_bus *bus, int devfn)
|
|
|
return dev;
|
|
|
}
|
|
|
|
|
|
+static void pcie_report_downtraining(struct pci_dev *dev)
|
|
|
+{
|
|
|
+ if (!pci_is_pcie(dev))
|
|
|
+ return;
|
|
|
+
|
|
|
+ /* Look from the device up to avoid downstream ports with no devices */
|
|
|
+ if ((pci_pcie_type(dev) != PCI_EXP_TYPE_ENDPOINT) &&
|
|
|
+ (pci_pcie_type(dev) != PCI_EXP_TYPE_LEG_END) &&
|
|
|
+ (pci_pcie_type(dev) != PCI_EXP_TYPE_UPSTREAM))
|
|
|
+ return;
|
|
|
+
|
|
|
+ /* Multi-function PCIe devices share the same link/status */
|
|
|
+ if (PCI_FUNC(dev->devfn) != 0 || dev->is_virtfn)
|
|
|
+ return;
|
|
|
+
|
|
|
+ /* Print link status only if the device is constrained by the fabric */
|
|
|
+ __pcie_print_link_status(dev, false);
|
|
|
+}
|
|
|
+
|
|
|
static void pci_init_capabilities(struct pci_dev *dev)
|
|
|
{
|
|
|
/* Enhanced Allocation */
|
|
@@ -2240,6 +2289,8 @@ static void pci_init_capabilities(struct pci_dev *dev)
|
|
|
/* Advanced Error Reporting */
|
|
|
pci_aer_init(dev);
|
|
|
|
|
|
+ pcie_report_downtraining(dev);
|
|
|
+
|
|
|
if (pci_probe_reset_function(dev) == 0)
|
|
|
dev->reset_fn = 1;
|
|
|
}
|