Message ID | 20240716042540.89639-2-kobayashi.da-06@fujitsu.com |
---|---|
State | Superseded |
Headers | show |
Series | Export cxl1.1 device link status register value to pci device sysfs. | expand |
Kobayashi,Daisuke wrote: > Add rcd_pcie_cap and its initialization to cache the offset of cxl1.1 > device link status information. By caching it, avoid the walking > memory map area to find the offset when output the register value. > > Reviewed-by: Jonathan Cameron <Jonathan.Cameron@huawei.com> > Signed-off-by: "Kobayashi,Daisuke" <kobayashi.da-06@fujitsu.com> [..] Apologies for the delay. I notice that patch2 has another port leak bug which means lets take the opportunity to make cxl_pci_find_port() usage safe in this patch. Fold in this cover letter update and change: "Given that this solution involves port lookups via cxl_pci_find_port() and multiple exit paths where that reference needs to be dropped, introduce a new put_cxl_port() scope-based-free handler." > @@ -503,6 +496,8 @@ static int cxl_rcrb_get_comp_regs(struct pci_dev *pdev, > static int cxl_pci_setup_regs(struct pci_dev *pdev, enum cxl_regloc_type type, > struct cxl_register_map *map) > { > + struct cxl_dport *dport; > + struct cxl_port *port; > int rc; > > rc = cxl_find_regblock(pdev, type, map); > @@ -512,11 +507,25 @@ static int cxl_pci_setup_regs(struct pci_dev *pdev, enum cxl_regloc_type type, > * is an RCH and try to extract the Component Registers from > * an RCRB. > */ > - if (rc && type == CXL_REGLOC_RBI_COMPONENT && is_cxl_restricted(pdev)) > - rc = cxl_rcrb_get_comp_regs(pdev, map); > + if (rc && type == CXL_REGLOC_RBI_COMPONENT && is_cxl_restricted(pdev)) { > + port = cxl_pci_find_port(pdev, &dport); ...then this can become: struct cxl_port *port __free(put_cxl_port) = cxl_pci_find_port(pdev, &dport); > + if (!port) > + return -EPROBE_DEFER; > > - if (rc) > + rc = cxl_rcrb_get_comp_regs(pdev, map, dport); > + if (rc) { > + put_device(&port->dev); > + return rc; > + } > + > + rc = cxl_dport_map_rcd_linkcap(pdev, dport); > + if (rc) { > + put_device(&port->dev); > + return rc; > + } ...then these calls to put_device() can be dropped.
Dan Williams wrote: > Kobayashi,Daisuke wrote: > > Add rcd_pcie_cap and its initialization to cache the offset of cxl1.1 > > device link status information. By caching it, avoid the walking > > memory map area to find the offset when output the register value. > > > > Reviewed-by: Jonathan Cameron <Jonathan.Cameron@huawei.com> > > Signed-off-by: "Kobayashi,Daisuke" <kobayashi.da-06@fujitsu.com> > [..] > > Apologies for the delay. > Thank you for your review. > I notice that patch2 has another port leak bug which means lets take the > opportunity to make cxl_pci_find_port() usage safe in this patch. > I will fix the port leak bug included in patch 2. > Fold in this cover letter update and change: > > "Given that this solution involves port lookups via cxl_pci_find_port() > and multiple exit paths where that reference needs to be dropped, > introduce a new put_cxl_port() scope-based-free handler." > I'll add it to the cover letter for this patch. The following two issues will also be fixed in the next patch. > > > @@ -503,6 +496,8 @@ static int cxl_rcrb_get_comp_regs(struct pci_dev > *pdev, > > static int cxl_pci_setup_regs(struct pci_dev *pdev, enum cxl_regloc_type > type, > > struct cxl_register_map *map) > > { > > + struct cxl_dport *dport; > > + struct cxl_port *port; > > int rc; > > > > rc = cxl_find_regblock(pdev, type, map); > > @@ -512,11 +507,25 @@ static int cxl_pci_setup_regs(struct pci_dev *pdev, > enum cxl_regloc_type type, > > * is an RCH and try to extract the Component Registers from > > * an RCRB. > > */ > > - if (rc && type == CXL_REGLOC_RBI_COMPONENT && > is_cxl_restricted(pdev)) > > - rc = cxl_rcrb_get_comp_regs(pdev, map); > > + if (rc && type == CXL_REGLOC_RBI_COMPONENT && > is_cxl_restricted(pdev)) { > > + port = cxl_pci_find_port(pdev, &dport); > > ...then this can become: > > struct cxl_port *port __free(put_cxl_port) = cxl_pci_find_port(pdev, > &dport); > > > + if (!port) > > + return -EPROBE_DEFER; > > > > - if (rc) > > + rc = cxl_rcrb_get_comp_regs(pdev, map, dport); > > + if (rc) { > > + put_device(&port->dev); > > + return rc; > > + } > > + > > + rc = cxl_dport_map_rcd_linkcap(pdev, dport); > > + if (rc) { > > + put_device(&port->dev); > > + return rc; > > + } > > ...then these calls to put_device() can be dropped.
diff --git a/drivers/cxl/core/core.h b/drivers/cxl/core/core.h index 3b64fb1b9ed0..95d94ec14eb6 100644 --- a/drivers/cxl/core/core.h +++ b/drivers/cxl/core/core.h @@ -75,6 +75,11 @@ resource_size_t __rcrb_to_component(struct device *dev, enum cxl_rcrb which); u16 cxl_rcrb_to_aer(struct device *dev, resource_size_t rcrb); +#define PCI_RCRB_CAP_LIST_ID_MASK GENMASK(7, 0) +#define PCI_RCRB_CAP_HDR_ID_MASK GENMASK(7, 0) +#define PCI_RCRB_CAP_HDR_NEXT_MASK GENMASK(15, 8) +#define PCI_CAP_EXP_SIZEOF 0x3c + extern struct rw_semaphore cxl_dpa_rwsem; extern struct rw_semaphore cxl_region_rwsem; diff --git a/drivers/cxl/core/regs.c b/drivers/cxl/core/regs.c index 372786f80955..b1ab0d9bcbcb 100644 --- a/drivers/cxl/core/regs.c +++ b/drivers/cxl/core/regs.c @@ -505,6 +505,62 @@ u16 cxl_rcrb_to_aer(struct device *dev, resource_size_t rcrb) return offset; } +static resource_size_t cxl_rcrb_to_linkcap(struct device *dev, struct cxl_dport *dport) +{ + resource_size_t rcrb = dport->rcrb.base; + void __iomem *addr; + u32 cap_hdr; + u16 offset; + + if (!request_mem_region(rcrb, SZ_4K, "CXL RCRB")) + return CXL_RESOURCE_NONE; + + addr = ioremap(rcrb, SZ_4K); + if (!addr) { + dev_err(dev, "Failed to map region %pr\n", addr); + release_mem_region(rcrb, SZ_4K); + return CXL_RESOURCE_NONE; + } + + offset = FIELD_GET(PCI_RCRB_CAP_LIST_ID_MASK, readw(addr + PCI_CAPABILITY_LIST)); + cap_hdr = readl(addr + offset); + while ((FIELD_GET(PCI_RCRB_CAP_HDR_ID_MASK, cap_hdr)) != PCI_CAP_ID_EXP) { + offset = FIELD_GET(PCI_RCRB_CAP_HDR_NEXT_MASK, cap_hdr); + if (offset == 0 || offset > SZ_4K) { + offset = 0; + break; + } + cap_hdr = readl(addr + offset); + } + + iounmap(addr); + release_mem_region(rcrb, SZ_4K); + if (!offset) + return CXL_RESOURCE_NONE; + + return offset; +} + +int cxl_dport_map_rcd_linkcap(struct pci_dev *pdev, struct cxl_dport *dport) +{ + void __iomem *dport_pcie_cap = NULL; + resource_size_t pos; + struct cxl_rcrb_info *ri; + + ri = &dport->rcrb; + pos = cxl_rcrb_to_linkcap(&pdev->dev, dport); + if (pos == CXL_RESOURCE_NONE) + return -ENXIO; + + dport_pcie_cap = devm_cxl_iomap_block(&pdev->dev, + ri->base + pos, + PCI_CAP_EXP_SIZEOF); + dport->regs.rcd_pcie_cap = dport_pcie_cap; + + return 0; +} +EXPORT_SYMBOL_NS_GPL(cxl_dport_map_rcd_linkcap, CXL); + resource_size_t __rcrb_to_component(struct device *dev, struct cxl_rcrb_info *ri, enum cxl_rcrb which) { diff --git a/drivers/cxl/cxl.h b/drivers/cxl/cxl.h index 003feebab79b..839c5a20cc33 100644 --- a/drivers/cxl/cxl.h +++ b/drivers/cxl/cxl.h @@ -230,6 +230,14 @@ struct cxl_regs { struct_group_tagged(cxl_rch_regs, rch_regs, void __iomem *dport_aer; ); + + /* + * RCD upstream port specific PCIe cap register + * @pcie_cap: CXL 3.0 8.2.1.2 RCD Upstream Port RCRB + */ + struct_group_tagged(cxl_rcd_regs, rcd_regs, + void __iomem *rcd_pcie_cap; + ); }; struct cxl_reg_map { @@ -299,6 +307,7 @@ int cxl_setup_regs(struct cxl_register_map *map); struct cxl_dport; resource_size_t cxl_rcd_component_reg_phys(struct device *dev, struct cxl_dport *dport); +int cxl_dport_map_rcd_linkcap(struct pci_dev *pdev, struct cxl_dport *dport); #define CXL_RESOURCE_NONE ((resource_size_t) -1) #define CXL_TARGET_STRLEN 20 diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c index 2ff361e756d6..5e7e66f28420 100644 --- a/drivers/cxl/pci.c +++ b/drivers/cxl/pci.c @@ -471,10 +471,9 @@ static bool is_cxl_restricted(struct pci_dev *pdev) } static int cxl_rcrb_get_comp_regs(struct pci_dev *pdev, - struct cxl_register_map *map) + struct cxl_register_map *map, + struct cxl_dport *dport) { - struct cxl_port *port; - struct cxl_dport *dport; resource_size_t component_reg_phys; *map = (struct cxl_register_map) { @@ -482,14 +481,8 @@ static int cxl_rcrb_get_comp_regs(struct pci_dev *pdev, .resource = CXL_RESOURCE_NONE, }; - port = cxl_pci_find_port(pdev, &dport); - if (!port) - return -EPROBE_DEFER; - component_reg_phys = cxl_rcd_component_reg_phys(&pdev->dev, dport); - put_device(&port->dev); - if (component_reg_phys == CXL_RESOURCE_NONE) return -ENXIO; @@ -503,6 +496,8 @@ static int cxl_rcrb_get_comp_regs(struct pci_dev *pdev, static int cxl_pci_setup_regs(struct pci_dev *pdev, enum cxl_regloc_type type, struct cxl_register_map *map) { + struct cxl_dport *dport; + struct cxl_port *port; int rc; rc = cxl_find_regblock(pdev, type, map); @@ -512,11 +507,25 @@ static int cxl_pci_setup_regs(struct pci_dev *pdev, enum cxl_regloc_type type, * is an RCH and try to extract the Component Registers from * an RCRB. */ - if (rc && type == CXL_REGLOC_RBI_COMPONENT && is_cxl_restricted(pdev)) - rc = cxl_rcrb_get_comp_regs(pdev, map); + if (rc && type == CXL_REGLOC_RBI_COMPONENT && is_cxl_restricted(pdev)) { + port = cxl_pci_find_port(pdev, &dport); + if (!port) + return -EPROBE_DEFER; - if (rc) + rc = cxl_rcrb_get_comp_regs(pdev, map, dport); + if (rc) { + put_device(&port->dev); + return rc; + } + + rc = cxl_dport_map_rcd_linkcap(pdev, dport); + if (rc) { + put_device(&port->dev); + return rc; + } + } else if (rc) { return rc; + } return cxl_setup_regs(map); }