| #include "writefrudata.hpp" |
| |
| #include <ipmid/api.h> |
| #include <unistd.h> |
| |
| #include <phosphor-logging/log.hpp> |
| #include <sdbusplus/bus.hpp> |
| |
| #include <cstdio> |
| #include <cstring> |
| |
| void register_netfn_storage_write_fru() __attribute__((constructor)); |
| |
| sd_bus* ipmid_get_sd_bus_connection(void); |
| |
| using namespace phosphor::logging; |
| |
| ///------------------------------------------------------- |
| // Called by IPMI netfn router for write fru data command |
| //-------------------------------------------------------- |
| ipmi_ret_t ipmiStorageWriteFruData(ipmi_netfn_t /*netfn*/, ipmi_cmd_t /*cmd*/, |
| ipmi_request_t request, |
| ipmi_response_t response, |
| ipmi_data_len_t dataLen, |
| ipmi_context_t /*context*/) |
| { |
| FILE* fp = NULL; |
| char fruFilename[16] = {0}; |
| size_t offset = 0; |
| size_t len = 0; |
| ipmi_ret_t rc = IPMI_CC_INVALID; |
| const char* mode = NULL; |
| |
| // From the payload, extract the header that has fruid and the offsets |
| auto reqptr = static_cast<write_fru_data_t*>(request); |
| |
| // Maintaining a temporary file to pump the data |
| std::sprintf(fruFilename, "%s%02x", "/tmp/ipmifru", reqptr->frunum); |
| |
| offset = ((size_t)reqptr->offsetms) << 8 | reqptr->offsetls; |
| |
| // Length is the number of request bytes minus the header itself. |
| // The header contains an extra byte to indicate the start of |
| // the data (so didn't need to worry about word/byte boundaries) |
| // hence the -1... |
| len = ((size_t)*dataLen) - (sizeof(write_fru_data_t) - 1); |
| |
| // On error there is no response data for this command. |
| *dataLen = 0; |
| |
| #ifdef __IPMI__DEBUG__ |
| log<level::DEBUG>("IPMI WRITE-FRU-DATA", entry("FILE=%s", fruFilename), |
| entry("OFFSET=%d", offset), entry("LENGTH=%d", len)); |
| #endif |
| |
| if (access(fruFilename, F_OK) == -1) |
| { |
| mode = "wb"; |
| } |
| else |
| { |
| mode = "rb+"; |
| } |
| |
| if ((fp = std::fopen(fruFilename, mode)) != NULL) |
| { |
| if (std::fseek(fp, offset, SEEK_SET)) |
| { |
| log<level::ERR>("Seek into fru file failed", |
| entry("FILE=%s", fruFilename), |
| entry("ERRNO=%s", std::strerror(errno))); |
| std::fclose(fp); |
| return rc; |
| } |
| |
| if (std::fwrite(&reqptr->data, len, 1, fp) != 1) |
| { |
| log<level::ERR>("Write into fru file failed", |
| entry("FILE=%s", fruFilename), |
| entry("ERRNO=%s", std::strerror(errno))); |
| std::fclose(fp); |
| return rc; |
| } |
| |
| std::fclose(fp); |
| } |
| else |
| { |
| log<level::ERR>("Error trying to write to fru file", |
| entry("FILE=%s", fruFilename)); |
| return rc; |
| } |
| |
| // If we got here then set the resonse byte |
| // to the number of bytes written |
| std::memcpy(response, &len, 1); |
| *dataLen = 1; |
| rc = IPMI_CC_OK; |
| |
| // Get the reference to global sd_bus object |
| sd_bus* bus_type = ipmid_get_sd_bus_connection(); |
| |
| // We received some bytes. It may be full or partial. Send a valid |
| // FRU file to the inventory controller on DBus for the correct number |
| sdbusplus::bus_t bus{bus_type}; |
| bool bmcOnlyFru = false; |
| validateFRUArea(reqptr->frunum, fruFilename, bus, bmcOnlyFru); |
| |
| return rc; |
| } |
| |
| //------------------------------------------------------- |
| // Registering WRITE FRU DATA command handler with daemon |
| //------------------------------------------------------- |
| void register_netfn_storage_write_fru() |
| { |
| std::printf("Registering NetFn:[0x%X], Cmd:[0x%X]\n", NETFUN_STORAGE, |
| IPMI_CMD_WRITE_FRU_DATA); |
| |
| ipmi_register_callback(NETFUN_STORAGE, IPMI_CMD_WRITE_FRU_DATA, NULL, |
| ipmiStorageWriteFruData, SYSTEM_INTERFACE); |
| } |