[PATCH v2 09/12] cxl: Extend devm_cxl_enumerate_ports() to support restricted devices (RCDs)

From: Robert Richter
Date: Tue Oct 18 2022 - 09:25:29 EST


The PCIe Software View of an RCH and RCD is different to VH mode. An
RCD is paired with an RCH and shows up as RCiEP. Its downstream and
upstream ports are hidden to the PCI hierarchy. This different PCI
topology requires a different handling of RCHs.

Extend devm_cxl_enumerate_ports() to support restricted devices
(RCDs). If an RCD is detected all to do is to search its corresponding
RCH's port and attach the EP to it. Update cxl_mem_find_port() for
proper removal of the EP in delete_endpoint().

Signed-off-by: Robert Richter <rrichter@xxxxxxx>
---
drivers/cxl/core/port.c | 45 ++++++++++++++++++++++++++++++++++++++++-
1 file changed, 44 insertions(+), 1 deletion(-)

diff --git a/drivers/cxl/core/port.c b/drivers/cxl/core/port.c
index 4b15481426f7..35f8fa98904e 100644
--- a/drivers/cxl/core/port.c
+++ b/drivers/cxl/core/port.c
@@ -1384,16 +1384,56 @@ static int add_port_attach_ep(struct cxl_memdev *cxlmd,
return rc;
}

+static inline bool is_cxl_restricted(struct cxl_memdev *cxlmd)
+{
+ struct device *parent = cxlmd->dev.parent;
+ if (!dev_is_pci(parent))
+ return false;
+ return pci_pcie_type(to_pci_dev(parent)) == PCI_EXP_TYPE_RC_END;
+}
+
+static int restricted_host_enumerate_port(struct cxl_memdev *cxlmd)
+{
+ struct device *dev, *dport_dev, *uport_dev;
+ int count;
+
+ if (!is_cxl_restricted(cxlmd))
+ return 0;
+
+ /*
+ * The cxlmd is an RCD, the dport_dev of it is the PCI device
+ * and the uport_dev is its host bridge which is the parent of
+ * the PCI device.
+ */
+ dev = &cxlmd->dev; /* cxlmd */
+ dport_dev = dev->parent; /* pci_dev */
+ uport_dev = dev->parent->parent; /* pci_host_bridge */
+
+ count = find_port_attach_ep(cxlmd, uport_dev, dport_dev, dev);
+
+ /* If missing the host is not yet ready. */
+ if (!count)
+ return -EAGAIN;
+
+ return count;
+}
+
int devm_cxl_enumerate_ports(struct cxl_memdev *cxlmd)
{
struct device *dev = &cxlmd->dev;
struct device *iter;
- int rc;
+ int count, rc;

rc = devm_add_action_or_reset(&cxlmd->dev, cxl_detach_ep, cxlmd);
if (rc)
return rc;

+ count = restricted_host_enumerate_port(cxlmd);
+ if (count < 0)
+ return count;
+ if (count)
+ return 0;
+
/*
* Scan for and add all cxl_ports in this device's ancestry.
* Repeat until no more ports are added. Abort if a port add
@@ -1445,6 +1485,9 @@ EXPORT_SYMBOL_NS_GPL(devm_cxl_enumerate_ports, CXL);
struct cxl_port *cxl_mem_find_port(struct cxl_memdev *cxlmd,
struct cxl_dport **dport)
{
+ if (is_cxl_restricted(cxlmd))
+ return find_cxl_port(cxlmd->dev.parent, dport);
+
return find_cxl_port(grandparent(&cxlmd->dev), dport);
}
EXPORT_SYMBOL_NS_GPL(cxl_mem_find_port, CXL);
--
2.30.2