On 5/23/26 2:42 AM, Anisa Su wrote:
> From: Ira Weiny <[email protected]>
>
> Device partitions have an implied order which is made more complex by
> the addition of a dynamic partition.
>
> Remove the ram special case information calls in favor of generic calls
> with a check ahead of time to ensure the preservation of the implied
> partition order.
>
> Signed-off-by: Ira Weiny <[email protected]>
>
> ---
> Changes::
> [anisa: rebase]
> [davidlohr: core/hdm.c: return -EINVAL instead of 0 in cxl_dpa_setup
> if partitions are out of order]
> ---
> drivers/cxl/core/hdm.c | 11 ++++++++++-
> drivers/cxl/core/memdev.c | 32 +++++++++-----------------------
> drivers/cxl/cxlmem.h | 9 +++------
> drivers/cxl/mem.c | 2 +-
> 4 files changed, 23 insertions(+), 31 deletions(-)
>
> diff --git a/drivers/cxl/core/hdm.c b/drivers/cxl/core/hdm.c
> index 28974adaab75..7a5812971f8f 100644
> --- a/drivers/cxl/core/hdm.c
> +++ b/drivers/cxl/core/hdm.c
> @@ -464,6 +464,7 @@ static const char *cxl_mode_name(enum cxl_partition_mode
> mode)
> int cxl_dpa_setup(struct cxl_dev_state *cxlds, const struct cxl_dpa_info
> *info)
> {
> struct device *dev = cxlds->dev;
> + int i;
>
> guard(rwsem_write)(&cxl_rwsem.dpa);
>
> @@ -476,9 +477,17 @@ int cxl_dpa_setup(struct cxl_dev_state *cxlds, const
> struct cxl_dpa_info *info)
> return 0;
> }
>
> + /* Verify partitions are in expected order. */
> + for (i = 1; i < info->nr_partitions; i++) {
> + if (cxlds->part[i].mode < cxlds->part[i-1].mode) {
I think we need to check info->part[i].mode and not cxlds here. cxlds mode is
assigned later in this function.
DJ
> + dev_err(dev, "Partition order mismatch\n");
> + return -EINVAL;
> + }
> + }
> +
> cxlds->dpa_res = DEFINE_RES_MEM(0, info->size);
>
> - for (int i = 0; i < info->nr_partitions; i++) {
> + for (i = 0; i < info->nr_partitions; i++) {
> const struct cxl_dpa_part_info *part = &info->part[i];
> int rc;
>
> diff --git a/drivers/cxl/core/memdev.c b/drivers/cxl/core/memdev.c
> index 80e65690eb77..71602820f896 100644
> --- a/drivers/cxl/core/memdev.c
> +++ b/drivers/cxl/core/memdev.c
> @@ -75,20 +75,12 @@ static ssize_t label_storage_size_show(struct device *dev,
> }
> static DEVICE_ATTR_RO(label_storage_size);
>
> -static resource_size_t cxl_ram_size(struct cxl_dev_state *cxlds)
> -{
> - /* Static RAM is only expected at partition 0. */
> - if (cxlds->part[0].mode != CXL_PARTMODE_RAM)
> - return 0;
> - return resource_size(&cxlds->part[0].res);
> -}
> -
> static ssize_t ram_size_show(struct device *dev, struct device_attribute
> *attr,
> char *buf)
> {
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> struct cxl_dev_state *cxlds = cxlmd->cxlds;
> - unsigned long long len = cxl_ram_size(cxlds);
> + unsigned long long len = cxl_part_size(cxlds, CXL_PARTMODE_RAM);
>
> return sysfs_emit(buf, "%#llx\n", len);
> }
> @@ -101,7 +93,7 @@ static ssize_t pmem_size_show(struct device *dev, struct
> device_attribute *attr,
> {
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> struct cxl_dev_state *cxlds = cxlmd->cxlds;
> - unsigned long long len = cxl_pmem_size(cxlds);
> + unsigned long long len = cxl_part_size(cxlds, CXL_PARTMODE_PMEM);
>
> return sysfs_emit(buf, "%#llx\n", len);
> }
> @@ -424,10 +416,11 @@ static struct attribute *cxl_memdev_attributes[] = {
> NULL,
> };
>
> -static struct cxl_dpa_perf *to_pmem_perf(struct cxl_dev_state *cxlds)
> +static struct cxl_dpa_perf *part_perf(struct cxl_dev_state *cxlds,
> + enum cxl_partition_mode mode)
> {
> for (int i = 0; i < cxlds->nr_partitions; i++)
> - if (cxlds->part[i].mode == CXL_PARTMODE_PMEM)
> + if (cxlds->part[i].mode == mode)
> return &cxlds->part[i].perf;
> return NULL;
> }
> @@ -438,7 +431,7 @@ static ssize_t pmem_qos_class_show(struct device *dev,
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> struct cxl_dev_state *cxlds = cxlmd->cxlds;
>
> - return sysfs_emit(buf, "%d\n", to_pmem_perf(cxlds)->qos_class);
> + return sysfs_emit(buf, "%d\n", part_perf(cxlds,
> CXL_PARTMODE_PMEM)->qos_class);
> }
>
> static struct device_attribute dev_attr_pmem_qos_class =
> @@ -450,20 +443,13 @@ static struct attribute *cxl_memdev_pmem_attributes[] =
> {
> NULL,
> };
>
> -static struct cxl_dpa_perf *to_ram_perf(struct cxl_dev_state *cxlds)
> -{
> - if (cxlds->part[0].mode != CXL_PARTMODE_RAM)
> - return NULL;
> - return &cxlds->part[0].perf;
> -}
> -
> static ssize_t ram_qos_class_show(struct device *dev,
> struct device_attribute *attr, char *buf)
> {
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> struct cxl_dev_state *cxlds = cxlmd->cxlds;
>
> - return sysfs_emit(buf, "%d\n", to_ram_perf(cxlds)->qos_class);
> + return sysfs_emit(buf, "%d\n", part_perf(cxlds,
> CXL_PARTMODE_RAM)->qos_class);
> }
>
> static struct device_attribute dev_attr_ram_qos_class =
> @@ -499,7 +485,7 @@ static umode_t cxl_ram_visible(struct kobject *kobj,
> struct attribute *a, int n)
> {
> struct device *dev = kobj_to_dev(kobj);
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> - struct cxl_dpa_perf *perf = to_ram_perf(cxlmd->cxlds);
> + struct cxl_dpa_perf *perf = part_perf(cxlmd->cxlds, CXL_PARTMODE_RAM);
>
> if (a == &dev_attr_ram_qos_class.attr &&
> (!perf || perf->qos_class == CXL_QOS_CLASS_INVALID))
> @@ -518,7 +504,7 @@ static umode_t cxl_pmem_visible(struct kobject *kobj,
> struct attribute *a, int n
> {
> struct device *dev = kobj_to_dev(kobj);
> struct cxl_memdev *cxlmd = to_cxl_memdev(dev);
> - struct cxl_dpa_perf *perf = to_pmem_perf(cxlmd->cxlds);
> + struct cxl_dpa_perf *perf = part_perf(cxlmd->cxlds, CXL_PARTMODE_PMEM);
>
> if (a == &dev_attr_pmem_qos_class.attr &&
> (!perf || perf->qos_class == CXL_QOS_CLASS_INVALID))
> diff --git a/drivers/cxl/cxlmem.h b/drivers/cxl/cxlmem.h
> index cee936fb3d03..10175ca3b7ee 100644
> --- a/drivers/cxl/cxlmem.h
> +++ b/drivers/cxl/cxlmem.h
> @@ -383,14 +383,11 @@ struct cxl_security_state {
>
> #define CXL_MAX_DC_PARTITIONS 8
>
> -static inline resource_size_t cxl_pmem_size(struct cxl_dev_state *cxlds)
> +static inline resource_size_t cxl_part_size(struct cxl_dev_state *cxlds,
> + enum cxl_partition_mode mode)
> {
> - /*
> - * Static PMEM may be at partition index 0 when there is no static RAM
> - * capacity.
> - */
> for (int i = 0; i < cxlds->nr_partitions; i++)
> - if (cxlds->part[i].mode == CXL_PARTMODE_PMEM)
> + if (cxlds->part[i].mode == mode)
> return resource_size(&cxlds->part[i].res);
> return 0;
> }
> diff --git a/drivers/cxl/mem.c b/drivers/cxl/mem.c
> index fcffe24dcb42..f19e08279ec7 100644
> --- a/drivers/cxl/mem.c
> +++ b/drivers/cxl/mem.c
> @@ -114,7 +114,7 @@ static int cxl_mem_probe(struct device *dev)
> return -ENXIO;
> }
>
> - if (cxl_pmem_size(cxlds) && IS_ENABLED(CONFIG_CXL_PMEM)) {
> + if (cxl_part_size(cxlds, CXL_PARTMODE_PMEM) &&
> IS_ENABLED(CONFIG_CXL_PMEM)) {
> rc = devm_cxl_add_nvdimm(dev, parent_port, cxlmd);
> if (rc) {
> if (rc == -ENODEV)