]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
PCI: Enable AtomicOps only if Root Port supports them
authorGerd Bayer <gbayer@linux.ibm.com>
Mon, 30 Mar 2026 13:09:45 +0000 (15:09 +0200)
committerBjorn Helgaas <bhelgaas@google.com>
Fri, 3 Apr 2026 21:39:31 +0000 (16:39 -0500)
When inspecting the config space of a Connect-X physical function in an
s390 system after it was initialized by the mlx5_core device driver, we
found the function to be enabled to request AtomicOps despite the Root Port
lacking support for completing them:

  00:00.1 Ethernet controller: Mellanox Technologies MT2894 Family [ConnectX-6 Lx]
          Subsystem: Mellanox Technologies Device 0002
          DevCtl2: Completion Timeout: 50us to 50ms, TimeoutDis-
                   AtomicOpsCtl: ReqEn+

On s390 and many virtualized guests, the Endpoint is visible but the Root
Port is not.  In this case, pci_enable_atomic_ops_to_root() previously
enabled AtomicOps in the Endpoint even though it can't tell whether the
Root Port supports them as a completer.

Change pci_enable_atomic_ops_to_root() to fail if there's no Root Port or
the Root Port doesn't support AtomicOps.

Fixes: 430a23689dea ("PCI: Add pci_enable_atomic_ops_to_root()")
Reported-by: Alexander Schmidt <alexs@linux.ibm.com>
Signed-off-by: Gerd Bayer <gbayer@linux.ibm.com>
[bhelgaas: commit log, check RP first to simplify flow]
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Link: https://patch.msgid.link/20260330-fix_pciatops-v7-2-f601818417e8@linux.ibm.com
drivers/pci/pci.c

index 135e5b591df405e87e7f520a618d7e2ccba55ce1..3a4ba587042bfce9038ee919c94203dde6a009a9 100644 (file)
@@ -3675,8 +3675,7 @@ void pci_acs_init(struct pci_dev *dev)
  */
 int pci_enable_atomic_ops_to_root(struct pci_dev *dev, u32 cap_mask)
 {
-       struct pci_bus *bus = dev->bus;
-       struct pci_dev *bridge;
+       struct pci_dev *root, *bridge;
        u32 cap, ctl2;
 
        /*
@@ -3705,35 +3704,35 @@ int pci_enable_atomic_ops_to_root(struct pci_dev *dev, u32 cap_mask)
                return -EINVAL;
        }
 
-       while (bus->parent) {
-               bridge = bus->self;
+       root = pcie_find_root_port(dev);
+       if (!root)
+               return -EINVAL;
 
-               pcie_capability_read_dword(bridge, PCI_EXP_DEVCAP2, &cap);
+       pcie_capability_read_dword(root, PCI_EXP_DEVCAP2, &cap);
+       if ((cap & cap_mask) != cap_mask)
+               return -EINVAL;
 
+       bridge = pci_upstream_bridge(dev);
+       while (bridge != root) {
                switch (pci_pcie_type(bridge)) {
-               /* Ensure switch ports support AtomicOp routing */
                case PCI_EXP_TYPE_UPSTREAM:
-               case PCI_EXP_TYPE_DOWNSTREAM:
-                       if (!(cap & PCI_EXP_DEVCAP2_ATOMIC_ROUTE))
-                               return -EINVAL;
-                       break;
-
-               /* Ensure root port supports all the sizes we care about */
-               case PCI_EXP_TYPE_ROOT_PORT:
-                       if ((cap & cap_mask) != cap_mask)
-                               return -EINVAL;
-                       break;
-               }
-
-               /* Ensure upstream ports don't block AtomicOps on egress */
-               if (pci_pcie_type(bridge) == PCI_EXP_TYPE_UPSTREAM) {
+                       /* Upstream ports must not block AtomicOps on egress */
                        pcie_capability_read_dword(bridge, PCI_EXP_DEVCTL2,
                                                   &ctl2);
                        if (ctl2 & PCI_EXP_DEVCTL2_ATOMIC_EGRESS_BLOCK)
                                return -EINVAL;
+                       fallthrough;
+
+               /* All switch ports need to route AtomicOps */
+               case PCI_EXP_TYPE_DOWNSTREAM:
+                       pcie_capability_read_dword(bridge, PCI_EXP_DEVCAP2,
+                                                  &cap);
+                       if (!(cap & PCI_EXP_DEVCAP2_ATOMIC_ROUTE))
+                               return -EINVAL;
+                       break;
                }
 
-               bus = bus->parent;
+               bridge = pci_upstream_bridge(bridge);
        }
 
        pcie_capability_set_word(dev, PCI_EXP_DEVCTL2,