anitazha / rpms / ndctl

Forked from rpms/ndctl 2 years ago
Clone

Blame 0192-libcxl-Introduce-libcxl-region-and-mapping-objects.patch

Jeff Moyer 2c91dc
From d25dc6d7956bc022d7e4c4453416c52368df291d Mon Sep 17 00:00:00 2001
Jeff Moyer 2c91dc
From: Vishal Verma <vishal.l.verma@intel.com>
Jeff Moyer 2c91dc
Date: Mon, 15 Aug 2022 13:22:07 -0600
Jeff Moyer 2c91dc
Subject: [PATCH 192/217] libcxl: Introduce libcxl region and mapping objects
Jeff Moyer 2c91dc
Jeff Moyer 2c91dc
Add a cxl_region object to libcxl that represents a CXL region. CXL
Jeff Moyer 2c91dc
regions are made up of one or more cxl_memdev 'targets'. The
Jeff Moyer 2c91dc
relationship between a target and a region is conveyed with a
Jeff Moyer 2c91dc
cxl_memdev_mapping object.
Jeff Moyer 2c91dc
Jeff Moyer 2c91dc
CXL regions are childeren of root decoders, and are organized as such.
Jeff Moyer 2c91dc
Mapping objects are childeren of a CXL region.  Introduce the two
Jeff Moyer 2c91dc
classes of objects themselves, and common accessors related to them.
Jeff Moyer 2c91dc
Jeff Moyer 2c91dc
Link: https://lore.kernel.org/r/20220815192214.545800-5-vishal.l.verma@intel.com
Jeff Moyer 2c91dc
Cc: Dan Williams <dan.j.williams@intel.com>
Jeff Moyer 2c91dc
Reviewed-by: Dan Williams <dan.j.williams@intel.com>
Jeff Moyer 2c91dc
Signed-off-by: Vishal Verma <vishal.l.verma@intel.com>
Jeff Moyer 2c91dc
---
Jeff Moyer 2c91dc
 .clang-format      |   2 +
Jeff Moyer 2c91dc
 cxl/lib/libcxl.c   | 442 ++++++++++++++++++++++++++++++++++++++++++++-
Jeff Moyer 2c91dc
 cxl/lib/libcxl.sym |  20 ++
Jeff Moyer 2c91dc
 cxl/lib/private.h  |  35 ++++
Jeff Moyer 2c91dc
 cxl/libcxl.h       |  41 +++++
Jeff Moyer 2c91dc
 5 files changed, 530 insertions(+), 10 deletions(-)
Jeff Moyer 2c91dc
Jeff Moyer 2c91dc
diff --git a/.clang-format b/.clang-format
Jeff Moyer 2c91dc
index 7254a1b..b6169e1 100644
Jeff Moyer 2c91dc
--- a/.clang-format
Jeff Moyer 2c91dc
+++ b/.clang-format
Jeff Moyer 2c91dc
@@ -86,6 +86,8 @@ ForEachMacros:
Jeff Moyer 2c91dc
   - 'cxl_dport_foreach'
Jeff Moyer 2c91dc
   - 'cxl_endpoint_foreach'
Jeff Moyer 2c91dc
   - 'cxl_port_foreach_all'
Jeff Moyer 2c91dc
+  - 'cxl_region_foreach'
Jeff Moyer 2c91dc
+  - 'cxl_region_foreach_safe'
Jeff Moyer 2c91dc
   - 'daxctl_dev_foreach'
Jeff Moyer 2c91dc
   - 'daxctl_mapping_foreach'
Jeff Moyer 2c91dc
   - 'daxctl_region_foreach'
Jeff Moyer 2c91dc
diff --git a/cxl/lib/libcxl.c b/cxl/lib/libcxl.c
Jeff Moyer 2c91dc
index 145c6ba..ad98188 100644
Jeff Moyer 2c91dc
--- a/cxl/lib/libcxl.c
Jeff Moyer 2c91dc
+++ b/cxl/lib/libcxl.c
Jeff Moyer 2c91dc
@@ -79,6 +79,38 @@ static void free_target(struct cxl_target *target, struct list_head *head)
Jeff Moyer 2c91dc
 	free(target);
