@@ -85,6 +85,29 @@ OPTIONS
}
]
----
+-I::
+--partition::
+ Include partition information in the memdev listing. Example listing:
+----
+# cxl list -m mem0 -I
+[
+ {
+ "memdev":"mem0",
+ "pmem_size":0,
+ "ram_size":273535729664,
+ "partition_info":{
+ "active_volatile_bytes":273535729664,
+ "active_persistent_bytes":0,
+ "next_volatile_bytes":0,
+ "next_persistent_bytes":0,
+ "total_bytes":273535729664,
+ "volatile_only_bytes":0,
+ "persistent_only_bytes":0,
+ "partition_alignment_bytes":268435456
+ }
+ }
+]
+----
include::human-option.txt[]
@@ -1,5 +1,6 @@
// SPDX-License-Identifier: GPL-2.0
// Copyright (C) 2015-2020 Intel Corporation. All rights reserved.
+#include <limits.h>
#include <util/json.h>
#include <uuid/uuid.h>
#include <cxl/libcxl.h>
@@ -183,6 +184,114 @@ err_jobj:
return NULL;
}
+/*
+ * Present complete view of memdev partition by presenting fields from
+ * both GET_PARTITION_INFO and IDENTIFY mailbox commands.
+ */
+static struct json_object *util_cxl_memdev_partition_to_json(struct cxl_memdev *memdev,
+ unsigned long flags)
+{
+ struct json_object *jobj = NULL;
+ struct json_object *jpart;
+ unsigned long long cap;
+ struct cxl_cmd *cmd;
+ int rc;
+
+ jpart = json_object_new_object();
+ if (!jpart)
+ return NULL;
+ if (!memdev)
+ goto err_jobj;
+
+ /* Retrieve partition info in GET_PARTITION_INFO mbox cmd */
+ cmd = cxl_cmd_new_get_partition_info(memdev);
+ if (!cmd)
+ goto err_jobj;
+
+ rc = cxl_cmd_submit(cmd);
+ if (rc < 0)
+ goto err_cmd;
+ rc = cxl_cmd_get_mbox_status(cmd);
+ if (rc != 0)
+ goto err_cmd;
+
+ cap = cxl_cmd_partition_info_get_active_volatile_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "active_volatile_bytes", jobj);
+ }
+ cap = cxl_cmd_partition_info_get_active_persistent_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "active_persistent_bytes", jobj);
+ }
+ cap = cxl_cmd_partition_info_get_next_volatile_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "next_volatile_bytes", jobj);
+ }
+ cap = cxl_cmd_partition_info_get_next_persistent_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "next_persistent_bytes", jobj);
+ }
+ cxl_cmd_unref(cmd);
+
+ /* Retrieve partition info in the IDENTIFY mbox cmd */
+ cmd = cxl_cmd_new_identify(memdev);
+ if (!cmd)
+ goto err_jobj;
+
+ rc = cxl_cmd_submit(cmd);
+ if (rc < 0)
+ goto err_cmd;
+ rc = cxl_cmd_get_mbox_status(cmd);
+ if (rc != 0)
+ goto err_cmd;
+
+ cap = cxl_cmd_identify_get_total_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart, "total_bytes", jobj);
+ }
+ cap = cxl_cmd_identify_get_volatile_only_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "volatile_only_bytes", jobj);
+ }
+ cap = cxl_cmd_identify_get_persistent_only_bytes(cmd);
+ if (cap != ULLONG_MAX) {
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart,
+ "persistent_only_bytes", jobj);
+ }
+ cap = cxl_cmd_identify_get_partition_align(cmd);
+ jobj = util_json_object_size(cap, flags);
+ if (jobj)
+ json_object_object_add(jpart, "partition_alignment_bytes", jobj);
+
+ cxl_cmd_unref(cmd);
+ return jpart;
+
+err_cmd:
+ cxl_cmd_unref(cmd);
+err_jobj:
+ json_object_put(jpart);
+ return NULL;
+}
+
struct json_object *util_cxl_memdev_to_json(struct cxl_memdev *memdev,
unsigned long flags)
{
@@ -210,5 +319,10 @@ struct json_object *util_cxl_memdev_to_json(struct cxl_memdev *memdev,
if (jobj)
json_object_object_add(jdev, "health", jobj);
}
+ if (flags & UTIL_JSON_PARTITION) {
+ jobj = util_cxl_memdev_partition_to_json(memdev, flags);
+ if (jobj)
+ json_object_object_add(jdev, "partition_info", jobj);
+ }
return jdev;
}
@@ -19,6 +19,7 @@ static struct {
bool idle;
bool human;
bool health;
+ bool partition;
} list;
static unsigned long listopts_to_flags(void)
@@ -31,6 +32,8 @@ static unsigned long listopts_to_flags(void)
flags |= UTIL_JSON_HUMAN;
if (list.health)
flags |= UTIL_JSON_HEALTH;
+ if (list.partition)
+ flags |= UTIL_JSON_PARTITION;
return flags;
}
@@ -64,6 +67,8 @@ int cmd_list(int argc, const char **argv, struct cxl_ctx *ctx)
"use human friendly number formats "),
OPT_BOOLEAN('H', "health", &list.health,
"include memory device health information "),
+ OPT_BOOLEAN('I', "partition", &list.partition,
+ "include memory device partition information "),
OPT_END(),
};
const char * const u[] = {
@@ -17,6 +17,7 @@ enum util_json_flags {
UTIL_JSON_FIRMWARE = (1 << 8),
UTIL_JSON_DAX_MAPPINGS = (1 << 9),
UTIL_JSON_HEALTH = (1 << 10),
+ UTIL_JSON_PARTITION = (1 << 11),
};
struct json_object;