#include "config.h" #include "file_io.hpp" #include "libpldm/base.h" #include "file_io_by_type.hpp" #include "file_table.hpp" #include "utils.hpp" #include "xyz/openbmc_project/Common/error.hpp" #include #include #include #include #include #include #include #include #include namespace pldm { using namespace pldm::responder::utils; using namespace sdbusplus::xyz::openbmc_project::Common::Error; namespace responder { namespace fs = std::filesystem; namespace dma { /** @struct AspeedXdmaOp * * Structure representing XDMA operation */ struct AspeedXdmaOp { uint64_t hostAddr; //!< the DMA address on the host side, configured by //!< PCI subsystem. uint32_t len; //!< the size of the transfer in bytes, it should be a //!< multiple of 16 bytes uint32_t upstream; //!< boolean indicating the direction of the DMA //!< operation, true means a transfer from BMC to host. }; constexpr auto xdmaDev = "/dev/aspeed-xdma"; int DMA::transferHostDataToSocket(int fd, uint32_t length, uint64_t address) { static const size_t pageSize = getpagesize(); uint32_t numPages = length / pageSize; uint32_t pageAlignedLength = numPages * pageSize; if (length > pageAlignedLength) { pageAlignedLength += pageSize; } auto mmapCleanup = [pageAlignedLength](void* vgaMem) { munmap(vgaMem, pageAlignedLength); }; int dmaFd = -1; int rc = 0; dmaFd = open(xdmaDev, O_RDWR); if (dmaFd < 0) { rc = -errno; std::cerr << "Failed to open the XDMA device, RC=" << rc << "\n"; return rc; } pldm::utils::CustomFD xdmaFd(dmaFd); void* vgaMem; vgaMem = mmap(nullptr, pageAlignedLength, PROT_READ, MAP_SHARED, xdmaFd(), 0); if (MAP_FAILED == vgaMem) { rc = -errno; std::cerr << "Failed to mmap the XDMA device, RC=" << rc << "\n"; return rc; } std::unique_ptr vgaMemPtr(vgaMem, mmapCleanup); AspeedXdmaOp xdmaOp; xdmaOp.upstream = 0; xdmaOp.hostAddr = address; xdmaOp.len = length; rc = write(xdmaFd(), &xdmaOp, sizeof(xdmaOp)); if (rc < 0) { rc = -errno; std::cerr << "Failed to execute the DMA operation, RC=" << rc << " ADDRESS=" << address << " LENGTH=" << length << "\n"; return rc; } rc = writeToUnixSocket(fd, static_cast(vgaMemPtr.get()), length); if (rc < 0) { rc = -errno; close(fd); std::cerr << "closing socket" << std::endl; return rc; } return 0; } int DMA::transferDataHost(int fd, uint32_t offset, uint32_t length, uint64_t address, bool upstream) { static const size_t pageSize = getpagesize(); uint32_t numPages = length / pageSize; uint32_t pageAlignedLength = numPages * pageSize; if (length > pageAlignedLength) { pageAlignedLength += pageSize; } auto mmapCleanup = [pageAlignedLength](void* vgaMem) { munmap(vgaMem, pageAlignedLength); }; int dmaFd = -1; int rc = 0; dmaFd = open(xdmaDev, O_RDWR); if (dmaFd < 0) { rc = -errno; std::cerr << "Failed to open the XDMA device, RC=" << rc << "\n"; return rc; } pldm::utils::CustomFD xdmaFd(dmaFd); void* vgaMem; vgaMem = mmap(nullptr, pageAlignedLength, upstream ? PROT_WRITE : PROT_READ, MAP_SHARED, xdmaFd(), 0); if (MAP_FAILED == vgaMem) { rc = -errno; std::cerr << "Failed to mmap the XDMA device, RC=" << rc << "\n"; return rc; } std::unique_ptr vgaMemPtr(vgaMem, mmapCleanup); if (upstream) { rc = lseek(fd, offset, SEEK_SET); if (rc == -1) { std::cerr << "lseek failed, ERROR=" << errno << ", UPSTREAM=" << upstream << ", OFFSET=" << offset << "\n"; return rc; } // Writing to the VGA memory should be aligned at page boundary, // otherwise write data into a buffer aligned at page boundary and // then write to the VGA memory. std::vector buffer{}; buffer.resize(pageAlignedLength); rc = read(fd, buffer.data(), length); if (rc == -1) { std::cerr << "file read failed, ERROR=" << errno << ", UPSTREAM=" << upstream << ", LENGTH=" << length << ", OFFSET=" << offset << "\n"; return rc; } if (rc != static_cast(length)) { std::cerr << "mismatch between number of characters to read and " << "the length read, LENGTH=" << length << " COUNT=" << rc << "\n"; return -1; } memcpy(static_cast(vgaMemPtr.get()), buffer.data(), pageAlignedLength); } AspeedXdmaOp xdmaOp; xdmaOp.upstream = upstream ? 1 : 0; xdmaOp.hostAddr = address; xdmaOp.len = length; rc = write(xdmaFd(), &xdmaOp, sizeof(xdmaOp)); if (rc < 0) { rc = -errno; std::cerr << "Failed to execute the DMA operation, RC=" << rc << " UPSTREAM=" << upstream << " ADDRESS=" << address << " LENGTH=" << length << "\n"; return rc; } if (!upstream) { rc = lseek(fd, offset, SEEK_SET); if (rc == -1) { std::cerr << "lseek failed, ERROR=" << errno << ", UPSTREAM=" << upstream << ", OFFSET=" << offset << "\n"; return rc; } rc = write(fd, static_cast(vgaMemPtr.get()), length); if (rc == -1) { std::cerr << "file write failed, ERROR=" << errno << ", UPSTREAM=" << upstream << ", LENGTH=" << length << ", OFFSET=" << offset << "\n"; return rc; } } return 0; } } // namespace dma namespace oem_ibm { Response Handler::readFileIntoMemory(const pldm_msg* request, size_t payloadLength) { uint32_t fileHandle = 0; uint32_t offset = 0; uint32_t length = 0; uint64_t address = 0; Response response((sizeof(pldm_msg_hdr) + PLDM_RW_FILE_MEM_RESP_BYTES), 0); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_RW_FILE_MEM_REQ_BYTES) { encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_READ_FILE_INTO_MEMORY, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } decode_rw_file_memory_req(request, payloadLength, &fileHandle, &offset, &length, &address); using namespace pldm::filetable; auto& table = buildFileTable(FILE_TABLE_JSON); FileEntry value{}; try { value = table.at(fileHandle); } catch (std::exception& e) { std::cerr << "File handle does not exist in the file table, HANDLE=" << fileHandle << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_READ_FILE_INTO_MEMORY, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } if (!fs::exists(value.fsPath)) { std::cerr << "File does not exist, HANDLE=" << fileHandle << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_READ_FILE_INTO_MEMORY, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } auto fileSize = fs::file_size(value.fsPath); if (offset >= fileSize) { std::cerr << "Offset exceeds file size, OFFSET=" << offset << " FILE_SIZE=" << fileSize << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_READ_FILE_INTO_MEMORY, PLDM_DATA_OUT_OF_RANGE, 0, responsePtr); return response; } if (offset + length > fileSize) { length = fileSize - offset; } if (length % dma::minSize) { std::cerr << "Read length is not a multiple of DMA minSize, LENGTH=" << length << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_READ_FILE_INTO_MEMORY, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } using namespace dma; DMA intf; return transferAll(&intf, PLDM_READ_FILE_INTO_MEMORY, value.fsPath, offset, length, address, true, request->hdr.instance_id); } Response Handler::writeFileFromMemory(const pldm_msg* request, size_t payloadLength) { uint32_t fileHandle = 0; uint32_t offset = 0; uint32_t length = 0; uint64_t address = 0; Response response(sizeof(pldm_msg_hdr) + PLDM_RW_FILE_MEM_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_RW_FILE_MEM_REQ_BYTES) { encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } decode_rw_file_memory_req(request, payloadLength, &fileHandle, &offset, &length, &address); if (length % dma::minSize) { std::cerr << "Write length is not a multiple of DMA minSize, LENGTH=" << length << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } using namespace pldm::filetable; auto& table = buildFileTable(FILE_TABLE_JSON); FileEntry value{}; try { value = table.at(fileHandle); } catch (std::exception& e) { std::cerr << "File handle does not exist in the file table, HANDLE=" << fileHandle << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } if (!fs::exists(value.fsPath)) { std::cerr << "File does not exist, HANDLE=" << fileHandle << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } auto fileSize = fs::file_size(value.fsPath); if (offset >= fileSize) { std::cerr << "Offset exceeds file size, OFFSET=" << offset << " FILE_SIZE=" << fileSize << "\n"; encode_rw_file_memory_resp(request->hdr.instance_id, PLDM_WRITE_FILE_FROM_MEMORY, PLDM_DATA_OUT_OF_RANGE, 0, responsePtr); return response; } using namespace dma; DMA intf; return transferAll(&intf, PLDM_WRITE_FILE_FROM_MEMORY, value.fsPath, offset, length, address, false, request->hdr.instance_id); } Response Handler::getFileTable(const pldm_msg* request, size_t payloadLength) { uint32_t transferHandle = 0; uint8_t transferFlag = 0; uint8_t tableType = 0; Response response(sizeof(pldm_msg_hdr) + PLDM_GET_FILE_TABLE_MIN_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_GET_FILE_TABLE_REQ_BYTES) { encode_get_file_table_resp(request->hdr.instance_id, PLDM_ERROR_INVALID_LENGTH, 0, 0, nullptr, 0, responsePtr); return response; } auto rc = decode_get_file_table_req(request, payloadLength, &transferHandle, &transferFlag, &tableType); if (rc) { encode_get_file_table_resp(request->hdr.instance_id, rc, 0, 0, nullptr, 0, responsePtr); return response; } if (tableType != PLDM_FILE_ATTRIBUTE_TABLE) { encode_get_file_table_resp(request->hdr.instance_id, PLDM_INVALID_FILE_TABLE_TYPE, 0, 0, nullptr, 0, responsePtr); return response; } using namespace pldm::filetable; auto table = buildFileTable(FILE_TABLE_JSON); auto attrTable = table(); response.resize(response.size() + attrTable.size()); responsePtr = reinterpret_cast(response.data()); if (attrTable.empty()) { encode_get_file_table_resp(request->hdr.instance_id, PLDM_FILE_TABLE_UNAVAILABLE, 0, 0, nullptr, 0, responsePtr); return response; } encode_get_file_table_resp(request->hdr.instance_id, PLDM_SUCCESS, 0, PLDM_START_AND_END, attrTable.data(), attrTable.size(), responsePtr); return response; } Response Handler::readFile(const pldm_msg* request, size_t payloadLength) { uint32_t fileHandle = 0; uint32_t offset = 0; uint32_t length = 0; Response response(sizeof(pldm_msg_hdr) + PLDM_READ_FILE_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_READ_FILE_REQ_BYTES) { encode_read_file_resp(request->hdr.instance_id, PLDM_ERROR_INVALID_LENGTH, length, responsePtr); return response; } auto rc = decode_read_file_req(request, payloadLength, &fileHandle, &offset, &length); if (rc) { encode_read_file_resp(request->hdr.instance_id, rc, 0, responsePtr); return response; } using namespace pldm::filetable; auto& table = buildFileTable(FILE_TABLE_JSON); FileEntry value{}; try { value = table.at(fileHandle); } catch (std::exception& e) { std::cerr << "File handle does not exist in the file table, HANDLE=" << fileHandle << "\n"; encode_read_file_resp(request->hdr.instance_id, PLDM_INVALID_FILE_HANDLE, length, responsePtr); return response; } if (!fs::exists(value.fsPath)) { std::cerr << "File does not exist, HANDLE=" << fileHandle << "\n"; encode_read_file_resp(request->hdr.instance_id, PLDM_INVALID_FILE_HANDLE, length, responsePtr); return response; } auto fileSize = fs::file_size(value.fsPath); if (offset >= fileSize) { std::cerr << "Offset exceeds file size, OFFSET=" << offset << " FILE_SIZE=" << fileSize << "\n"; encode_read_file_resp(request->hdr.instance_id, PLDM_DATA_OUT_OF_RANGE, length, responsePtr); return response; } if (offset + length > fileSize) { length = fileSize - offset; } response.resize(response.size() + length); responsePtr = reinterpret_cast(response.data()); auto fileDataPos = reinterpret_cast(responsePtr); fileDataPos += sizeof(pldm_msg_hdr) + sizeof(uint8_t) + sizeof(length); std::ifstream stream(value.fsPath, std::ios::in | std::ios::binary); stream.seekg(offset); stream.read(fileDataPos, length); encode_read_file_resp(request->hdr.instance_id, PLDM_SUCCESS, length, responsePtr); return response; } Response Handler::writeFile(const pldm_msg* request, size_t payloadLength) { uint32_t fileHandle = 0; uint32_t offset = 0; uint32_t length = 0; size_t fileDataOffset = 0; Response response(sizeof(pldm_msg_hdr) + PLDM_WRITE_FILE_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength < PLDM_WRITE_FILE_REQ_BYTES) { encode_write_file_resp(request->hdr.instance_id, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } auto rc = decode_write_file_req(request, payloadLength, &fileHandle, &offset, &length, &fileDataOffset); if (rc) { encode_write_file_resp(request->hdr.instance_id, rc, 0, responsePtr); return response; } using namespace pldm::filetable; auto& table = buildFileTable(FILE_TABLE_JSON); FileEntry value{}; try { value = table.at(fileHandle); } catch (std::exception& e) { std::cerr << "File handle does not exist in the file table, HANDLE=" << fileHandle << "\n"; encode_write_file_resp(request->hdr.instance_id, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } if (!fs::exists(value.fsPath)) { std::cerr << "File does not exist, HANDLE=" << fileHandle << "\n"; encode_write_file_resp(request->hdr.instance_id, PLDM_INVALID_FILE_HANDLE, 0, responsePtr); return response; } auto fileSize = fs::file_size(value.fsPath); if (offset >= fileSize) { std::cerr << "Offset exceeds file size, OFFSET=" << offset << " FILE_SIZE=" << fileSize << "\n"; encode_write_file_resp(request->hdr.instance_id, PLDM_DATA_OUT_OF_RANGE, 0, responsePtr); return response; } auto fileDataPos = reinterpret_cast(request->payload) + fileDataOffset; std::ofstream stream(value.fsPath, std::ios::in | std::ios::out | std::ios::binary); stream.seekp(offset); stream.write(fileDataPos, length); encode_write_file_resp(request->hdr.instance_id, PLDM_SUCCESS, length, responsePtr); return response; } Response rwFileByTypeIntoMemory(uint8_t cmd, const pldm_msg* request, size_t payloadLength, oem_platform::Handler* oemPlatformHandler) { Response response( sizeof(pldm_msg_hdr) + PLDM_RW_FILE_BY_TYPE_MEM_RESP_BYTES, 0); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_RW_FILE_BY_TYPE_MEM_REQ_BYTES) { encode_rw_file_by_type_memory_resp(request->hdr.instance_id, cmd, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } uint16_t fileType{}; uint32_t fileHandle{}; uint32_t offset{}; uint32_t length{}; uint64_t address{}; auto rc = decode_rw_file_by_type_memory_req(request, payloadLength, &fileType, &fileHandle, &offset, &length, &address); if (rc != PLDM_SUCCESS) { encode_rw_file_by_type_memory_resp(request->hdr.instance_id, cmd, rc, 0, responsePtr); return response; } if (length % dma::minSize) { std::cerr << "Length is not a multiple of DMA minSize, LENGTH=" << length << "\n"; encode_rw_file_by_type_memory_resp(request->hdr.instance_id, cmd, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } std::unique_ptr handler{}; try { handler = getHandlerByType(fileType, fileHandle); } catch (const InternalFailure& e) { std::cerr << "unknown file type, TYPE=" << fileType << "\n"; encode_rw_file_by_type_memory_resp(request->hdr.instance_id, cmd, PLDM_INVALID_FILE_TYPE, 0, responsePtr); return response; } rc = cmd == PLDM_WRITE_FILE_BY_TYPE_FROM_MEMORY ? handler->writeFromMemory(offset, length, address, oemPlatformHandler) : handler->readIntoMemory(offset, length, address, oemPlatformHandler); encode_rw_file_by_type_memory_resp(request->hdr.instance_id, cmd, rc, length, responsePtr); return response; } Response Handler::writeFileByTypeFromMemory(const pldm_msg* request, size_t payloadLength) { return rwFileByTypeIntoMemory(PLDM_WRITE_FILE_BY_TYPE_FROM_MEMORY, request, payloadLength, oemPlatformHandler); } Response Handler::readFileByTypeIntoMemory(const pldm_msg* request, size_t payloadLength) { return rwFileByTypeIntoMemory(PLDM_READ_FILE_BY_TYPE_INTO_MEMORY, request, payloadLength, oemPlatformHandler); } Response Handler::writeFileByType(const pldm_msg* request, size_t payloadLength) { Response response(sizeof(pldm_msg_hdr) + PLDM_RW_FILE_BY_TYPE_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength < PLDM_RW_FILE_BY_TYPE_REQ_BYTES) { encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_WRITE_FILE_BY_TYPE, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } uint16_t fileType{}; uint32_t fileHandle{}; uint32_t offset{}; uint32_t length{}; auto rc = decode_rw_file_by_type_req(request, payloadLength, &fileType, &fileHandle, &offset, &length); if (rc != PLDM_SUCCESS) { encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_WRITE_FILE_BY_TYPE, rc, 0, responsePtr); return response; } std::unique_ptr handler{}; try { handler = getHandlerByType(fileType, fileHandle); } catch (const InternalFailure& e) { std::cerr << "unknown file type, TYPE=" << fileType << "\n"; encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_WRITE_FILE_BY_TYPE, PLDM_INVALID_FILE_TYPE, 0, responsePtr); return response; } rc = handler->write(reinterpret_cast( request->payload + PLDM_RW_FILE_BY_TYPE_REQ_BYTES), offset, length, oemPlatformHandler); encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_WRITE_FILE_BY_TYPE, rc, length, responsePtr); return response; } Response Handler::readFileByType(const pldm_msg* request, size_t payloadLength) { Response response(sizeof(pldm_msg_hdr) + PLDM_RW_FILE_BY_TYPE_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_RW_FILE_BY_TYPE_REQ_BYTES) { encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_READ_FILE_BY_TYPE, PLDM_ERROR_INVALID_LENGTH, 0, responsePtr); return response; } uint16_t fileType{}; uint32_t fileHandle{}; uint32_t offset{}; uint32_t length{}; auto rc = decode_rw_file_by_type_req(request, payloadLength, &fileType, &fileHandle, &offset, &length); if (rc != PLDM_SUCCESS) { encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_READ_FILE_BY_TYPE, rc, 0, responsePtr); return response; } std::unique_ptr handler{}; try { handler = getHandlerByType(fileType, fileHandle); } catch (const InternalFailure& e) { std::cerr << "unknown file type, TYPE=" << fileType << "\n"; encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_READ_FILE_BY_TYPE, PLDM_INVALID_FILE_TYPE, 0, responsePtr); return response; } rc = handler->read(offset, length, response, oemPlatformHandler); responsePtr = reinterpret_cast(response.data()); encode_rw_file_by_type_resp(request->hdr.instance_id, PLDM_READ_FILE_BY_TYPE, rc, length, responsePtr); return response; } Response Handler::fileAck(const pldm_msg* request, size_t payloadLength) { Response response(sizeof(pldm_msg_hdr) + PLDM_FILE_ACK_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_FILE_ACK_REQ_BYTES) { encode_file_ack_resp(request->hdr.instance_id, PLDM_ERROR_INVALID_LENGTH, responsePtr); return response; } uint16_t fileType{}; uint32_t fileHandle{}; uint8_t fileStatus{}; auto rc = decode_file_ack_req(request, payloadLength, &fileType, &fileHandle, &fileStatus); if (rc != PLDM_SUCCESS) { encode_file_ack_resp(request->hdr.instance_id, rc, responsePtr); return response; } std::unique_ptr handler{}; try { handler = getHandlerByType(fileType, fileHandle); } catch (const InternalFailure& e) { encode_file_ack_resp(request->hdr.instance_id, PLDM_INVALID_FILE_TYPE, responsePtr); return response; } rc = handler->fileAck(fileStatus); encode_file_ack_resp(request->hdr.instance_id, rc, responsePtr); return response; } Response Handler::getAlertStatus(const pldm_msg* request, size_t payloadLength) { Response response(sizeof(pldm_msg_hdr) + PLDM_GET_ALERT_STATUS_RESP_BYTES); auto responsePtr = reinterpret_cast(response.data()); if (payloadLength != PLDM_GET_ALERT_STATUS_REQ_BYTES) { return CmdHandler::ccOnlyResponse(request, PLDM_ERROR_INVALID_LENGTH); } uint8_t versionId{}; auto rc = decode_get_alert_status_req(request, payloadLength, &versionId); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } if (versionId != 0) { return CmdHandler::ccOnlyResponse(request, PLDM_HOST_UNSUPPORTED_FORMAT_VERSION); } constexpr uint32_t rackEntry = 0xFF000030; constexpr uint32_t priCecNode = 0x00008030; rc = encode_get_alert_status_resp(request->hdr.instance_id, PLDM_SUCCESS, rackEntry, priCecNode, responsePtr, PLDM_GET_ALERT_STATUS_RESP_BYTES); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } return response; } Response Handler::newFileAvailable(const pldm_msg* request, size_t payloadLength) { Response response(sizeof(pldm_msg_hdr) + PLDM_NEW_FILE_RESP_BYTES); if (payloadLength != PLDM_NEW_FILE_REQ_BYTES) { return CmdHandler::ccOnlyResponse(request, PLDM_ERROR_INVALID_LENGTH); } uint16_t fileType{}; uint32_t fileHandle{}; uint64_t length{}; auto rc = decode_new_file_req(request, payloadLength, &fileType, &fileHandle, &length); if (rc != PLDM_SUCCESS) { return CmdHandler::ccOnlyResponse(request, rc); } std::unique_ptr handler{}; try { handler = getHandlerByType(fileType, fileHandle); } catch (const InternalFailure& e) { std::cerr << "unknown file type, TYPE=" << fileType << "\n"; return CmdHandler::ccOnlyResponse(request, PLDM_INVALID_FILE_TYPE); } rc = handler->newFileAvailable(length); auto responsePtr = reinterpret_cast(response.data()); encode_new_file_resp(request->hdr.instance_id, rc, responsePtr); return response; } } // namespace oem_ibm } // namespace responder } // namespace pldm