#include "config.h" #include "libpldm/base.h" #include "libpldm/bios.h" #include "libpldm/fru.h" #include "libpldm/platform.h" #include "libpldm/requester/pldm.h" #include "base.hpp" #include "common/utils.hpp" #include "libpldmresponder/pdr.hpp" #include #include #include #include #include #include #ifdef OEM_IBM #include "libpldm/file_io.h" #include "libpldm/host.h" #endif namespace pldm { using Type = uint8_t; namespace responder { using Cmd = std::vector; static const std::map capabilities{ {PLDM_BASE, {PLDM_GET_TID, PLDM_GET_PLDM_VERSION, PLDM_GET_PLDM_TYPES, PLDM_GET_PLDM_COMMANDS}}, {PLDM_PLATFORM, {PLDM_GET_PDR, PLDM_SET_STATE_EFFECTER_STATES, PLDM_SET_EVENT_RECEIVER, PLDM_GET_SENSOR_READING, PLDM_GET_STATE_SENSOR_READINGS, PLDM_SET_NUMERIC_EFFECTER_VALUE, PLDM_GET_NUMERIC_EFFECTER_VALUE, PLDM_PLATFORM_EVENT_MESSAGE}}, {PLDM_BIOS, {PLDM_GET_DATE_TIME, PLDM_SET_DATE_TIME, PLDM_GET_BIOS_TABLE, PLDM_GET_BIOS_ATTRIBUTE_CURRENT_VALUE_BY_HANDLE, PLDM_SET_BIOS_ATTRIBUTE_CURRENT_VALUE, PLDM_SET_BIOS_TABLE}}, {PLDM_FRU, {PLDM_GET_FRU_RECORD_TABLE_METADATA, PLDM_GET_FRU_RECORD_TABLE, PLDM_GET_FRU_RECORD_BY_OPTION}}, #ifdef OEM_IBM {PLDM_OEM, {PLDM_HOST_GET_ALERT_STATUS, PLDM_GET_FILE_TABLE, PLDM_READ_FILE, PLDM_WRITE_FILE, PLDM_READ_FILE_INTO_MEMORY, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_READ_FILE_BY_TYPE_INTO_MEMORY, PLDM_WRITE_FILE_BY_TYPE_FROM_MEMORY, PLDM_NEW_FILE_AVAILABLE, PLDM_READ_FILE_BY_TYPE, PLDM_WRITE_FILE_BY_TYPE, PLDM_FILE_ACK}}, #endif }; static const std::map versions{ {PLDM_BASE, {0xF1, 0xF0, 0xF0, 0x00}}, {PLDM_PLATFORM, {0xF1, 0xF2, 0xF0, 0x00}}, {PLDM_BIOS, {0xF1, 0xF0, 0xF0, 0x00}}, {PLDM_FRU, {0xF1, 0xF0, 0xF0, 0x00}}, #ifdef OEM_IBM {PLDM_OEM, {0xF1, 0xF0, 0xF0, 0x00}}, #endif }; namespace base { Response Handler::getPLDMTypes(const pldm_msg* request, size_t /*payloadLength*/) { // DSP0240 has this as a bitfield8[N], where N = 0 to 7 std::array types{}; for (const auto& type : capabilities) { auto index = type.first / 8; // = * 8 + auto bit = type.first - (index * 8); types[index].byte |= 1 << bit; } Response response(sizeof(pldm_msg_hdr) + PLDM_GET_TYPES_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); auto rc = encode_get_types_resp(request->hdr.instance_id, PLDM_SUCCESS, types.data(), responsePtr); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } return response; } Response Handler::getPLDMCommands(const pldm_msg* request, size_t payloadLength) { ver32_t version{}; Type type; Response response(sizeof(pldm_msg_hdr) + PLDM_GET_COMMANDS_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); auto rc = decode_get_commands_req(request, payloadLength, &type, &version); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } // DSP0240 has this as a bitfield8[N], where N = 0 to 31 std::array cmds{}; if (capabilities.find(type) == capabilities.end()) { return CmdHandler::ccOnlyResponse(request, PLDM_ERROR_INVALID_PLDM_TYPE); } for (const auto& cmd : capabilities.at(type)) { auto index = cmd / 8; // = * 8 + auto bit = cmd - (index * 8); cmds[index].byte |= 1 << bit; } rc = encode_get_commands_resp(request->hdr.instance_id, PLDM_SUCCESS, cmds.data(), responsePtr); if (rc != PLDM_SUCCESS) { return ccOnlyResponse(request, rc); } return response; } Response Handler::getPLDMVersion(const pldm_msg* request, size_t payloadLength) { uint32_t transferHandle; Type type; uint8_t transferFlag; Response response(sizeof(pldm_msg_hdr) + PLDM_GET_VERSION_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); uint8_t rc = decode_get_version_req(request, payloadLength, &transferHandle, &transferFlag, &type); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } ver32_t version{}; auto search = versions.find(type); if (search == versions.end()) { return CmdHandler::ccOnlyResponse(request, PLDM_ERROR_INVALID_PLDM_TYPE); } memcpy(&version, &(search->second), sizeof(version)); rc = encode_get_version_resp(request->hdr.instance_id, PLDM_SUCCESS, 0, PLDM_START_AND_END, &version, sizeof(pldm_version), responsePtr); if (rc != PLDM_SUCCESS) { return ccOnlyResponse(request, rc); } return response; } void Handler::processSetEventReceiver( sdeventplus::source::EventBase& /*source */) { survEvent.reset(); std::vector requestMsg(sizeof(pldm_msg_hdr) + PLDM_SET_EVENT_RECEIVER_REQ_BYTES); auto request = reinterpret_cast(requestMsg.data()); auto instanceId = requester.getInstanceId(eid); uint8_t eventMessageGlobalEnable = PLDM_EVENT_MESSAGE_GLOBAL_ENABLE_ASYNC_KEEP_ALIVE; uint8_t transportProtocolType = PLDM_TRANSPORT_PROTOCOL_TYPE_MCTP; uint8_t eventReceiverAddressInfo = pldm::responder::pdr::BmcMctpEid; uint16_t heartbeatTimer = HEARTBEAT_TIMEOUT; auto rc = encode_set_event_receiver_req( instanceId, eventMessageGlobalEnable, transportProtocolType, eventReceiverAddressInfo, heartbeatTimer, request); if (rc != PLDM_SUCCESS) { requester.markFree(eid, instanceId); std::cerr << "Failed to encode_set_event_receiver_req, rc = " << std::hex << std::showbase << rc << std::endl; return; } auto processSetEventReceiverResponse = [](mctp_eid_t /*eid*/, const pldm_msg* response, size_t respMsgLen) { if (response == nullptr || !respMsgLen) { std::cerr << "Failed to receive response for " "setEventReceiver command \n"; return; } uint8_t completionCode{}; auto rc = decode_set_event_receiver_resp(response, respMsgLen, &completionCode); if (rc || completionCode) { std::cerr << "Failed to decode setEventReceiver command response," << " rc=" << rc << "cc=" << (uint8_t)completionCode << "\n"; pldm::utils::reportError( "xyz.openbmc_project.bmc.pldm.InternalFailure"); } }; rc = handler->registerRequest( eid, instanceId, PLDM_PLATFORM, PLDM_SET_EVENT_RECEIVER, std::move(requestMsg), std::move(processSetEventReceiverResponse)); if (rc != PLDM_SUCCESS) { std::cerr << "Failed to send the setEventReceiver request" << "\n"; } if (oemPlatformHandler) { oemPlatformHandler->countSetEventReceiver(); oemPlatformHandler->checkAndDisableWatchDog(); } } Response Handler::getTID(const pldm_msg* request, size_t /*payloadLength*/) { // assigned 1 to the bmc as the PLDM terminus uint8_t tid = 1; Response response(sizeof(pldm_msg_hdr) + PLDM_GET_TID_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); auto rc = encode_get_tid_resp(request->hdr.instance_id, PLDM_SUCCESS, tid, responsePtr); if (rc != PLDM_SUCCESS) { return ccOnlyResponse(request, rc); } survEvent = std::make_unique( event, std::bind_front(&Handler::processSetEventReceiver, this)); return response; } } // namespace base } // namespace responder } // namespace pldm