1 /* 2 * Copyright 2018 Google Inc. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include "config.h" 18 19 #include "ipmi.hpp" 20 #include "process.hpp" 21 #include "utils.hpp" 22 23 #include <host-ipmid/ipmid-api.h> 24 25 #include <cstdio> 26 #include <host-ipmid/iana.hpp> 27 #include <host-ipmid/oemopenbmc.hpp> 28 #include <host-ipmid/oemrouter.hpp> 29 #include <memory> 30 #include <phosphor-logging/log.hpp> 31 32 namespace blobs 33 { 34 35 using namespace phosphor::logging; 36 37 static ipmi_ret_t handleBlobCommand(ipmi_cmd_t cmd, const uint8_t* reqBuf, 38 uint8_t* replyCmdBuf, size_t* dataLen) 39 { 40 /* It's holding at least a sub-command. The OEN is trimmed from the bytes 41 * before this is called. 42 */ 43 if ((*dataLen) < 1) 44 { 45 return IPMI_CC_REQ_DATA_LEN_INVALID; 46 } 47 48 /* on failure rc is set to the corresponding IPMI error. */ 49 ipmi_ret_t rc = IPMI_CC_OK; 50 Crc16 crc; 51 IpmiBlobHandler command = 52 validateBlobCommand(&crc, reqBuf, replyCmdBuf, dataLen, &rc); 53 if (command == nullptr) 54 { 55 return rc; 56 } 57 58 return processBlobCommand(command, getBlobManager(), &crc, reqBuf, 59 replyCmdBuf, dataLen); 60 } 61 62 /* TODO: this should come from the makefile or recipe... */ 63 constexpr auto expectedHandlerPath = "/usr/lib/blob-ipmid"; 64 65 void setupBlobGlobalHandler() __attribute__((constructor)); 66 67 void setupBlobGlobalHandler() 68 { 69 oem::Router* oemRouter = oem::mutableRouter(); 70 std::fprintf(stderr, 71 "Registering OEM:[%#08X], Cmd:[%#04X] for Blob Commands\n", 72 oem::obmcOemNumber, oem::Cmd::blobTransferCmd); 73 74 oemRouter->registerHandler(oem::obmcOemNumber, oem::Cmd::blobTransferCmd, 75 handleBlobCommand); 76 77 /* Install handlers. */ 78 try 79 { 80 loadLibraries(expectedHandlerPath); 81 } 82 catch (const std::exception& e) 83 { 84 log<level::ERR>("ERROR loading blob handlers", 85 entry("ERROR=%s", e.what())); 86 } 87 } 88 } // namespace blobs 89