blob: 3cc3fb78a80270702712d8ec656f96eab42d5130 [file] [log] [blame]
/*
* Copyright 2018 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 <memory>
#include "host-ipmid/ipmid-api.h"
#include "host-ipmid/oemrouter.hpp"
#include "config.h"
#include "flash-ipmi.hpp"
#include "ipmi.hpp"
static constexpr auto stagingPath = STAGING_PATH;
/* TODO: Once OEM IPMI number placement is settled, point to that. */
namespace oem
{
namespace google
{
constexpr int number = 11129;
constexpr int flashOverBTCmd = 127;
} // namespace google
} // namespace oem
/* We have one instance of the FlashUpdate object tracking commands. */
std::unique_ptr<FlashUpdate> flashUpdateSingleton;
static ipmi_ret_t flashControl(ipmi_cmd_t cmd, const uint8_t* reqBuf,
uint8_t* replyCmdBuf, size_t* dataLen)
{
size_t requestLength = (*dataLen);
/* Verify it's at least as long as the shortest message. */
if (requestLength < 1)
{
return IPMI_CC_INVALID;
}
auto subCmd = static_cast<FlashSubCmds>(reqBuf[0]);
/* Validate the minimum request length for the command. */
if (!validateRequestLength(subCmd, requestLength))
{
return IPMI_CC_INVALID;
}
auto handler = getCommandHandler(subCmd);
if (handler)
{
return handler(flashUpdateSingleton.get(), reqBuf, replyCmdBuf,
dataLen);
}
return IPMI_CC_INVALID;
}
static ipmi_ret_t ipmiFlashControl(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)
{
/* request_t, response_t are void*. ipmi_data_len_t is a size_t* */
auto reqBuf = static_cast<const uint8_t*>(request);
auto replyCmdBuf = static_cast<uint8_t*>(response);
return flashControl(cmd, reqBuf, replyCmdBuf, dataLen);
}
void setupGlobalOemFlashControl() __attribute__((constructor));
void setupGlobalOemFlashControl()
{
flashUpdateSingleton = std::make_unique<FlashUpdate>(stagingPath);
#ifdef ENABLE_GOOGLE
oem::Router* router = oem::mutableRouter();
fprintf(stderr, "Registering OEM:[%#08X], Cmd:[%#04X] for Flash Update\n",
oem::google::number, oem::google::flashOverBTCmd);
router->registerHandler(oem::google::number, oem::google::flashOverBTCmd,
flashControl);
#endif
ipmi_register_callback(NETFUN_FIRMWARE, oem::google::flashOverBTCmd, NULL,
ipmiFlashControl, PRIVILEGE_USER);
}