blob: 10ac2fee8658a3484dcedf35db648acf24b7f1fa [file] [log] [blame]
/*
* Copyright 2017 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <cstring>
#include <unordered_map>
#include "flash-ipmi.hpp"
#include "ipmi.hpp"
IpmiFlashHandler getCommandHandler(FlashSubCmds command)
{
static const std::unordered_map<FlashSubCmds, IpmiFlashHandler>
subHandlers = {
{FlashSubCmds::flashStartTransfer, startTransfer},
{FlashSubCmds::flashDataBlock, dataBlock},
{FlashSubCmds::flashDataFinish, dataFinish},
{FlashSubCmds::flashStartHash, startHash},
{FlashSubCmds::flashHashData, hashBlock},
{FlashSubCmds::flashHashFinish, hashFinish},
{FlashSubCmds::flashDataVerify, dataVerify},
{FlashSubCmds::flashAbort, abortUpdate},
};
auto results = subHandlers.find(command);
if (results == subHandlers.end())
{
return nullptr;
}
return results->second;
}
bool validateRequestLength(FlashSubCmds command, size_t requestLen)
{
static const std::unordered_map<FlashSubCmds, size_t> minimumLengths = {
{FlashSubCmds::flashStartTransfer, sizeof(struct StartTx)},
{FlashSubCmds::flashDataBlock, sizeof(struct ChunkHdr) + 1},
{FlashSubCmds::flashStartHash, sizeof(struct StartTx)},
{FlashSubCmds::flashHashData, sizeof(struct ChunkHdr) + 1},
};
auto results = minimumLengths.find(command);
if (results == minimumLengths.end())
{
/* Valid length by default if we don't care. */
return true;
}
/* If the request is shorter than the minimum, it's invalid. */
if (requestLen < results->second)
{
return false;
}
return true;
}
ipmi_ret_t startTransfer(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
auto request = reinterpret_cast<const struct StartTx*>(reqBuf);
if (!updater->start(request->length))
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t dataBlock(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
struct ChunkHdr hdr;
std::memcpy(&hdr, reqBuf, sizeof(hdr));
auto requestLength = *dataLen;
/* Grab the bytes from the packet. */
auto bytesLength = requestLength - sizeof(struct ChunkHdr);
std::vector<uint8_t> bytes(bytesLength);
std::memcpy(bytes.data(), &reqBuf[sizeof(struct ChunkHdr)], bytesLength);
if (!updater->flashData(hdr.offset, bytes))
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t dataFinish(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
if (!updater->flashFinish())
{
return IPMI_CC_INVALID;
}
/* TODO: If all commands return this on success, handle it in one place. */
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t startHash(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
auto request = reinterpret_cast<const struct StartTx*>(reqBuf);
if (!updater->startHash(request->length))
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t hashBlock(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
struct ChunkHdr hdr;
std::memcpy(&hdr, reqBuf, sizeof(hdr));
auto requestLength = *dataLen;
/* Grab the bytes from the packet. */
auto bytesLength = requestLength - sizeof(struct ChunkHdr);
std::vector<uint8_t> bytes(bytesLength);
std::memcpy(bytes.data(), &reqBuf[sizeof(struct ChunkHdr)], bytesLength);
/* TODO: Refactor this and dataBlock for re-use. */
if (!updater->hashData(hdr.offset, bytes))
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t hashFinish(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
if (!updater->hashFinish())
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t dataVerify(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
if (!updater->startDataVerification())
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}
ipmi_ret_t abortUpdate(UpdateInterface* updater, const uint8_t* reqBuf,
uint8_t* replyBuf, size_t* dataLen)
{
/* TODO: May make sense to work all the pass-through commands into one
* piece of code
*/
if (!updater->abortUpdate())
{
return IPMI_CC_INVALID;
}
/* We were successful and set the response byte to 0. */
replyBuf[0] = 0x00;
(*dataLen) = 1;
return IPMI_CC_OK;
}