// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (c) 2022 Solidigm. * * Authors: haro.panosyan@solidigm.com * leonardo.da.cunha@solidigm.com */ #include #include "ocp-utils.h" #include "nvme-print.h" static const __u8 OCP_FID_CLEAR_FW_ACTIVATION_HISTORY = 0xC1; static const __u8 OCP_FID_CLEAR_PCIE_CORRECTABLE_ERROR_COUNTERS = 0xC3; static int ocp_clear_feature(int argc, char **argv, const char *desc, const __u8 fid) { __u32 result = 0; __u32 clear = 1 << 31; struct nvme_dev *dev; __u8 uuid_index = 0; bool uuid = true; int err; OPT_ARGS(opts) = { OPT_FLAG("no-uuid", 'n', NULL, "Skip UUID index search (UUID index not required for OCP 1.0)"), OPT_END() }; err = parse_and_open(&dev, argc, argv, desc, opts); if (err) return err; if (opts[0].seen) uuid = false; if (uuid) { /* OCP 2.0 requires UUID index support */ err = ocp_get_uuid_index(dev, &uuid_index); if (err || !uuid_index) { fprintf(stderr, "ERROR: No OCP UUID index found\n"); goto close_dev; } } struct nvme_set_features_args args = { .result = &result, .data = NULL, .args_size = sizeof(args), .fd = dev_fd(dev), .timeout = NVME_DEFAULT_IOCTL_TIMEOUT, .nsid = 0, .cdw11 = clear, .cdw12 = 0, .cdw13 = 0, .cdw15 = 0, .data_len = 0, .save = 0, .uuidx = uuid_index, .fid = fid, }; err = nvme_set_features(&args); if (err == 0) printf("Success : %s\n", desc); else if (err > 0) nvme_show_status(err); else printf("Fail : %s\n", desc); close_dev: /* Redundant close() to make static code analysis happy */ close(dev->direct.fd); dev_close(dev); return err; } int ocp_clear_fw_update_history(int argc, char **argv, struct command *cmd, struct plugin *plugin) { const char *desc = "OCP Clear Firmware Update History"; return ocp_clear_feature(argc, argv, desc, OCP_FID_CLEAR_FW_ACTIVATION_HISTORY); } int ocp_clear_pcie_correctable_errors(int argc, char **argv, struct command *cmd, struct plugin *plugin) { const char *desc = "OCP Clear PCIe Correctable Error Counters"; return ocp_clear_feature(argc, argv, desc, OCP_FID_CLEAR_PCIE_CORRECTABLE_ERROR_COUNTERS); }