]> git.itanic.dy.fi Git - linux-stable/commitdiff
cxl/rch: Prepare for caching the MMIO mapped PCIe AER capability
authorDan Williams <dan.j.williams@intel.com>
Thu, 22 Jun 2023 20:54:59 +0000 (15:54 -0500)
committerDan Williams <dan.j.williams@intel.com>
Sun, 25 Jun 2023 18:35:26 +0000 (11:35 -0700)
Prepare cxl_probe_rcrb() for retrieving more than just the component
register block. The RCH AER handling code wants to get back to the AER
capability that happens to be MMIO mapped rather then configuration
cycles.

Move RCRB specific downstream port data, like the RCRB base and the
AER capability offset, into its own data structure ('struct
cxl_rcrb_info') for cxl_probe_rcrb() to fill. Extend 'struct
cxl_dport' to include a 'struct cxl_rcrb_info' attribute.

This centralizes all RCRB scanning in one routine.

Co-developed-by: Robert Richter <rrichter@amd.com>
Signed-off-by: Robert Richter <rrichter@amd.com>
Signed-off-by: Terry Bowman <terry.bowman@amd.com>
Reviewed-by: Jonathan Cameron <Jonathan.Cameron@huawei.com>
Link: https://lore.kernel.org/r/20230622205523.85375-4-terry.bowman@amd.com
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
drivers/cxl/core/core.h
drivers/cxl/core/port.c
drivers/cxl/core/regs.c
drivers/cxl/cxl.h
tools/testing/cxl/test/mock.c

index bd0a5788c696b9b2a5b8b72d0032db7484618c30..b001669a513343f68393706ac94fa5e1aeb4052b 100644 (file)
@@ -68,7 +68,9 @@ enum cxl_rcrb {
        CXL_RCRB_DOWNSTREAM,
        CXL_RCRB_UPSTREAM,
 };
-resource_size_t __rcrb_to_component(struct device *dev, resource_size_t rcrb,
+struct cxl_rcrb_info;
+resource_size_t __rcrb_to_component(struct device *dev,
+                                   struct cxl_rcrb_info *ri,
                                    enum cxl_rcrb which);
 
 extern struct rw_semaphore cxl_dpa_rwsem;
index 45f5299af7a6fe8e2a38b2d550ad280d03755cff..76888c75dae450acf5923d3f855e993b67c2cdd3 100644 (file)
@@ -939,7 +939,8 @@ __devm_cxl_add_dport(struct cxl_port *port, struct device *dport_dev,
                return ERR_PTR(-ENOMEM);
 
        if (rcrb != CXL_RESOURCE_NONE) {
-               component_reg_phys = __rcrb_to_component(dport_dev, rcrb,
+               dport->rcrb.base = rcrb;
+               component_reg_phys = __rcrb_to_component(dport_dev, &dport->rcrb,
                                                         CXL_RCRB_DOWNSTREAM);
                if (component_reg_phys == CXL_RESOURCE_NONE) {
                        dev_warn(dport_dev, "Invalid Component Registers in RCRB");
@@ -957,7 +958,6 @@ __devm_cxl_add_dport(struct cxl_port *port, struct device *dport_dev,
        dport->port_id = port_id;
        dport->component_reg_phys = component_reg_phys;
        dport->port = port;
-       dport->rcrb = rcrb;
 
        cond_cxl_root_lock(port);
        rc = add_dport(port, dport);
index 564dd430258ab13adbfafc4781771d2abab3fb01..6c4b33133918efc0e1824fde2ecb5c15af7e3a83 100644 (file)
@@ -332,10 +332,11 @@ int cxl_find_regblock(struct pci_dev *pdev, enum cxl_regloc_type type,
 }
 EXPORT_SYMBOL_NS_GPL(cxl_find_regblock, CXL);
 
-resource_size_t __rcrb_to_component(struct device *dev, resource_size_t rcrb,
+resource_size_t __rcrb_to_component(struct device *dev, struct cxl_rcrb_info *ri,
                                    enum cxl_rcrb which)
 {
        resource_size_t component_reg_phys;
+       resource_size_t rcrb = ri->base;
        void __iomem *addr;
        u32 bar0, bar1;
        u16 cmd;
@@ -400,6 +401,6 @@ resource_size_t cxl_rcd_component_reg_phys(struct device *dev,
 {
        if (!dport->rch)
                return CXL_RESOURCE_NONE;
-       return __rcrb_to_component(dev, dport->rcrb, CXL_RCRB_UPSTREAM);
+       return __rcrb_to_component(dev, &dport->rcrb, CXL_RCRB_UPSTREAM);
 }
 EXPORT_SYMBOL_NS_GPL(cxl_rcd_component_reg_phys, CXL);
index 28888bb0c0885922ec3094459f26189614ee96bb..7c8674079f1a8e3f26b8aa4361c964ec3ad69cdf 100644 (file)
@@ -582,12 +582,17 @@ cxl_find_dport_by_dev(struct cxl_port *port, const struct device *dport_dev)
        return xa_load(&port->dports, (unsigned long)dport_dev);
 }
 
+struct cxl_rcrb_info {
+       resource_size_t base;
+       u16 aer_cap;
+};
+
 /**
  * struct cxl_dport - CXL downstream port
  * @dport: PCI bridge or firmware device representing the downstream link
  * @port_id: unique hardware identifier for dport in decoder target list
  * @component_reg_phys: downstream port component registers
- * @rcrb: base address for the Root Complex Register Block
+ * @rcrb: Data about the Root Complex Register Block layout
  * @rch: Indicate whether this dport was enumerated in RCH or VH mode
  * @port: reference to cxl_port that contains this downstream port
  */
@@ -595,7 +600,7 @@ struct cxl_dport {
        struct device *dport;
        int port_id;
        resource_size_t component_reg_phys;
-       resource_size_t rcrb;
+       struct cxl_rcrb_info rcrb;
        bool rch;
        struct cxl_port *port;
 };
index 30119a16ae856f7f156564bee6833f34dc9493ba..dbeef5c6f606d887fdd0abe6fcb54e637a27d4cb 100644 (file)
@@ -271,8 +271,10 @@ struct cxl_dport *__wrap_devm_cxl_add_rch_dport(struct cxl_port *port,
        if (ops && ops->is_mock_port(dport_dev)) {
                dport = devm_cxl_add_dport(port, dport_dev, port_id,
                                           CXL_RESOURCE_NONE);
-               if (!IS_ERR(dport))
+               if (!IS_ERR(dport)) {
+                       dport->rcrb.base = rcrb;
                        dport->rch = true;
+               }
        } else
                dport = devm_cxl_add_rch_dport(port, dport_dev, port_id, rcrb);
        put_cxl_mock_ops(index);