852 lines
32 KiB
C
852 lines
32 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
#include "util/types.h"
|
||
|
#include "common.h"
|
||
|
#include "nvme-print.h"
|
||
|
#include "ocp-print.h"
|
||
|
#include "ocp-hardware-component-log.h"
|
||
|
#include "ocp-fw-activation-history.h"
|
||
|
#include "ocp-smart-extended-log.h"
|
||
|
#include "ocp-telemetry-decode.h"
|
||
|
#include "ocp-nvme.h"
|
||
|
|
||
|
static void print_hwcomp_desc_json(struct hwcomp_desc_entry *e, struct json_object *r)
|
||
|
{
|
||
|
obj_add_str(r, "Description", hwcomp_id_to_string(le32_to_cpu(e->desc->id)));
|
||
|
obj_add_nprix64(r, "Date/Lot Size", e->date_lot_size);
|
||
|
obj_add_nprix64(r, "Additional Information Size", e->add_info_size);
|
||
|
obj_add_uint_0nx(r, "Identifier", le32_to_cpu(e->desc->id), 8);
|
||
|
obj_add_0nprix64(r, "Manufacture", le64_to_cpu(e->desc->mfg), 16);
|
||
|
obj_add_0nprix64(r, "Revision", le64_to_cpu(e->desc->rev), 16);
|
||
|
obj_add_0nprix64(r, "Manufacture Code", le64_to_cpu(e->desc->mfg_code), 16);
|
||
|
obj_add_byte_array(r, "Date/Lot Code", e->date_lot_code, e->date_lot_size);
|
||
|
obj_add_byte_array(r, "Additional Information", e->add_info, e->add_info_size);
|
||
|
}
|
||
|
|
||
|
static void print_hwcomp_desc_list_json(struct json_object *r, struct hwcomp_desc_entry *e,
|
||
|
bool list, int num)
|
||
|
{
|
||
|
_cleanup_free_ char *k = NULL;
|
||
|
|
||
|
if (asprintf(&k, "Component %d", num) < 0)
|
||
|
return;
|
||
|
|
||
|
if (list) {
|
||
|
obj_add_str(r, k, hwcomp_id_to_string(le32_to_cpu(e->desc->id)));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
print_hwcomp_desc_json(e, obj_create_array_obj(r, k));
|
||
|
}
|
||
|
|
||
|
static void print_hwcomp_descs_json(struct hwcomp_desc *desc, long double log_size, __u32 id,
|
||
|
bool list, struct json_object *r)
|
||
|
{
|
||
|
size_t date_lot_code_offset = sizeof(struct hwcomp_desc);
|
||
|
struct hwcomp_desc_entry e = { desc };
|
||
|
int num = 1;
|
||
|
|
||
|
while (log_size > 0) {
|
||
|
e.date_lot_size = le64_to_cpu(e.desc->date_lot_size) * sizeof(__le32);
|
||
|
e.date_lot_code = e.date_lot_size ?
|
||
|
(__u8 *)e.desc + date_lot_code_offset : NULL;
|
||
|
e.add_info_size = le64_to_cpu(e.desc->add_info_size) * sizeof(__le32);
|
||
|
e.add_info = e.add_info_size ? e.date_lot_code ?
|
||
|
e.date_lot_code + e.date_lot_size :
|
||
|
(__u8 *)e.desc + date_lot_code_offset : NULL;
|
||
|
if (!id || id == le32_to_cpu(e.desc->id))
|
||
|
print_hwcomp_desc_list_json(r, &e, list, num++);
|
||
|
e.desc_size = date_lot_code_offset + e.date_lot_size + e.add_info_size;
|
||
|
e.desc = (struct hwcomp_desc *)((__u8 *)e.desc + e.desc_size);
|
||
|
log_size -= e.desc_size;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void json_hwcomp_log(struct hwcomp_log *log, __u32 id, bool list)
|
||
|
{
|
||
|
struct json_object *r = json_create_object();
|
||
|
|
||
|
long double log_size = uint128_t_to_double(le128_to_cpu(log->size)) * sizeof(__le32);
|
||
|
|
||
|
obj_add_uint_02x(r, "Log Identifier", LID_HWCOMP);
|
||
|
obj_add_uint_0x(r, "Log Page Version", le16_to_cpu(log->ver));
|
||
|
obj_add_byte_array(r, "Reserved2", log->rsvd2, ARRAY_SIZE(log->rsvd2));
|
||
|
obj_add_byte_array(r, "Log page GUID", log->guid, ARRAY_SIZE(log->guid));
|
||
|
obj_add_nprix64(r, "Hardware Component Log Size", (unsigned long long)log_size);
|
||
|
obj_add_byte_array(r, "Reserved48", log->rsvd48, ARRAY_SIZE(log->rsvd48));
|
||
|
print_hwcomp_descs_json(log->desc, log_size, id, list,
|
||
|
obj_create_array_obj(r, "Component Descriptions"));
|
||
|
|
||
|
json_print(r);
|
||
|
}
|
||
|
|
||
|
static void json_fw_activation_history(const struct fw_activation_history *fw_history)
|
||
|
{
|
||
|
struct json_object *root = json_create_object();
|
||
|
|
||
|
json_object_add_value_uint(root, "log identifier", fw_history->log_id);
|
||
|
json_object_add_value_uint(root, "valid entries", le32_to_cpu(fw_history->valid_entries));
|
||
|
|
||
|
struct json_object *entries = json_create_array();
|
||
|
|
||
|
for (int index = 0; index < le32_to_cpu(fw_history->valid_entries); index++) {
|
||
|
const struct fw_activation_history_entry *entry = &fw_history->entries[index];
|
||
|
struct json_object *entry_obj = json_create_object();
|
||
|
|
||
|
json_object_add_value_uint(entry_obj, "version number", entry->ver_num);
|
||
|
json_object_add_value_uint(entry_obj, "entry length", entry->entry_length);
|
||
|
json_object_add_value_uint(entry_obj, "activation count",
|
||
|
le16_to_cpu(entry->activation_count));
|
||
|
json_object_add_value_uint64(entry_obj, "timestamp",
|
||
|
(0x0000FFFFFFFFFFFF & le64_to_cpu(entry->timestamp)));
|
||
|
json_object_add_value_uint(entry_obj, "power cycle count",
|
||
|
le64_to_cpu(entry->power_cycle_count));
|
||
|
|
||
|
struct json_object *fw = json_object_new_string_len(entry->previous_fw,
|
||
|
sizeof(entry->previous_fw));
|
||
|
|
||
|
json_object_add_value_object(entry_obj, "previous firmware", fw);
|
||
|
|
||
|
fw = json_object_new_string_len(entry->new_fw, sizeof(entry->new_fw));
|
||
|
|
||
|
json_object_add_value_object(entry_obj, "new firmware", fw);
|
||
|
json_object_add_value_uint(entry_obj, "slot number", entry->slot_number);
|
||
|
json_object_add_value_uint(entry_obj, "commit action type", entry->commit_action);
|
||
|
json_object_add_value_uint(entry_obj, "result", le16_to_cpu(entry->result));
|
||
|
|
||
|
json_array_add_value_object(entries, entry_obj);
|
||
|
}
|
||
|
|
||
|
json_object_add_value_array(root, "entries", entries);
|
||
|
|
||
|
json_object_add_value_uint(root, "log page version",
|
||
|
le16_to_cpu(fw_history->log_page_version));
|
||
|
|
||
|
char guid[2 * sizeof(fw_history->log_page_guid) + 3] = { 0 };
|
||
|
|
||
|
sprintf(guid, "0x%"PRIx64"%"PRIx64"",
|
||
|
le64_to_cpu(fw_history->log_page_guid[1]),
|
||
|
le64_to_cpu(fw_history->log_page_guid[0]));
|
||
|
json_object_add_value_string(root, "log page guid", guid);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
json_free_object(root);
|
||
|
|
||
|
printf("\n");
|
||
|
}
|
||
|
|
||
|
static void json_smart_extended_log(void *data)
|
||
|
{
|
||
|
struct json_object *root;
|
||
|
struct json_object *pmuw;
|
||
|
struct json_object *pmur;
|
||
|
uint16_t smart_log_ver = 0;
|
||
|
__u8 *log_data = data;
|
||
|
char guid[40];
|
||
|
|
||
|
root = json_create_object();
|
||
|
pmuw = json_create_object();
|
||
|
pmur = json_create_object();
|
||
|
|
||
|
json_object_add_value_uint64(pmuw, "hi",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_PMUW + 8] & 0xFFFFFFFFFFFFFFFF));
|
||
|
json_object_add_value_uint64(pmuw, "lo",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_PMUW] & 0xFFFFFFFFFFFFFFFF));
|
||
|
json_object_add_value_object(root, "Physical media units written", pmuw);
|
||
|
json_object_add_value_uint64(pmur, "hi",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_PMUR + 8] & 0xFFFFFFFFFFFFFFFF));
|
||
|
json_object_add_value_uint64(pmur, "lo",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_PMUR] & 0xFFFFFFFFFFFFFFFF));
|
||
|
json_object_add_value_object(root, "Physical media units read", pmur);
|
||
|
json_object_add_value_uint64(root, "Bad user nand blocks - Raw",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_BUNBR] & 0x0000FFFFFFFFFFFF));
|
||
|
json_object_add_value_uint(root, "Bad user nand blocks - Normalized",
|
||
|
(uint16_t)le16_to_cpu(*(uint16_t *)&log_data[SCAO_BUNBN]));
|
||
|
json_object_add_value_uint64(root, "Bad system nand blocks - Raw",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_BSNBR] & 0x0000FFFFFFFFFFFF));
|
||
|
json_object_add_value_uint(root, "Bad system nand blocks - Normalized",
|
||
|
(uint16_t)le16_to_cpu(*(uint16_t *)&log_data[SCAO_BSNBN]));
|
||
|
json_object_add_value_uint64(root, "XOR recovery count",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_XRC]));
|
||
|
json_object_add_value_uint64(root, "Uncorrectable read error count",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_UREC]));
|
||
|
json_object_add_value_uint64(root, "Soft ecc error count",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_SEEC]));
|
||
|
json_object_add_value_uint(root, "End to end detected errors",
|
||
|
(uint32_t)le32_to_cpu(*(uint32_t *)&log_data[SCAO_EEDC]));
|
||
|
json_object_add_value_uint(root, "End to end corrected errors",
|
||
|
(uint32_t)le32_to_cpu(*(uint32_t *)&log_data[SCAO_EECE]));
|
||
|
json_object_add_value_uint(root, "System data percent used",
|
||
|
(__u8)log_data[SCAO_SDPU]);
|
||
|
json_object_add_value_uint64(root, "Refresh counts",
|
||
|
(uint64_t)(le64_to_cpu(*(uint64_t *)&log_data[SCAO_RFSC]) & 0x00FFFFFFFFFFFFFF));
|
||
|
json_object_add_value_uint(root, "Max User data erase counts",
|
||
|
(uint32_t)le32_to_cpu(*(uint32_t *)&log_data[SCAO_MXUDEC]));
|
||
|
json_object_add_value_uint(root, "Min User data erase counts",
|
||
|
(uint32_t)le32_to_cpu(*(uint32_t *)&log_data[SCAO_MNUDEC]));
|
||
|
json_object_add_value_uint(root, "Number of Thermal throttling events",
|
||
|
(__u8)log_data[SCAO_NTTE]);
|
||
|
json_object_add_value_uint(root, "Current throttling status",
|
||
|
(__u8)log_data[SCAO_CTS]);
|
||
|
json_object_add_value_uint64(root, "PCIe correctable error count",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_PCEC]));
|
||
|
json_object_add_value_uint(root, "Incomplete shutdowns",
|
||
|
(uint32_t)le32_to_cpu(*(uint32_t *)&log_data[SCAO_ICS]));
|
||
|
json_object_add_value_uint(root, "Percent free blocks",
|
||
|
(__u8)log_data[SCAO_PFB]);
|
||
|
json_object_add_value_uint(root, "Capacitor health",
|
||
|
(uint16_t)le16_to_cpu(*(uint16_t *)&log_data[SCAO_CPH]));
|
||
|
json_object_add_value_uint64(root, "Unaligned I/O",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_UIO]));
|
||
|
json_object_add_value_uint64(root, "Security Version Number",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_SVN]));
|
||
|
json_object_add_value_uint64(root, "NUSE - Namespace utilization",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_NUSE]));
|
||
|
json_object_add_value_uint128(root, "PLP start count",
|
||
|
le128_to_cpu(&log_data[SCAO_PSC]));
|
||
|
json_object_add_value_uint128(root, "Endurance estimate",
|
||
|
le128_to_cpu(&log_data[SCAO_EEST]));
|
||
|
smart_log_ver = (uint16_t)le16_to_cpu(*(uint16_t *)&log_data[SCAO_LPV]);
|
||
|
|
||
|
json_object_add_value_uint(root, "Log page version", smart_log_ver);
|
||
|
|
||
|
memset((void *)guid, 0, 40);
|
||
|
sprintf((char *)guid, "0x%"PRIx64"%"PRIx64"",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_LPG + 8]),
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data[SCAO_LPG]));
|
||
|
json_object_add_value_string(root, "Log page GUID", guid);
|
||
|
|
||
|
switch (smart_log_ver) {
|
||
|
case 0 ... 1:
|
||
|
break;
|
||
|
default:
|
||
|
case 4:
|
||
|
json_object_add_value_uint(root, "NVMe Command Set Errata Version",
|
||
|
(__u8)log_data[SCAO_NCSEV]);
|
||
|
json_object_add_value_uint(root, "Lowest Permitted Firmware Revision",
|
||
|
le64_to_cpu(*(uint64_t *)&log_data[SCAO_PSCC]));
|
||
|
fallthrough;
|
||
|
case 2 ... 3:
|
||
|
json_object_add_value_uint(root, "Errata Version Field",
|
||
|
(__u8)log_data[SCAO_EVF]);
|
||
|
json_object_add_value_uint(root, "Point Version Field",
|
||
|
le16_to_cpu(*(uint16_t *)&log_data[SCAO_PVF]));
|
||
|
json_object_add_value_uint(root, "Minor Version Field",
|
||
|
le16_to_cpu(*(uint16_t *)&log_data[SCAO_MIVF]));
|
||
|
json_object_add_value_uint(root, "Major Version Field",
|
||
|
(__u8)log_data[SCAO_MAVF]);
|
||
|
json_object_add_value_uint(root, "NVMe Base Errata Version",
|
||
|
(__u8)log_data[SCAO_NBEV]);
|
||
|
json_object_add_value_uint(root, "PCIe Link Retraining Count",
|
||
|
le64_to_cpu(*(uint64_t *)&log_data[SCAO_PLRC]));
|
||
|
json_object_add_value_uint(root, "Power State Change Count",
|
||
|
le64_to_cpu(*(uint64_t *)&log_data[SCAO_PSCC]));
|
||
|
}
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_telemetry_log(struct ocp_telemetry_parse_options *options)
|
||
|
{
|
||
|
print_ocp_telemetry_json(options);
|
||
|
}
|
||
|
|
||
|
static void json_c3_log(struct nvme_dev *dev, struct ssd_latency_monitor_log *log_data)
|
||
|
{
|
||
|
struct json_object *root;
|
||
|
char ts_buf[128];
|
||
|
char buf[128];
|
||
|
int i, j;
|
||
|
char *operation[3] = {"Trim", "Write", "Read"};
|
||
|
|
||
|
root = json_create_object();
|
||
|
|
||
|
json_object_add_value_uint(root, "Feature Status",
|
||
|
log_data->feature_status);
|
||
|
json_object_add_value_uint(root, "Active Bucket Timer",
|
||
|
C3_ACTIVE_BUCKET_TIMER_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_bucket_timer));
|
||
|
json_object_add_value_uint(root, "Active Bucket Timer Threshold",
|
||
|
C3_ACTIVE_BUCKET_TIMER_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_bucket_timer_threshold));
|
||
|
json_object_add_value_uint(root, "Active Threshold A",
|
||
|
C3_ACTIVE_THRESHOLD_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_threshold_a + 1));
|
||
|
json_object_add_value_uint(root, "Active Threshold B",
|
||
|
C3_ACTIVE_THRESHOLD_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_threshold_b + 1));
|
||
|
json_object_add_value_uint(root, "Active Threshold C",
|
||
|
C3_ACTIVE_THRESHOLD_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_threshold_c + 1));
|
||
|
json_object_add_value_uint(root, "Active Threshold D",
|
||
|
C3_ACTIVE_THRESHOLD_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_threshold_d + 1));
|
||
|
json_object_add_value_uint(root, "Active Latency Configuration",
|
||
|
le16_to_cpu(log_data->active_latency_config));
|
||
|
json_object_add_value_uint(root, "Active Latency Minimum Window",
|
||
|
C3_MINIMUM_WINDOW_INCREMENT *
|
||
|
le16_to_cpu(log_data->active_latency_min_window));
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Active Bucket Counter: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
json_object_add_value_uint(bucket, operation[j],
|
||
|
le32_to_cpu(log_data->active_bucket_counter[i][j+1]));
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Active Latency Time Stamp: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
if (le64_to_cpu(log_data->active_latency_timestamp[3-i][j]) == -1) {
|
||
|
json_object_add_value_string(bucket, operation[j], "NA");
|
||
|
} else {
|
||
|
convert_ts(le64_to_cpu(log_data->active_latency_timestamp[3-i][j]),
|
||
|
ts_buf);
|
||
|
json_object_add_value_string(bucket, operation[j], ts_buf);
|
||
|
}
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Active Measured Latency: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
json_object_add_value_uint(bucket, operation[j],
|
||
|
le16_to_cpu(log_data->active_measured_latency[3-i][j]));
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
json_object_add_value_uint(root, "Active Latency Stamp Units",
|
||
|
le16_to_cpu(log_data->active_latency_stamp_units));
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Static Bucket Counter: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
json_object_add_value_uint(bucket, operation[j],
|
||
|
le32_to_cpu(log_data->static_bucket_counter[i][j+1]));
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Static Latency Time Stamp: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
if (le64_to_cpu(log_data->static_latency_timestamp[3-i][j]) == -1) {
|
||
|
json_object_add_value_string(bucket, operation[j], "NA");
|
||
|
} else {
|
||
|
convert_ts(le64_to_cpu(log_data->static_latency_timestamp[3-i][j]),
|
||
|
ts_buf);
|
||
|
json_object_add_value_string(bucket, operation[j], ts_buf);
|
||
|
}
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < C3_BUCKET_NUM; i++) {
|
||
|
struct json_object *bucket;
|
||
|
|
||
|
bucket = json_create_object();
|
||
|
sprintf(buf, "Static Measured Latency: Bucket %d", i);
|
||
|
for (j = 2; j >= 0; j--) {
|
||
|
json_object_add_value_uint(bucket, operation[j],
|
||
|
le16_to_cpu(log_data->static_measured_latency[3-i][j]));
|
||
|
}
|
||
|
json_object_add_value_object(root, buf, bucket);
|
||
|
}
|
||
|
|
||
|
json_object_add_value_uint(root, "Static Latency Stamp Units",
|
||
|
le16_to_cpu(log_data->static_latency_stamp_units));
|
||
|
json_object_add_value_uint(root, "Debug Log Trigger Enable",
|
||
|
le16_to_cpu(log_data->debug_log_trigger_enable));
|
||
|
json_object_add_value_uint(root, "Debug Log Measured Latency",
|
||
|
le16_to_cpu(log_data->debug_log_measured_latency));
|
||
|
if (le64_to_cpu(log_data->debug_log_latency_stamp) == -1) {
|
||
|
json_object_add_value_string(root, "Debug Log Latency Time Stamp", "NA");
|
||
|
} else {
|
||
|
convert_ts(le64_to_cpu(log_data->debug_log_latency_stamp), ts_buf);
|
||
|
json_object_add_value_string(root, "Debug Log Latency Time Stamp", ts_buf);
|
||
|
}
|
||
|
json_object_add_value_uint(root, "Debug Log Pointer",
|
||
|
le16_to_cpu(log_data->debug_log_ptr));
|
||
|
json_object_add_value_uint(root, "Debug Counter Trigger Source",
|
||
|
le16_to_cpu(log_data->debug_log_counter_trigger));
|
||
|
json_object_add_value_uint(root, "Debug Log Stamp Units",
|
||
|
le16_to_cpu(log_data->debug_log_stamp_units));
|
||
|
json_object_add_value_uint(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
char guid[(GUID_LEN * 2) + 1];
|
||
|
char *ptr = &guid[0];
|
||
|
|
||
|
for (i = GUID_LEN - 1; i >= 0; i--)
|
||
|
ptr += sprintf(ptr, "%02X", log_data->log_page_guid[i]);
|
||
|
|
||
|
json_object_add_value_string(root, "Log Page GUID", guid);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_c5_log(struct nvme_dev *dev, struct unsupported_requirement_log *log_data)
|
||
|
{
|
||
|
int j;
|
||
|
struct json_object *root;
|
||
|
char unsup_req_list_str[40];
|
||
|
char guid_buf[GUID_LEN];
|
||
|
char *guid = guid_buf;
|
||
|
|
||
|
root = json_create_object();
|
||
|
|
||
|
json_object_add_value_int(root, "Number Unsupported Req IDs",
|
||
|
le16_to_cpu(log_data->unsupported_count));
|
||
|
|
||
|
memset((void *)unsup_req_list_str, 0, 40);
|
||
|
for (j = 0; j < le16_to_cpu(log_data->unsupported_count); j++) {
|
||
|
sprintf((char *)unsup_req_list_str, "Unsupported Requirement List %d", j);
|
||
|
json_object_add_value_string(root, unsup_req_list_str,
|
||
|
(char *)log_data->unsupported_req_list[j]);
|
||
|
}
|
||
|
|
||
|
json_object_add_value_int(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
memset((void *)guid, 0, GUID_LEN);
|
||
|
for (j = GUID_LEN - 1; j >= 0; j--)
|
||
|
guid += sprintf(guid, "%02x", log_data->log_page_guid[j]);
|
||
|
json_object_add_value_string(root, "Log page GUID", guid_buf);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_c1_log(struct ocp_error_recovery_log_page *log_data)
|
||
|
{
|
||
|
struct json_object *root;
|
||
|
|
||
|
root = json_create_object();
|
||
|
char guid[64];
|
||
|
|
||
|
json_object_add_value_int(root, "Panic Reset Wait Time",
|
||
|
le16_to_cpu(log_data->panic_reset_wait_time));
|
||
|
json_object_add_value_int(root, "Panic Reset Action", log_data->panic_reset_action);
|
||
|
json_object_add_value_int(root, "Device Recovery Action 1",
|
||
|
log_data->device_recover_action_1);
|
||
|
json_object_add_value_int(root, "Panic ID", le32_to_cpu(log_data->panic_id));
|
||
|
json_object_add_value_int(root, "Device Capabilities",
|
||
|
le32_to_cpu(log_data->device_capabilities));
|
||
|
json_object_add_value_int(root, "Vendor Specific Recovery Opcode",
|
||
|
log_data->vendor_specific_recovery_opcode);
|
||
|
json_object_add_value_int(root, "Vendor Specific Command CDW12",
|
||
|
le32_to_cpu(log_data->vendor_specific_command_cdw12));
|
||
|
json_object_add_value_int(root, "Vendor Specific Command CDW13",
|
||
|
le32_to_cpu(log_data->vendor_specific_command_cdw13));
|
||
|
json_object_add_value_int(root, "Vendor Specific Command Timeout",
|
||
|
log_data->vendor_specific_command_timeout);
|
||
|
json_object_add_value_int(root, "Device Recovery Action 2",
|
||
|
log_data->device_recover_action_2);
|
||
|
json_object_add_value_int(root, "Device Recovery Action 2 Timeout",
|
||
|
log_data->device_recover_action_2_timeout);
|
||
|
json_object_add_value_int(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
memset((void *)guid, 0, 64);
|
||
|
sprintf((char *)guid, "0x%"PRIx64"%"PRIx64"",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data->log_page_guid[8]),
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data->log_page_guid[0]));
|
||
|
json_object_add_value_string(root, "Log page GUID", guid);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_c4_log(struct ocp_device_capabilities_log_page *log_data)
|
||
|
{
|
||
|
struct json_object *root = json_create_object();
|
||
|
char guid[64];
|
||
|
int i;
|
||
|
|
||
|
json_object_add_value_int(root, "PCI Express Ports", le16_to_cpu(log_data->pcie_exp_port));
|
||
|
json_object_add_value_int(root, "OOB Management Support",
|
||
|
le16_to_cpu(log_data->oob_management_support));
|
||
|
json_object_add_value_int(root, "Write Zeroes Command Support",
|
||
|
le16_to_cpu(log_data->wz_cmd_support));
|
||
|
json_object_add_value_int(root, "Sanitize Command Support",
|
||
|
le16_to_cpu(log_data->sanitize_cmd_support));
|
||
|
json_object_add_value_int(root, "Dataset Management Command Support",
|
||
|
le16_to_cpu(log_data->dsm_cmd_support));
|
||
|
json_object_add_value_int(root, "Write Uncorrectable Command Support",
|
||
|
le16_to_cpu(log_data->wu_cmd_support));
|
||
|
json_object_add_value_int(root, "Fused Operation Support",
|
||
|
le16_to_cpu(log_data->fused_operation_support));
|
||
|
json_object_add_value_int(root, "Minimum Valid DSSD Power State",
|
||
|
le16_to_cpu(log_data->min_valid_dssd_pwr_state));
|
||
|
for (i = 0; i <= 127; i++)
|
||
|
json_object_add_value_int(root, "DSSD Power State Descriptors",
|
||
|
log_data->dssd_pwr_state_desc[i]);
|
||
|
json_object_add_value_int(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
memset((void *)guid, 0, 64);
|
||
|
sprintf((char *)guid, "0x%"PRIx64"%"PRIx64"",
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data->log_page_guid[8]),
|
||
|
(uint64_t)le64_to_cpu(*(uint64_t *)&log_data->log_page_guid[0]));
|
||
|
json_object_add_value_string(root, "Log page GUID", guid);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_c9_log(struct telemetry_str_log_format *log_data, __u8 *log_data_buf,
|
||
|
int total_log_page_size)
|
||
|
{
|
||
|
struct json_object *root = json_create_object();
|
||
|
char res_arr[48];
|
||
|
char *res = res_arr;
|
||
|
char guid_buf[GUID_LEN];
|
||
|
char *guid = guid_buf;
|
||
|
char fifo_arr[16];
|
||
|
char *fifo = fifo_arr;
|
||
|
char buf[128];
|
||
|
//calculating the index value for array
|
||
|
__le64 stat_id_index = (log_data->sitsz * 4) / 16;
|
||
|
__le64 eve_id_index = (log_data->estsz * 4) / 16;
|
||
|
__le64 vu_eve_index = (log_data->vu_eve_st_sz * 4) / 16;
|
||
|
__le64 ascii_table_index = (log_data->asctsz * 4);
|
||
|
//Calculating the offset for dynamic fields.
|
||
|
__le64 stat_id_str_table_ofst = log_data->sits * 4;
|
||
|
__le64 event_str_table_ofst = log_data->ests * 4;
|
||
|
__le64 vu_event_str_table_ofst = log_data->vu_eve_sts * 4;
|
||
|
__le64 ascii_table_ofst = log_data->ascts * 4;
|
||
|
struct statistics_id_str_table_entry stat_id_str_table_arr[stat_id_index];
|
||
|
struct event_id_str_table_entry event_id_str_table_arr[eve_id_index];
|
||
|
struct vu_event_id_str_table_entry vu_event_id_str_table_arr[vu_eve_index];
|
||
|
__u8 ascii_table_info_arr[ascii_table_index];
|
||
|
char ascii_buf[ascii_table_index];
|
||
|
char *ascii = ascii_buf;
|
||
|
int j;
|
||
|
|
||
|
json_object_add_value_int(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
memset((__u8 *)res, 0, 15);
|
||
|
for (j = 0; j < 15; j++)
|
||
|
res += sprintf(res, "%d", log_data->reserved1[j]);
|
||
|
json_object_add_value_string(root, "Reserved", res_arr);
|
||
|
|
||
|
memset((void *)guid, 0, GUID_LEN);
|
||
|
for (j = GUID_LEN - 1; j >= 0; j--)
|
||
|
guid += sprintf(guid, "%02x", log_data->log_page_guid[j]);
|
||
|
json_object_add_value_string(root, "Log page GUID", guid_buf);
|
||
|
|
||
|
json_object_add_value_int(root, "Telemetry String Log Size", le64_to_cpu(log_data->sls));
|
||
|
|
||
|
memset((__u8 *)res, 0, 24);
|
||
|
for (j = 0; j < 24; j++)
|
||
|
res += sprintf(res, "%d", log_data->reserved2[j]);
|
||
|
json_object_add_value_string(root, "Reserved", res_arr);
|
||
|
|
||
|
json_object_add_value_int(root, "Statistics Identifier String Table Start",
|
||
|
le64_to_cpu(log_data->sits));
|
||
|
json_object_add_value_int(root, "Event String Table Start", le64_to_cpu(log_data->ests));
|
||
|
json_object_add_value_int(root, "Event String Table Size", le64_to_cpu(log_data->estsz));
|
||
|
json_object_add_value_int(root, "VU Event String Table Start",
|
||
|
le64_to_cpu(log_data->vu_eve_sts));
|
||
|
json_object_add_value_int(root, "VU Event String Table Size",
|
||
|
le64_to_cpu(log_data->vu_eve_st_sz));
|
||
|
json_object_add_value_int(root, "ASCII Table Start",
|
||
|
le64_to_cpu(log_data->ascts));
|
||
|
json_object_add_value_int(root, "ASCII Table Size",
|
||
|
le64_to_cpu(log_data->asctsz));
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo1[j]);
|
||
|
json_object_add_value_string(root, "FIFO 1 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo2[j]);
|
||
|
json_object_add_value_string(root, "FIFO 2 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo3[j]);
|
||
|
json_object_add_value_string(root, "FIFO 3 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo4[j]);
|
||
|
json_object_add_value_string(root, "FIFO 4 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo5[j]);
|
||
|
json_object_add_value_string(root, "FIFO 5 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo6[j]);
|
||
|
json_object_add_value_string(root, "FIFO 6 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo7[j]);
|
||
|
json_object_add_value_string(root, "FIFO 7 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo8[j]);
|
||
|
json_object_add_value_string(root, "FIFO 8 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo9[j]);
|
||
|
json_object_add_value_string(root, "FIFO 9 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo10[j]);
|
||
|
json_object_add_value_string(root, "FIFO 10 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo11[j]);
|
||
|
json_object_add_value_string(root, "FIFO 11 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo12[j]);
|
||
|
json_object_add_value_string(root, "FIFO 12 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo13[j]);
|
||
|
json_object_add_value_string(root, "FIFO 13 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo14[j]);
|
||
|
json_object_add_value_string(root, "FIFO 14 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo15[j]);
|
||
|
json_object_add_value_string(root, "FIFO 15 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((void *)fifo, 0, 16);
|
||
|
for (j = 0; j < 16; j++)
|
||
|
fifo += sprintf(fifo, "%c", log_data->fifo16[j]);
|
||
|
json_object_add_value_string(root, "FIFO 16 ASCII String", fifo_arr);
|
||
|
|
||
|
memset((__u8 *)res, 0, 48);
|
||
|
for (j = 0; j < 48; j++)
|
||
|
res += sprintf(res, "%d", log_data->reserved3[j]);
|
||
|
json_object_add_value_string(root, "Reserved", res_arr);
|
||
|
|
||
|
if (log_data->sitsz != 0) {
|
||
|
|
||
|
memcpy(stat_id_str_table_arr,
|
||
|
(__u8 *)log_data_buf + stat_id_str_table_ofst,
|
||
|
(log_data->sitsz * 4));
|
||
|
struct json_object *stat_table = json_create_object();
|
||
|
|
||
|
for (j = 0; j < stat_id_index; j++) {
|
||
|
struct json_object *entry = json_create_object();
|
||
|
|
||
|
json_object_add_value_uint(entry, "Vendor Specific Statistic Identifier",
|
||
|
le16_to_cpu(stat_id_str_table_arr[j].vs_si));
|
||
|
json_object_add_value_uint(entry, "Reserved",
|
||
|
le64_to_cpu(stat_id_str_table_arr[j].reserved1));
|
||
|
json_object_add_value_uint(entry, "ASCII ID Length",
|
||
|
le64_to_cpu(stat_id_str_table_arr[j].ascii_id_len));
|
||
|
json_object_add_value_uint(entry, "ASCII ID offset",
|
||
|
le64_to_cpu(stat_id_str_table_arr[j].ascii_id_ofst));
|
||
|
json_object_add_value_uint(entry, "Reserved2",
|
||
|
le64_to_cpu(stat_id_str_table_arr[j].reserved2));
|
||
|
sprintf(buf, "Statistics Identifier String Table %d", j);
|
||
|
json_object_add_value_object(stat_table, buf, entry);
|
||
|
}
|
||
|
|
||
|
json_object_add_value_object(root,
|
||
|
"Statistics Identifier String Table", stat_table);
|
||
|
}
|
||
|
|
||
|
if (log_data->estsz != 0) {
|
||
|
struct json_object *eve_table = json_create_object();
|
||
|
|
||
|
memcpy(event_id_str_table_arr,
|
||
|
(__u8 *)log_data_buf + event_str_table_ofst,
|
||
|
(log_data->estsz * 4));
|
||
|
for (j = 0; j < eve_id_index; j++) {
|
||
|
struct json_object *entry = json_create_object();
|
||
|
|
||
|
json_object_add_value_int(entry, "Debug Event Class",
|
||
|
le16_to_cpu(event_id_str_table_arr[j].deb_eve_class));
|
||
|
json_object_add_value_int(entry, "Event Identifier",
|
||
|
le16_to_cpu(event_id_str_table_arr[j].ei));
|
||
|
json_object_add_value_int(entry, "ASCII ID Length",
|
||
|
le64_to_cpu(event_id_str_table_arr[j].ascii_id_len));
|
||
|
json_object_add_value_int(entry, "ASCII ID offset",
|
||
|
le64_to_cpu(event_id_str_table_arr[j].ascii_id_ofst));
|
||
|
json_object_add_value_int(entry, "Reserved",
|
||
|
le64_to_cpu(event_id_str_table_arr[j].reserved2));
|
||
|
sprintf(buf, "Event Identifier String Table Entry %d", j);
|
||
|
json_object_add_value_object(eve_table, buf, entry);
|
||
|
}
|
||
|
json_object_add_value_object(root,
|
||
|
"Event Identifier String Table Entry",
|
||
|
eve_table);
|
||
|
}
|
||
|
|
||
|
if (log_data->vu_eve_st_sz != 0) {
|
||
|
struct json_object *vu_eve_table = json_create_object();
|
||
|
|
||
|
memcpy(vu_event_id_str_table_arr,
|
||
|
(__u8 *)log_data_buf + vu_event_str_table_ofst,
|
||
|
(log_data->vu_eve_st_sz * 4));
|
||
|
for (j = 0; j < vu_eve_index; j++) {
|
||
|
struct json_object *entry = json_create_object();
|
||
|
|
||
|
json_object_add_value_int(entry, "Debug Event Class",
|
||
|
le16_to_cpu(vu_event_id_str_table_arr[j].deb_eve_class));
|
||
|
json_object_add_value_int(entry, "VU Event Identifier",
|
||
|
le16_to_cpu(vu_event_id_str_table_arr[j].vu_ei));
|
||
|
json_object_add_value_int(entry, "ASCII ID Length",
|
||
|
le64_to_cpu(vu_event_id_str_table_arr[j].ascii_id_len));
|
||
|
json_object_add_value_int(entry, "ASCII ID offset",
|
||
|
le64_to_cpu(vu_event_id_str_table_arr[j].ascii_id_ofst));
|
||
|
json_object_add_value_int(entry, "Reserved",
|
||
|
le64_to_cpu(vu_event_id_str_table_arr[j].reserved));
|
||
|
sprintf(buf, "VU Event Identifier String Table Entry %d", j);
|
||
|
json_object_add_value_object(vu_eve_table, buf, entry);
|
||
|
}
|
||
|
json_object_add_value_object(root,
|
||
|
"VU Event Identifier String Table Entry",
|
||
|
vu_eve_table);
|
||
|
}
|
||
|
|
||
|
if (log_data->asctsz != 0) {
|
||
|
memcpy(ascii_table_info_arr,
|
||
|
(__u8 *)log_data_buf + ascii_table_ofst,
|
||
|
(log_data->asctsz * 4));
|
||
|
memset((void *)ascii, 0, ascii_table_index);
|
||
|
for (j = 0; j < ascii_table_index; j++)
|
||
|
ascii += sprintf(ascii, "%c", ascii_table_info_arr[j]);
|
||
|
json_object_add_value_string(root, "ASCII Table", ascii_buf);
|
||
|
}
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static void json_c7_log(struct nvme_dev *dev, struct tcg_configuration_log *log_data)
|
||
|
{
|
||
|
int j;
|
||
|
struct json_object *root;
|
||
|
char guid_buf[GUID_LEN];
|
||
|
char *guid = guid_buf;
|
||
|
char res_arr[458];
|
||
|
char *res = res_arr;
|
||
|
|
||
|
root = json_create_object();
|
||
|
|
||
|
json_object_add_value_int(root, "State", log_data->state);
|
||
|
memset((__u8 *)res, 0, 3);
|
||
|
for (j = 0; j < 3; j++)
|
||
|
res += sprintf(res, "%d", log_data->rsvd1[j]);
|
||
|
json_object_add_value_string(root, "Reserved1", res_arr);
|
||
|
json_object_add_value_int(root, "Locking SP Activation Count",
|
||
|
log_data->locking_sp_act_count);
|
||
|
json_object_add_value_int(root, "Tper Revert Count",
|
||
|
log_data->locking_sp_rev_count);
|
||
|
json_object_add_value_int(root, "Number of Locking Objects",
|
||
|
log_data->no_of_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Single User Mode Locking Objects",
|
||
|
log_data->no_of_single_um_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Range Provisioned Locking Objects",
|
||
|
log_data->no_of_range_prov_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Namespace Provisioned Locking Objects",
|
||
|
log_data->no_of_ns_prov_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Read Locked Locking Objects",
|
||
|
log_data->no_of_read_lock_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Write Locked Locking Objects",
|
||
|
log_data->no_of_write_lock_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Read Unlocked Locking Objects",
|
||
|
log_data->no_of_read_unlock_locking_obj);
|
||
|
json_object_add_value_int(root, "Number of Write Unlocked Locking Objects",
|
||
|
log_data->no_of_write_unlock_locking_obj);
|
||
|
json_object_add_value_int(root, "Reserved2", log_data->rsvd2);
|
||
|
|
||
|
json_object_add_value_int(root, "SID Authentication Try Count",
|
||
|
le32_to_cpu(log_data->sid_auth_try_count));
|
||
|
json_object_add_value_int(root, "SID Authentication Try Limit",
|
||
|
le32_to_cpu(log_data->sid_auth_try_limit));
|
||
|
json_object_add_value_int(root, "Programmatic TCG Reset Count",
|
||
|
le32_to_cpu(log_data->pro_tcg_rc));
|
||
|
json_object_add_value_int(root, "Programmatic Reset Lock Count",
|
||
|
le32_to_cpu(log_data->pro_rlc));
|
||
|
json_object_add_value_int(root, "TCG Error Count", le32_to_cpu(log_data->tcg_ec));
|
||
|
|
||
|
memset((__u8 *)res, 0, 458);
|
||
|
for (j = 0; j < 458; j++)
|
||
|
res += sprintf(res, "%d", log_data->rsvd3[j]);
|
||
|
json_object_add_value_string(root, "Reserved3", res_arr);
|
||
|
|
||
|
json_object_add_value_int(root, "Log Page Version",
|
||
|
le16_to_cpu(log_data->log_page_version));
|
||
|
|
||
|
memset((void *)guid, 0, GUID_LEN);
|
||
|
for (j = GUID_LEN - 1; j >= 0; j--)
|
||
|
guid += sprintf(guid, "%02x", log_data->log_page_guid[j]);
|
||
|
json_object_add_value_string(root, "Log page GUID", guid_buf);
|
||
|
|
||
|
json_print_object(root, NULL);
|
||
|
printf("\n");
|
||
|
|
||
|
json_free_object(root);
|
||
|
}
|
||
|
|
||
|
static struct ocp_print_ops json_print_ops = {
|
||
|
.hwcomp_log = json_hwcomp_log,
|
||
|
.fw_act_history = json_fw_activation_history,
|
||
|
.smart_extended_log = json_smart_extended_log,
|
||
|
.telemetry_log = json_telemetry_log,
|
||
|
.c3_log = json_c3_log,
|
||
|
.c5_log = json_c5_log,
|
||
|
.c1_log = json_c1_log,
|
||
|
.c4_log = json_c4_log,
|
||
|
.c9_log = json_c9_log,
|
||
|
.c7_log = json_c7_log,
|
||
|
};
|
||
|
|
||
|
struct ocp_print_ops *ocp_get_json_print_ops(nvme_print_flags_t flags)
|
||
|
{
|
||
|
json_print_ops.flags = flags;
|
||
|
return &json_print_ops;
|
||
|
}
|