/** * DOC: cxl core region * * CXL Regions represent mapped memory capacity in system physical address * space. Whereas the CXL Root Decoders identify the bounds of potential CXL * Memory ranges, Regions represent the active mapped capacity by the HDM * Decoder Capability structures throughout the Host Bridges, Switches, and * Endpoints in the topology. * * Region configuration has ordering constraints. UUID may be set at any time * but is only visible for persistent regions. * 1. Interleave granularity * 2. Interleave size * 3. Decoder targets
*/
staticvoid cxl_region_decode_reset(struct cxl_region *cxlr, int count)
{ struct cxl_region_params *p = &cxlr->params; int i;
/* * Before region teardown attempt to flush, evict any data cached for * this region, or scream loudly about missing arch / platform support * for CXL teardown.
*/
cxl_region_invalidate_memregion(cxlr);
ACQUIRE(rwsem_write_kill, rwsem)(&cxl_rwsem.region); if ((rc = ACQUIRE_ERR(rwsem_write_kill, &rwsem))) return rc;
/* Already in the requested state? */ if (p->state >= CXL_CONFIG_COMMIT) return 0;
/* Not ready to commit? */ if (p->state < CXL_CONFIG_ACTIVE) return -ENXIO;
/* * Invalidate caches before region setup to drop any speculative * consumption of this address space
*/
rc = cxl_region_invalidate_memregion(cxlr); if (rc) return rc;
rc = cxl_region_decode_commit(cxlr); if (rc) return rc;
if (commit) {
rc = __commit(cxlr); if (rc) return rc; return len;
}
rc = queue_reset(cxlr); if (rc) return rc;
/* * Unmap the region and depend the reset-pending state to ensure * it does not go active again until post reset
*/
device_release_driver(&cxlr->dev);
/* * With the reset pending take cxl_rwsem.region unconditionally * to ensure the reset gets handled before returning.
*/
guard(rwsem_write)(&cxl_rwsem.region);
/* * Revalidate that the reset is still pending in case another * thread already handled this reset.
*/ if (p->state == CXL_CONFIG_RESET_PENDING) {
cxl_region_decode_reset(cxlr, p->interleave_ways);
p->state = CXL_CONFIG_ACTIVE;
}
/* * Support tooling that expects to find a 'uuid' attribute for all * regions regardless of mode.
*/ if (a == &dev_attr_uuid.attr && cxlr->mode != CXL_PARTMODE_PMEM) return 0444; return a->mode;
}
/* * Even for x3, x6, and x12 interleaves the region interleave must be a * power of 2 multiple of the host bridge interleave.
*/ if (!is_power_of_2(val / cxld->interleave_ways) ||
(val % cxld->interleave_ways)) {
dev_dbg(&cxlr->dev, "invalid interleave: %d\n", val); return -EINVAL;
}
ACQUIRE(rwsem_write_kill, rwsem)(&cxl_rwsem.region); if ((rc = ACQUIRE_ERR(rwsem_write_kill, &rwsem))) return rc;
if (p->state >= CXL_CONFIG_INTERLEAVE_ACTIVE) return -EBUSY;
save = p->interleave_ways;
p->interleave_ways = val;
rc = sysfs_update_group(&cxlr->dev.kobj, get_cxl_region_target_group()); if (rc) {
p->interleave_ways = save; return rc;
}
rc = granularity_to_eig(val, &ig); if (rc) return rc;
/* * When the host-bridge is interleaved, disallow region granularity != * root granularity. Regions with a granularity less than the root * interleave result in needing multiple endpoints to support a single * slot in the interleave (possible to support in the future). Regions * with a granularity greater than the root interleave result in invalid * DPA translations (invalid to support).
*/ if (cxld->interleave_ways > 1 && val != cxld->interleave_granularity) return -EINVAL;
ACQUIRE(rwsem_write_kill, rwsem)(&cxl_rwsem.region); if ((rc = ACQUIRE_ERR(rwsem_write_kill, &rwsem))) return rc;
if (p->state >= CXL_CONFIG_INTERLEAVE_ACTIVE) return -EBUSY;
/* Nothing to do... */ if (p->res && resource_size(p->res) == size) return 0;
/* To change size the old size must be freed first */ if (p->res) return -EBUSY;
if (p->state >= CXL_CONFIG_INTERLEAVE_ACTIVE) return -EBUSY;
/* ways, granularity and uuid (if PMEM) need to be set before HPA */ if (!p->interleave_ways || !p->interleave_granularity ||
(cxlr->mode == CXL_PARTMODE_PMEM && uuid_is_null(&p->uuid))) return -ENXIO;
div64_u64_rem(size, (u64)SZ_256M * p->interleave_ways, &remainder); if (remainder) return -EINVAL;
res = alloc_free_mem_region(cxlrd->res, size, SZ_256M,
dev_name(&cxlr->dev)); if (IS_ERR(res)) {
dev_dbg(&cxlr->dev, "HPA allocation error (%ld) for size:%pap in %s %pr\n",
PTR_ERR(res), &size, cxlrd->res->name, cxlrd->res); return PTR_ERR(res);
}
if (device_is_registered(&cxlr->dev))
lockdep_assert_held_write(&cxl_rwsem.region); if (p->res) { /* * Autodiscovered regions may not have been able to insert their * resource.
*/ if (p->res->parent)
remove_resource(p->res);
kfree(p->res);
p->res = NULL;
}
}
/* * if port->commit_end is not the only free decoder, then out of * order shutdown has occurred, block further allocations until * that is resolved
*/ if (((cxld->flags & CXL_DECODER_F_ENABLE) == 0)) return -EBUSY; return 0;
}
if (cxld->region) {
dev_dbg(dev->parent, "next decoder to commit (%s) is already reserved (%s)\n",
dev_name(dev), dev_name(&cxld->region->dev)); return 0;
}
rc = device_for_each_child_reverse_from(dev->parent, dev, NULL,
check_commit_order); if (rc) {
dev_dbg(dev->parent, "unable to allocate %s due to out of order shutdown\n",
dev_name(dev)); return 0;
} return 1;
}
staticbool region_res_match_cxl_range(conststruct cxl_region_params *p, conststruct range *range)
{ if (!p->res) returnfalse;
/* * If an extended linear cache region then the CXL range is assumed * to be fronted by the DRAM range in current known implementation. * This assumption will be made until a variant implementation exists.
*/ return p->res->start + p->cache_size == range->start &&
p->res->end == range->end;
}
/** * cxl_port_pick_region_decoder() - assign or lookup a decoder for a region * @port: a port in the ancestry of the endpoint implied by @cxled * @cxled: endpoint decoder to be, or currently, mapped by @port * @cxlr: region to establish, or validate, decode @port * * In the region creation path cxl_port_pick_region_decoder() is an * allocator to find a free port. In the region assembly path, it is * recalling the decoder that platform firmware picked for validation * purposes. * * The result is recorded in a 'struct cxl_region_ref' in @port.
*/ staticstruct cxl_decoder *
cxl_port_pick_region_decoder(struct cxl_port *port, struct cxl_endpoint_decoder *cxled, struct cxl_region *cxlr)
{ struct device *dev;
if (port == cxled_to_port(cxled)) return &cxled->cxld;
if (test_bit(CXL_REGION_F_AUTO, &cxlr->flags))
dev = device_find_child(&port->dev, &cxlr->params,
match_auto_decoder); else
dev = device_find_child(&port->dev, NULL, match_free_decoder); if (!dev) return NULL; /* * This decoder is pinned registered as long as the endpoint decoder is * registered, and endpoint decoder unregistration holds the * cxl_rwsem.region over unregister events, so no need to hold on to * this extra reference.
*/
put_device(dev); return to_cxl_decoder(dev);
}
/* * Allow the out of order assembly of auto-discovered regions. * Per CXL Spec 3.1 8.2.4.20.12 software must commit decoders * in HPA order. Confirm that the decoder with the lesser HPA * starting address has the lesser id.
*/
dev_dbg(&cxld->dev, "check for HPA violation %s:%d < %s:%d\n",
dev_name(&cxld->dev), cxld->id,
dev_name(&cxld_iter->dev), cxld_iter->id);
/* * Endpoints should already match the region type, but backstop that * assumption with an assertion. Switch-decoders change mapping-type * based on what is mapped when they are assigned to a region.
*/
dev_WARN_ONCE(&cxlr->dev,
port == cxled_to_port(cxled) &&
cxld->target_type != cxlr->type, "%s:%s mismatch decoder type %d -> %d\n",
dev_name(&cxled_to_memdev(cxled)->dev),
dev_name(&cxld->dev), cxld->target_type, cxlr->type);
cxld->target_type = cxlr->type;
cxl_rr->decoder = cxld; return 0;
}
/** * cxl_port_attach_region() - track a region's interest in a port by endpoint * @port: port to add a new region reference 'struct cxl_region_ref' * @cxlr: region to attach to @port * @cxled: endpoint decoder used to create or further pin a region reference * @pos: interleave position of @cxled in @cxlr * * The attach event is an opportunity to validate CXL decode setup * constraints and record metadata needed for programming HDM decoders, * in particular decoder target lists. * * The steps are: * * - validate that there are no other regions with a higher HPA already * associated with @port * - establish a region reference if one is not already present * * - additionally allocate a decoder instance that will host @cxlr on * @port * * - pin the region reference by the endpoint * - account for how many entries in @port's target list are needed to * cover all of the added endpoints.
*/ staticint cxl_port_attach_region(struct cxl_port *port, struct cxl_region *cxlr, struct cxl_endpoint_decoder *cxled, int pos)
{ struct cxl_memdev *cxlmd = cxled_to_memdev(cxled); struct cxl_ep *ep = cxl_ep_load(port, cxlmd); struct cxl_region_ref *cxl_rr; bool nr_targets_inc = false; struct cxl_decoder *cxld; unsignedlong index; int rc = -EBUSY;
lockdep_assert_held_write(&cxl_rwsem.region);
cxl_rr = cxl_rr_load(port, cxlr); if (cxl_rr) { struct cxl_ep *ep_iter; int found = 0;
/* * Walk the existing endpoints that have been attached to * @cxlr at @port and see if they share the same 'next' port * in the downstream direction. I.e. endpoints that share common * upstream switch.
*/
xa_for_each(&cxl_rr->endpoints, index, ep_iter) { if (ep_iter == ep) continue; if (ep_iter->next == ep->next) {
found++; break;
}
}
/* * New target port, or @port is an endpoint port that always * accounts its own local decode as a target.
*/ if (!found || !ep->next) {
cxl_rr->nr_targets++;
nr_targets_inc = true;
}
} else { struct cxl_decoder *cxld;
cxld = cxl_port_pick_region_decoder(port, cxled, cxlr); if (!cxld) {
dev_dbg(&cxlr->dev, "%s: no decoder available\n",
dev_name(&port->dev)); return -EBUSY;
}
cxl_rr = alloc_region_ref(port, cxlr, cxled, cxld); if (IS_ERR(cxl_rr)) {
dev_dbg(&cxlr->dev, "%s: failed to allocate region reference\n",
dev_name(&port->dev)); return PTR_ERR(cxl_rr);
}
nr_targets_inc = true;
/* * the number of targets should not exceed the target_count * of the decoder
*/ if (is_switch_decoder(&cxld->dev)) { struct cxl_switch_decoder *cxlsd;
cxl_rr = cxl_rr_load(port, cxlr); if (!cxl_rr) return;
/* * Endpoint ports do not carry cxl_ep references, and they * never target more than one endpoint by definition
*/ if (cxl_rr->decoder == &cxled->cxld)
cxl_rr->nr_eps--; else
ep = xa_erase(&cxl_rr->endpoints, (unsignedlong)cxled); if (ep) { struct cxl_ep *ep_iter; unsignedlong index; int found = 0;
cxl_rr->nr_eps--;
xa_for_each(&cxl_rr->endpoints, index, ep_iter) { if (ep_iter->next == ep->next) {
found++; break;
}
} if (!found)
cxl_rr->nr_targets--;
}
if (cxl_rr->nr_eps == 0)
free_region_ref(cxl_rr);
}
/* * If this position wants to share a dport with the last endpoint mapped * then that endpoint, at index 'position - distance', must also be * mapped by this dport.
*/ if (pos < distance) {
dev_dbg(&cxlr->dev, "%s:%s: cannot host %s:%s at %d\n",
dev_name(port->uport_dev), dev_name(&port->dev),
dev_name(&cxlmd->dev), dev_name(&cxled->cxld.dev), pos); return -ENXIO;
}
cxled_peer = p->targets[pos - distance];
cxlmd_peer = cxled_to_memdev(cxled_peer);
ep_peer = cxl_ep_load(port, cxlmd_peer); if (ep->dport != ep_peer->dport) {
dev_dbg(&cxlr->dev, "%s:%s: %s:%s pos %d mismatched peer %s:%s\n",
dev_name(port->uport_dev), dev_name(&port->dev),
dev_name(&cxlmd->dev), dev_name(&cxled->cxld.dev), pos,
dev_name(&cxlmd_peer->dev),
dev_name(&cxled_peer->cxld.dev)); return -ENXIO;
}
return 0;
}
staticint check_interleave_cap(struct cxl_decoder *cxld, int iw, int ig)
{ struct cxl_port *port = to_cxl_port(cxld->dev.parent); struct cxl_hdm *cxlhdm = dev_get_drvdata(&port->dev); unsignedint interleave_mask;
u8 eiw;
u16 eig; int high_pos, low_pos;
if (!test_bit(iw, &cxlhdm->iw_cap_mask)) return -ENXIO; /* * Per CXL specification r3.1(8.2.4.20.13 Decoder Protection), * if eiw < 8: * DPAOFFSET[51: eig + 8] = HPAOFFSET[51: eig + 8 + eiw] * DPAOFFSET[eig + 7: 0] = HPAOFFSET[eig + 7: 0] * * when the eiw is 0, all the bits of HPAOFFSET[51: 0] are used, the * interleave bits are none. * * if eiw >= 8: * DPAOFFSET[51: eig + 8] = HPAOFFSET[51: eig + eiw] / 3 * DPAOFFSET[eig + 7: 0] = HPAOFFSET[eig + 7: 0] * * when the eiw is 8, all the bits of HPAOFFSET[51: 0] are used, the * interleave bits are none.
*/
ways_to_eiw(iw, &eiw); if (eiw == 0 || eiw == 8) return 0;
/* * While root level decoders support x3, x6, x12, switch level * decoders only support powers of 2 up to x16.
*/ if (!is_power_of_2(cxl_rr->nr_targets)) {
dev_dbg(&cxlr->dev, "%s:%s: invalid target count %d\n",
dev_name(port->uport_dev), dev_name(&port->dev),
cxl_rr->nr_targets); return -EINVAL;
}
cxlsd = to_cxl_switch_decoder(&cxld->dev); if (cxl_rr->nr_targets_set) { int i, distance = 1; struct cxl_region_ref *cxl_rr_iter;
/* * The "distance" between peer downstream ports represents which * endpoint positions in the region interleave a given port can * host. * * For example, at the root of a hierarchy the distance is * always 1 as every index targets a different host-bridge. At * each subsequent switch level those ports map every Nth region * position where N is the width of the switch == distance.
*/ do {
cxl_rr_iter = cxl_rr_load(iter, cxlr);
distance *= cxl_rr_iter->nr_targets;
iter = to_cxl_port(iter->dev.parent);
} while (!is_cxl_root(iter));
distance *= cxlrd->cxlsd.cxld.interleave_ways;
for (i = 0; i < cxl_rr->nr_targets_set; i++) if (ep->dport == cxlsd->target[i]) {
rc = check_last_peer(cxled, ep, cxl_rr,
distance); if (rc) return rc; goto out_target_set;
} goto add_target;
}
if (is_cxl_root(parent_port)) { /* * Root decoder IG is always set to value in CFMWS which * may be different than this region's IG. We can use the * region's IG here since interleave_granularity_store() * does not allow interleaved host-bridges with * root IG != region IG.
*/
parent_ig = p->interleave_granularity;
parent_iw = cxlrd->cxlsd.cxld.interleave_ways; /* * For purposes of address bit routing, use power-of-2 math for * switch ports.
*/ if (!is_power_of_2(parent_iw))
parent_iw /= 3;
} else { struct cxl_region_ref *parent_rr; struct cxl_decoder *parent_cxld;
/* * In the auto-discovery case skip automatic teardown since the * address space is already active
*/ if (test_bit(CXL_REGION_F_AUTO, &cxlr->flags)) return;
for (i = 0; i < p->nr_targets; i++) {
cxled = p->targets[i];
cxlmd = cxled_to_memdev(cxled);
cxlds = cxlmd->cxlds;
if (cxlds->rcd) continue;
iter = cxled_to_port(cxled); while (!is_cxl_root(to_cxl_port(iter->dev.parent)))
iter = to_cxl_port(iter->dev.parent);
for (ep = cxl_ep_load(iter, cxlmd); iter;
iter = ep->next, ep = cxl_ep_load(iter, cxlmd))
cxl_port_reset_targets(iter, cxlr);
}
}
if (cxled->state != CXL_DECODER_STATE_AUTO) {
dev_err(&cxlr->dev, "%s: unable to add decoder to autodetected region\n",
dev_name(&cxled->cxld.dev)); return -EINVAL;
}
if (pos >= 0) {
dev_dbg(&cxlr->dev, "%s: expected auto position, not %d\n",
dev_name(&cxled->cxld.dev), pos); return -EINVAL;
}
if (p->nr_targets >= p->interleave_ways) {
dev_err(&cxlr->dev, "%s: no more target slots available\n",
dev_name(&cxled->cxld.dev)); return -ENXIO;
}
/* * Temporarily record the endpoint decoder into the target array. Yes, * this means that userspace can view devices in the wrong position * before the region activates, and must be careful to understand when * it might be racing region autodiscovery.
*/
pos = p->nr_targets;
p->targets[pos] = cxled;
cxled->pos = pos;
p->nr_targets++;
staticint find_pos_and_ways(struct cxl_port *port, struct range *range, int *pos, int *ways)
{ struct cxl_switch_decoder *cxlsd; struct cxl_port *parent; struct device *dev; int rc = -ENXIO;
parent = parent_port_of(port); if (!parent) return rc;
dev = device_find_child(&parent->dev, range,
match_switch_decoder_by_range); if (!dev) {
dev_err(port->uport_dev, "failed to find decoder mapping %#llx-%#llx\n",
range->start, range->end); return rc;
}
cxlsd = to_cxl_switch_decoder(dev);
*ways = cxlsd->cxld.interleave_ways;
for (int i = 0; i < *ways; i++) { if (cxlsd->target[i] == port->parent_dport) {
*pos = i;
rc = 0; break;
}
}
put_device(dev);
if (rc)
dev_err(port->uport_dev, "failed to find %s:%s in target list of %s\n",
dev_name(&port->dev),
dev_name(port->parent_dport->dport_dev),
dev_name(&cxlsd->cxld.dev));
return rc;
}
/** * cxl_calc_interleave_pos() - calculate an endpoint position in a region * @cxled: endpoint decoder member of given region * * The endpoint position is calculated by traversing the topology from * the endpoint to the root decoder and iteratively applying this * calculation: * * position = position * parent_ways + parent_pos; * * ...where @position is inferred from switch and root decoder target lists. * * Return: position >= 0 on success * -ENXIO on failure
*/ staticint cxl_calc_interleave_pos(struct cxl_endpoint_decoder *cxled)
{ struct cxl_port *iter, *port = cxled_to_port(cxled); struct cxl_memdev *cxlmd = cxled_to_memdev(cxled); struct range *range = &cxled->cxld.hpa_range; int parent_ways = 0, parent_pos = 0, pos = 0; int rc;
/* * Example: the expected interleave order of the 4-way region shown * below is: mem0, mem2, mem1, mem3 * * root_port * / \ * host_bridge_0 host_bridge_1 * | | | | * mem0 mem1 mem2 mem3 * * In the example the calculator will iterate twice. The first iteration * uses the mem position in the host-bridge and the ways of the host- * bridge to generate the first, or local, position. The second * iteration uses the host-bridge position in the root_port and the ways * of the root_port to refine the position. * * A trace of the calculation per endpoint looks like this: * mem0: pos = 0 * 2 + 0 mem2: pos = 0 * 2 + 0 * pos = 0 * 2 + 0 pos = 0 * 2 + 1 * pos: 0 pos: 1 * * mem1: pos = 0 * 2 + 1 mem3: pos = 0 * 2 + 1 * pos = 1 * 2 + 0 pos = 1 * 2 + 1 * pos: 2 pos = 3 * * Note that while this example is simple, the method applies to more * complex topologies, including those with switches.
*/
/* Iterate from endpoint to root_port refining the position */ for (iter = port; iter; iter = parent_port_of(iter)) { if (is_cxl_root(iter)) break;
rc = find_pos_and_ways(iter, range, &parent_pos, &parent_ways); if (rc) return rc;
staticint cxl_region_sort_targets(struct cxl_region *cxlr)
{ struct cxl_region_params *p = &cxlr->params; int i, rc = 0;
for (i = 0; i < p->nr_targets; i++) { struct cxl_endpoint_decoder *cxled = p->targets[i];
cxled->pos = cxl_calc_interleave_pos(cxled); /* * Record that sorting failed, but still continue to calc * cxled->pos so that follow-on code paths can reliably * do p->targets[cxled->pos] to self-reference their entry.
*/ if (cxled->pos < 0)
rc = -ENXIO;
} /* Keep the cxlr target list in interleave position order */
sort(p->targets, p->nr_targets, sizeof(p->targets[0]),
cmp_interleave_pos, NULL);
if (cxlds->part[cxled->part].mode != cxlr->mode) {
dev_dbg(&cxlr->dev, "%s region mode: %d mismatch\n",
dev_name(&cxled->cxld.dev), cxlr->mode); return -EINVAL;
}
/* all full of members, or interleave config not established? */ if (p->state > CXL_CONFIG_INTERLEAVE_ACTIVE) {
dev_dbg(&cxlr->dev, "region already active\n"); return -EBUSY;
}
if (test_bit(CXL_REGION_F_AUTO, &cxlr->flags)) { int i;
rc = cxl_region_attach_auto(cxlr, cxled, pos); if (rc) return rc;
/* await more targets to arrive... */ if (p->nr_targets < p->interleave_ways) return 0;
/* * All targets are here, which implies all PCI enumeration that * affects this region has been completed. Walk the topology to * sort the devices into their relative region decode position.
*/
rc = cxl_region_sort_targets(cxlr); if (rc) return rc;
for (i = 0; i < p->nr_targets; i++) {
cxled = p->targets[i];
ep_port = cxled_to_port(cxled);
dport = cxl_find_dport_by_dev(root_port,
ep_port->host_bridge);
rc = cxl_region_attach_position(cxlr, cxlrd, cxled,
dport, i); if (rc) return rc;
}
rc = cxl_region_setup_targets(cxlr); if (rc) return rc;
/* * If target setup succeeds in the autodiscovery case * then the region is already committed.
*/
p->state = CXL_CONFIG_COMMIT;
cxl_region_shared_upstream_bandwidth_update(cxlr);
return 0;
}
rc = cxl_region_validate_position(cxlr, cxled, pos); if (rc) return rc;
if (p->nr_targets != p->interleave_ways) return 0;
/* * Test the auto-discovery position calculator function * against this successfully created user-defined region. * A fail message here means that this interleave config * will fail when presented as CXL_REGION_F_AUTO.
*/ for (int i = 0; i < p->nr_targets; i++) { struct cxl_endpoint_decoder *cxled = p->targets[i]; int test_pos;
/* * Cleanup a decoder's interest in a region. There are 2 cases to * handle, removing an unknown @cxled from a known position in a region * (detach_target()) or removing a known @cxled from an unknown @cxlr * (cxld_unregister()) * * When the detachment finds a region release the region driver.
*/ int cxl_decoder_detach(struct cxl_region *cxlr, struct cxl_endpoint_decoder *cxled, int pos, enum cxl_detach_mode mode)
{ struct cxl_region *detach;
/* when the decoder is being destroyed lock unconditionally */ if (mode == DETACH_INVALIDATE) {
guard(rwsem_write)(&cxl_rwsem.region);
detach = __cxl_decoder_detach(cxlr, cxled, pos, mode);
} else { int rc;
staticvoid cxl_region_release(struct device *dev)
{ struct cxl_root_decoder *cxlrd = to_cxl_root_decoder(dev->parent); struct cxl_region *cxlr = to_cxl_region(dev); int id = atomic_read(&cxlrd->region_id);
/* * Try to reuse the recently idled id rather than the cached * next id to prevent the region id space from increasing * unnecessarily.
*/ if (cxlr->id < id) if (atomic_try_cmpxchg(&cxlrd->region_id, &id, cxlr->id)) {
memregion_free(id); goto out;
}
/* * Now that region sysfs is shutdown, the parameter block is now * read-only, so no need to hold the region rwsem to access the * region parameters.
*/ for (i = 0; i < p->interleave_ways; i++)
detach_target(cxlr, i);
dev = &cxlr->dev;
device_initialize(dev);
lockdep_set_class(&dev->mutex, &cxl_region_key);
dev->parent = &cxlrd->cxlsd.cxld.dev; /* * Keep root decoder pinned through cxl_region_release to fixup * region id allocations
*/
get_device(dev->parent);
device_set_pm_not_required(dev);
dev->bus = &cxl_bus_type;
dev->type = &cxl_region_type;
cxlr->id = id;
return cxlr;
}
staticbool cxl_region_update_coordinates(struct cxl_region *cxlr, int nid)
{ int cset = 0; int rc;
for (int i = 0; i < ACCESS_COORDINATE_MAX; i++) { if (cxlr->coord[i].read_bandwidth) {
rc = 0; if (cxl_need_node_perf_attrs_update(nid))
node_set_perf_attrs(nid, &cxlr->coord[i], i); else
rc = cxl_update_hmat_access_coordinates(nid, cxlr, i);
if (rc == 0)
cset++;
}
}
if (!cset) returnfalse;
rc = sysfs_update_group(&cxlr->dev.kobj, get_cxl_region_access0_group()); if (rc)
dev_dbg(&cxlr->dev, "Failed to update access0 group\n");
rc = sysfs_update_group(&cxlr->dev.kobj, get_cxl_region_access1_group()); if (rc)
dev_dbg(&cxlr->dev, "Failed to update access1 group\n");
returntrue;
}
staticint cxl_region_perf_attrs_callback(struct notifier_block *nb, unsignedlong action, void *arg)
{ struct cxl_region *cxlr = container_of(nb, struct cxl_region,
node_notifier); struct node_notify *nn = arg; int nid = nn->nid; int region_nid;
if (action != NODE_ADDED_FIRST_MEMORY) return NOTIFY_DONE;
/* * No need to hold cxl_rwsem.region; region parameters are stable * within the cxl_region driver.
*/
region_nid = phys_to_target_node(cxlr->params.res->start); if (nid != region_nid)
--> --------------------
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.