On 5/23/26 2:50 AM, Anisa Su wrote:
> From: Ira Weiny <[email protected]>
>
> Dynamic capacity partitions are exposed as a singular dynamic ram
> partition.
>
> Add CXL library support to read this partition information.
>
> Signed-off-by: Ira Weiny <[email protected]>
Missing Anisa sign off.
Can probably squash this and the next commit so the usage is shown for the
reviewer.
DJ
> ---
> Documentation/cxl/lib/libcxl.txt | 6 +++--
> cxl/lib/libcxl.c | 43 ++++++++++++++++++++++++++++++++
> cxl/lib/libcxl.sym | 4 +++
> cxl/lib/private.h | 3 +++
> cxl/libcxl.h | 10 +++++++-
> 5 files changed, 63 insertions(+), 3 deletions(-)
>
> diff --git a/Documentation/cxl/lib/libcxl.txt
> b/Documentation/cxl/lib/libcxl.txt
> index 5c3ebd4..9921ac1 100644
> --- a/Documentation/cxl/lib/libcxl.txt
> +++ b/Documentation/cxl/lib/libcxl.txt
> @@ -74,6 +74,7 @@ int cxl_memdev_get_major(struct cxl_memdev *memdev);
> int cxl_memdev_get_minor(struct cxl_memdev *memdev);
> unsigned long long cxl_memdev_get_pmem_size(struct cxl_memdev *memdev);
> unsigned long long cxl_memdev_get_ram_size(struct cxl_memdev *memdev);
> +unsigned long long cxl_memdev_get_dynamic_ram_a_size(struct cxl_memdev
> *memdev);
> const char *cxl_memdev_get_firmware_version(struct cxl_memdev *memdev);
> size_t cxl_memdev_get_label_size(struct cxl_memdev *memdev);
> int cxl_memdev_nvdimm_bridge_active(struct cxl_memdev *memdev);
> @@ -93,8 +94,8 @@ The character device node for command submission can be
> found by default
> at /dev/cxl/mem%d, or created with a major / minor returned from
> cxl_memdev_get_{major,minor}().
>
> -The 'pmem_size' and 'ram_size' attributes return the current
> -provisioning of DPA (Device Physical Address / local capacity) in the
> +The 'pmem_size', 'ram_size', and 'dynamic_ram_a_size' attributes return the
> +current provisioning of DPA (Device Physical Address / local capacity) in the
> device.
>
> cxl_memdev_get_numa_node() returns the affinitized CPU node number if
> @@ -453,6 +454,7 @@ enum cxl_decoder_mode {
> CXL_DECODER_MODE_MIXED,
> CXL_DECODER_MODE_PMEM,
> CXL_DECODER_MODE_RAM,
> + CXL_DECODER_MODE_DYNAMIC_RAM_A,
> };
> enum cxl_decoder_mode cxl_decoder_get_mode(struct cxl_decoder *decoder);
> int cxl_decoder_set_mode(struct cxl_decoder *decoder, enum cxl_decoder_mode
> mode);
> diff --git a/cxl/lib/libcxl.c b/cxl/lib/libcxl.c
> index e55a7b4..be0bc03 100644
> --- a/cxl/lib/libcxl.c
> +++ b/cxl/lib/libcxl.c
> @@ -501,6 +501,9 @@ CXL_EXPORT bool cxl_region_qos_class_mismatch(struct
> cxl_region *region)
> } else if (region->mode == CXL_DECODER_MODE_PMEM) {
> if (root_decoder->qos_class != memdev->pmem_qos_class)
> return true;
> + } else if (region->mode == CXL_DECODER_MODE_DYNAMIC_RAM_A) {
> + if (root_decoder->qos_class !=
> memdev->dynamic_ram_a_qos_class)
> + return true;
> }
> }
>
> @@ -1426,6 +1429,10 @@ static void *add_cxl_memdev(void *parent, int id,
> const char *cxlmem_base)
> if (sysfs_read_attr(ctx, path, buf) == 0)
> memdev->ram_size = strtoull(buf, NULL, 0);
>
> + sprintf(path, "%s/dynamic_ram_a/size", cxlmem_base);
> + if (sysfs_read_attr(ctx, path, buf) == 0)
> + memdev->dynamic_ram_a_size = strtoull(buf, NULL, 0);
> +
> sprintf(path, "%s/pmem/qos_class", cxlmem_base);
> if (sysfs_read_attr(ctx, path, buf) < 0)
> memdev->pmem_qos_class = CXL_QOS_CLASS_NONE;
> @@ -1438,6 +1445,12 @@ static void *add_cxl_memdev(void *parent, int id,
> const char *cxlmem_base)
> else
> memdev->ram_qos_class = atoi(buf);
>
> + sprintf(path, "%s/dynamic_ram_a/qos_class", cxlmem_base);
> + if (sysfs_read_attr(ctx, path, buf) < 0)
> + memdev->dynamic_ram_a_qos_class = CXL_QOS_CLASS_NONE;
> + else
> + memdev->dynamic_ram_a_qos_class = atoi(buf);
> +
> sprintf(path, "%s/payload_max", cxlmem_base);
> if (sysfs_read_attr(ctx, path, buf) == 0) {
> memdev->payload_max = strtoull(buf, NULL, 0);
> @@ -1685,6 +1698,11 @@ CXL_EXPORT unsigned long long
> cxl_memdev_get_ram_size(struct cxl_memdev *memdev)
> return memdev->ram_size;
> }
>
> +CXL_EXPORT unsigned long long cxl_memdev_get_dynamic_ram_a_size(struct
> cxl_memdev *memdev)
> +{
> + return memdev->dynamic_ram_a_size;
> +}
> +
> CXL_EXPORT int cxl_memdev_get_pmem_qos_class(struct cxl_memdev *memdev)
> {
> return memdev->pmem_qos_class;
> @@ -1695,6 +1713,11 @@ CXL_EXPORT int cxl_memdev_get_ram_qos_class(struct
> cxl_memdev *memdev)
> return memdev->ram_qos_class;
> }
>
> +CXL_EXPORT int cxl_memdev_get_dynamic_ram_a_qos_class(struct cxl_memdev
> *memdev)
> +{
> + return memdev->dynamic_ram_a_qos_class;
> +}
> +
> CXL_EXPORT const char *cxl_memdev_get_firmware_verison(struct cxl_memdev
> *memdev)
> {
> return memdev->firmware_version;
> @@ -2465,6 +2488,8 @@ static void *add_cxl_decoder(void *parent, int id,
> const char *cxldecoder_base)
> decoder->mode = CXL_DECODER_MODE_MIXED;
> else if (strcmp(buf, "none") == 0)
> decoder->mode = CXL_DECODER_MODE_NONE;
> + else if (strcmp(buf, "dynamic_ram_a") == 0)
> + decoder->mode = CXL_DECODER_MODE_DYNAMIC_RAM_A;
> else
> decoder->mode = CXL_DECODER_MODE_MIXED;
> } else
> @@ -2504,6 +2529,7 @@ static void *add_cxl_decoder(void *parent, int id,
> const char *cxldecoder_base)
> case CXL_PORT_SWITCH:
> decoder->pmem_capable = true;
> decoder->volatile_capable = true;
> + decoder->dynamic_ram_a_capable = true;
> decoder->mem_capable = true;
> decoder->accelmem_capable = true;
> sprintf(path, "%s/locked", cxldecoder_base);
> @@ -2528,6 +2554,7 @@ static void *add_cxl_decoder(void *parent, int id,
> const char *cxldecoder_base)
> { "cap_type3", &decoder->mem_capable },
> { "cap_ram", &decoder->volatile_capable },
> { "cap_pmem", &decoder->pmem_capable },
> + { "cap_dynamic_ram_a", &decoder->dynamic_ram_a_capable
> },
> { "locked", &decoder->locked },
> };
>
> @@ -2778,6 +2805,9 @@ CXL_EXPORT int cxl_decoder_set_mode(struct cxl_decoder
> *decoder,
> case CXL_DECODER_MODE_RAM:
> sprintf(buf, "ram");
> break;
> + case CXL_DECODER_MODE_DYNAMIC_RAM_A:
> + sprintf(buf, "dynamic_ram_a");
> + break;
> default:
> err(ctx, "%s: unsupported mode: %d\n",
> cxl_decoder_get_devname(decoder), mode);
> @@ -2829,6 +2859,11 @@ CXL_EXPORT bool cxl_decoder_is_volatile_capable(struct
> cxl_decoder *decoder)
> return decoder->volatile_capable;
> }
>
> +CXL_EXPORT bool cxl_decoder_is_dynamic_ram_a_capable(struct cxl_decoder
> *decoder)
> +{
> + return decoder->dynamic_ram_a_capable;
> +}
> +
> CXL_EXPORT bool cxl_decoder_is_mem_capable(struct cxl_decoder *decoder)
> {
> return decoder->mem_capable;
> @@ -2903,6 +2938,8 @@ static struct cxl_region
> *cxl_decoder_create_region(struct cxl_decoder *decoder,
> sprintf(path, "%s/create_pmem_region", decoder->dev_path);
> else if (mode == CXL_DECODER_MODE_RAM)
> sprintf(path, "%s/create_ram_region", decoder->dev_path);
> + else if (mode == CXL_DECODER_MODE_DYNAMIC_RAM_A)
> + sprintf(path, "%s/create_dynamic_ram_a_region",
> decoder->dev_path);
>
> rc = sysfs_read_attr(ctx, path, buf);
> if (rc < 0) {
> @@ -2954,6 +2991,12 @@ cxl_decoder_create_ram_region(struct cxl_decoder
> *decoder)
> return cxl_decoder_create_region(decoder, CXL_DECODER_MODE_RAM);
> }
>
> +CXL_EXPORT struct cxl_region *
> +cxl_decoder_create_dynamic_ram_a_region(struct cxl_decoder *decoder)
> +{
> + return cxl_decoder_create_region(decoder,
> CXL_DECODER_MODE_DYNAMIC_RAM_A);
> +}
> +
> CXL_EXPORT int cxl_decoder_get_nr_targets(struct cxl_decoder *decoder)
> {
> return decoder->nr_targets;
> diff --git a/cxl/lib/libcxl.sym b/cxl/lib/libcxl.sym
> index ed4429f..258bdd3 100644
> --- a/cxl/lib/libcxl.sym
> +++ b/cxl/lib/libcxl.sym
> @@ -294,6 +294,10 @@ global:
> cxl_memdev_get_fwctl;
> cxl_fwctl_get_major;
> cxl_fwctl_get_minor;
> + cxl_memdev_get_dynamic_ram_a_size;
> + cxl_memdev_get_dynamic_ram_a_qos_class;
> + cxl_decoder_is_dynamic_ram_a_capable;
> + cxl_decoder_create_dynamic_ram_a_region;
> } LIBECXL_8;
>
> LIBCXL_10 {
> diff --git a/cxl/lib/private.h b/cxl/lib/private.h
> index d2d71fc..37b7b06 100644
> --- a/cxl/lib/private.h
> +++ b/cxl/lib/private.h
> @@ -52,8 +52,10 @@ struct cxl_memdev {
> struct list_node list;
> unsigned long long pmem_size;
> unsigned long long ram_size;
> + unsigned long long dynamic_ram_a_size;
> int ram_qos_class;
> int pmem_qos_class;
> + int dynamic_ram_a_qos_class;
> int payload_max;
> size_t lsa_size;
> struct kmod_module *module;
> @@ -159,6 +161,7 @@ struct cxl_decoder {
> unsigned int interleave_granularity;
> bool pmem_capable;
> bool volatile_capable;
> + bool dynamic_ram_a_capable;
> bool mem_capable;
> bool accelmem_capable;
> bool locked;
> diff --git a/cxl/libcxl.h b/cxl/libcxl.h
> index e91af90..fd41122 100644
> --- a/cxl/libcxl.h
> +++ b/cxl/libcxl.h
> @@ -75,8 +75,10 @@ struct cxl_fwctl *cxl_memdev_get_fwctl(struct cxl_memdev
> *memdev);
> struct cxl_ctx *cxl_memdev_get_ctx(struct cxl_memdev *memdev);
> unsigned long long cxl_memdev_get_pmem_size(struct cxl_memdev *memdev);
> unsigned long long cxl_memdev_get_ram_size(struct cxl_memdev *memdev);
> +unsigned long long cxl_memdev_get_dynamic_ram_a_size(struct cxl_memdev
> *memdev);
> int cxl_memdev_get_pmem_qos_class(struct cxl_memdev *memdev);
> int cxl_memdev_get_ram_qos_class(struct cxl_memdev *memdev);
> +int cxl_memdev_get_dynamic_ram_a_qos_class(struct cxl_memdev *memdev);
> const char *cxl_memdev_get_firmware_verison(struct cxl_memdev *memdev);
> bool cxl_memdev_fw_update_in_progress(struct cxl_memdev *memdev);
> size_t cxl_memdev_fw_update_get_remaining(struct cxl_memdev *memdev);
> @@ -210,6 +212,7 @@ enum cxl_decoder_mode {
> CXL_DECODER_MODE_MIXED,
> CXL_DECODER_MODE_PMEM,
> CXL_DECODER_MODE_RAM,
> + CXL_DECODER_MODE_DYNAMIC_RAM_A,
> };
>
> static inline const char *cxl_decoder_mode_name(enum cxl_decoder_mode mode)
> @@ -219,9 +222,10 @@ static inline const char *cxl_decoder_mode_name(enum
> cxl_decoder_mode mode)
> [CXL_DECODER_MODE_MIXED] = "mixed",
> [CXL_DECODER_MODE_PMEM] = "pmem",
> [CXL_DECODER_MODE_RAM] = "ram",
> + [CXL_DECODER_MODE_DYNAMIC_RAM_A] = "dynamic_ram_a",
> };
>
> - if (mode < CXL_DECODER_MODE_NONE || mode > CXL_DECODER_MODE_RAM)
> + if (mode < CXL_DECODER_MODE_NONE || mode >
> CXL_DECODER_MODE_DYNAMIC_RAM_A)
> mode = CXL_DECODER_MODE_NONE;
> return names[mode];
> }
> @@ -235,6 +239,8 @@ cxl_decoder_mode_from_ident(const char *ident)
> return CXL_DECODER_MODE_RAM;
> else if (strcmp(ident, "pmem") == 0)
> return CXL_DECODER_MODE_PMEM;
> + else if (strcmp(ident, "dynamic_ram_a") == 0)
> + return CXL_DECODER_MODE_DYNAMIC_RAM_A;
> return CXL_DECODER_MODE_NONE;
> }
>
> @@ -264,6 +270,7 @@ cxl_decoder_get_target_type(struct cxl_decoder *decoder);
> bool cxl_decoder_is_pmem_capable(struct cxl_decoder *decoder);
> bool cxl_decoder_is_volatile_capable(struct cxl_decoder *decoder);
> bool cxl_decoder_is_mem_capable(struct cxl_decoder *decoder);
> +bool cxl_decoder_is_dynamic_ram_a_capable(struct cxl_decoder *decoder);
> bool cxl_decoder_is_accelmem_capable(struct cxl_decoder *decoder);
> bool cxl_decoder_is_locked(struct cxl_decoder *decoder);
> unsigned int
> @@ -272,6 +279,7 @@ unsigned int cxl_decoder_get_interleave_ways(struct
> cxl_decoder *decoder);
> struct cxl_region *cxl_decoder_get_region(struct cxl_decoder *decoder);
> struct cxl_region *cxl_decoder_create_pmem_region(struct cxl_decoder
> *decoder);
> struct cxl_region *cxl_decoder_create_ram_region(struct cxl_decoder
> *decoder);
> +struct cxl_region *cxl_decoder_create_dynamic_ram_a_region(struct
> cxl_decoder *decoder);
> struct cxl_decoder *cxl_decoder_get_by_name(struct cxl_ctx *ctx,
> const char *ident);
> struct cxl_memdev *cxl_decoder_get_memdev(struct cxl_decoder *decoder);