| /* |
| * Copyright (c) 2018 Intel Corporation. |
| * Copyright (c) 2018-present Facebook. |
| * |
| * 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 <fcntl.h> |
| #include <ipmid/api.h> |
| #include <sys/stat.h> |
| #include <unistd.h> |
| |
| #include <appcommands.hpp> |
| #include <commandutils.hpp> |
| #include <nlohmann/json.hpp> |
| #include <phosphor-logging/log.hpp> |
| #include <sdbusplus/message/types.hpp> |
| #include <ipmid/api.hpp> |
| #include <ipmid/api-types.hpp> |
| |
| #include <fstream> |
| #include <iomanip> |
| #include <iostream> |
| #include <sstream> |
| |
| namespace ipmi |
| { |
| |
| static void registerAPPFunctions() __attribute__((constructor)); |
| static constexpr size_t GUID_SIZE = 16; |
| // TODO Make offset and location runtime configurable to ensure we |
| // can make each define their own locations. |
| static constexpr off_t OFFSET_SYS_GUID = 0x17F0; |
| static constexpr const char* FRU_EEPROM = "/sys/bus/i2c/devices/6-0054/eeprom"; |
| |
| // TODO: Need to store this info after identifying proper storage |
| static uint8_t globEna = 0x09; |
| static SysInfoParam sysInfoParams; |
| nlohmann::json appData __attribute__((init_priority(101))); |
| |
| int sendBicCmd(uint8_t, uint8_t, uint8_t, std::vector<uint8_t>&, |
| std::vector<uint8_t>&); |
| |
| static inline auto responseSystemInfoParamterNotSupportCommand() |
| { |
| return response(IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED); |
| } |
| |
| void printGUID(uint8_t* guid, off_t offset) |
| { |
| std::cout << "Read GUID from offset : " << offset << " :\n"; |
| for (size_t i = 0; i < GUID_SIZE; i++) |
| { |
| int data = guid[i]; |
| std::cout << std::hex << data << " "; |
| } |
| std::cout << std::endl; |
| } |
| |
| int getGUID(off_t offset, uint8_t* guid) |
| { |
| int fd = -1; |
| ssize_t bytes_rd; |
| int ret = 0; |
| |
| errno = 0; |
| |
| // Check if file is present |
| if (access(FRU_EEPROM, F_OK) == -1) |
| { |
| std::cerr << "Unable to access: " << FRU_EEPROM << std::endl; |
| return errno; |
| } |
| |
| // Open the file |
| fd = open(FRU_EEPROM, O_RDONLY); |
| if (fd == -1) |
| { |
| std::cerr << "Unable to open: " << FRU_EEPROM << std::endl; |
| return errno; |
| } |
| |
| // seek to the offset |
| lseek(fd, offset, SEEK_SET); |
| |
| // Read bytes from location |
| bytes_rd = read(fd, guid, GUID_SIZE); |
| if (bytes_rd != GUID_SIZE) |
| { |
| phosphor::logging::log<phosphor::logging::level::ERR>( |
| "GUID read data from EEPROM failed"); |
| ret = errno; |
| } |
| else |
| { |
| printGUID(guid, offset); |
| } |
| close(fd); |
| return ret; |
| } |
| |
| int getSystemGUID(uint8_t* guid) |
| { |
| return getGUID(OFFSET_SYS_GUID, guid); |
| } |
| |
| //---------------------------------------------------------------------- |
| // Get Self Test Results (IPMI/Section 20.4) (CMD_APP_GET_SELFTEST_RESULTS) |
| //---------------------------------------------------------------------- |
| ipmi_ret_t ipmiAppGetSTResults(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t, |
| ipmi_response_t response, |
| ipmi_data_len_t data_len, ipmi_context_t) |
| { |
| uint8_t* res = reinterpret_cast<uint8_t*>(response); |
| |
| // TODO: Following data needs to be updated based on self-test results |
| *res++ = 0x55; // Self-Test result |
| *res++ = 0x00; // Extra error info in case of failure |
| |
| *data_len = 2; |
| |
| return IPMI_CC_OK; |
| } |
| |
| //---------------------------------------------------------------------- |
| // Manufacturing Test On (IPMI/Section 20.5) (CMD_APP_MFR_TEST_ON) |
| //---------------------------------------------------------------------- |
| ipmi_ret_t ipmiAppMfrTestOn(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t request, |
| ipmi_response_t, ipmi_data_len_t data_len, |
| ipmi_context_t) |
| { |
| uint8_t* req = reinterpret_cast<uint8_t*>(request); |
| std::string mfrTest = "sled-cycle"; |
| ipmi_ret_t rc = IPMI_CC_OK; |
| |
| if (!memcmp(req, mfrTest.data(), mfrTest.length()) && |
| (*data_len == mfrTest.length())) |
| { |
| /* sled-cycle the BMC */ |
| auto ret = system("/usr/sbin/power-util sled-cycle"); |
| if (ret) |
| { |
| rc = IPMI_CC_UNSPECIFIED_ERROR; |
| } |
| } |
| else |
| { |
| rc = IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED; |
| } |
| |
| *data_len = 0; |
| |
| return rc; |
| } |
| |
| //---------------------------------------------------------------------- |
| // Set Global Enables (CMD_APP_SET_GLOBAL_ENABLES) |
| //---------------------------------------------------------------------- |
| ipmi_ret_t ipmiAppSetGlobalEnables(ipmi_netfn_t, ipmi_cmd_t, |
| ipmi_request_t request, ipmi_response_t, |
| ipmi_data_len_t data_len, ipmi_context_t) |
| { |
| uint8_t* req = reinterpret_cast<uint8_t*>(request); |
| |
| globEna = *req; |
| *data_len = 0; |
| |
| return IPMI_CC_OK; |
| } |
| |
| //---------------------------------------------------------------------- |
| // Get Global Enables (CMD_APP_GET_GLOBAL_ENABLES) |
| //---------------------------------------------------------------------- |
| ipmi_ret_t ipmiAppGetGlobalEnables(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t, |
| ipmi_response_t response, |
| ipmi_data_len_t data_len, ipmi_context_t) |
| { |
| uint8_t* res = reinterpret_cast<uint8_t*>(response); |
| |
| *data_len = 1; |
| *res++ = globEna; |
| |
| return IPMI_CC_OK; |
| } |
| |
| //---------------------------------------------------------------------- |
| // Clear Message flags (IPMI/Section 22.3) (CMD_APP_CLEAR_MESSAGE_FLAGS) |
| //---------------------------------------------------------------------- |
| ipmi_ret_t ipmiAppClearMsgFlags(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t, |
| ipmi_response_t, ipmi_data_len_t data_len, |
| ipmi_context_t) |
| { |
| // Do Nothing and just return success |
| *data_len = 0; |
| |
| return IPMI_CC_OK; |
| } |
| |
| //---------------------------------------------------------------------- |
| // Get System GUID (CMD_APP_GET_SYS_GUID) |
| //---------------------------------------------------------------------- |
| #if BIC_ENABLED |
| ipmi::RspType<std::vector<uint8_t>> |
| ipmiAppGetSysGUID(ipmi::Context::ptr ctx, std::vector<uint8_t> reqData) |
| |
| { |
| std::vector<uint8_t> respData; |
| |
| uint8_t bicAddr = (uint8_t)ctx->hostIdx << 2; |
| |
| if (sendBicCmd(ctx->netFn, ctx->cmd, bicAddr, reqData, respData)) |
| return ipmi::responseUnspecifiedError(); |
| |
| return ipmi::responseSuccess(respData); |
| } |
| |
| #else |
| ipmi_ret_t ipmiAppGetSysGUID(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t, |
| ipmi_response_t response, ipmi_data_len_t data_len, |
| ipmi_context_t) |
| { |
| uint8_t* res = reinterpret_cast<uint8_t*>(response); |
| if (getSystemGUID(res)) |
| { |
| return IPMI_CC_UNSPECIFIED_ERROR; |
| } |
| *data_len = GUID_SIZE; |
| return IPMI_CC_OK; |
| } |
| |
| #endif |
| |
| //---------------------------------------------------------------------- |
| // Platform specific functions for storing app data |
| //---------------------------------------------------------------------- |
| |
| void flush_app_data() |
| { |
| std::ofstream file(JSON_APP_DATA_FILE); |
| file << appData; |
| file.close(); |
| return; |
| } |
| |
| static int platSetSysFWVer(uint8_t* ver, const std::string key) |
| { |
| std::stringstream ss; |
| int i; |
| |
| /* TODO: implement byte 1: Set selector |
| * byte 2: encodeing, currently only supported |
| * ASCII which is value 0, UTF and unicode are |
| * not supported yet. |
| */ |
| if (ver[1] & 0x0f) |
| return -1; |
| |
| for (i = 3; i < 3 + ver[2]; i++) |
| { |
| ss << (char)ver[i]; |
| } |
| |
| appData[key] = ss.str(); |
| flush_app_data(); |
| |
| return 0; |
| } |
| |
| static int platGetSysFWVer(std::vector<uint8_t>& respData, |
| const std::string key) |
| { |
| int len = -1; |
| |
| if (!appData.contains(std::string(key))) |
| { |
| return -1; |
| } |
| std::string str = appData[key].get<std::string>(); |
| |
| respData.push_back(0); // byte 1: Set selector not supported |
| respData.push_back(0); // byte 2: Only ASCII supported |
| |
| len = str.length(); |
| respData.push_back(len); // byte 3: Size of version |
| |
| for (auto c : str) |
| { |
| respData.push_back(c); |
| } |
| |
| // Remaining byte fill to 0 |
| for (int i = 0; i < SIZE_SYSFW_VER - (len + 3); i++) |
| { |
| respData.push_back(0); |
| } |
| |
| return (len + 3); |
| } |
| |
| //---------------------------------------------------------------------- |
| // Set Sys Info Params (IPMI/Sec 22.14a) (CMD_APP_SET_SYS_INFO_PARAMS) |
| //---------------------------------------------------------------------- |
| ipmi::RspType<uint8_t> ipmiAppSetSysInfoParams(ipmi::Context::ptr ctx, |
| std::vector<uint8_t> req) |
| { |
| |
| uint8_t param = req[0]; |
| uint8_t req_len = req.size(); |
| std::optional<size_t> hostId = findHost(ctx->hostIdx); |
| |
| if (!hostId) |
| { |
| phosphor::logging::log<phosphor::logging::level::ERR>( |
| "Invalid Host Id received"); |
| return ipmi::responseInvalidCommand(); |
| } |
| |
| switch (param) |
| { |
| case SYS_INFO_PARAM_SET_IN_PROG: |
| sysInfoParams.set_in_prog = req[1]; |
| break; |
| case SYS_INFO_PARAM_SYSFW_VER: |
| { |
| memcpy(sysInfoParams.sysfw_ver, &req[1], SIZE_SYSFW_VER); |
| std::string version_key = KEY_SYSFW_VER + std::to_string(*hostId); |
| if (platSetSysFWVer(sysInfoParams.sysfw_ver, version_key)) |
| return ipmi::responseSystemInfoParamterNotSupportCommand(); |
| break; |
| } |
| case SYS_INFO_PARAM_SYS_NAME: |
| memcpy(sysInfoParams.sys_name, &req[1], SIZE_SYS_NAME); |
| break; |
| case SYS_INFO_PARAM_PRI_OS_NAME: |
| memcpy(sysInfoParams.pri_os_name, &req[1], SIZE_OS_NAME); |
| break; |
| case SYS_INFO_PARAM_PRESENT_OS_NAME: |
| memcpy(sysInfoParams.present_os_name, &req[1], SIZE_OS_NAME); |
| break; |
| case SYS_INFO_PARAM_PRESENT_OS_VER: |
| memcpy(sysInfoParams.present_os_ver, &req[1], SIZE_OS_VER); |
| break; |
| case SYS_INFO_PARAM_BMC_URL: |
| memcpy(sysInfoParams.bmc_url, &req[1], SIZE_BMC_URL); |
| break; |
| case SYS_INFO_PARAM_OS_HV_URL: |
| memcpy(sysInfoParams.os_hv_url, &req[1], SIZE_OS_HV_URL); |
| break; |
| case SYS_INFO_PARAM_BIOS_CURRENT_BOOT_LIST: |
| memcpy(sysInfoParams.bios_current_boot_list, &req[1], req_len); |
| appData[KEY_BIOS_BOOT_LEN] = req_len; |
| flush_app_data(); |
| break; |
| case SYS_INFO_PARAM_BIOS_FIXED_BOOT_DEVICE: |
| if (SIZE_BIOS_FIXED_BOOT_DEVICE != req_len) |
| break; |
| memcpy(sysInfoParams.bios_fixed_boot_device, &req[1], |
| SIZE_BIOS_FIXED_BOOT_DEVICE); |
| break; |
| case SYS_INFO_PARAM_BIOS_RSTR_DFLT_SETTING: |
| if (SIZE_BIOS_RSTR_DFLT_SETTING != req_len) |
| break; |
| memcpy(sysInfoParams.bios_rstr_dflt_setting, &req[1], |
| SIZE_BIOS_RSTR_DFLT_SETTING); |
| break; |
| case SYS_INFO_PARAM_LAST_BOOT_TIME: |
| if (SIZE_LAST_BOOT_TIME != req_len) |
| break; |
| memcpy(sysInfoParams.last_boot_time, &req[1], SIZE_LAST_BOOT_TIME); |
| break; |
| default: |
| return ipmi::responseSystemInfoParamterNotSupportCommand(); |
| break; |
| } |
| |
| return ipmi::responseSuccess(); |
| } |
| |
| //---------------------------------------------------------------------- |
| // Get Sys Info Params (IPMI/Sec 22.14b) (CMD_APP_GET_SYS_INFO_PARAMS) |
| //---------------------------------------------------------------------- |
| ipmi::RspType<std::vector<uint8_t>> |
| ipmiAppGetSysInfoParams(ipmi::Context::ptr ctx, uint8_t, uint8_t param, |
| uint8_t, uint8_t) |
| { |
| int len; |
| std::vector<uint8_t> respData; |
| respData.push_back(1); // Parameter revision |
| |
| std::optional<size_t> hostId = findHost(ctx->hostIdx); |
| |
| if (!hostId) |
| { |
| phosphor::logging::log<phosphor::logging::level::ERR>( |
| "Invalid Host Id received"); |
| return ipmi::responseInvalidCommand(); |
| } |
| |
| switch (param) |
| { |
| case SYS_INFO_PARAM_SET_IN_PROG: |
| respData.push_back(sysInfoParams.set_in_prog); |
| break; |
| case SYS_INFO_PARAM_SYSFW_VER: |
| { |
| std::string version_key = KEY_SYSFW_VER + std::to_string(*hostId); |
| if ((platGetSysFWVer(respData, version_key)) < 0) |
| return ipmi::responseSystemInfoParamterNotSupportCommand(); |
| break; |
| } |
| case SYS_INFO_PARAM_SYS_NAME: |
| respData.insert(respData.end(), std::begin(sysInfoParams.sys_name), |
| std::end(sysInfoParams.sys_name)); |
| break; |
| case SYS_INFO_PARAM_PRI_OS_NAME: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.pri_os_name), |
| std::end(sysInfoParams.pri_os_name)); |
| break; |
| case SYS_INFO_PARAM_PRESENT_OS_NAME: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.present_os_name), |
| std::end(sysInfoParams.present_os_name)); |
| break; |
| case SYS_INFO_PARAM_PRESENT_OS_VER: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.present_os_ver), |
| std::end(sysInfoParams.present_os_ver)); |
| break; |
| case SYS_INFO_PARAM_BMC_URL: |
| respData.insert(respData.end(), std::begin(sysInfoParams.bmc_url), |
| std::end(sysInfoParams.bmc_url)); |
| break; |
| case SYS_INFO_PARAM_OS_HV_URL: |
| respData.insert(respData.end(), std::begin(sysInfoParams.os_hv_url), |
| std::end(sysInfoParams.os_hv_url)); |
| break; |
| case SYS_INFO_PARAM_BIOS_CURRENT_BOOT_LIST: |
| len = appData[KEY_BIOS_BOOT_LEN].get<uint8_t>(); |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.bios_current_boot_list), |
| std::begin(sysInfoParams.bios_current_boot_list) + |
| len); |
| break; |
| case SYS_INFO_PARAM_BIOS_FIXED_BOOT_DEVICE: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.bios_fixed_boot_device), |
| std::end(sysInfoParams.bios_fixed_boot_device)); |
| break; |
| case SYS_INFO_PARAM_BIOS_RSTR_DFLT_SETTING: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.bios_rstr_dflt_setting), |
| std::end(sysInfoParams.bios_rstr_dflt_setting)); |
| break; |
| case SYS_INFO_PARAM_LAST_BOOT_TIME: |
| respData.insert(respData.end(), |
| std::begin(sysInfoParams.last_boot_time), |
| std::end(sysInfoParams.last_boot_time)); |
| break; |
| default: |
| return ipmi::responseSystemInfoParamterNotSupportCommand(); |
| break; |
| } |
| |
| return ipmi::responseSuccess(respData); |
| } |
| |
| void registerAPPFunctions() |
| { |
| /* Get App data stored in json file */ |
| std::ifstream file(JSON_APP_DATA_FILE); |
| if (file) |
| { |
| file >> appData; |
| file.close(); |
| } |
| |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_GET_SELFTEST_RESULTS, NULL, |
| ipmiAppGetSTResults, |
| PRIVILEGE_USER); // Get Self Test Results |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_MFR_TEST_ON, NULL, |
| ipmiAppMfrTestOn, |
| PRIVILEGE_USER); // Manufacturing Test On |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_SET_GLOBAL_ENABLES, NULL, |
| ipmiAppSetGlobalEnables, |
| PRIVILEGE_USER); // Set Global Enables |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_GET_GLOBAL_ENABLES, NULL, |
| ipmiAppGetGlobalEnables, |
| PRIVILEGE_USER); // Get Global Enables |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_CLEAR_MESSAGE_FLAGS, NULL, |
| ipmiAppClearMsgFlags, |
| PRIVILEGE_USER); // Clear Message flags |
| #if BIC_ENABLED |
| ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp, |
| ipmi::app::cmdGetSystemGuid, ipmi::Privilege::User, |
| ipmiAppGetSysGUID); |
| #else |
| ipmiPrintAndRegister(NETFUN_APP, CMD_APP_GET_SYS_GUID, NULL, |
| ipmiAppGetSysGUID, |
| PRIVILEGE_USER); // Get System GUID |
| #endif |
| ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp, |
| ipmi::app::cmdSetSystemInfoParameters, |
| ipmi::Privilege::User, ipmiAppSetSysInfoParams); |
| |
| ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp, |
| ipmi::app::cmdGetSystemInfoParameters, |
| ipmi::Privilege::User, ipmiAppGetSysInfoParams); |
| return; |
| } |
| |
| } // namespace ipmi |