Commit d02034b4 authored by Robert Richter's avatar Robert Richter Committed by Dan Williams

cxl/acpi: Directly bind the CEDT detected CHBCR to the Host Bridge's port

During a Host Bridge's downstream port enumeration the CHBS entries in
the CEDT table are parsed, its Component Register base address
extracted and then stored in struct cxl_dport. The CHBS may contain
either the RCRB (RCH mode) or the Host Bridge's Component Registers
(CHBCR, VH mode). The RCRB further contains the CXL downstream port
register base address, while in VH mode the CXL Downstream Switch
Ports are visible in the PCI hierarchy and the DP's component regs are
disovered using the CXL DVSEC register locator capability. The
Component Registers derived from the CHBS for both modes are different
and thus also must be treated differently. That is, in RCH mode, the
component regs base should be bound to the dport, but in VH mode to
the CXL host bridge's port object.

The current implementation stores the CHBCR in addition in struct
cxl_dport and copies it later from there to struct cxl_port. As a
result, the dport contains the wrong Component Registers base address
and, e.g. the RAS capability of a CXL Root Port cannot be detected.

To fix the CHBCR binding, attach it directly to the Host Bridge's
@cxl_port structure. Do this during port creation of the Host Bridge
in add_host_bridge_uport(). Factor out CHBS parsing code in
add_host_bridge_dport() and use it in both functions.
Co-developed-by: default avatarDan Williams <dan.j.williams@intel.com>
Signed-off-by: default avatarRobert Richter <rrichter@amd.com>
Signed-off-by: default avatarTerry Bowman <terry.bowman@amd.com>
Reviewed-by: default avatarJonathan Cameron <Jonathan.Cameron@huawei.com>
Link: https://lore.kernel.org/r/20230622205523.85375-10-terry.bowman@amd.comSigned-off-by: default avatarDan Williams <dan.j.williams@intel.com>
parent f44c7b7a
...@@ -335,13 +335,13 @@ struct cxl_chbs_context { ...@@ -335,13 +335,13 @@ struct cxl_chbs_context {
u32 cxl_version; u32 cxl_version;
}; };
static int cxl_get_chbs(union acpi_subtable_headers *header, void *arg, static int cxl_get_chbs_iter(union acpi_subtable_headers *header, void *arg,
const unsigned long end) const unsigned long end)
{ {
struct cxl_chbs_context *ctx = arg; struct cxl_chbs_context *ctx = arg;
struct acpi_cedt_chbs *chbs; struct acpi_cedt_chbs *chbs;
if (ctx->base) if (ctx->base != CXL_RESOURCE_NONE)
return 0; return 0;
chbs = (struct acpi_cedt_chbs *) header; chbs = (struct acpi_cedt_chbs *) header;
...@@ -350,8 +350,6 @@ static int cxl_get_chbs(union acpi_subtable_headers *header, void *arg, ...@@ -350,8 +350,6 @@ static int cxl_get_chbs(union acpi_subtable_headers *header, void *arg,
return 0; return 0;
ctx->cxl_version = chbs->cxl_version; ctx->cxl_version = chbs->cxl_version;
ctx->base = CXL_RESOURCE_NONE;
if (!chbs->base) if (!chbs->base)
return 0; return 0;
...@@ -364,11 +362,35 @@ static int cxl_get_chbs(union acpi_subtable_headers *header, void *arg, ...@@ -364,11 +362,35 @@ static int cxl_get_chbs(union acpi_subtable_headers *header, void *arg,
return 0; return 0;
} }
static int cxl_get_chbs(struct device *dev, struct acpi_device *hb,
struct cxl_chbs_context *ctx)
{
unsigned long long uid;
int rc;
rc = acpi_evaluate_integer(hb->handle, METHOD_NAME__UID, NULL, &uid);
if (rc != AE_OK) {
dev_err(dev, "unable to retrieve _UID\n");
return -ENOENT;
}
dev_dbg(dev, "UID found: %lld\n", uid);
*ctx = (struct cxl_chbs_context) {
.dev = dev,
.uid = uid,
.base = CXL_RESOURCE_NONE,
.cxl_version = UINT_MAX,
};
acpi_table_parse_cedt(ACPI_CEDT_TYPE_CHBS, cxl_get_chbs_iter, ctx);
return 0;
}
static int add_host_bridge_dport(struct device *match, void *arg) static int add_host_bridge_dport(struct device *match, void *arg)
{ {
acpi_status rc; acpi_status rc;
struct device *bridge; struct device *bridge;
unsigned long long uid;
struct cxl_dport *dport; struct cxl_dport *dport;
struct cxl_chbs_context ctx; struct cxl_chbs_context ctx;
struct acpi_pci_root *pci_root; struct acpi_pci_root *pci_root;
...@@ -379,41 +401,38 @@ static int add_host_bridge_dport(struct device *match, void *arg) ...@@ -379,41 +401,38 @@ static int add_host_bridge_dport(struct device *match, void *arg)
if (!hb) if (!hb)
return 0; return 0;
rc = acpi_evaluate_integer(hb->handle, METHOD_NAME__UID, NULL, &uid); rc = cxl_get_chbs(match, hb, &ctx);
if (rc != AE_OK) { if (rc)
dev_err(match, "unable to retrieve _UID\n"); return rc;
return -ENODEV;
}
dev_dbg(match, "UID found: %lld\n", uid);
ctx = (struct cxl_chbs_context) {
.dev = match,
.uid = uid,
};
acpi_table_parse_cedt(ACPI_CEDT_TYPE_CHBS, cxl_get_chbs, &ctx);
if (!ctx.base) { if (ctx.cxl_version == UINT_MAX) {
dev_warn(match, "No CHBS found for Host Bridge (UID %lld)\n", dev_warn(match, "No CHBS found for Host Bridge (UID %lld)\n",
uid); ctx.uid);
return 0; return 0;
} }
if (ctx.base == CXL_RESOURCE_NONE) { if (ctx.base == CXL_RESOURCE_NONE) {
dev_warn(match, "CHBS invalid for Host Bridge (UID %lld)\n", dev_warn(match, "CHBS invalid for Host Bridge (UID %lld)\n",
uid); ctx.uid);
return 0; return 0;
} }
pci_root = acpi_pci_find_root(hb->handle); pci_root = acpi_pci_find_root(hb->handle);
bridge = pci_root->bus->bridge; bridge = pci_root->bus->bridge;
/*
* In RCH mode, bind the component regs base to the dport. In
* VH mode it will be bound to the CXL host bridge's port
* object later in add_host_bridge_uport().
*/
if (ctx.cxl_version == ACPI_CEDT_CHBS_VERSION_CXL11) { if (ctx.cxl_version == ACPI_CEDT_CHBS_VERSION_CXL11) {
dev_dbg(match, "RCRB found for UID %lld: %pa\n", uid, &ctx.base); dev_dbg(match, "RCRB found for UID %lld: %pa\n", ctx.uid,
dport = devm_cxl_add_rch_dport(root_port, bridge, uid, ctx.base); &ctx.base);
dport = devm_cxl_add_rch_dport(root_port, bridge, ctx.uid,
ctx.base);
} else { } else {
dev_dbg(match, "CHBCR found for UID %lld: %pa\n", uid, &ctx.base); dport = devm_cxl_add_dport(root_port, bridge, ctx.uid,
dport = devm_cxl_add_dport(root_port, bridge, uid, ctx.base); CXL_RESOURCE_NONE);
} }
if (IS_ERR(dport)) if (IS_ERR(dport))
...@@ -435,6 +454,8 @@ static int add_host_bridge_uport(struct device *match, void *arg) ...@@ -435,6 +454,8 @@ static int add_host_bridge_uport(struct device *match, void *arg)
struct cxl_dport *dport; struct cxl_dport *dport;
struct cxl_port *port; struct cxl_port *port;
struct device *bridge; struct device *bridge;
struct cxl_chbs_context ctx;
resource_size_t component_reg_phys;
int rc; int rc;
if (!hb) if (!hb)
...@@ -453,12 +474,26 @@ static int add_host_bridge_uport(struct device *match, void *arg) ...@@ -453,12 +474,26 @@ static int add_host_bridge_uport(struct device *match, void *arg)
return 0; return 0;
} }
rc = cxl_get_chbs(match, hb, &ctx);
if (rc)
return rc;
if (ctx.cxl_version == ACPI_CEDT_CHBS_VERSION_CXL11) {
dev_warn(bridge,
"CXL CHBS version mismatch, skip port registration\n");
return 0;
}
component_reg_phys = ctx.base;
if (component_reg_phys != CXL_RESOURCE_NONE)
dev_dbg(match, "CHBCR found for UID %lld: %pa\n",
ctx.uid, &component_reg_phys);
rc = devm_cxl_register_pci_bus(host, bridge, pci_root->bus); rc = devm_cxl_register_pci_bus(host, bridge, pci_root->bus);
if (rc) if (rc)
return rc; return rc;
port = devm_cxl_add_port(host, bridge, dport->component_reg_phys, port = devm_cxl_add_port(host, bridge, component_reg_phys, dport);
dport);
if (IS_ERR(port)) if (IS_ERR(port))
return PTR_ERR(port); return PTR_ERR(port);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment