appcommands: support multi-host set/get fw version
Support multi-host set/get fw version in IPMI set/get system info command
Test case:
Get bios version from different host success.
Signed-off-by: Bonnie Lo <Bonnie_Lo@wiwynn.com>
Change-Id: Id2e1598bbe8f7d834fe3c31710384a7dbbe537e0
diff --git a/src/appcommands.cpp b/src/appcommands.cpp
index a4aa3cc..e0ae4ed 100644
--- a/src/appcommands.cpp
+++ b/src/appcommands.cpp
@@ -51,6 +51,11 @@
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";
@@ -247,7 +252,7 @@
return;
}
-static int platSetSysFWVer(uint8_t* ver)
+static int platSetSysFWVer(uint8_t* ver, const std::string key)
{
std::stringstream ss;
int i;
@@ -265,23 +270,39 @@
ss << (char)ver[i];
}
- appData[KEY_SYSFW_VER] = ss.str();
+ appData[key] = ss.str();
flush_app_data();
return 0;
}
-static int platGetSysFWVer(uint8_t* ver)
+static int platGetSysFWVer(std::vector<uint8_t>& respData,
+ const std::string key)
{
- std::string str = appData[KEY_SYSFW_VER].get<std::string>();
- int len;
+ int len = -1;
- *ver++ = 0; // byte 1: Set selector not supported
- *ver++ = 0; // byte 2: Only ASCII supported
+ 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();
- *ver++ = len;
- memcpy(ver, str.data(), len);
+ 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);
}
@@ -289,16 +310,20 @@
//----------------------------------------------------------------------
// Set Sys Info Params (IPMI/Sec 22.14a) (CMD_APP_SET_SYS_INFO_PARAMS)
//----------------------------------------------------------------------
-ipmi_ret_t ipmiAppSetSysInfoParams(ipmi_netfn_t, ipmi_cmd_t,
- ipmi_request_t request, ipmi_response_t,
- ipmi_data_len_t data_len, ipmi_context_t)
+ipmi::RspType<uint8_t> ipmiAppSetSysInfoParams(ipmi::Context::ptr ctx,
+ std::vector<uint8_t> req)
{
- uint8_t* req = reinterpret_cast<uint8_t*>(request);
uint8_t param = req[0];
- uint8_t req_len = *data_len;
+ uint8_t req_len = req.size();
+ std::optional<size_t> hostId = findHost(ctx->hostIdx);
- *data_len = 0;
+ if (!hostId)
+ {
+ phosphor::logging::log<phosphor::logging::level::ERR>(
+ "Invalid Host Id received");
+ return ipmi::responseInvalidCommand();
+ }
switch (param)
{
@@ -306,10 +331,13 @@
sysInfoParams.set_in_prog = req[1];
break;
case SYS_INFO_PARAM_SYSFW_VER:
+ {
memcpy(sysInfoParams.sysfw_ver, &req[1], SIZE_SYSFW_VER);
- if (platSetSysFWVer(sysInfoParams.sysfw_ver))
- return IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED;
+ 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;
@@ -351,89 +379,100 @@
memcpy(sysInfoParams.last_boot_time, &req[1], SIZE_LAST_BOOT_TIME);
break;
default:
- return IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED;
+ return ipmi::responseSystemInfoParamterNotSupportCommand();
break;
}
- return IPMI_CC_OK;
+ return ipmi::responseSuccess();
}
//----------------------------------------------------------------------
// Get Sys Info Params (IPMI/Sec 22.14b) (CMD_APP_GET_SYS_INFO_PARAMS)
//----------------------------------------------------------------------
-ipmi_ret_t ipmiAppGetSysInfoParams(ipmi_netfn_t, ipmi_cmd_t,
- ipmi_request_t request,
- ipmi_response_t response,
- ipmi_data_len_t data_len, ipmi_context_t)
+ipmi::RspType<std::vector<uint8_t>>
+ ipmiAppGetSysInfoParams(ipmi::Context::ptr ctx, uint8_t, uint8_t param,
+ uint8_t, uint8_t)
{
- uint8_t* req = reinterpret_cast<uint8_t*>(request);
- uint8_t* res = reinterpret_cast<uint8_t*>(response);
-
- uint8_t param = req[1];
int len;
+ std::vector<uint8_t> respData;
+ respData.push_back(1); // Parameter revision
- *res++ = 1; // Parameter revision
- *data_len = 1;
+ 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:
- *res++ = sysInfoParams.set_in_prog;
- *data_len += 1;
+ respData.push_back(sysInfoParams.set_in_prog);
break;
case SYS_INFO_PARAM_SYSFW_VER:
- if ((len = platGetSysFWVer(res)) < 0)
- return IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED;
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.sys_name, SIZE_SYS_NAME);
- *data_len += SIZE_SYS_NAME;
+ respData.insert(respData.end(), std::begin(sysInfoParams.sys_name),
+ std::end(sysInfoParams.sys_name));
break;
case SYS_INFO_PARAM_PRI_OS_NAME:
- memcpy(res, sysInfoParams.pri_os_name, SIZE_OS_NAME);
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.present_os_name, SIZE_OS_NAME);
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.present_os_ver, SIZE_OS_VER);
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.bmc_url, SIZE_BMC_URL);
- *data_len += SIZE_BMC_URL;
+ respData.insert(respData.end(), std::begin(sysInfoParams.bmc_url),
+ std::end(sysInfoParams.bmc_url));
break;
case SYS_INFO_PARAM_OS_HV_URL:
- memcpy(res, sysInfoParams.os_hv_url, SIZE_OS_HV_URL);
- *data_len += SIZE_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>();
- memcpy(res, sysInfoParams.bios_current_boot_list, len);
- *data_len += len;
+ 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:
- memcpy(res, sysInfoParams.bios_fixed_boot_device,
- SIZE_BIOS_FIXED_BOOT_DEVICE);
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.bios_rstr_dflt_setting,
- SIZE_BIOS_RSTR_DFLT_SETTING);
- *data_len += SIZE_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:
- memcpy(res, sysInfoParams.last_boot_time, SIZE_LAST_BOOT_TIME);
- *data_len += SIZE_LAST_BOOT_TIME;
+ respData.insert(respData.end(),
+ std::begin(sysInfoParams.last_boot_time),
+ std::end(sysInfoParams.last_boot_time));
break;
default:
- return IPMI_CC_SYSTEM_INFO_PARAMETER_NOT_SUPPORTED;
+ return ipmi::responseSystemInfoParamterNotSupportCommand();
break;
}
- return IPMI_CC_OK;
+
+ return ipmi::responseSuccess(respData);
}
void registerAPPFunctions()
@@ -470,12 +509,13 @@
ipmiAppGetSysGUID,
PRIVILEGE_USER); // Get System GUID
#endif
- ipmiPrintAndRegister(NETFUN_APP, CMD_APP_SET_SYS_INFO_PARAMS, NULL,
- ipmiAppSetSysInfoParams,
- PRIVILEGE_USER); // Set Sys Info Params
- ipmiPrintAndRegister(NETFUN_APP, CMD_APP_GET_SYS_INFO_PARAMS, NULL,
- ipmiAppGetSysInfoParams,
- PRIVILEGE_USER); // Get Sys Info Params
+ 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;
}