| /* |
| * 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 <memory> |
| |
| #include "host-ipmid/ipmid-api.h" |
| #include "host-ipmid/oemrouter.hpp" |
| |
| #include "flash-ipmi.hpp" |
| #include "ipmi.hpp" |
| |
| /* 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) |
| { |
| /* Verify it's at least as long as the shortest message. */ |
| if ((*dataLen) < 1) |
| { |
| return IPMI_CC_INVALID; |
| } |
| |
| auto subCmd = static_cast<FlashSubCmds>(reqBuf[0]); |
| |
| /* Validate the minimum request length for the command. */ |
| if (!validateRequestLength(subCmd, *dataLen)) |
| { |
| return IPMI_CC_INVALID; |
| } |
| |
| /* TODO: This could be cleaner to just have a function pointer table, may |
| * transition in later patchset. |
| */ |
| switch (subCmd) |
| { |
| case FlashSubCmds::flashStartTransfer: |
| return startTransfer(flashUpdateSingleton.get(), reqBuf, |
| replyCmdBuf, dataLen); |
| case FlashSubCmds::flashDataBlock: |
| return dataBlock(flashUpdateSingleton.get(), reqBuf, replyCmdBuf, |
| dataLen); |
| case FlashSubCmds::flashDataFinish: |
| return dataFinish(flashUpdateSingleton.get(), reqBuf, replyCmdBuf, |
| dataLen); |
| case FlashSubCmds::flashStartHash: |
| return startHash(flashUpdateSingleton.get(), reqBuf, replyCmdBuf, |
| dataLen); |
| case FlashSubCmds::flashHashData: |
| return hashBlock(flashUpdateSingleton.get(), reqBuf, replyCmdBuf, |
| dataLen); |
| case FlashSubCmds::flashHashFinish: |
| return hashFinish(flashUpdateSingleton.get(), reqBuf, replyCmdBuf, |
| dataLen); |
| default: |
| return IPMI_CC_INVALID; |
| } |
| |
| 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>(); |
| |
| #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); |
| } |