Jeff Moyer 2c91dc
 }
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
+static void free_region(struct cxl_region *region, struct list_head *head)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_memdev_mapping *mapping, *_m;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	list_for_each_safe(&region->mappings, mapping, _m, list) {
Jeff Moyer 2c91dc
+		list_del_from(&region->mappings, &mapping->list);
Jeff Moyer 2c91dc
+		free(mapping);
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+	if (head)
Jeff Moyer 2c91dc
+		list_del_from(head, &region->list);
Jeff Moyer 2c91dc
+	kmod_module_unref(region->module);
Jeff Moyer 2c91dc
+	free(region->dev_buf);
Jeff Moyer 2c91dc
+	free(region->dev_path);
Jeff Moyer 2c91dc
+	free(region);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static void free_stale_regions(struct cxl_decoder *decoder)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_region *region, *_r;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	list_for_each_safe(&decoder->stale_regions, region, _r, list)
Jeff Moyer 2c91dc
+		free_region(region, &decoder->stale_regions);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static void free_regions(struct cxl_decoder *decoder)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_region *region, *_r;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	list_for_each_safe(&decoder->regions, region, _r, list)
Jeff Moyer 2c91dc
+		free_region(region, &decoder->regions);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
 static void free_decoder(struct cxl_decoder *decoder, struct list_head *head)
Jeff Moyer 2c91dc
 {
Jeff Moyer 2c91dc
 	struct cxl_target *target, *_t;
Jeff Moyer 2c91dc
@@ -87,6 +119,8 @@ static void free_decoder(struct cxl_decoder *decoder, struct list_head *head)
Jeff Moyer 2c91dc
 		list_del_from(head, &decoder->list);
Jeff Moyer 2c91dc
 	list_for_each_safe(&decoder->targets, target, _t, list)
Jeff Moyer 2c91dc
 		free_target(target, &decoder->targets);
Jeff Moyer 2c91dc
+	free_regions(decoder);
Jeff Moyer 2c91dc
+	free_stale_regions(decoder);
Jeff Moyer 2c91dc
 	free(decoder->dev_buf);
Jeff Moyer 2c91dc
 	free(decoder->dev_path);
Jeff Moyer 2c91dc
 	free(decoder);
Jeff Moyer 2c91dc
@@ -304,6 +338,402 @@ CXL_EXPORT void cxl_set_log_priority(struct cxl_ctx *ctx, int priority)
Jeff Moyer 2c91dc
 	ctx->ctx.log_priority = priority;
Jeff Moyer 2c91dc
 }
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
+static int is_enabled(const char *drvpath)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct stat st;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (lstat(drvpath, &st) < 0 || !S_ISLNK(st.st_mode))
Jeff Moyer 2c91dc
+		return 0;
Jeff Moyer 2c91dc
+	else
Jeff Moyer 2c91dc
+		return 1;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT int cxl_region_is_enabled(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_region_get_ctx(region);
Jeff Moyer 2c91dc
+	char *path = region->dev_buf;
Jeff Moyer 2c91dc
+	int len = region->buf_len;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (snprintf(path, len, "%s/driver", region->dev_path) >= len) {
Jeff Moyer 2c91dc
+		err(ctx, "%s: buffer too small!\n", cxl_region_get_devname(region));
Jeff Moyer 2c91dc
+		return 0;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return is_enabled(path);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT int cxl_region_disable(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	const char *devname = cxl_region_get_devname(region);
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_region_get_ctx(region);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	util_unbind(region->dev_path, ctx);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (cxl_region_is_enabled(region)) {
Jeff Moyer 2c91dc
+		err(ctx, "%s: failed to disable\n", devname);
Jeff Moyer 2c91dc
+		return -EBUSY;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	dbg(ctx, "%s: disabled\n", devname);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return 0;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT int cxl_region_enable(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	const char *devname = cxl_region_get_devname(region);
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_region_get_ctx(region);
Jeff Moyer 2c91dc
+	char *path = region->dev_buf;
Jeff Moyer 2c91dc
+	int len = region->buf_len;
Jeff Moyer 2c91dc
+	char buf[SYSFS_ATTR_SIZE];
Jeff Moyer 2c91dc
+	u64 resource = ULLONG_MAX;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (cxl_region_is_enabled(region))
Jeff Moyer 2c91dc
+		return 0;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	util_bind(devname, region->module, "cxl", ctx);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (!cxl_region_is_enabled(region)) {
Jeff Moyer 2c91dc
+		err(ctx, "%s: failed to enable\n", devname);
Jeff Moyer 2c91dc
+		return -ENXIO;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	/*
Jeff Moyer 2c91dc
+	 * Currently 'resource' is the only attr that may change after enabling.
Jeff Moyer 2c91dc
+	 * Just refresh it here. If there are additional resources that need
Jeff Moyer 2c91dc
+	 * to be refreshed here later, split these out into a common helper
Jeff Moyer 2c91dc
+	 * for this and add_cxl_region()
Jeff Moyer 2c91dc
+	 */
Jeff Moyer 2c91dc
+	if (snprintf(path, len, "%s/resource", region->dev_path) >= len) {
Jeff Moyer 2c91dc
+		err(ctx, "%s: buffer too small!\n", devname);
Jeff Moyer 2c91dc
+		return 0;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) == 0)
Jeff Moyer 2c91dc
+		resource = strtoull(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (resource < ULLONG_MAX)
Jeff Moyer 2c91dc
+		region->start = resource;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	dbg(ctx, "%s: enabled\n", devname);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return 0;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static void *add_cxl_region(void *parent, int id, const char *cxlregion_base)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	const char *devname = devpath_to_devname(cxlregion_base);
Jeff Moyer 2c91dc
+	char *path = calloc(1, strlen(cxlregion_base) + 100);
Jeff Moyer 2c91dc
+	struct cxl_region *region, *region_dup, *_r;
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder = parent;
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_decoder_get_ctx(decoder);
Jeff Moyer 2c91dc
+	char buf[SYSFS_ATTR_SIZE];
Jeff Moyer 2c91dc
+	u64 resource = ULLONG_MAX;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	dbg(ctx, "%s: base: \'%s\'\n", devname, cxlregion_base);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (!path)
Jeff Moyer 2c91dc
+		return NULL;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	region = calloc(1, sizeof(*region));
Jeff Moyer 2c91dc
+	if (!region)
Jeff Moyer 2c91dc
+		goto err;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	region->id = id;
Jeff Moyer 2c91dc
+	region->ctx = ctx;
Jeff Moyer 2c91dc
+	region->decoder = decoder;
Jeff Moyer 2c91dc
+	list_head_init(&region->mappings);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	region->dev_path = strdup(cxlregion_base);
Jeff Moyer 2c91dc
+	if (!region->dev_path)
Jeff Moyer 2c91dc
+		goto err;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	region->dev_buf = calloc(1, strlen(cxlregion_base) + 50);
Jeff Moyer 2c91dc
+	if (!region->dev_buf)
Jeff Moyer 2c91dc
+		goto err;
Jeff Moyer 2c91dc
+	region->buf_len = strlen(cxlregion_base) + 50;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/size", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) < 0)
Jeff Moyer 2c91dc
+		region->size = ULLONG_MAX;
Jeff Moyer 2c91dc
+	else
Jeff Moyer 2c91dc
+		region->size = strtoull(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/resource", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) == 0)
Jeff Moyer 2c91dc
+		resource = strtoull(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (resource < ULLONG_MAX)
Jeff Moyer 2c91dc
+		region->start = resource;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/uuid", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) < 0)
Jeff Moyer 2c91dc
+		goto err;
Jeff Moyer 2c91dc
+	if (strlen(buf) && uuid_parse(buf, region->uuid) < 0) {
Jeff Moyer 2c91dc
+		dbg(ctx, "%s:%s\n", path, buf);
Jeff Moyer 2c91dc
+		goto err;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/interleave_granularity", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) < 0)
Jeff Moyer 2c91dc
+		region->interleave_granularity = UINT_MAX;
Jeff Moyer 2c91dc
+	else
Jeff Moyer 2c91dc
+		region->interleave_granularity = strtoul(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/interleave_ways", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) < 0)
Jeff Moyer 2c91dc
+		region->interleave_ways = UINT_MAX;
Jeff Moyer 2c91dc
+	else
Jeff Moyer 2c91dc
+		region->interleave_ways = strtoul(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/commit", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) < 0)
Jeff Moyer 2c91dc
+		region->decode_state = CXL_DECODE_UNKNOWN;
Jeff Moyer 2c91dc
+	else
Jeff Moyer 2c91dc
+		region->decode_state = strtoul(buf, NULL, 0);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sprintf(path, "%s/modalias", cxlregion_base);
Jeff Moyer 2c91dc
+	if (sysfs_read_attr(ctx, path, buf) == 0)
Jeff Moyer 2c91dc
+		region->module = util_modalias_to_module(ctx, buf);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	cxl_region_foreach_safe(decoder, region_dup, _r)
Jeff Moyer 2c91dc
+		if (region_dup->id == region->id) {
Jeff Moyer 2c91dc
+			list_del_from(&decoder->regions, &region_dup->list);
Jeff Moyer 2c91dc
+			list_add_tail(&decoder->stale_regions,
Jeff Moyer 2c91dc
+				      &region_dup->list);
Jeff Moyer 2c91dc
+			break;
Jeff Moyer 2c91dc
+		}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	list_add(&decoder->regions, &region->list);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return region;
Jeff Moyer 2c91dc
+err:
Jeff Moyer 2c91dc
+	free(region->dev_path);
Jeff Moyer 2c91dc
+	free(region->dev_buf);
Jeff Moyer 2c91dc
+	free(region);
Jeff Moyer 2c91dc
+	free(path);
Jeff Moyer 2c91dc
+	return NULL;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static void cxl_regions_init(struct cxl_decoder *decoder)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_port *port = cxl_decoder_get_port(decoder);
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_decoder_get_ctx(decoder);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (decoder->regions_init)
Jeff Moyer 2c91dc
+		return;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	/* Only root port decoders may have child regions */
Jeff Moyer 2c91dc
+	if (!cxl_port_is_root(port))
Jeff Moyer 2c91dc
+		return;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	decoder->regions_init = 1;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	sysfs_device_parse(ctx, decoder->dev_path, "region", decoder,
Jeff Moyer 2c91dc
+			   add_cxl_region);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_region *cxl_region_get_first(struct cxl_decoder *decoder)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	cxl_regions_init(decoder);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return list_top(&decoder->regions, struct cxl_region, list);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_region *cxl_region_get_next(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder = region->decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return list_next(&decoder->regions, region, list);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_ctx *cxl_region_get_ctx(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->ctx;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_decoder *cxl_region_get_decoder(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->decoder;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT int cxl_region_get_id(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->id;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT const char *cxl_region_get_devname(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return devpath_to_devname(region->dev_path);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT void cxl_region_get_uuid(struct cxl_region *region, uuid_t uu)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	memcpy(uu, region->uuid, sizeof(uuid_t));
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT unsigned long long cxl_region_get_size(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->size;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT unsigned long long cxl_region_get_resource(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->start;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT unsigned int
Jeff Moyer 2c91dc
+cxl_region_get_interleave_ways(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->interleave_ways;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT int cxl_region_decode_is_committed(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return (region->decode_state == CXL_DECODE_COMMIT) ? 1 : 0;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT unsigned int
Jeff Moyer 2c91dc
+cxl_region_get_interleave_granularity(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return region->interleave_granularity;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static struct cxl_decoder *__cxl_port_match_decoder(struct cxl_port *port,
Jeff Moyer 2c91dc
+						    const char *ident)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	cxl_decoder_foreach(port, decoder)
Jeff Moyer 2c91dc
+		if (strcmp(cxl_decoder_get_devname(decoder), ident) == 0)
Jeff Moyer 2c91dc
+			return decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return NULL;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static struct cxl_decoder *cxl_port_find_decoder(struct cxl_port *port,
Jeff Moyer 2c91dc
+						 const char *ident)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+	struct cxl_endpoint *ep;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	/* First, check decoders directly under @port */
Jeff Moyer 2c91dc
+	decoder = __cxl_port_match_decoder(port, ident);
Jeff Moyer 2c91dc
+	if (decoder)
Jeff Moyer 2c91dc
+		return decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	/* Next, iterate over the endpoints under @port */
Jeff Moyer 2c91dc
+	cxl_endpoint_foreach(port, ep) {
Jeff Moyer 2c91dc
+		decoder = __cxl_port_match_decoder(cxl_endpoint_get_port(ep),
Jeff Moyer 2c91dc
+						   ident);
Jeff Moyer 2c91dc
+		if (decoder)
Jeff Moyer 2c91dc
+			return decoder;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return NULL;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static struct cxl_decoder *cxl_decoder_get_by_name(struct cxl_ctx *ctx,
Jeff Moyer 2c91dc
+						   const char *ident)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_bus *bus;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	cxl_bus_foreach(ctx, bus) {
Jeff Moyer 2c91dc
+		struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+		struct cxl_port *port, *top;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		port = cxl_bus_get_port(bus);
Jeff Moyer 2c91dc
+		decoder = cxl_port_find_decoder(port, ident);
Jeff Moyer 2c91dc
+		if (decoder)
Jeff Moyer 2c91dc
+			return decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		top = port;
Jeff Moyer 2c91dc
+		cxl_port_foreach_all (top, port) {
Jeff Moyer 2c91dc
+			decoder = cxl_port_find_decoder(port, ident);
Jeff Moyer 2c91dc
+			if (decoder)
Jeff Moyer 2c91dc
+				return decoder;
Jeff Moyer 2c91dc
+		}
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return NULL;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+static void cxl_mappings_init(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	const char *devname = cxl_region_get_devname(region);
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx = cxl_region_get_ctx(region);
Jeff Moyer 2c91dc
+	char *mapping_path, buf[SYSFS_ATTR_SIZE];
Jeff Moyer 2c91dc
+	unsigned int i;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	if (region->mappings_init)
Jeff Moyer 2c91dc
+		return;
Jeff Moyer 2c91dc
+	region->mappings_init = 1;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	mapping_path = calloc(1, strlen(region->dev_path) + 100);
Jeff Moyer 2c91dc
+	if (!mapping_path) {
Jeff Moyer 2c91dc
+		err(ctx, "%s: allocation failure\n", devname);
Jeff Moyer 2c91dc
+		return;
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	for (i = 0; i < region->interleave_ways; i++) {
Jeff Moyer 2c91dc
+		struct cxl_memdev_mapping *mapping;
Jeff Moyer 2c91dc
+		struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		sprintf(mapping_path, "%s/target%d", region->dev_path, i);
Jeff Moyer 2c91dc
+		if (sysfs_read_attr(ctx, mapping_path, buf) < 0) {
Jeff Moyer 2c91dc
+			err(ctx, "%s: failed to read target%d\n", devname, i);
Jeff Moyer 2c91dc
+			continue;
Jeff Moyer 2c91dc
+		}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		decoder = cxl_decoder_get_by_name(ctx, buf);
Jeff Moyer 2c91dc
+		if (!decoder) {
Jeff Moyer 2c91dc
+			err(ctx, "%s target%d: %s lookup failure\n",
Jeff Moyer 2c91dc
+			    devname, i, buf);
Jeff Moyer 2c91dc
+			continue;
Jeff Moyer 2c91dc
+		}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		mapping = calloc(1, sizeof(*mapping));
Jeff Moyer 2c91dc
+		if (!mapping) {
Jeff Moyer 2c91dc
+			err(ctx, "%s target%d: allocation failure\n", devname, i);
Jeff Moyer 2c91dc
+			continue;
Jeff Moyer 2c91dc
+		}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+		mapping->region = region;
Jeff Moyer 2c91dc
+		mapping->decoder = decoder;
Jeff Moyer 2c91dc
+		mapping->position = i;
Jeff Moyer 2c91dc
+		list_add(&region->mappings, &mapping->list);
Jeff Moyer 2c91dc
+	}
Jeff Moyer 2c91dc
+	free(mapping_path);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_memdev_mapping *
Jeff Moyer 2c91dc
+cxl_mapping_get_first(struct cxl_region *region)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	cxl_mappings_init(region);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return list_top(&region->mappings, struct cxl_memdev_mapping, list);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_memdev_mapping *
Jeff Moyer 2c91dc
+cxl_mapping_get_next(struct cxl_memdev_mapping *mapping)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	struct cxl_region *region = mapping->region;
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+	return list_next(&region->mappings, mapping, list);
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT struct cxl_decoder *
Jeff Moyer 2c91dc
+cxl_mapping_get_decoder(struct cxl_memdev_mapping *mapping)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return mapping->decoder;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+CXL_EXPORT unsigned int
Jeff Moyer 2c91dc
+cxl_mapping_get_position(struct cxl_memdev_mapping *mapping)
Jeff Moyer 2c91dc
+{
Jeff Moyer 2c91dc
+	return mapping->position;
Jeff Moyer 2c91dc
+}
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
 static void *add_cxl_pmem(void *parent, int id, const char *br_base)
Jeff Moyer 2c91dc
 {
Jeff Moyer 2c91dc
 	const char *devname = devpath_to_devname(br_base);
Jeff Moyer 2c91dc
@@ -681,16 +1111,6 @@ CXL_EXPORT size_t cxl_memdev_get_label_size(struct cxl_memdev *memdev)
Jeff Moyer 2c91dc
 	return memdev->lsa_size;
Jeff Moyer 2c91dc
 }
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
-static int is_enabled(const char *drvpath)
Jeff Moyer 2c91dc
-{
Jeff Moyer 2c91dc
-	struct stat st;
Jeff Moyer 2c91dc
-
Jeff Moyer 2c91dc
-	if (lstat(drvpath, &st) < 0 || !S_ISLNK(st.st_mode))
Jeff Moyer 2c91dc
-		return 0;
Jeff Moyer 2c91dc
-	else
Jeff Moyer 2c91dc
-		return 1;
Jeff Moyer 2c91dc
-}
Jeff Moyer 2c91dc
-
Jeff Moyer 2c91dc
 CXL_EXPORT int cxl_memdev_is_enabled(struct cxl_memdev *memdev)
Jeff Moyer 2c91dc
 {
Jeff Moyer 2c91dc
 	struct cxl_ctx *ctx = cxl_memdev_get_ctx(memdev);
Jeff Moyer 2c91dc
@@ -940,6 +1360,8 @@ static void *add_cxl_decoder(void *parent, int id, const char *cxldecoder_base)
Jeff Moyer 2c91dc
 	decoder->ctx = ctx;
Jeff Moyer 2c91dc
 	decoder->port = port;
Jeff Moyer 2c91dc
 	list_head_init(&decoder->targets);
Jeff Moyer 2c91dc
+	list_head_init(&decoder->regions);
Jeff Moyer 2c91dc
+	list_head_init(&decoder->stale_regions);
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
 	decoder->dev_path = strdup(cxldecoder_base);
Jeff Moyer 2c91dc
 	if (!decoder->dev_path)
Jeff Moyer 2c91dc
diff --git a/cxl/lib/libcxl.sym b/cxl/lib/libcxl.sym
Jeff Moyer 2c91dc
index 7712de0..e410298 100644
Jeff Moyer 2c91dc
--- a/cxl/lib/libcxl.sym
Jeff Moyer 2c91dc
+++ b/cxl/lib/libcxl.sym
Jeff Moyer 2c91dc
@@ -177,4 +177,24 @@ global:
Jeff Moyer 2c91dc
 	cxl_decoder_get_prev;
Jeff Moyer 2c91dc
 	cxl_decoder_set_dpa_size;
Jeff Moyer 2c91dc
 	cxl_decoder_set_mode;
Jeff Moyer 2c91dc
+	cxl_region_get_first;
Jeff Moyer 2c91dc
+	cxl_region_get_next;
Jeff Moyer 2c91dc
+	cxl_region_decode_is_committed;
Jeff Moyer 2c91dc
+	cxl_region_is_enabled;
Jeff Moyer 2c91dc
+	cxl_region_disable;
Jeff Moyer 2c91dc
+	cxl_region_enable;
Jeff Moyer 2c91dc
+	cxl_region_get_ctx;
Jeff Moyer 2c91dc
+	cxl_region_get_decoder;
Jeff Moyer 2c91dc
+	cxl_region_get_id;
Jeff Moyer 2c91dc
+	cxl_region_get_devname;
Jeff Moyer 2c91dc
+	cxl_region_get_uuid;
Jeff Moyer 2c91dc
+	cxl_region_get_size;
Jeff Moyer 2c91dc
+	cxl_region_get_resource;
Jeff Moyer 2c91dc
+	cxl_region_get_interleave_ways;
Jeff Moyer 2c91dc
+	cxl_region_get_interleave_granularity;
Jeff Moyer 2c91dc
+	cxl_mapping_get_first;
Jeff Moyer 2c91dc
+	cxl_mapping_get_next;
Jeff Moyer 2c91dc
+	cxl_mapping_get_decoder;
Jeff Moyer 2c91dc
+	cxl_mapping_get_region;
Jeff Moyer 2c91dc
+	cxl_mapping_get_position;
Jeff Moyer 2c91dc
 } LIBCXL_2;
Jeff Moyer 2c91dc
diff --git a/cxl/lib/private.h b/cxl/lib/private.h
Jeff Moyer 2c91dc
index 832a815..5e2fdd5 100644
Jeff Moyer 2c91dc
--- a/cxl/lib/private.h
Jeff Moyer 2c91dc
+++ b/cxl/lib/private.h
Jeff Moyer 2c91dc
@@ -116,7 +116,42 @@ struct cxl_decoder {
Jeff Moyer 2c91dc
 	bool accelmem_capable;
Jeff Moyer 2c91dc
 	bool locked;
Jeff Moyer 2c91dc
 	enum cxl_decoder_target_type target_type;
Jeff Moyer 2c91dc
+	int regions_init;
Jeff Moyer 2c91dc
 	struct list_head targets;
Jeff Moyer 2c91dc
+	struct list_head regions;
Jeff Moyer 2c91dc
+	struct list_head stale_regions;
Jeff Moyer 2c91dc
+};
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+enum cxl_decode_state {
Jeff Moyer 2c91dc
+	CXL_DECODE_UNKNOWN = -1,
Jeff Moyer 2c91dc
+	CXL_DECODE_RESET = 0,
Jeff Moyer 2c91dc
+	CXL_DECODE_COMMIT,
Jeff Moyer 2c91dc
+};
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+struct cxl_region {
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+	struct list_node list;
Jeff Moyer 2c91dc
+	int mappings_init;
Jeff Moyer 2c91dc
+	struct cxl_ctx *ctx;
Jeff Moyer 2c91dc
+	void *dev_buf;
Jeff Moyer 2c91dc
+	size_t buf_len;
Jeff Moyer 2c91dc
+	char *dev_path;
Jeff Moyer 2c91dc
+	int id;
Jeff Moyer 2c91dc
+	uuid_t uuid;
Jeff Moyer 2c91dc
+	u64 start;
Jeff Moyer 2c91dc
+	u64 size;
Jeff Moyer 2c91dc
+	unsigned int interleave_ways;
Jeff Moyer 2c91dc
+	unsigned int interleave_granularity;
Jeff Moyer 2c91dc
+	enum cxl_decode_state decode_state;
Jeff Moyer 2c91dc
+	struct kmod_module *module;
Jeff Moyer 2c91dc
+	struct list_head mappings;
Jeff Moyer 2c91dc
+};
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+struct cxl_memdev_mapping {
Jeff Moyer 2c91dc
+	struct cxl_region *region;
Jeff Moyer 2c91dc
+	struct cxl_decoder *decoder;
Jeff Moyer 2c91dc
+	unsigned int position;
Jeff Moyer 2c91dc
+	struct list_node list;
Jeff Moyer 2c91dc
 };
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
 enum cxl_cmd_query_status {
Jeff Moyer 2c91dc
diff --git a/cxl/libcxl.h b/cxl/libcxl.h
Jeff Moyer 2c91dc
index c1f8d14..19d94e4 100644
Jeff Moyer 2c91dc
--- a/cxl/libcxl.h
Jeff Moyer 2c91dc
+++ b/cxl/libcxl.h
Jeff Moyer 2c91dc
@@ -237,6 +237,47 @@ int cxl_memdev_is_enabled(struct cxl_memdev *memdev);
Jeff Moyer 2c91dc
 	for (endpoint = cxl_endpoint_get_first(port); endpoint != NULL;        \
Jeff Moyer 2c91dc
 	     endpoint = cxl_endpoint_get_next(endpoint))
Jeff Moyer 2c91dc
 
Jeff Moyer 2c91dc
+struct cxl_region;
Jeff Moyer 2c91dc
+struct cxl_region *cxl_region_get_first(struct cxl_decoder *decoder);
Jeff Moyer 2c91dc
+struct cxl_region *cxl_region_get_next(struct cxl_region *region);
Jeff Moyer 2c91dc
+int cxl_region_decode_is_committed(struct cxl_region *region);
Jeff Moyer 2c91dc
+int cxl_region_is_enabled(struct cxl_region *region);
Jeff Moyer 2c91dc
+int cxl_region_disable(struct cxl_region *region);
Jeff Moyer 2c91dc
+int cxl_region_enable(struct cxl_region *region);
Jeff Moyer 2c91dc
+struct cxl_ctx *cxl_region_get_ctx(struct cxl_region *region);
Jeff Moyer 2c91dc
+struct cxl_decoder *cxl_region_get_decoder(struct cxl_region *region);
Jeff Moyer 2c91dc
+int cxl_region_get_id(struct cxl_region *region);
Jeff Moyer 2c91dc
+const char *cxl_region_get_devname(struct cxl_region *region);
Jeff Moyer 2c91dc
+void cxl_region_get_uuid(struct cxl_region *region, uuid_t uu);
Jeff Moyer 2c91dc
+unsigned long long cxl_region_get_size(struct cxl_region *region);
Jeff Moyer 2c91dc
+unsigned long long cxl_region_get_resource(struct cxl_region *region);
Jeff Moyer 2c91dc
+unsigned int cxl_region_get_interleave_ways(struct cxl_region *region);
Jeff Moyer 2c91dc
+unsigned int cxl_region_get_interleave_granularity(struct cxl_region *region);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+#define cxl_region_foreach(decoder, region)                                    \
Jeff Moyer 2c91dc
+	for (region = cxl_region_get_first(decoder); region != NULL;           \
Jeff Moyer 2c91dc
+	     region = cxl_region_get_next(region))
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+#define cxl_region_foreach_safe(decoder, region, _region)                      \
Jeff Moyer 2c91dc
+	for (region = cxl_region_get_first(decoder),                           \
Jeff Moyer 2c91dc
+	     _region = region ? cxl_region_get_next(region) : NULL;            \
Jeff Moyer 2c91dc
+	     region != NULL;                                                   \
Jeff Moyer 2c91dc
+	     region = _region,                                                 \
Jeff Moyer 2c91dc
+	     _region = _region ? cxl_region_get_next(_region) : NULL)
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+struct cxl_memdev_mapping;
Jeff Moyer 2c91dc
+struct cxl_memdev_mapping *cxl_mapping_get_first(struct cxl_region *region);
Jeff Moyer 2c91dc
+struct cxl_memdev_mapping *
Jeff Moyer 2c91dc
+cxl_mapping_get_next(struct cxl_memdev_mapping *mapping);
Jeff Moyer 2c91dc
+struct cxl_decoder *cxl_mapping_get_decoder(struct cxl_memdev_mapping *mapping);
Jeff Moyer 2c91dc
+struct cxl_region *cxl_mapping_get_region(struct cxl_memdev_mapping *mapping);
Jeff Moyer 2c91dc
+unsigned int cxl_mapping_get_position(struct cxl_memdev_mapping *mapping);
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
+#define cxl_mapping_foreach(region, mapping) \
Jeff Moyer 2c91dc
+        for (mapping = cxl_mapping_get_first(region); \
Jeff Moyer 2c91dc
+             mapping != NULL; \
Jeff Moyer 2c91dc
+             mapping = cxl_mapping_get_next(mapping))
Jeff Moyer 2c91dc
+
Jeff Moyer 2c91dc
 struct cxl_cmd;
Jeff Moyer 2c91dc
 const char *cxl_cmd_get_devname(struct cxl_cmd *cmd);
Jeff Moyer 2c91dc
 struct cxl_cmd *cxl_cmd_new_raw(struct cxl_memdev *memdev, int opcode);
Jeff Moyer 2c91dc
-- 
Jeff Moyer 2c91dc
2.27.0
Jeff Moyer 2c91dc