| #include "common/types.hpp" |
| #include "pldm_cmd_helper.hpp" |
| |
| #include <libpldm/entity.h> |
| #include <libpldm/state_set.h> |
| |
| #include <cstddef> |
| #include <map> |
| |
| #ifdef OEM_IBM |
| #include "oem/ibm/oem_ibm_state_set.hpp" |
| #endif |
| |
| using namespace pldm::utils; |
| |
| namespace pldmtool |
| { |
| namespace platform |
| { |
| namespace |
| { |
| using namespace pldmtool::helper; |
| |
| static const std::map<uint8_t, std::string> sensorPresState{ |
| {PLDM_SENSOR_UNKNOWN, "Sensor Unknown"}, |
| {PLDM_SENSOR_NORMAL, "Sensor Normal"}, |
| {PLDM_SENSOR_WARNING, "Sensor Warning"}, |
| {PLDM_SENSOR_CRITICAL, "Sensor Critical"}, |
| {PLDM_SENSOR_FATAL, "Sensor Fatal"}, |
| {PLDM_SENSOR_LOWERWARNING, "Sensor Lower Warning"}, |
| {PLDM_SENSOR_LOWERCRITICAL, "Sensor Lower Critical"}, |
| {PLDM_SENSOR_LOWERFATAL, "Sensor Lower Fatal"}, |
| {PLDM_SENSOR_UPPERWARNING, "Sensor Upper Warning"}, |
| {PLDM_SENSOR_UPPERCRITICAL, "Sensor Upper Critical"}, |
| {PLDM_SENSOR_UPPERFATAL, "Sensor Upper Fatal"}}; |
| |
| static const std::map<uint8_t, std::string> sensorOpState{ |
| {PLDM_SENSOR_ENABLED, "Sensor Enabled"}, |
| {PLDM_SENSOR_DISABLED, "Sensor Disabled"}, |
| {PLDM_SENSOR_UNAVAILABLE, "Sensor Unavailable"}, |
| {PLDM_SENSOR_STATUSUNKOWN, "Sensor Status Unknown"}, |
| {PLDM_SENSOR_FAILED, "Sensor Failed"}, |
| {PLDM_SENSOR_INITIALIZING, "Sensor Sensor Intializing"}, |
| {PLDM_SENSOR_SHUTTINGDOWN, "Sensor Shutting down"}, |
| {PLDM_SENSOR_INTEST, "Sensor Intest"}}; |
| |
| std::vector<std::unique_ptr<CommandInterface>> commands; |
| |
| } // namespace |
| |
| using ordered_json = nlohmann::ordered_json; |
| |
| class GetPDR : public CommandInterface |
| { |
| public: |
| ~GetPDR() = default; |
| GetPDR() = delete; |
| GetPDR(const GetPDR&) = delete; |
| GetPDR(GetPDR&&) = default; |
| GetPDR& operator=(const GetPDR&) = delete; |
| GetPDR& operator=(GetPDR&&) = delete; |
| |
| using CommandInterface::CommandInterface; |
| |
| explicit GetPDR(const char* type, const char* name, CLI::App* app) : |
| CommandInterface(type, name, app) |
| { |
| auto pdrOptionGroup = app->add_option_group( |
| "Required Option", |
| "Retrieve individual PDR, all PDRs, PDRs of a requested type or retrieve all PDRs of the requested terminusID"); |
| pdrOptionGroup->add_option( |
| "-d,--data", recordHandle, |
| "retrieve individual PDRs from a PDR Repository\n" |
| "eg: The recordHandle value for the PDR to be retrieved and 0 " |
| "means get first PDR in the repository."); |
| pdrRecType = ""; |
| pdrOptionGroup->add_option("-t, --type", pdrRecType, |
| "retrieve all PDRs of the requested type\n" |
| "supported types:\n" |
| "[terminusLocator, stateSensor, " |
| "numericEffecter, stateEffecter, " |
| "EntityAssociation, fruRecord, ... ]"); |
| |
| getPDRGroupOption = pdrOptionGroup->add_option( |
| "-i, --terminusID", pdrTerminus, |
| "retrieve all PDRs of the requested terminusID\n" |
| "supported IDs:\n [1, 2, 208...]"); |
| |
| allPDRs = false; |
| pdrOptionGroup->add_flag("-a, --all", allPDRs, |
| "retrieve all PDRs from a PDR repository"); |
| |
| pdrOptionGroup->require_option(1); |
| } |
| |
| void parseGetPDROptions() |
| { |
| optTIDSet = false; |
| if (getPDRGroupOption->count() > 0) |
| { |
| optTIDSet = true; |
| getPDRs(); |
| } |
| } |
| |
| void getPDRs() |
| { |
| // start the array |
| std::cout << "["; |
| |
| recordHandle = 0; |
| do |
| { |
| CommandInterface::exec(); |
| } while (recordHandle != 0); |
| |
| // close the array |
| std::cout << "]\n"; |
| |
| if (handleFound) |
| { |
| recordHandle = 0; |
| uint32_t prevRecordHandle = 0; |
| do |
| { |
| CommandInterface::exec(); |
| if (recordHandle == prevRecordHandle) |
| { |
| return; |
| } |
| prevRecordHandle = recordHandle; |
| } while (recordHandle != 0); |
| } |
| } |
| |
| void exec() override |
| { |
| if (allPDRs || !pdrRecType.empty()) |
| { |
| if (!pdrRecType.empty()) |
| { |
| std::transform(pdrRecType.begin(), pdrRecType.end(), |
| pdrRecType.begin(), tolower); |
| } |
| |
| // start the array |
| std::cout << "[\n"; |
| |
| // Retrieve all PDR records starting from the first |
| recordHandle = 0; |
| uint32_t prevRecordHandle = 0; |
| std::map<uint32_t, uint32_t> recordsSeen; |
| do |
| { |
| CommandInterface::exec(); |
| // recordHandle is updated to nextRecord when |
| // CommandInterface::exec() is successful. |
| // In case of any error, return. |
| if (recordHandle == prevRecordHandle) |
| { |
| return; |
| } |
| |
| // check for circular references. |
| auto result = recordsSeen.emplace(recordHandle, |
| prevRecordHandle); |
| if (!result.second) |
| { |
| std::cerr |
| << "Record handle " << recordHandle |
| << " has multiple references: " << result.first->second |
| << ", " << prevRecordHandle << "\n"; |
| return; |
| } |
| prevRecordHandle = recordHandle; |
| |
| if (recordHandle != 0) |
| { |
| // close the array |
| std::cout << ","; |
| } |
| } while (recordHandle != 0); |
| |
| // close the array |
| std::cout << "]\n"; |
| } |
| else |
| { |
| CommandInterface::exec(); |
| } |
| } |
| |
| std::pair<int, std::vector<uint8_t>> createRequestMsg() override |
| { |
| std::vector<uint8_t> requestMsg(sizeof(pldm_msg_hdr) + |
| PLDM_GET_PDR_REQ_BYTES); |
| auto request = reinterpret_cast<pldm_msg*>(requestMsg.data()); |
| |
| auto rc = encode_get_pdr_req(instanceId, recordHandle, 0, |
| PLDM_GET_FIRSTPART, UINT16_MAX, 0, request, |
| PLDM_GET_PDR_REQ_BYTES); |
| return {rc, requestMsg}; |
| } |
| |
| void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override |
| { |
| uint8_t completionCode = 0; |
| uint8_t recordData[UINT16_MAX] = {0}; |
| uint32_t nextRecordHndl = 0; |
| uint32_t nextDataTransferHndl = 0; |
| uint8_t transferFlag = 0; |
| uint16_t respCnt = 0; |
| uint8_t transferCRC = 0; |
| |
| auto rc = decode_get_pdr_resp( |
| responsePtr, payloadLength, &completionCode, &nextRecordHndl, |
| &nextDataTransferHndl, &transferFlag, &respCnt, recordData, |
| sizeof(recordData), &transferCRC); |
| |
| if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS) |
| { |
| std::cerr << "Response Message Error: " |
| << "rc=" << rc << ",cc=" << (int)completionCode |
| << std::endl; |
| return; |
| } |
| |
| if (optTIDSet && !handleFound) |
| { |
| terminusHandle = getTerminusHandle(recordData, pdrTerminus); |
| if (terminusHandle.has_value()) |
| { |
| recordHandle = 0; |
| return; |
| } |
| else |
| { |
| recordHandle = nextRecordHndl; |
| return; |
| } |
| } |
| |
| else |
| { |
| printPDRMsg(nextRecordHndl, respCnt, recordData, terminusHandle); |
| recordHandle = nextRecordHndl; |
| } |
| } |
| |
| private: |
| const std::map<pldm::pdr::EntityType, std::string> entityType = { |
| {PLDM_ENTITY_UNSPECIFIED, "Unspecified"}, |
| {PLDM_ENTITY_OTHER, "Other"}, |
| {PLDM_ENTITY_NETWORK, "Network"}, |
| {PLDM_ENTITY_GROUP, "Group"}, |
| {PLDM_ENTITY_REMOTE_MGMT_COMM_DEVICE, |
| "Remote Management Communication Device"}, |
| {PLDM_ENTITY_EXTERNAL_ENVIRONMENT, "External Environment"}, |
| {PLDM_ENTITY_COMM_CHANNEL, " Communication Channel"}, |
| {PLDM_ENTITY_TERMINUS, "PLDM Terminus"}, |
| {PLDM_ENTITY_PLATFORM_EVENT_LOG, " Platform Event Log"}, |
| {PLDM_ENTITY_KEYPAD, "keypad"}, |
| {PLDM_ENTITY_SWITCH, "Switch"}, |
| {PLDM_ENTITY_PUSHBUTTON, "Pushbutton"}, |
| {PLDM_ENTITY_DISPLAY, "Display"}, |
| {PLDM_ENTITY_INDICATOR, "Indicator"}, |
| {PLDM_ENTITY_SYS_MGMT_SW, "System Management Software"}, |
| {PLDM_ENTITY_SYS_FIRMWARE, "System Firmware"}, |
| {PLDM_ENTITY_OPERATING_SYS, "Operating System"}, |
| {PLDM_ENTITY_VIRTUAL_MACHINE_MANAGER, "Virtual Machine Manager"}, |
| {PLDM_ENTITY_OS_LOADER, "OS Loader"}, |
| {PLDM_ENTITY_DEVICE_DRIVER, "Device Driver"}, |
| {PLDM_ENTITY_MGMT_CONTROLLER_FW, "Management Controller Firmware"}, |
| {PLDM_ENTITY_SYSTEM_CHASSIS, "System chassis (main enclosure)"}, |
| {PLDM_ENTITY_SUB_CHASSIS, "Sub-chassis"}, |
| {PLDM_ENTITY_DISK_DRIVE_BAY, "Disk Drive Bay"}, |
| {PLDM_ENTITY_PERIPHERAL_BAY, "Peripheral Bay"}, |
| {PLDM_ENTITY_DEVICE_BAY, "Device bay"}, |
| {PLDM_ENTITY_DOOR, "Door"}, |
| {PLDM_ENTITY_ACCESS_PANEL, "Access Panel"}, |
| {PLDM_ENTITY_COVER, "Cover"}, |
| {PLDM_ENTITY_BOARD, "Board"}, |
| {PLDM_ENTITY_CARD, "Card"}, |
| {PLDM_ENTITY_MODULE, "Module"}, |
| {PLDM_ENTITY_SYS_MGMT_MODULE, "System management module"}, |
| {PLDM_ENTITY_SYS_BOARD, "System Board"}, |
| {PLDM_ENTITY_MEMORY_BOARD, "Memory Board"}, |
| {PLDM_ENTITY_MEMORY_MODULE, "Memory Module"}, |
| {PLDM_ENTITY_PROC_MODULE, "Processor Module"}, |
| {PLDM_ENTITY_ADD_IN_CARD, "Add-in Card"}, |
| {PLDM_ENTITY_CHASSIS_FRONT_PANEL_BOARD, |
| "Chassis front panel board(control panel)"}, |
| {PLDM_ENTITY_BACK_PANEL_BOARD, "Back panel board"}, |
| {PLDM_ENTITY_POWER_MGMT, "Power management board"}, |
| {PLDM_ENTITY_POWER_SYS_BOARD, "Power system board"}, |
| {PLDM_ENTITY_DRIVE_BACKPLANE, "Drive backplane"}, |
| {PLDM_ENTITY_SYS_INTERNAL_EXPANSION_BOARD, |
| "System internal expansion board"}, |
| {PLDM_ENTITY_OTHER_SYS_BOARD, "Other system board"}, |
| {PLDM_ENTITY_CHASSIS_BACK_PANEL_BOARD, "Chassis back panel board"}, |
| {PLDM_ENTITY_PROCESSING_BLADE, "Processing blade"}, |
| {PLDM_ENTITY_CONNECTIVITY_SWITCH, "Connectivity switch"}, |
| {PLDM_ENTITY_PROC_MEMORY_MODULE, "Processor/Memory Module"}, |
| {PLDM_ENTITY_IO_MODULE, "I/O Module"}, |
| {PLDM_ENTITY_PROC_IO_MODULE, "Processor I/O Module"}, |
| {PLDM_ENTITY_COOLING_DEVICE, "Cooling device"}, |
| {PLDM_ENTITY_COOLING_SUBSYSTEM, "Cooling subsystem"}, |
| {PLDM_ENTITY_COOLING_UNIT, "Cooling Unit"}, |
| {PLDM_ENTITY_FAN, "Fan"}, |
| {PLDM_ENTITY_PELTIER_COOLING_DEVICE, "Peltier Cooling Device"}, |
| {PLDM_ENTITY_LIQUID_COOLING_DEVICE, "Liquid Cooling Device"}, |
| {PLDM_ENTITY_LIQUID_COOLING_SUBSYSTEM, "Liquid Colling Subsystem"}, |
| {PLDM_ENTITY_OTHER_STORAGE_DEVICE, "Other Storage Device"}, |
| {PLDM_ENTITY_FLOPPY_DRIVE, "Floppy Drive"}, |
| {PLDM_ENTITY_FIXED_DISK_HARD_DRIVE, "Hard Drive"}, |
| {PLDM_ENTITY_CD_DRIVE, "CD Drive"}, |
| {PLDM_ENTITY_CD_DVD_DRIVE, "CD/DVD Drive"}, |
| {PLDM_ENTITY_OTHER_SILICON_STORAGE_DEVICE, |
| "Other Silicon Storage Device"}, |
| {PLDM_ENTITY_SOLID_STATE_SRIVE, "Solid State Drive"}, |
| {PLDM_ENTITY_POWER_SUPPLY, "Power supply"}, |
| {PLDM_ENTITY_BATTERY, "Battery"}, |
| {PLDM_ENTITY_SUPER_CAPACITOR, "Super Capacitor"}, |
| {PLDM_ENTITY_POWER_CONVERTER, "Power Converter"}, |
| {PLDM_ENTITY_DC_DC_CONVERTER, "DC-DC Converter"}, |
| {PLDM_ENTITY_AC_MAINS_POWER_SUPPLY, "AC mains power supply"}, |
| {PLDM_ENTITY_DC_MAINS_POWER_SUPPLY, "DC mains power supply"}, |
| {PLDM_ENTITY_PROC, "Processor"}, |
| {PLDM_ENTITY_CHIPSET_COMPONENT, "Chipset Component"}, |
| {PLDM_ENTITY_MGMT_CONTROLLER, "Management Controller"}, |
| {PLDM_ENTITY_PERIPHERAL_CONTROLLER, "Peripheral Controller"}, |
| {PLDM_ENTITY_SEEPROM, "SEEPROM"}, |
| {PLDM_ENTITY_NVRAM_CHIP, "NVRAM Chip"}, |
| {PLDM_ENTITY_FLASH_MEMORY_CHIP, "FLASH Memory chip"}, |
| {PLDM_ENTITY_MEMORY_CHIP, "Memory Chip"}, |
| {PLDM_ENTITY_MEMORY_CONTROLLER, "Memory Controller"}, |
| {PLDM_ENTITY_NETWORK_CONTROLLER, "Network Controller"}, |
| {PLDM_ENTITY_IO_CONTROLLER, "I/O Controller"}, |
| {PLDM_ENTITY_SOUTH_BRIDGE, "South Bridge"}, |
| {PLDM_ENTITY_REAL_TIME_CLOCK, "Real Time Clock (RTC)"}, |
| {PLDM_ENTITY_FPGA_CPLD_DEVICE, "FPGA/CPLD Configurable Logic Device"}, |
| {PLDM_ENTITY_OTHER_BUS, "Other Bus"}, |
| {PLDM_ENTITY_SYS_BUS, "System Bus"}, |
| {PLDM_ENTITY_I2C_BUS, "I2C Bus"}, |
| {PLDM_ENTITY_SMBUS_BUS, "SMBus Bus"}, |
| {PLDM_ENTITY_SPI_BUS, "SPI Bus"}, |
| {PLDM_ENTITY_PCI_BUS, "PCI Bus"}, |
| {PLDM_ENTITY_PCI_EXPRESS_BUS, "PCI Express Bus"}, |
| {PLDM_ENTITY_PECI_BUS, "PECI Bus"}, |
| {PLDM_ENTITY_LPC_BUS, "LPC Bus"}, |
| {PLDM_ENTITY_USB_BUS, "USB Bus"}, |
| {PLDM_ENTITY_FIREWIRE_BUS, "FireWire Bus"}, |
| {PLDM_ENTITY_SCSI_BUS, "SCSI Bus"}, |
| {PLDM_ENTITY_SATA_SAS_BUS, "SATA/SAS Bus"}, |
| {PLDM_ENTITY_PROC_FRONT_SIDE_BUS, "Processor/Front-side Bus"}, |
| {PLDM_ENTITY_INTER_PROC_BUS, "Inter-processor Bus"}, |
| {PLDM_ENTITY_CONNECTOR, "Connector"}, |
| {PLDM_ENTITY_SLOT, "Slot"}, |
| {PLDM_ENTITY_CABLE, "Cable(electrical or optical)"}, |
| {PLDM_ENTITY_INTERCONNECT, "Interconnect"}, |
| {PLDM_ENTITY_PLUG, "Plug"}, |
| {PLDM_ENTITY_SOCKET, "Socket"}, |
| }; |
| |
| const std::map<uint16_t, std::string> stateSet = { |
| {PLDM_STATE_SET_HEALTH_STATE, "Health State"}, |
| {PLDM_STATE_SET_AVAILABILITY, "Availability"}, |
| {PLDM_STATE_SET_PREDICTIVE_CONDITION, "Predictive Condition"}, |
| {PLDM_STATE_SET_REDUNDANCY_STATUS, "Redundancy Status"}, |
| {PLDM_STATE_SET_HEALTH_REDUNDANCY_TREND, "Health/Redundancy Trend"}, |
| {PLDM_STATE_SET_GROUP_RESOURCE_LEVEL, "Group Resource Level"}, |
| {PLDM_STATE_SET_REDUNDANCY_ENTITY_ROLE, "Redundancy Entity Role"}, |
| {PLDM_STATE_SET_OPERATIONAL_STATUS, "Operational Status"}, |
| {PLDM_STATE_SET_OPERATIONAL_STRESS_STATUS, "Operational Stress Status"}, |
| {PLDM_STATE_SET_OPERATIONAL_FAULT_STATUS, "Operational Fault Status"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS, |
| "Operational Running Status"}, |
| {PLDM_STATE_SET_OPERATIONAL_CONNECTION_STATUS, |
| "Operational Connection Status"}, |
| {PLDM_STATE_SET_PRESENCE, "Presence"}, |
| {PLDM_STATE_SET_PERFORMANCE, "Performance"}, |
| {PLDM_STATE_SET_CONFIGURATION_STATE, "Configuration State"}, |
| {PLDM_STATE_SET_CHANGED_CONFIGURATION, "Changed Configuration"}, |
| {PLDM_STATE_SET_IDENTIFY_STATE, "Identify State"}, |
| {PLDM_STATE_SET_VERSION, "Version"}, |
| {PLDM_STATE_SET_ALARM_STATE, "Alarm State"}, |
| {PLDM_STATE_SET_DEVICE_INITIALIZATION, "Device Initialization"}, |
| {PLDM_STATE_SET_THERMAL_TRIP, "Thermal Trip"}, |
| {PLDM_STATE_SET_HEARTBEAT, "Heartbeat"}, |
| {PLDM_STATE_SET_LINK_STATE, "Link State"}, |
| {PLDM_STATE_SET_SMOKE_STATE, "Smoke State"}, |
| {PLDM_STATE_SET_HUMIDITY_STATE, "Humidity State"}, |
| {PLDM_STATE_SET_DOOR_STATE, "Door State"}, |
| {PLDM_STATE_SET_SWITCH_STATE, "Switch State"}, |
| {PLDM_STATE_SET_LOCK_STATE, "Lock State"}, |
| {PLDM_STATE_SET_PHYSICAL_SECURITY, "Physical Security"}, |
| {PLDM_STATE_SET_DOCK_AUTHORIZATION, "Dock Authorization"}, |
| {PLDM_STATE_SET_HW_SECURITY, "Hardware Security"}, |
| {PLDM_STATE_SET_PHYSICAL_COMM_CONNECTION, |
| "Physical Communication Connection"}, |
| {PLDM_STATE_SET_COMM_LEASH_STATUS, "Communication Leash Status"}, |
| {PLDM_STATE_SET_FOREIGN_NW_DETECTION_STATUS, |
| "Foreign Network Detection Status"}, |
| {PLDM_STATE_SET_PASSWORD_PROTECTED_ACCESS_SECURITY, |
| "Password-Protected Access Security"}, |
| {PLDM_STATE_SET_SECURITY_ACCESS_PRIVILEGE_LEVEL, |
| "Security Access –PrivilegeLevel"}, |
| {PLDM_STATE_SET_SESSION_AUDIT, "PLDM Session Audit"}, |
| {PLDM_STATE_SET_SW_TERMINATION_STATUS, "Software Termination Status"}, |
| {PLDM_STATE_SET_STORAGE_MEDIA_ACTIVITY, "Storage Media Activity"}, |
| {PLDM_STATE_SET_BOOT_RESTART_CAUSE, "Boot/Restart Cause"}, |
| {PLDM_STATE_SET_BOOT_RESTART_REQUEST, "Boot/Restart Request"}, |
| {PLDM_STATE_SET_ENTITY_BOOT_STATUS, "Entity Boot Status"}, |
| {PLDM_STATE_SET_BOOT_ERROR_STATUS, "Boot ErrorStatus"}, |
| {PLDM_STATE_SET_BOOT_PROGRESS, "Boot Progress"}, |
| {PLDM_STATE_SET_SYS_FIRMWARE_HANG, "System Firmware Hang"}, |
| {PLDM_STATE_SET_POST_ERRORS, "POST Errors"}, |
| {PLDM_STATE_SET_LOG_FILL_STATUS, "Log Fill Status"}, |
| {PLDM_STATE_SET_LOG_FILTER_STATUS, "Log Filter Status"}, |
| {PLDM_STATE_SET_LOG_TIMESTAMP_CHANGE, "Log Timestamp Change"}, |
| {PLDM_STATE_SET_INTERRUPT_REQUESTED, "Interrupt Requested"}, |
| {PLDM_STATE_SET_INTERRUPT_RECEIVED, "Interrupt Received"}, |
| {PLDM_STATE_SET_DIAGNOSTIC_INTERRUPT_REQUESTED, |
| "Diagnostic Interrupt Requested"}, |
| {PLDM_STATE_SET_DIAGNOSTIC_INTERRUPT_RECEIVED, |
| "Diagnostic Interrupt Received"}, |
| {PLDM_STATE_SET_IO_CHANNEL_CHECK_NMI_REQUESTED, |
| "I/O Channel Check NMI Requested"}, |
| {PLDM_STATE_SET_IO_CHANNEL_CHECK_NMI_RECEIVED, |
| "I/O Channel Check NMI Received"}, |
| {PLDM_STATE_SET_FATAL_NMI_REQUESTED, "Fatal NMI Requested"}, |
| {PLDM_STATE_SET_FATAL_NMI_RECEIVED, "Fatal NMI Received"}, |
| {PLDM_STATE_SET_SOFTWARE_NMI_REQUESTED, "Software NMI Requested"}, |
| {PLDM_STATE_SET_SOFTWARE_NMI_RECEIVED, "Software NMI Received"}, |
| {PLDM_STATE_SET_SMI_REQUESTED, "SMI Requested"}, |
| {PLDM_STATE_SET_SMI_RECEIVED, "SMI Received"}, |
| {PLDM_STATE_SET_PCI_PERR_REQUESTED, "PCI PERR Requested"}, |
| {PLDM_STATE_SET_PCI_PERR_RECEIVED, "PCI PERR Received"}, |
| {PLDM_STATE_SET_PCI_SERR_REQUESTED, "PCI SERR Requested "}, |
| {PLDM_STATE_SET_PCI_SERR_RECEIVED, "PCI SERR Received"}, |
| {PLDM_STATE_SET_BUS_ERROR_STATUS, "Bus Error Status"}, |
| {PLDM_STATE_SET_WATCHDOG_STATUS, "Watchdog Status"}, |
| {PLDM_STATE_SET_POWER_SUPPLY_STATE, "Power Supply State"}, |
| {PLDM_STATE_SET_DEVICE_POWER_STATE, "Device Power State"}, |
| {PLDM_STATE_SET_ACPI_POWER_STATE, "ACPI Power State"}, |
| {PLDM_STATE_SET_BACKUP_POWER_SOURCE, "Backup Power Source"}, |
| {PLDM_STATE_SET_SYSTEM_POWER_STATE, "System Power State "}, |
| {PLDM_STATE_SET_BATTERY_ACTIVITY, "Battery Activity"}, |
| {PLDM_STATE_SET_BATTERY_STATE, "Battery State"}, |
| {PLDM_STATE_SET_PROC_POWER_STATE, "Processor Power State"}, |
| {PLDM_STATE_SET_POWER_PERFORMANCE_STATE, "Power-Performance State"}, |
| {PLDM_STATE_SET_PROC_ERROR_STATUS, "Processor Error Status"}, |
| {PLDM_STATE_SET_BIST_FAILURE_STATUS, "BIST FailureStatus"}, |
| {PLDM_STATE_SET_IBIST_FAILURE_STATUS, "IBIST FailureStatus"}, |
| {PLDM_STATE_SET_PROC_HANG_IN_POST, "Processor Hang in POST"}, |
| {PLDM_STATE_SET_PROC_STARTUP_FAILURE, "Processor Startup Failure"}, |
| {PLDM_STATE_SET_UNCORRECTABLE_CPU_ERROR, "Uncorrectable CPU Error"}, |
| {PLDM_STATE_SET_MACHINE_CHECK_ERROR, "Machine Check Error"}, |
| {PLDM_STATE_SET_CORRECTED_MACHINE_CHECK, "Corrected Machine Check"}, |
| {PLDM_STATE_SET_CACHE_STATUS, "Cache Status"}, |
| {PLDM_STATE_SET_MEMORY_ERROR_STATUS, "Memory Error Status"}, |
| {PLDM_STATE_SET_REDUNDANT_MEMORY_ACTIVITY_STATUS, |
| "Redundant Memory Activity Status"}, |
| {PLDM_STATE_SET_ERROR_DETECTION_STATUS, "Error Detection Status"}, |
| {PLDM_STATE_SET_STUCK_BIT_STATUS, "Stuck Bit Status"}, |
| {PLDM_STATE_SET_SCRUB_STATUS, "Scrub Status"}, |
| {PLDM_STATE_SET_SLOT_OCCUPANCY, "Slot Occupancy"}, |
| {PLDM_STATE_SET_SLOT_STATE, "Slot State"}, |
| }; |
| |
| const std::array<std::string_view, 4> sensorInit = { |
| "noInit", "useInitPDR", "enableSensor", "disableSensor"}; |
| |
| const std::array<std::string_view, 4> effecterInit = { |
| "noInit", "useInitPDR", "enableEffecter", "disableEffecter"}; |
| |
| const std::map<uint8_t, std::string> pdrType = { |
| {PLDM_TERMINUS_LOCATOR_PDR, "Terminus Locator PDR"}, |
| {PLDM_NUMERIC_SENSOR_PDR, "Numeric Sensor PDR"}, |
| {PLDM_NUMERIC_SENSOR_INITIALIZATION_PDR, |
| "Numeric Sensor Initialization PDR"}, |
| {PLDM_STATE_SENSOR_PDR, "State Sensor PDR"}, |
| {PLDM_STATE_SENSOR_INITIALIZATION_PDR, |
| "State Sensor Initialization PDR"}, |
| {PLDM_SENSOR_AUXILIARY_NAMES_PDR, "Sensor Auxiliary Names PDR"}, |
| {PLDM_OEM_UNIT_PDR, "OEM Unit PDR"}, |
| {PLDM_OEM_STATE_SET_PDR, "OEM State Set PDR"}, |
| {PLDM_NUMERIC_EFFECTER_PDR, "Numeric Effecter PDR"}, |
| {PLDM_NUMERIC_EFFECTER_INITIALIZATION_PDR, |
| "Numeric Effecter Initialization PDR"}, |
| {PLDM_STATE_EFFECTER_PDR, "State Effecter PDR"}, |
| {PLDM_STATE_EFFECTER_INITIALIZATION_PDR, |
| "State Effecter Initialization PDR"}, |
| {PLDM_EFFECTER_AUXILIARY_NAMES_PDR, "Effecter Auxiliary Names PDR"}, |
| {PLDM_EFFECTER_OEM_SEMANTIC_PDR, "Effecter OEM Semantic PDR"}, |
| {PLDM_PDR_ENTITY_ASSOCIATION, "Entity Association PDR"}, |
| {PLDM_ENTITY_AUXILIARY_NAMES_PDR, "Entity Auxiliary Names PDR"}, |
| {PLDM_OEM_ENTITY_ID_PDR, "OEM Entity ID PDR"}, |
| {PLDM_INTERRUPT_ASSOCIATION_PDR, "Interrupt Association PDR"}, |
| {PLDM_EVENT_LOG_PDR, "PLDM Event Log PDR"}, |
| {PLDM_PDR_FRU_RECORD_SET, "FRU Record Set PDR"}, |
| {PLDM_OEM_DEVICE_PDR, "OEM Device PDR"}, |
| {PLDM_OEM_PDR, "OEM PDR"}, |
| }; |
| |
| static inline const std::map<uint8_t, std::string> setThermalTrip{ |
| {PLDM_STATE_SET_THERMAL_TRIP_STATUS_NORMAL, "Normal"}, |
| {PLDM_STATE_SET_THERMAL_TRIP_STATUS_THERMAL_TRIP, "Thermal Trip"}}; |
| |
| static inline const std::map<uint8_t, std::string> setIdentifyState{ |
| {PLDM_STATE_SET_IDENTIFY_STATE_UNASSERTED, "Identify State Unasserted"}, |
| {PLDM_STATE_SET_IDENTIFY_STATE_ASSERTED, "Identify State Asserted"}}; |
| |
| static inline const std::map<uint8_t, std::string> setBootProgressState{ |
| {PLDM_STATE_SET_BOOT_PROG_STATE_NOT_ACTIVE, "Boot Not Active"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_COMPLETED, "Boot Completed"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_MEM_INITIALIZATION, |
| "Memory Initialization"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_SEC_PROC_INITIALIZATION, |
| "Secondary Processor(s) Initialization"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_PCI_RESORUCE_CONFIG, |
| "PCI Resource Configuration"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_STARTING_OP_SYS, |
| "Starting Operating System"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_BASE_BOARD_INITIALIZATION, |
| "Baseboard Initialization"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_PRIMARY_PROC_INITIALIZATION, |
| "Primary Processor Initialization"}, |
| {PLDM_STATE_SET_BOOT_PROG_STATE_OSSTART, "OSStart"}}; |
| |
| static inline const std::map<uint8_t, std::string> setOpFaultStatus{ |
| {PLDM_STATE_SET_OPERATIONAL_FAULT_STATUS_NORMAL, "Normal"}, |
| {PLDM_STATE_SET_OPERATIONAL_FAULT_STATUS_STRESSED, "Stressed"}}; |
| |
| static inline const std::map<uint8_t, std::string> setSysPowerState{ |
| {PLDM_STATE_SET_SYS_POWER_STATE_OFF_SOFT_GRACEFUL, |
| "Off-Soft Graceful"}}; |
| |
| static inline const std::map<uint8_t, std::string> setSWTerminationStatus{ |
| {PLDM_SW_TERM_GRACEFUL_RESTART_REQUESTED, |
| "Graceful Restart Requested"}}; |
| |
| static inline const std::map<uint8_t, std::string> setAvailability{ |
| {PLDM_STATE_SET_AVAILABILITY_REBOOTING, "Rebooting"}}; |
| |
| static inline const std::map<uint8_t, std::string> setHealthState{ |
| {PLDM_STATE_SET_HEALTH_STATE_NORMAL, "Normal"}, |
| {PLDM_STATE_SET_HEALTH_STATE_NON_CRITICAL, "Non-Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_CRITICAL, "Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_FATAL, "Fatal"}, |
| {PLDM_STATE_SET_HEALTH_STATE_UPPER_NON_CRITICAL, "Upper Non-Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_LOWER_NON_CRITICAL, "Lower Non-Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_UPPER_CRITICAL, "Upper Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_LOWER_CRITICAL, "Lower Critical"}, |
| {PLDM_STATE_SET_HEALTH_STATE_UPPER_FATAL, "Upper Fatal"}, |
| {PLDM_STATE_SET_HEALTH_STATE_LOWER_FATAL, "Lower Fatal"}}; |
| |
| static inline const std::map<uint8_t, std::string> |
| setOperationalRunningState{ |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_STARTING, "Starting"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_STOPPING, "Stopping"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_STOPPED, "Stopped"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_IN_SERVICE, |
| "In Service"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_ABORTED, "Aborted"}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS_DORMANT, "Dormant"}}; |
| |
| static inline const std::map<uint16_t, const std::map<uint8_t, std::string>> |
| populatePStateMaps{ |
| {PLDM_STATE_SET_THERMAL_TRIP, setThermalTrip}, |
| {PLDM_STATE_SET_IDENTIFY_STATE, setIdentifyState}, |
| {PLDM_STATE_SET_BOOT_PROGRESS, setBootProgressState}, |
| {PLDM_STATE_SET_OPERATIONAL_FAULT_STATUS, setOpFaultStatus}, |
| {PLDM_STATE_SET_SYSTEM_POWER_STATE, setSysPowerState}, |
| {PLDM_STATE_SET_SW_TERMINATION_STATUS, setSWTerminationStatus}, |
| {PLDM_STATE_SET_AVAILABILITY, setAvailability}, |
| {PLDM_STATE_SET_HEALTH_STATE, setHealthState}, |
| {PLDM_STATE_SET_OPERATIONAL_RUNNING_STATUS, |
| setOperationalRunningState}, |
| }; |
| |
| const std::map<std::string, uint8_t> strToPdrType = { |
| {"terminuslocator", PLDM_TERMINUS_LOCATOR_PDR}, |
| {"statesensor", PLDM_STATE_SENSOR_PDR}, |
| {"numericeffecter", PLDM_NUMERIC_EFFECTER_PDR}, |
| {"stateeffecter", PLDM_STATE_EFFECTER_PDR}, |
| {"entityassociation", PLDM_PDR_ENTITY_ASSOCIATION}, |
| {"frurecord", PLDM_PDR_FRU_RECORD_SET}, |
| // Add other types |
| }; |
| |
| bool isLogicalBitSet(const uint16_t entity_type) |
| { |
| return entity_type & 0x8000; |
| } |
| |
| uint16_t getEntityTypeForLogicalEntity(const uint16_t logical_entity_type) |
| { |
| return logical_entity_type & 0x7FFF; |
| } |
| |
| std::string getEntityName(pldm::pdr::EntityType type) |
| { |
| uint16_t entityNumber = type; |
| std::string entityName = "[Physical] "; |
| |
| if (isLogicalBitSet(type)) |
| { |
| entityName = "[Logical] "; |
| entityNumber = getEntityTypeForLogicalEntity(type); |
| } |
| |
| try |
| { |
| return entityName + entityType.at(entityNumber); |
| } |
| catch (const std::out_of_range& e) |
| { |
| auto OemString = |
| std::to_string(static_cast<unsigned>(entityNumber)); |
| if (type >= PLDM_OEM_ENTITY_TYPE_START && |
| type <= PLDM_OEM_ENTITY_TYPE_END) |
| { |
| #ifdef OEM_IBM |
| if (OemIBMEntityType.contains(entityNumber)) |
| { |
| return entityName + OemIBMEntityType.at(entityNumber) + |
| "(OEM)"; |
| } |
| #endif |
| return entityName + OemString + "(OEM)"; |
| } |
| return OemString; |
| } |
| } |
| |
| std::string getStateSetName(uint16_t id) |
| { |
| auto typeString = std::to_string(id); |
| try |
| { |
| return stateSet.at(id) + "(" + typeString + ")"; |
| } |
| catch (const std::out_of_range& e) |
| { |
| return typeString; |
| } |
| } |
| |
| std::vector<std::string> |
| getStateSetPossibleStateNames(uint16_t stateId, |
| const std::vector<uint8_t>& value) |
| { |
| std::vector<std::string> data{}; |
| std::map<uint8_t, std::string> stateNameMaps; |
| |
| for (auto& s : value) |
| { |
| std::map<uint8_t, std::string> stateNameMaps; |
| auto pstr = std::to_string(s); |
| |
| #ifdef OEM_IBM |
| if (stateId >= PLDM_OEM_STATE_SET_ID_START && |
| stateId < PLDM_OEM_STATE_SET_ID_END) |
| { |
| if (populateOemIBMStateMaps.contains(stateId)) |
| { |
| const std::map<uint8_t, std::string> stateNames = |
| populateOemIBMStateMaps.at(stateId); |
| stateNameMaps.insert(stateNames.begin(), stateNames.end()); |
| } |
| } |
| #endif |
| if (populatePStateMaps.contains(stateId)) |
| { |
| const std::map<uint8_t, std::string> stateNames = |
| populatePStateMaps.at(stateId); |
| stateNameMaps.insert(stateNames.begin(), stateNames.end()); |
| } |
| if (stateNameMaps.contains(s)) |
| { |
| data.push_back(stateNameMaps.at(s) + "(" + pstr + ")"); |
| } |
| else |
| { |
| data.push_back(pstr); |
| } |
| } |
| return data; |
| } |
| |
| std::string getPDRType(uint8_t type) |
| { |
| auto typeString = std::to_string(type); |
| try |
| { |
| return pdrType.at(type); |
| } |
| catch (const std::out_of_range& e) |
| { |
| return typeString; |
| } |
| } |
| |
| void printCommonPDRHeader(const pldm_pdr_hdr* hdr, ordered_json& output) |
| { |
| output["recordHandle"] = hdr->record_handle; |
| output["PDRHeaderVersion"] = unsigned(hdr->version); |
| output["PDRType"] = getPDRType(hdr->type); |
| output["recordChangeNumber"] = hdr->record_change_num; |
| output["dataLength"] = hdr->length; |
| } |
| |
| std::vector<uint8_t> printPossibleStates(uint8_t possibleStatesSize, |
| const bitfield8_t* states) |
| { |
| uint8_t possibleStatesPos{}; |
| std::vector<uint8_t> data{}; |
| auto printStates = [&possibleStatesPos, &data](const bitfield8_t& val) { |
| std::stringstream pstates; |
| for (int i = 0; i < CHAR_BIT; i++) |
| { |
| if (val.byte & (1 << i)) |
| { |
| pstates << (possibleStatesPos * CHAR_BIT + i); |
| data.push_back( |
| static_cast<uint8_t>(std::stoi(pstates.str()))); |
| pstates.str(""); |
| } |
| } |
| possibleStatesPos++; |
| }; |
| std::for_each(states, states + possibleStatesSize, printStates); |
| return data; |
| } |
| |
| void printStateSensorPDR(const uint8_t* data, ordered_json& output) |
| { |
| auto pdr = reinterpret_cast<const pldm_state_sensor_pdr*>(data); |
| output["PLDMTerminusHandle"] = pdr->terminus_handle; |
| output["sensorID"] = pdr->sensor_id; |
| output["entityType"] = getEntityName(pdr->entity_type); |
| output["entityInstanceNumber"] = pdr->entity_instance; |
| output["containerID"] = pdr->container_id; |
| output["sensorInit"] = sensorInit[pdr->sensor_init]; |
| output["sensorAuxiliaryNamesPDR"] = |
| (pdr->sensor_auxiliary_names_pdr ? true : false); |
| output["compositeSensorCount"] = unsigned(pdr->composite_sensor_count); |
| |
| auto statesPtr = pdr->possible_states; |
| auto compCount = pdr->composite_sensor_count; |
| |
| while (compCount--) |
| { |
| auto state = reinterpret_cast<const state_sensor_possible_states*>( |
| statesPtr); |
| output.emplace(("stateSetID[" + std::to_string(compCount) + "]"), |
| getStateSetName(state->state_set_id)); |
| output.emplace( |
| ("possibleStatesSize[" + std::to_string(compCount) + "]"), |
| state->possible_states_size); |
| output.emplace( |
| ("possibleStates[" + std::to_string(compCount) + "]"), |
| getStateSetPossibleStateNames( |
| state->state_set_id, |
| printPossibleStates(state->possible_states_size, |
| state->states))); |
| |
| if (compCount) |
| { |
| statesPtr += sizeof(state_sensor_possible_states) + |
| state->possible_states_size - 1; |
| } |
| } |
| } |
| |
| void printPDRFruRecordSet(uint8_t* data, ordered_json& output) |
| { |
| if (data == NULL) |
| { |
| return; |
| } |
| |
| data += sizeof(pldm_pdr_hdr); |
| pldm_pdr_fru_record_set* pdr = |
| reinterpret_cast<pldm_pdr_fru_record_set*>(data); |
| if (!pdr) |
| { |
| std::cerr << "Failed to get the FRU record set PDR" << std::endl; |
| return; |
| } |
| |
| output["PLDMTerminusHandle"] = unsigned(pdr->terminus_handle); |
| output["FRURecordSetIdentifier"] = unsigned(pdr->fru_rsi); |
| output["entityType"] = getEntityName(pdr->entity_type); |
| output["entityInstanceNumber"] = unsigned(pdr->entity_instance_num); |
| output["containerID"] = unsigned(pdr->container_id); |
| } |
| |
| void printPDREntityAssociation(uint8_t* data, ordered_json& output) |
| { |
| const std::map<uint8_t, const char*> assocationType = { |
| {PLDM_ENTITY_ASSOCIAION_PHYSICAL, "Physical"}, |
| {PLDM_ENTITY_ASSOCIAION_LOGICAL, "Logical"}, |
| }; |
| |
| if (data == NULL) |
| { |
| return; |
| } |
| |
| data += sizeof(pldm_pdr_hdr); |
| pldm_pdr_entity_association* pdr = |
| reinterpret_cast<pldm_pdr_entity_association*>(data); |
| if (!pdr) |
| { |
| std::cerr << "Failed to get the PDR eneity association" |
| << std::endl; |
| return; |
| } |
| |
| output["containerID"] = int(pdr->container_id); |
| if (assocationType.contains(pdr->association_type)) |
| { |
| output["associationType"] = |
| assocationType.at(pdr->association_type); |
| } |
| else |
| { |
| std::cout << "Get associationType failed.\n"; |
| } |
| output["containerEntityType"] = |
| getEntityName(pdr->container.entity_type); |
| output["containerEntityInstanceNumber"] = |
| int(pdr->container.entity_instance_num); |
| output["containerEntityContainerID"] = |
| int(pdr->container.entity_container_id); |
| output["containedEntityCount"] = |
| static_cast<unsigned>(pdr->num_children); |
| |
| auto child = reinterpret_cast<pldm_entity*>(&pdr->children[0]); |
| for (int i = 0; i < pdr->num_children; ++i) |
| { |
| output.emplace("containedEntityType[" + std::to_string(i + 1) + "]", |
| getEntityName(child->entity_type)); |
| output.emplace("containedEntityInstanceNumber[" + |
| std::to_string(i + 1) + "]", |
| unsigned(child->entity_instance_num)); |
| output.emplace("containedEntityContainerID[" + |
| std::to_string(i + 1) + "]", |
| unsigned(child->entity_container_id)); |
| |
| ++child; |
| } |
| } |
| |
| void printNumericEffecterPDR(uint8_t* data, ordered_json& output) |
| { |
| struct pldm_numeric_effecter_value_pdr* pdr = |
| (struct pldm_numeric_effecter_value_pdr*)data; |
| if (!pdr) |
| { |
| std::cerr << "Failed to get numeric effecter PDR" << std::endl; |
| return; |
| } |
| |
| output["PLDMTerminusHandle"] = int(pdr->terminus_handle); |
| output["effecterID"] = int(pdr->effecter_id); |
| output["entityType"] = int(pdr->entity_type); |
| output["entityInstanceNumber"] = int(pdr->entity_instance); |
| output["containerID"] = int(pdr->container_id); |
| output["effecterSemanticID"] = int(pdr->effecter_semantic_id); |
| output["effecterInit"] = unsigned(pdr->effecter_init); |
| output["effecterAuxiliaryNames"] = |
| (unsigned(pdr->effecter_auxiliary_names) ? true : false); |
| output["baseUnit"] = unsigned(pdr->base_unit); |
| output["unitModifier"] = unsigned(pdr->unit_modifier); |
| output["rateUnit"] = unsigned(pdr->rate_unit); |
| output["baseOEMUnitHandle"] = unsigned(pdr->base_oem_unit_handle); |
| output["auxUnit"] = unsigned(pdr->aux_unit); |
| output["auxUnitModifier"] = unsigned(pdr->aux_unit_modifier); |
| output["auxrateUnit"] = unsigned(pdr->aux_rate_unit); |
| output["auxOEMUnitHandle"] = unsigned(pdr->aux_oem_unit_handle); |
| output["isLinear"] = (unsigned(pdr->is_linear) ? true : false); |
| output["effecterDataSize"] = unsigned(pdr->effecter_data_size); |
| output["resolution"] = unsigned(pdr->resolution); |
| output["offset"] = unsigned(pdr->offset); |
| output["accuracy"] = unsigned(pdr->accuracy); |
| output["plusTolerance"] = unsigned(pdr->plus_tolerance); |
| output["minusTolerance"] = unsigned(pdr->minus_tolerance); |
| output["stateTransitionInterval"] = |
| unsigned(pdr->state_transition_interval); |
| output["TransitionInterval"] = unsigned(pdr->transition_interval); |
| |
| switch (pdr->effecter_data_size) |
| { |
| case PLDM_EFFECTER_DATA_SIZE_UINT8: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_u8); |
| output["minSettable"] = unsigned(pdr->min_settable.value_u8); |
| break; |
| case PLDM_EFFECTER_DATA_SIZE_SINT8: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_s8); |
| output["minSettable"] = unsigned(pdr->min_settable.value_s8); |
| break; |
| case PLDM_EFFECTER_DATA_SIZE_UINT16: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_u16); |
| output["minSettable"] = unsigned(pdr->min_settable.value_u16); |
| break; |
| case PLDM_EFFECTER_DATA_SIZE_SINT16: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_s16); |
| output["minSettable"] = unsigned(pdr->min_settable.value_s16); |
| break; |
| case PLDM_EFFECTER_DATA_SIZE_UINT32: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_u32); |
| output["minSettable"] = unsigned(pdr->min_settable.value_u32); |
| break; |
| case PLDM_EFFECTER_DATA_SIZE_SINT32: |
| output["maxSettable"] = unsigned(pdr->max_settable.value_s32); |
| output["minSettable"] = unsigned(pdr->min_settable.value_s32); |
| break; |
| default: |
| break; |
| } |
| |
| output["rangeFieldFormat"] = unsigned(pdr->range_field_format); |
| output["rangeFieldSupport"] = unsigned(pdr->range_field_support.byte); |
| |
| switch (pdr->range_field_format) |
| { |
| case PLDM_RANGE_FIELD_FORMAT_UINT8: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_u8); |
| output["normalMax"] = unsigned(pdr->normal_max.value_u8); |
| output["normalMin"] = unsigned(pdr->normal_min.value_u8); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_u8); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_u8); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_SINT8: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_s8); |
| output["normalMax"] = unsigned(pdr->normal_max.value_s8); |
| output["normalMin"] = unsigned(pdr->normal_min.value_s8); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_s8); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_s8); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_UINT16: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_u16); |
| output["normalMax"] = unsigned(pdr->normal_max.value_u16); |
| output["normalMin"] = unsigned(pdr->normal_min.value_u16); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_u16); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_u16); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_SINT16: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_s16); |
| output["normalMax"] = unsigned(pdr->normal_max.value_s16); |
| output["normalMin"] = unsigned(pdr->normal_min.value_s16); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_s16); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_s16); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_UINT32: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_u32); |
| output["normalMax"] = unsigned(pdr->normal_max.value_u32); |
| output["normalMin"] = unsigned(pdr->normal_min.value_u32); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_u32); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_u32); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_SINT32: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_s32); |
| output["normalMax"] = unsigned(pdr->normal_max.value_s32); |
| output["normalMin"] = unsigned(pdr->normal_min.value_s32); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_s32); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_s32); |
| break; |
| case PLDM_RANGE_FIELD_FORMAT_REAL32: |
| output["nominalValue"] = unsigned(pdr->nominal_value.value_f32); |
| output["normalMax"] = unsigned(pdr->normal_max.value_f32); |
| output["normalMin"] = unsigned(pdr->normal_min.value_f32); |
| output["ratedMax"] = unsigned(pdr->rated_max.value_f32); |
| output["ratedMin"] = unsigned(pdr->rated_min.value_f32); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| void printStateEffecterPDR(const uint8_t* data, ordered_json& output) |
| { |
| auto pdr = reinterpret_cast<const pldm_state_effecter_pdr*>(data); |
| |
| output["PLDMTerminusHandle"] = pdr->terminus_handle; |
| output["effecterID"] = pdr->effecter_id; |
| output["entityType"] = getEntityName(pdr->entity_type); |
| output["entityInstanceNumber"] = pdr->entity_instance; |
| output["containerID"] = pdr->container_id; |
| output["effecterSemanticID"] = pdr->effecter_semantic_id; |
| output["effecterInit"] = effecterInit[pdr->effecter_init]; |
| output["effecterDescriptionPDR"] = (pdr->has_description_pdr ? true |
| : false); |
| output["compositeEffecterCount"] = |
| unsigned(pdr->composite_effecter_count); |
| |
| auto statesPtr = pdr->possible_states; |
| auto compEffCount = pdr->composite_effecter_count; |
| |
| while (compEffCount--) |
| { |
| auto state = |
| reinterpret_cast<const state_effecter_possible_states*>( |
| statesPtr); |
| output.emplace(("stateSetID[" + std::to_string(compEffCount) + "]"), |
| getStateSetName(state->state_set_id)); |
| output.emplace( |
| ("possibleStatesSize[" + std::to_string(compEffCount) + "]"), |
| state->possible_states_size); |
| output.emplace( |
| ("possibleStates[" + std::to_string(compEffCount) + "]"), |
| getStateSetPossibleStateNames( |
| state->state_set_id, |
| printPossibleStates(state->possible_states_size, |
| state->states))); |
| |
| if (compEffCount) |
| { |
| statesPtr += sizeof(state_effecter_possible_states) + |
| state->possible_states_size - 1; |
| } |
| } |
| } |
| |
| bool checkTerminusHandle(const uint8_t* data, |
| std::optional<uint16_t> terminusHandle) |
| { |
| struct pldm_pdr_hdr* pdr = (struct pldm_pdr_hdr*)data; |
| |
| if (pdr->type == PLDM_TERMINUS_LOCATOR_PDR) |
| { |
| auto tlpdr = |
| reinterpret_cast<const pldm_terminus_locator_pdr*>(data); |
| |
| if (tlpdr->terminus_handle != terminusHandle) |
| { |
| return true; |
| } |
| } |
| else if (pdr->type == PLDM_STATE_SENSOR_PDR) |
| { |
| auto sensor = reinterpret_cast<const pldm_state_sensor_pdr*>(data); |
| |
| if (sensor->terminus_handle != terminusHandle) |
| { |
| return true; |
| } |
| } |
| else if (pdr->type == PLDM_NUMERIC_EFFECTER_PDR) |
| { |
| auto numericEffecter = |
| reinterpret_cast<const pldm_numeric_effecter_value_pdr*>(data); |
| |
| if (numericEffecter->terminus_handle != terminusHandle) |
| { |
| return true; |
| } |
| } |
| |
| else if (pdr->type == PLDM_STATE_EFFECTER_PDR) |
| { |
| auto stateEffecter = |
| reinterpret_cast<const pldm_state_effecter_pdr*>(data); |
| if (stateEffecter->terminus_handle != terminusHandle) |
| { |
| return true; |
| } |
| } |
| else if (pdr->type == PLDM_PDR_FRU_RECORD_SET) |
| { |
| data += sizeof(pldm_pdr_hdr); |
| auto fru = reinterpret_cast<const pldm_pdr_fru_record_set*>(data); |
| |
| if (fru->terminus_handle != terminusHandle) |
| { |
| return true; |
| } |
| } |
| else |
| { |
| // Entity association PDRs does not have terminus handle |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void printTerminusLocatorPDR(const uint8_t* data, ordered_json& output) |
| { |
| const std::array<std::string_view, 4> terminusLocatorType = { |
| "UID", "MCTP_EID", "SMBusRelative", "systemSoftware"}; |
| |
| auto pdr = reinterpret_cast<const pldm_terminus_locator_pdr*>(data); |
| |
| output["PLDMTerminusHandle"] = pdr->terminus_handle; |
| output["validity"] = (pdr->validity ? "valid" : "notValid"); |
| output["TID"] = unsigned(pdr->tid); |
| output["containerID"] = pdr->container_id; |
| output["terminusLocatorType"] = |
| terminusLocatorType[pdr->terminus_locator_type]; |
| output["terminusLocatorValueSize"] = |
| unsigned(pdr->terminus_locator_value_size); |
| |
| if (pdr->terminus_locator_type == PLDM_TERMINUS_LOCATOR_TYPE_MCTP_EID) |
| { |
| auto locatorValue = |
| reinterpret_cast<const pldm_terminus_locator_type_mctp_eid*>( |
| pdr->terminus_locator_value); |
| output["EID"] = unsigned(locatorValue->eid); |
| } |
| } |
| |
| std::optional<uint16_t> getTerminusHandle(uint8_t* data, |
| std::optional<uint8_t> tid) |
| { |
| struct pldm_pdr_hdr* pdr = (struct pldm_pdr_hdr*)data; |
| if (pdr->type == PLDM_TERMINUS_LOCATOR_PDR) |
| { |
| auto pdr = reinterpret_cast<const pldm_terminus_locator_pdr*>(data); |
| if (pdr->tid == tid) |
| { |
| handleFound = true; |
| return pdr->terminus_handle; |
| } |
| } |
| return std::nullopt; |
| } |
| |
| void printPDRMsg(uint32_t& nextRecordHndl, const uint16_t respCnt, |
| uint8_t* data, std::optional<uint16_t> terminusHandle) |
| { |
| if (data == NULL) |
| { |
| std::cerr << "Failed to get PDR message" << std::endl; |
| return; |
| } |
| |
| ordered_json output; |
| output["nextRecordHandle"] = nextRecordHndl; |
| output["responseCount"] = respCnt; |
| |
| struct pldm_pdr_hdr* pdr = (struct pldm_pdr_hdr*)data; |
| if (!pdr) |
| { |
| return; |
| } |
| |
| if (!pdrRecType.empty()) |
| { |
| // Need to return if the requested PDR type |
| // is not supported |
| if (!strToPdrType.contains(pdrRecType)) |
| { |
| std::cerr << "PDR type '" << pdrRecType |
| << "' is not supported or invalid\n"; |
| // PDR type not supported, setting next record handle to 0 |
| // to avoid looping through all PDR records |
| nextRecordHndl = 0; |
| return; |
| } |
| |
| // Do not print PDR record if the current record |
| // PDR type does not match with requested type |
| if (pdr->type != strToPdrType.at(pdrRecType)) |
| { |
| return; |
| } |
| } |
| |
| if (pdrTerminus.has_value()) |
| { |
| if (checkTerminusHandle(data, terminusHandle)) |
| { |
| std::cerr << "The Terminus handle doesn't match return" |
| << std::endl; |
| return; |
| } |
| } |
| |
| printCommonPDRHeader(pdr, output); |
| |
| switch (pdr->type) |
| { |
| case PLDM_TERMINUS_LOCATOR_PDR: |
| printTerminusLocatorPDR(data, output); |
| break; |
| case PLDM_STATE_SENSOR_PDR: |
| printStateSensorPDR(data, output); |
| break; |
| case PLDM_NUMERIC_EFFECTER_PDR: |
| printNumericEffecterPDR(data, output); |
| break; |
| case PLDM_STATE_EFFECTER_PDR: |
| printStateEffecterPDR(data, output); |
| break; |
| case PLDM_PDR_ENTITY_ASSOCIATION: |
| printPDREntityAssociation(data, output); |
| break; |
| case PLDM_PDR_FRU_RECORD_SET: |
| printPDRFruRecordSet(data, output); |
| break; |
| default: |
| break; |
| } |
| pldmtool::helper::DisplayInJson(output); |
| } |
| |
| private: |
| bool optTIDSet = false; |
| uint32_t recordHandle; |
| bool allPDRs; |
| std::string pdrRecType; |
| std::optional<uint8_t> pdrTerminus; |
| std::optional<uint16_t> terminusHandle; |
| bool handleFound = false; |
| CLI::Option* getPDRGroupOption = nullptr; |
| }; |
| |
| class SetStateEffecter : public CommandInterface |
| { |
| public: |
| ~SetStateEffecter() = default; |
| SetStateEffecter() = delete; |
| SetStateEffecter(const SetStateEffecter&) = delete; |
| SetStateEffecter(SetStateEffecter&&) = default; |
| SetStateEffecter& operator=(const SetStateEffecter&) = delete; |
| SetStateEffecter& operator=(SetStateEffecter&&) = delete; |
| |
| // compositeEffecterCount(value: 0x01 to 0x08) * stateField(2) |
| static constexpr auto maxEffecterDataSize = 16; |
| |
| // compositeEffecterCount(value: 0x01 to 0x08) |
| static constexpr auto minEffecterCount = 1; |
| static constexpr auto maxEffecterCount = 8; |
| explicit SetStateEffecter(const char* type, const char* name, |
| CLI::App* app) : |
| CommandInterface(type, name, app) |
| { |
| app->add_option( |
| "-i, --id", effecterId, |
| "A handle that is used to identify and access the effecter") |
| ->required(); |
| app->add_option("-c, --count", effecterCount, |
| "The number of individual sets of effecter information") |
| ->required(); |
| app->add_option( |
| "-d,--data", effecterData, |
| "Set effecter state data\n" |
| "eg: requestSet0 effecterState0 noChange1 dummyState1 ...") |
| ->required(); |
| } |
| |
| std::pair<int, std::vector<uint8_t>> createRequestMsg() override |
| { |
| std::vector<uint8_t> requestMsg( |
| sizeof(pldm_msg_hdr) + PLDM_SET_STATE_EFFECTER_STATES_REQ_BYTES); |
| auto request = reinterpret_cast<pldm_msg*>(requestMsg.data()); |
| |
| if (effecterCount > maxEffecterCount || |
| effecterCount < minEffecterCount) |
| { |
| std::cerr << "Request Message Error: effecterCount size " |
| << effecterCount << "is invalid\n"; |
| auto rc = PLDM_ERROR_INVALID_DATA; |
| return {rc, requestMsg}; |
| } |
| |
| if (effecterData.size() > maxEffecterDataSize) |
| { |
| std::cerr << "Request Message Error: effecterData size " |
| << effecterData.size() << "is invalid\n"; |
| auto rc = PLDM_ERROR_INVALID_DATA; |
| return {rc, requestMsg}; |
| } |
| |
| auto stateField = parseEffecterData(effecterData, effecterCount); |
| if (!stateField) |
| { |
| std::cerr << "Failed to parse effecter data, effecterCount size " |
| << effecterCount << "\n"; |
| auto rc = PLDM_ERROR_INVALID_DATA; |
| return {rc, requestMsg}; |
| } |
| |
| auto rc = encode_set_state_effecter_states_req( |
| instanceId, effecterId, effecterCount, stateField->data(), request); |
| return {rc, requestMsg}; |
| } |
| |
| void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override |
| { |
| uint8_t completionCode = 0; |
| auto rc = decode_set_state_effecter_states_resp( |
| responsePtr, payloadLength, &completionCode); |
| |
| if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS) |
| { |
| std::cerr << "Response Message Error: " |
| << "rc=" << rc << ",cc=" << (int)completionCode << "\n"; |
| return; |
| } |
| |
| ordered_json data; |
| data["Response"] = "SUCCESS"; |
| pldmtool::helper::DisplayInJson(data); |
| } |
| |
| private: |
| uint16_t effecterId; |
| uint8_t effecterCount; |
| std::vector<uint8_t> effecterData; |
| }; |
| |
| class SetNumericEffecterValue : public CommandInterface |
| { |
| public: |
| ~SetNumericEffecterValue() = default; |
| SetNumericEffecterValue() = delete; |
| SetNumericEffecterValue(const SetNumericEffecterValue&) = delete; |
| SetNumericEffecterValue(SetNumericEffecterValue&&) = default; |
| SetNumericEffecterValue& operator=(const SetNumericEffecterValue&) = delete; |
| SetNumericEffecterValue& operator=(SetNumericEffecterValue&&) = delete; |
| |
| explicit SetNumericEffecterValue(const char* type, const char* name, |
| CLI::App* app) : |
| CommandInterface(type, name, app) |
| { |
| app->add_option( |
| "-i, --id", effecterId, |
| "A handle that is used to identify and access the effecter") |
| ->required(); |
| app->add_option("-s, --size", effecterDataSize, |
| "The bit width and format of the setting value for the " |
| "effecter. enum value: {uint8, sint8, uint16, sint16, " |
| "uint32, sint32}\n") |
| ->required(); |
| app->add_option("-d,--data", maxEffecterValue, |
| "The setting value of numeric effecter being " |
| "requested\n") |
| ->required(); |
| } |
| |
| std::pair<int, std::vector<uint8_t>> createRequestMsg() override |
| { |
| std::vector<uint8_t> requestMsg( |
| sizeof(pldm_msg_hdr) + |
| PLDM_SET_NUMERIC_EFFECTER_VALUE_MIN_REQ_BYTES + 3); |
| |
| uint8_t* effecterValue = (uint8_t*)&maxEffecterValue; |
| |
| auto request = reinterpret_cast<pldm_msg*>(requestMsg.data()); |
| size_t payload_length = PLDM_SET_NUMERIC_EFFECTER_VALUE_MIN_REQ_BYTES; |
| |
| if (effecterDataSize == PLDM_EFFECTER_DATA_SIZE_UINT16 || |
| effecterDataSize == PLDM_EFFECTER_DATA_SIZE_SINT16) |
| { |
| payload_length = PLDM_SET_NUMERIC_EFFECTER_VALUE_MIN_REQ_BYTES + 1; |
| } |
| if (effecterDataSize == PLDM_EFFECTER_DATA_SIZE_UINT32 || |
| effecterDataSize == PLDM_EFFECTER_DATA_SIZE_SINT32) |
| { |
| payload_length = PLDM_SET_NUMERIC_EFFECTER_VALUE_MIN_REQ_BYTES + 3; |
| } |
| auto rc = encode_set_numeric_effecter_value_req( |
| 0, effecterId, effecterDataSize, effecterValue, request, |
| payload_length); |
| |
| return {rc, requestMsg}; |
| } |
| |
| void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override |
| { |
| uint8_t completionCode = 0; |
| auto rc = decode_set_numeric_effecter_value_resp( |
| responsePtr, payloadLength, &completionCode); |
| |
| if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS) |
| { |
| std::cerr << "Response Message Error: " |
| << "rc=" << rc << ",cc=" << (int)completionCode |
| << std::endl; |
| return; |
| } |
| |
| ordered_json data; |
| data["Response"] = "SUCCESS"; |
| pldmtool::helper::DisplayInJson(data); |
| } |
| |
| private: |
| uint16_t effecterId; |
| uint8_t effecterDataSize; |
| uint64_t maxEffecterValue; |
| }; |
| |
| class GetStateSensorReadings : public CommandInterface |
| { |
| public: |
| ~GetStateSensorReadings() = default; |
| GetStateSensorReadings() = delete; |
| GetStateSensorReadings(const GetStateSensorReadings&) = delete; |
| GetStateSensorReadings(GetStateSensorReadings&&) = default; |
| GetStateSensorReadings& operator=(const GetStateSensorReadings&) = delete; |
| GetStateSensorReadings& operator=(GetStateSensorReadings&&) = delete; |
| |
| explicit GetStateSensorReadings(const char* type, const char* name, |
| CLI::App* app) : |
| CommandInterface(type, name, app) |
| { |
| app->add_option( |
| "-i, --sensor_id", sensorId, |
| "Sensor ID that is used to identify and access the sensor") |
| ->required(); |
| app->add_option("-r, --rearm", sensorRearm, |
| "Each bit location in this field corresponds to a " |
| "particular sensor") |
| ->required(); |
| } |
| |
| std::pair<int, std::vector<uint8_t>> createRequestMsg() override |
| { |
| std::vector<uint8_t> requestMsg( |
| sizeof(pldm_msg_hdr) + PLDM_GET_STATE_SENSOR_READINGS_REQ_BYTES); |
| auto request = reinterpret_cast<pldm_msg*>(requestMsg.data()); |
| |
| uint8_t reserved = 0; |
| bitfield8_t bf; |
| bf.byte = sensorRearm; |
| auto rc = encode_get_state_sensor_readings_req(instanceId, sensorId, bf, |
| reserved, request); |
| |
| return {rc, requestMsg}; |
| } |
| |
| void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override |
| { |
| uint8_t completionCode = 0; |
| uint8_t compSensorCount = 0; |
| std::array<get_sensor_state_field, 8> stateField{}; |
| auto rc = decode_get_state_sensor_readings_resp( |
| responsePtr, payloadLength, &completionCode, &compSensorCount, |
| stateField.data()); |
| |
| if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS) |
| { |
| std::cerr << "Response Message Error: " |
| << "rc=" << rc << ",cc=" << (int)completionCode |
| << std::endl; |
| return; |
| } |
| ordered_json output; |
| output["compositeSensorCount"] = (int)compSensorCount; |
| |
| for (size_t i = 0; i < compSensorCount; i++) |
| { |
| if (sensorOpState.contains(stateField[i].sensor_op_state)) |
| { |
| output.emplace(("sensorOpState[" + std::to_string(i) + "]"), |
| sensorOpState.at(stateField[i].sensor_op_state)); |
| } |
| |
| if (sensorPresState.contains(stateField[i].present_state)) |
| { |
| output.emplace(("presentState[" + std::to_string(i) + "]"), |
| sensorPresState.at(stateField[i].present_state)); |
| } |
| |
| if (sensorPresState.contains(stateField[i].previous_state)) |
| { |
| output.emplace( |
| ("previousState[" + std::to_string(i) + "]"), |
| sensorPresState.at(stateField[i].previous_state)); |
| } |
| |
| if (sensorPresState.contains(stateField[i].event_state)) |
| { |
| output.emplace(("eventState[" + std::to_string(i) + "]"), |
| sensorPresState.at(stateField[i].event_state)); |
| } |
| } |
| |
| pldmtool::helper::DisplayInJson(output); |
| } |
| |
| private: |
| uint16_t sensorId; |
| uint8_t sensorRearm; |
| }; |
| |
| void registerCommand(CLI::App& app) |
| { |
| auto platform = app.add_subcommand("platform", "platform type command"); |
| platform->require_subcommand(1); |
| |
| auto getPDR = platform->add_subcommand("GetPDR", |
| "get platform descriptor records"); |
| commands.push_back(std::make_unique<GetPDR>("platform", "getPDR", getPDR)); |
| |
| auto setStateEffecterStates = platform->add_subcommand( |
| "SetStateEffecterStates", "set effecter states"); |
| commands.push_back(std::make_unique<SetStateEffecter>( |
| "platform", "setStateEffecterStates", setStateEffecterStates)); |
| |
| auto setNumericEffecterValue = platform->add_subcommand( |
| "SetNumericEffecterValue", "set the value for a PLDM Numeric Effecter"); |
| commands.push_back(std::make_unique<SetNumericEffecterValue>( |
| "platform", "setNumericEffecterValue", setNumericEffecterValue)); |
| |
| auto getStateSensorReadings = platform->add_subcommand( |
| "GetStateSensorReadings", "get the state sensor readings"); |
| commands.push_back(std::make_unique<GetStateSensorReadings>( |
| "platform", "getStateSensorReadings", getStateSensorReadings)); |
| } |
| |
| void parseGetPDROption() |
| { |
| for (const auto& command : commands) |
| { |
| if (command.get()->getPLDMType() == "platform" && |
| command.get()->getCommandName() == "getPDR") |
| { |
| auto getPDR = dynamic_cast<GetPDR*>(command.get()); |
| getPDR->parseGetPDROptions(); |
| } |
| } |
| } |
| |
| } // namespace platform |
| } // namespace pldmtool |