// 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) { json_object_add_value_string(r, "Description", hwcomp_id_to_string(le32_to_cpu(e->desc->id))); json_object_add_nprix64(r, "Date/Lot Size", e->date_lot_size); json_object_add_nprix64(r, "Additional Information Size", e->add_info_size); json_object_add_uint_0nx(r, "Identifier", le32_to_cpu(e->desc->id), 8); json_object_add_0nprix64(r, "Manufacture", le64_to_cpu(e->desc->mfg), 16); json_object_add_0nprix64(r, "Revision", le64_to_cpu(e->desc->rev), 16); json_object_add_0nprix64(r, "Manufacture Code", le64_to_cpu(e->desc->mfg_code), 16); json_object_add_byte_array(r, "Date/Lot Code", e->date_lot_code, e->date_lot_size); json_object_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) { json_object_add_value_string(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) { long double log_bytes = uint128_t_to_double(le128_to_cpu(log->size)); struct json_object *r = json_create_object(); if (log->ver == 1) log_bytes *= sizeof(__le32); json_object_add_uint_02x(r, "Log Identifier", OCP_LID_HWCOMP); json_object_add_uint_0x(r, "Log Page Version", le16_to_cpu(log->ver)); json_object_add_byte_array(r, "Reserved2", log->rsvd2, ARRAY_SIZE(log->rsvd2)); json_object_add_byte_array(r, "Log page GUID", log->guid, ARRAY_SIZE(log->guid)); json_object_add_nprix64(r, "Hardware Component Log Size", (unsigned long long)log_bytes); json_object_add_byte_array(r, "Reserved48", log->rsvd48, ARRAY_SIZE(log->rsvd48)); print_hwcomp_descs_json(log->desc, log_bytes - offsetof(struct hwcomp_log, desc), 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", int48_to_long(entry->ts.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_v1(struct ocp_smart_extended_log *log) { struct json_object *root; struct json_object *pmuw; struct json_object *pmur; uint16_t smart_log_ver = 0; uint16_t dssd_version = 0; int i = 0; char guid[40]; char ascii_arr[65]; char *ascii = ascii_arr; root = json_create_object(); pmuw = json_create_object(); pmur = json_create_object(); json_object_add_value_uint64(pmuw, "hi", le64_to_cpu(*(uint64_t *)&log->physical_media_units_written[8])); json_object_add_value_uint64(pmuw, "lo", le64_to_cpu(*(uint64_t *)&log->physical_media_units_written)); json_object_add_value_object(root, "Physical media units written", pmuw); json_object_add_value_uint64(pmur, "hi", le64_to_cpu(*(uint64_t *)&log->physical_media_units_read[8])); json_object_add_value_uint64(pmur, "lo", le64_to_cpu(*(uint64_t *)&log->physical_media_units_read)); json_object_add_value_object(root, "Physical media units read", pmur); json_object_add_value_uint64(root, "Bad user nand blocks - Raw", int48_to_long(log->bad_user_nand_blocks_raw)); json_object_add_value_uint(root, "Bad user nand blocks - Normalized", le16_to_cpu(log->bad_user_nand_blocks_normalized)); json_object_add_value_uint64(root, "Bad system nand blocks - Raw", int48_to_long(log->bad_system_nand_blocks_raw)); json_object_add_value_uint(root, "Bad system nand blocks - Normalized", le16_to_cpu(log->bad_system_nand_blocks_normalized)); json_object_add_value_uint64(root, "XOR recovery count", le64_to_cpu(log->xor_recovery_count)); json_object_add_value_uint64(root, "Uncorrectable read error count", le64_to_cpu(log->uncorrectable_read_err_count)); json_object_add_value_uint64(root, "Soft ecc error count", le64_to_cpu(log->soft_ecc_err_count)); json_object_add_value_uint(root, "End to end detected errors", le32_to_cpu(log->end_to_end_detected_err)); json_object_add_value_uint(root, "End to end corrected errors", le32_to_cpu(log->end_to_end_corrected_err)); json_object_add_value_uint(root, "System data percent used", log->system_data_used_percent); json_object_add_value_uint64(root, "Refresh counts", int56_to_long(log->refresh_counts)); json_object_add_value_uint(root, "Max User data erase counts", le32_to_cpu(log->user_data_erase_count_max)); json_object_add_value_uint(root, "Min User data erase counts", le32_to_cpu(log->user_data_erase_count_min)); json_object_add_value_uint(root, "Number of Thermal throttling events", log->thermal_throttling_event_count); json_object_add_value_uint(root, "Current throttling status", log->thermal_throttling_current_status); json_object_add_value_uint64(root, "PCIe correctable error count", le64_to_cpu(log->pcie_correctable_err_count)); json_object_add_value_uint(root, "Incomplete shutdowns", le32_to_cpu(log->incomplete_shoutdowns)); json_object_add_value_uint(root, "Percent free blocks", log->percent_free_blocks); json_object_add_value_uint(root, "Capacitor health", le16_to_cpu(log->capacitor_health)); json_object_add_value_uint64(root, "Unaligned I/O", le64_to_cpu(log->unaligned_io)); json_object_add_value_uint64(root, "Security Version Number", le64_to_cpu(log->security_version)); json_object_add_value_uint64(root, "NUSE - Namespace utilization", le64_to_cpu(log->total_nuse)); json_object_add_value_uint128(root, "PLP start count", le128_to_cpu(log->plp_start_count)); json_object_add_value_uint128(root, "Endurance estimate", le128_to_cpu(log->endurance_estimate)); smart_log_ver = le16_to_cpu(log->log_page_version); json_object_add_value_uint(root, "Log page version", smart_log_ver); memset((void *)guid, 0, 40); sprintf((char *)guid, "0x%"PRIx64"%"PRIx64"", le64_to_cpu(*(uint64_t *)&log->log_page_guid[8]), le64_to_cpu(*(uint64_t *)&log->log_page_guid)); 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", log->nvme_cmdset_errata_version); json_object_add_value_uint(root, "Lowest Permitted Firmware Revision", le64_to_cpu(log->lowest_permitted_fw_rev)); json_object_add_value_uint(root, "NVMe Over Pcie Errata Version", log->nvme_over_pcie_errate_version); json_object_add_value_uint(root, "NVMe Mi Errata Version", log->nvme_mi_errata_version); json_object_add_value_uint(root, "Total media dies", le16_to_cpu(log->total_media_dies)); json_object_add_value_uint(root, "Total die failure tolerance", le16_to_cpu(log->total_die_failure_tolerance)); json_object_add_value_uint(root, "Media dies offline", le16_to_cpu(log->media_dies_offline)); json_object_add_value_uint(root, "Max temperature recorded", log->max_temperature_recorded); json_object_add_value_uint64(root, "Nand avg erase count", le64_to_cpu(log->nand_avg_erase_count)); json_object_add_value_uint(root, "Command timeouts", le32_to_cpu(log->command_timeouts)); json_object_add_value_uint(root, "Sys area program fail count raw", le32_to_cpu(log->sys_area_program_fail_count_raw)); json_object_add_value_uint(root, "Sys area program fail count noralized", log->sys_area_program_fail_count_normalized); json_object_add_value_uint(root, "Sys area uncorrectable read count raw", le32_to_cpu(log->sys_area_uncorr_read_count_raw)); json_object_add_value_uint(root, "Sys area uncorrectable read count noralized", log->sys_area_uncorr_read_count_normalized); json_object_add_value_uint(root, "Sys area erase fail count raw", le32_to_cpu(log->sys_area_erase_fail_count_raw)); json_object_add_value_uint(root, "Sys area erase fail count noralized", log->sys_area_erase_fail_count_normalized); json_object_add_value_uint(root, "Max peak power capability", le16_to_cpu(log->max_peak_power_capability)); json_object_add_value_uint(root, "Current max avg power", le16_to_cpu(log->current_max_avg_power)); json_object_add_value_uint64(root, "Lifetime power consumed", int48_to_long(log->lifetime_power_consumed)); memset((void *)ascii, 0, 65); for (i = 0; i < 8; i++) ascii += sprintf(ascii, "%c", log->dssd_firmware_revision[i]); json_object_add_value_string(root, "Dssd firmware revision", ascii_arr); json_object_add_value_string(root, "Dssd firmware build UUID", util_uuid_to_string(log->dssd_firmware_build_uuid)); ascii = ascii_arr; memset((void *)ascii, 0, 65); for (i = 0; i < 64; i++) ascii += sprintf(ascii, "%c", log->dssd_firmware_build_label[i]); json_object_add_value_string(root, "Dssd firmware build label", ascii_arr); fallthrough; case 2 ... 3: json_object_add_value_uint(root, "Errata Version Field", log->dssd_errata_version); memcpy(&dssd_version, log->dssd_point_version, sizeof(dssd_version)); json_object_add_value_uint(root, "Point Version Field", le16_to_cpu(dssd_version)); memcpy(&dssd_version, log->dssd_minor_version, sizeof(dssd_version)); json_object_add_value_uint(root, "Minor Version Field", le16_to_cpu(dssd_version)); json_object_add_value_uint(root, "Major Version Field", log->dssd_major_version); json_object_add_value_uint(root, "NVMe Base Errata Version", log->nvme_base_errata_version); json_object_add_value_uint(root, "PCIe Link Retraining Count", le64_to_cpu(log->pcie_link_retaining_count)); json_object_add_value_uint(root, "Power State Change Count", le64_to_cpu(log->power_state_change_count)); } json_print_object(root, NULL); printf("\n"); json_free_object(root); } static void json_smart_extended_log_v2(struct ocp_smart_extended_log *log) { struct json_object *root; struct json_object *pmuw; struct json_object *pmur; int i = 0; uint16_t smart_log_ver = 0; uint16_t dssd_version = 0; char guid[40]; char ascii_arr[65]; char *ascii = ascii_arr; root = json_create_object(); pmuw = json_create_object(); pmur = json_create_object(); json_object_add_value_uint64(pmuw, "hi", le64_to_cpu(*(uint64_t *)&log->physical_media_units_written[8])); json_object_add_value_uint64(pmuw, "lo", le64_to_cpu(*(uint64_t *)&log->physical_media_units_written)); json_object_add_value_object(root, "physical_media_units_written", pmuw); json_object_add_value_uint64(pmur, "hi", le64_to_cpu(*(uint64_t *)&log->physical_media_units_read[8])); json_object_add_value_uint64(pmur, "lo", le64_to_cpu(*(uint64_t *)&log->physical_media_units_read)); json_object_add_value_object(root, "physical_media_units_read", pmur); json_object_add_value_uint64(root, "bad_user_nand_blocks_raw", int48_to_long(log->bad_user_nand_blocks_raw)); json_object_add_value_uint(root, "bad_user_nand_blocks_normalized", le16_to_cpu(log->bad_user_nand_blocks_normalized)); json_object_add_value_uint64(root, "bad_system_nand_blocks_raw", int48_to_long(log->bad_system_nand_blocks_raw)); json_object_add_value_uint(root, "bad_system_nand_blocks_normalized", le16_to_cpu(log->bad_system_nand_blocks_normalized)); json_object_add_value_uint64(root, "xor_recovery_count", le64_to_cpu(log->xor_recovery_count)); json_object_add_value_uint64(root, "uncorrectable_read_errors", le64_to_cpu(log->uncorrectable_read_err_count)); json_object_add_value_uint64(root, "soft_ecc_error_count", le64_to_cpu(log->soft_ecc_err_count)); json_object_add_value_uint(root, "end_to_end_detected_errors", le32_to_cpu(log->end_to_end_detected_err)); json_object_add_value_uint(root, "end_to_end_corrected_errors", le32_to_cpu(log->end_to_end_corrected_err)); json_object_add_value_uint(root, "system_data_percent_used", log->system_data_used_percent); json_object_add_value_uint64(root, "refresh_count", int56_to_long(log->refresh_counts)); json_object_add_value_uint(root, "max_user_data_erase_count", le32_to_cpu(log->user_data_erase_count_max)); json_object_add_value_uint(root, "min_user_data_erase_count", le32_to_cpu(log->user_data_erase_count_min)); json_object_add_value_uint(root, "thermal_throttling_events", log->thermal_throttling_event_count); json_object_add_value_uint(root, "current_throttling_status", log->thermal_throttling_current_status); json_object_add_value_uint64(root, "pcie_correctable_errors", le64_to_cpu(log->pcie_correctable_err_count)); json_object_add_value_uint(root, "incomplete_shutdowns", le32_to_cpu(log->incomplete_shoutdowns)); json_object_add_value_uint(root, "percent_free_blocks", log->percent_free_blocks); json_object_add_value_uint(root, "capacitor_health", le16_to_cpu(log->capacitor_health)); json_object_add_value_uint64(root, "unaligned_io", le64_to_cpu(log->unaligned_io)); json_object_add_value_uint64(root, "security_version_number", le64_to_cpu(log->security_version)); json_object_add_value_uint64(root, "nuse_namespace_utilization", le64_to_cpu(log->total_nuse)); json_object_add_value_uint128(root, "plp_start_count", le128_to_cpu(log->plp_start_count)); json_object_add_value_uint128(root, "endurance_estimate", le128_to_cpu(log->endurance_estimate)); smart_log_ver = le16_to_cpu(log->log_page_version); json_object_add_value_uint(root, "log_page_version", smart_log_ver); memset((void *)guid, 0, 40); sprintf((char *)guid, "0x%"PRIx64"%"PRIx64"", le64_to_cpu(*(uint64_t *)&log->log_page_guid[8]), le64_to_cpu(*(uint64_t *)&log->log_page_guid)); 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", log->nvme_cmdset_errata_version); json_object_add_value_uint(root, "lowest_permitted_firmware_revision", le64_to_cpu(log->lowest_permitted_fw_rev)); json_object_add_value_uint(root, "nvme_over_pcie_errata_version", log->nvme_over_pcie_errate_version); json_object_add_value_uint(root, "nvme_mi_errata_version", log->nvme_mi_errata_version); json_object_add_value_uint(root, "total_media_dies", le16_to_cpu(log->total_media_dies)); json_object_add_value_uint(root, "total_die_failure_tolerance", le16_to_cpu(log->total_die_failure_tolerance)); json_object_add_value_uint(root, "media_dies_offline", le16_to_cpu(log->media_dies_offline)); json_object_add_value_uint(root, "max_temperature_recorded", log->max_temperature_recorded); json_object_add_value_uint64(root, "nand_avg_erase_count", le64_to_cpu(log->nand_avg_erase_count)); json_object_add_value_uint(root, "command_timeouts", le32_to_cpu(log->command_timeouts)); json_object_add_value_uint(root, "sys_area_program_fail_count_raw", le32_to_cpu(log->sys_area_program_fail_count_raw)); json_object_add_value_uint(root, "sys_area_program_fail_count_noralized", log->sys_area_program_fail_count_normalized); json_object_add_value_uint(root, "sys_area_uncorrectable_read_count_raw", le32_to_cpu(log->sys_area_uncorr_read_count_raw)); json_object_add_value_uint(root, "sys_area_uncorrectable_read_count_noralized", log->sys_area_uncorr_read_count_normalized); json_object_add_value_uint(root, "sys_area_erase_fail_count_raw", le32_to_cpu(log->sys_area_erase_fail_count_raw)); json_object_add_value_uint(root, "sys_area_erase_fail_count_noralized", log->sys_area_erase_fail_count_normalized); json_object_add_value_uint(root, "max_peak_power_capability", le16_to_cpu(log->max_peak_power_capability)); json_object_add_value_uint(root, "current_max_avg_power", le16_to_cpu(log->current_max_avg_power)); json_object_add_value_uint64(root, "lifetime_power_consumed", int48_to_long(log->lifetime_power_consumed)); memset((void *)ascii, 0, 65); for (i = 0; i < 8; i++) ascii += sprintf(ascii, "%c", log->dssd_firmware_revision[i]); json_object_add_value_string(root, "dssd_firmware_revision", ascii_arr); json_object_add_value_string(root, "dssd_firmware_build_uuid", util_uuid_to_string(log->dssd_firmware_build_uuid)); ascii = ascii_arr; memset((void *)ascii, 0, 65); for (i = 0; i < 64; i++) ascii += sprintf(ascii, "%c", log->dssd_firmware_build_label[i]); json_object_add_value_string(root, "dssd_firmware_build_label", ascii_arr); fallthrough; case 2 ... 3: json_object_add_value_uint(root, "errata_version_field", log->dssd_errata_version); memcpy(&dssd_version, log->dssd_point_version, sizeof(dssd_version)); json_object_add_value_uint(root, "point_version_field", le16_to_cpu(dssd_version)); memcpy(&dssd_version, log->dssd_minor_version, sizeof(dssd_version)); json_object_add_value_uint(root, "minor_version_field", le16_to_cpu(dssd_version)); json_object_add_value_uint(root, "major_version_field", log->dssd_major_version); json_object_add_value_uint(root, "nvme_base_errata_version", log->nvme_base_errata_version); json_object_add_value_uint(root, "pcie_link_retraining_count", le64_to_cpu(log->pcie_link_retaining_count)); json_object_add_value_uint(root, "power_state_change_count", le64_to_cpu(log->power_state_change_count)); } json_print_object(root, NULL); printf("\n"); json_free_object(root); } static void json_smart_extended_log(struct ocp_smart_extended_log *log, unsigned int version) { switch (version) { default: case 1: json_smart_extended_log_v1(log); break; case 2: json_smart_extended_log_v2(log); } } 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"}; __u16 log_page_version = le16_to_cpu(log_data->log_page_version); 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)); if (log_page_version >= 0x4) { strcpy(buf, "0x"); for (i = ARRAY_SIZE(log_data->latency_monitor_debug_log_size) - 1; i > 0 && (log_data->latency_monitor_debug_log_size[i] == 0); i--) ; while (i >= 0) { char hex_string[3]; sprintf(hex_string, "%02x", log_data->latency_monitor_debug_log_size[i--]); strcat(buf, hex_string); } json_object_add_value_string(root, "Debug Telemetry Log Size", buf); } 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", 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, 48); 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)); res = res_arr; memset((__u8 *)res, 0, 48); 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); fifo = 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); res = res_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; __u16 log_page_version = le16_to_cpu(log_data->log_page_version); 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->rsvd15); 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); if (log_page_version == 1) { res += sprintf(res, "%d%d", *(__u8 *)&log_data->no_of_ns_prov_locking_obj_ext, *((__u8 *)&log_data->no_of_ns_prov_locking_obj_ext + 1)); } else { json_object_add_value_int(root, "Number of Namespace Provisioned Locking Objects Extended", log_data->no_of_ns_prov_locking_obj_ext); } for (j = 0; j < 456; j++) res += sprintf(res, "%d", log_data->rsvd38[j]); json_object_add_value_string(root, "Reserved3", res_arr); json_object_add_value_int(root, "Log Page Version", 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; }