diff --git a/src/smbiosmdrv2.cpp b/src/smbiosmdrv2.cpp
deleted file mode 100644
index 80e8620..0000000
--- a/src/smbiosmdrv2.cpp
+++ /dev/null
@@ -1,1119 +0,0 @@
-/*
-// Copyright (c) 2018 Intel Corporation
-//
-// 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 <errno.h>
-
-#include <commandutils.hpp>
-#include <cstdint>
-#include <ipmid/api.hpp>
-#include <ipmid/utils.hpp>
-#include <phosphor-ipmi-host/ipmid.hpp>
-#include <phosphor-logging/elog-errors.hpp>
-#include <phosphor-logging/log.hpp>
-#include <smbiosmdrv2.hpp>
-#include <string>
-#include <vector>
-#include <xyz/openbmc_project/Common/error.hpp>
-
-constexpr const char* DBUS_PROPERTIES = "org.freedesktop.DBus.Properties";
-constexpr const char* MDRV2_PATH = "/xyz/openbmc_project/Smbios/MDR_V2";
-constexpr const char* MDRV2_INTERFACE = "xyz.openbmc_project.Smbios.MDR_V2";
-constexpr const int LAST_AGENT_INDEX = -1;
-constexpr const uint16_t LAST_AGENT_ID = 0xFFFF;
-
-static void register_netfn_smbiosmdrv2_functions() __attribute__((constructor));
-
-int gentLookup(const uint16_t& agentId, const std::string& service)
-{
-    int agentIndex = -1;
-
-    if (LAST_AGENT_ID == agentId)
-    {
-        return LAST_AGENT_INDEX;
-    }
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "AgentLookup");
-    method.append(agentId);
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get agent index, sdbusplus call failed");
-        return -1;
-    }
-    reply.read(agentIndex);
-
-    return agentIndex;
-}
-
-int findLockHandle(const uint16_t& lockHandle, const std::string& service)
-{
-    int idIndex = -1;
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "FindLockHandle");
-    method.append(lockHandle);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error find lock handle",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return -1;
-    }
-    reply.read(idIndex);
-
-    return idIndex;
-}
-
-int sdplusMdrv2GetProperty(const std::string& name,
-                           sdbusplus::message::variant<uint8_t>& value,
-                           const std::string& service)
-{
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, DBUS_PROPERTIES, "Get");
-    method.append(MDRV2_INTERFACE, name);
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get property, sdbusplus call failed");
-        return -1;
-    }
-    reply.read(value);
-
-    return 0;
-}
-
-int findDataId(const uint8_t* dataInfo, const size_t& len,
-               const std::string& service)
-{
-    int idIndex = -1;
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "FindIdIndex");
-    std::vector<uint8_t> info;
-    for (int index = 0; index < len; index++)
-    {
-        info.push_back(dataInfo[index]);
-    }
-    method.append(info);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error find id index",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return -1;
-    }
-    reply.read(idIndex);
-
-    return idIndex;
-}
-
-ipmi_ret_t cmd_mdr2_agent_status(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                 ipmi_request_t request,
-                                 ipmi_response_t response,
-                                 ipmi_data_len_t data_len,
-                                 ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiGetAgentStatus*>(request);
-    auto dataOut = reinterpret_cast<uint8_t*>(response);
-    std::vector<uint8_t> status;
-
-    if (*data_len != sizeof(MDRiiGetAgentStatus))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "AgentStatus");
-    method.append(requestData->dirVersion);
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get agent status",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(status);
-
-    if (status.size() != sizeof(MDRiiAgentStatusResponse))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Get agent status response length not valid");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    *data_len = static_cast<size_t>(status.size());
-    std::copy(&status[0], &status[*data_len], dataOut);
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_get_dir(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                            ipmi_request_t request, ipmi_response_t response,
-                            ipmi_data_len_t data_len, ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiGetDirRequest*>(request);
-    auto dataOut = reinterpret_cast<uint8_t*>(response);
-    std::vector<uint8_t> dirInfo;
-
-    if (*data_len != sizeof(MDRiiGetDirRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::variant<uint8_t> value = 0;
-    if (0 != sdplusMdrv2GetProperty("DirEntries", value, service))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error getting DirEnries");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    if (requestData->dirIndex > std::get<uint8_t>(value))
-    {
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "GetDir");
-
-    method.append(requestData->dirIndex);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get dir",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(dirInfo);
-
-    if (dirInfo.size() < sizeof(MDRiiGetDirResponse))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get dir, response length invalid");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    auto responseData = reinterpret_cast<MDRiiGetDirResponse*>(dirInfo.data());
-
-    *data_len = dirInfo.size();
-
-    if (*data_len > MAX_IPMI_BUFFER) // length + completion code should no more
-                                     // than MAX_IPMI_BUFFER
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Data length send from service is invalid");
-        *data_len = 0;
-        return IPMI_CC_RESPONSE_ERROR;
-    }
-
-    std::copy(&dirInfo[0], &dirInfo[*data_len], dataOut);
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_get_data_info(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                  ipmi_request_t request,
-                                  ipmi_response_t response,
-                                  ipmi_data_len_t data_len,
-                                  ipmi_context_t context)
-{
-    auto requestData =
-        reinterpret_cast<const MDRiiGetDataInfoRequest*>(request);
-    auto dataOut = reinterpret_cast<uint8_t*>(response);
-    std::vector<uint8_t> res;
-
-    if (*data_len < sizeof(MDRiiGetDataInfoRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex =
-        findDataId(requestData->dataSetInfo.dataInfo,
-                   sizeof(requestData->dataSetInfo.dataInfo), service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "GetDataInfo");
-
-    method.append(idIndex);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get data info",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(res);
-
-    if (res.size() != sizeof(MDRiiGetDataInfoResponse))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Get data info response length not invalid");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    *data_len = static_cast<size_t>(res.size());
-    std::copy(&res[0], &res[*data_len], dataOut);
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_lock_data(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                              ipmi_request_t request, ipmi_response_t response,
-                              ipmi_data_len_t data_len, ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiLockDataRequest*>(request);
-    auto responseData = reinterpret_cast<MDRiiLockDataResponse*>(response);
-
-    std::tuple<bool, uint8_t, uint16_t, uint32_t, uint32_t, uint32_t> res;
-
-    if (*data_len < sizeof(MDRiiLockDataRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex =
-        findDataId(requestData->dataSetInfo.dataInfo,
-                   sizeof(requestData->dataSetInfo.dataInfo), service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "LockData");
-
-    method.append((uint8_t)idIndex, requestData->timeout);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        if (reply.get_errno() == EBUSY)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Lock Data failed - cannot lock idIndex");
-            return IPMI_CC_PARAMETER_NOT_SUPPORT_IN_PRESENT_STATE;
-        }
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error lock data",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(res);
-
-    if (std::get<0>(res) == false)
-    {
-        return IPMI_CC_PARAMETER_NOT_SUPPORT_IN_PRESENT_STATE;
-    }
-
-    *data_len = sizeof(MDRiiLockDataResponse);
-
-    responseData->mdrVersion = std::get<1>(res);
-    responseData->lockHandle = std::get<2>(res);
-    responseData->dataLength = std::get<3>(res);
-    responseData->xferAddress = std::get<4>(res);
-    responseData->xferLength = std::get<5>(res);
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_unlock_data(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                ipmi_request_t request,
-                                ipmi_response_t response,
-                                ipmi_data_len_t data_len,
-                                ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiUnlockDataRequest*>(request);
-    std::string resStatus;
-
-    if (*data_len != sizeof(MDRiiUnlockDataRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex = findLockHandle(requestData->lockHandle, service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "UnLockData");
-    method.append((uint8_t)idIndex);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        if (reply.get_errno() == EBUSY)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Unlock Data failed - cannot unlock idIndex");
-            return IPMI_CC_PARAMETER_NOT_SUPPORT_IN_PRESENT_STATE;
-        }
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error unlock data",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(resStatus);
-
-    if (resStatus != "success")
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Agent unlock Invalid lock status.");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_get_data_block(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                   ipmi_request_t request,
-                                   ipmi_response_t response,
-                                   ipmi_data_len_t data_len,
-                                   ipmi_context_t context)
-{
-    auto requestData =
-        reinterpret_cast<const MDRiiGetDataBlockRequest*>(request);
-    auto responseData = reinterpret_cast<MDRiiGetDataBlockResponse*>(response);
-    std::tuple<uint8_t, uint32_t, uint32_t, std::vector<uint8_t>> res;
-    std::vector<uint8_t> resData;
-    uint8_t status = 1;
-
-    if (*data_len != sizeof(MDRiiGetDataBlockRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex = findLockHandle(requestData->lockHandle, service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "GetDataBlock");
-    method.append((uint8_t)idIndex, requestData->xferOffset,
-                  requestData->xferLength);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error get data block",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(res);
-
-    // Get the status of get data block, 0 means succeed
-    status = std::get<0>(res);
-    if (status == 1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Request data offset is outside of range.");
-        return IPMI_CC_CANNOT_RETURN_NUMBER_OF_REQUESTED_DATA_BYTES;
-    }
-    else if (status != 0)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Get data block unexpected error.");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    responseData->xferLength = std::get<1>(res);
-    if (responseData->xferLength > requestData->xferLength)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Get data block unexpected error.");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    responseData->checksum = std::get<2>(res);
-
-    resData = std::get<3>(res);
-
-    *data_len = sizeof(responseData->xferLength) +
-                sizeof(responseData->checksum) + resData.size();
-
-    if (*data_len > MAX_IPMI_BUFFER) // length + completion code should no more
-                                     // than MAX_IPMI_BUFFER
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Data length send from service is invalid");
-        *data_len = 0;
-        return IPMI_CC_RESPONSE_ERROR;
-    }
-
-    std::copy(resData.begin(), resData.end(), responseData->data);
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_send_dir(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                             ipmi_request_t request, ipmi_response_t response,
-                             ipmi_data_len_t data_len, ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiSendDirRequest*>(request);
-    std::vector<uint8_t> idVector;
-    bool teminate = false;
-
-    if (*data_len != sizeof(MDRiiSendDirRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    if ((requestData->dirIndex + requestData->returnedEntries) > maxDirEntries)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Too many directory entries");
-        return IPMI_CC_STORGE_LEAK;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "SendDir");
-    method.append(requestData->dirVersion, requestData->dirIndex,
-                  requestData->returnedEntries, requestData->remainingEntries);
-    uint8_t* reqPoint;
-    for (int index = 0; index < requestData->returnedEntries; index++)
-    {
-        reqPoint = (uint8_t*)&(requestData->data[index]);
-        std::copy(reqPoint, sizeof(Mdr2DirEntry) + reqPoint, idVector.data());
-    }
-    method.append(idVector);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error send dir",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(teminate);
-
-    *data_len = 1;
-    if (teminate == false)
-        *(static_cast<uint8_t*>(response)) = 0;
-    else
-        *(static_cast<uint8_t*>(response)) = 1;
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_data_info_offer(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                    ipmi_request_t request,
-                                    ipmi_response_t response,
-                                    ipmi_data_len_t data_len,
-                                    ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiOfferDataInfo*>(request);
-    auto dataOut = reinterpret_cast<uint8_t*>(response);
-    std::vector<uint8_t> dataInfo;
-
-    if (*data_len != sizeof(MDRiiOfferDataInfo))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "GetDataOffer");
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        if (reply.get_errno() == EBUSY)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data info offer failed - not available to update data "
-                "into agent at present");
-            return IPMI_CC_PARAMETER_NOT_SUPPORT_IN_PRESENT_STATE;
-        }
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error send data info offer",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(dataInfo);
-    if (dataInfo.size() != sizeof(MDRiiOfferDataInfoResponse))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error send data info offer, return length invalid");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    *data_len = dataInfo.size();
-    std::copy(dataInfo.begin(), dataInfo.end(), dataOut);
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_send_data_info(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                   ipmi_request_t request,
-                                   ipmi_response_t response,
-                                   ipmi_data_len_t data_len,
-                                   ipmi_context_t context)
-{
-    auto requestData =
-        reinterpret_cast<const MDRiiSendDataInfoRequest*>(request);
-    bool entryChanged = true;
-
-    if (*data_len != sizeof(MDRiiSendDataInfoRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    if (requestData->dataLength > smbiosTableStorageSize)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Requested data length is out of SMBIOS Table storage size.");
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex =
-        findDataId(requestData->dataSetInfo.dataInfo,
-                   sizeof(requestData->dataSetInfo.dataInfo), service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "SendDataInfo");
-
-    method.append((uint8_t)idIndex, requestData->validFlag,
-                  requestData->dataLength, requestData->dataVersion,
-                  requestData->timeStamp);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error send data info",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(entryChanged);
-
-    *data_len = 1;
-
-    if (entryChanged)
-    {
-        *(static_cast<uint8_t*>(response)) = 1;
-    }
-    else
-    {
-        *(static_cast<uint8_t*>(response)) = 0;
-    }
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_data_start(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                               ipmi_request_t request, ipmi_response_t response,
-                               ipmi_data_len_t data_len, ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiDataStartRequest*>(request);
-    auto responseData = reinterpret_cast<MDRiiDataStartResponse*>(response);
-    std::vector<uint8_t> idVector;
-
-    if (*data_len != sizeof(MDRiiDataStartRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    if (requestData->dataLength > smbiosTableStorageSize)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Requested data length is out of SMBIOS Table storage size.");
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    if ((requestData->xferLength + requestData->xferAddress) > mdriiSMSize)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid data address and size");
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex =
-        findDataId(requestData->dataSetInfo.dataInfo,
-                   sizeof(requestData->dataSetInfo.dataInfo), service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "DataStart");
-
-    for (uint8_t infoIndex = 0; infoIndex < sizeof(DataIdStruct); infoIndex++)
-    {
-        idVector.push_back(requestData->dataSetInfo.dataInfo[infoIndex]);
-    }
-    method.append((uint8_t)idIndex, idVector, requestData->dataLength,
-                  requestData->xferAddress, requestData->xferLength,
-                  requestData->timeout);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        int errNumber = reply.get_errno();
-        if (errNumber == ENOMEM)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data start failed - cannot map share memory");
-            return IPMI_CC_UNSPECIFIED_ERROR;
-        }
-        else if (errNumber == EINVAL)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Invalid data address and size");
-            return IPMI_CC_PARM_OUT_OF_RANGE;
-        }
-        else
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Error Send Data Start",
-                phosphor::logging::entry("SERVICE=%s", service.c_str()),
-                phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-            return IPMI_CC_UNSPECIFIED_ERROR;
-        }
-    }
-    uint8_t xferStartAck = 0;
-    uint16_t sessionHandle = 0;
-    reply.read(xferStartAck, sessionHandle);
-    responseData->sessionHandle = sessionHandle;
-    responseData->xferStartAck = xferStartAck;
-    if (responseData->xferStartAck == 0)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Send data start unexpected error");
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-
-    *data_len = sizeof(MDRiiDataStartResponse);
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_data_done(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                              ipmi_request_t request, ipmi_response_t response,
-                              ipmi_data_len_t data_len, ipmi_context_t context)
-{
-    auto requestData = reinterpret_cast<const MDRiiDataDoneRequest*>(request);
-    std::string resStatus;
-
-    if (*data_len != sizeof(MDRiiDataDoneRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex = findLockHandle(requestData->lockHandle, service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "DataDone");
-    method.append((uint8_t)idIndex);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        if (reply.get_errno() == EBUSY)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data done failed - cannot unlock idIndex");
-            return IPMI_CC_DESTINATION_UNAVAILABLE;
-        }
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Error Send Data done",
-            phosphor::logging::entry("SERVICE=%s", service.c_str()),
-            phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-        return IPMI_CC_UNSPECIFIED_ERROR;
-    }
-    reply.read(resStatus);
-
-    if (resStatus != "success")
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Data done failure.");
-        return IPMI_CC_DESTINATION_UNAVAILABLE;
-    }
-
-    return IPMI_CC_OK;
-}
-
-ipmi_ret_t cmd_mdr2_send_data_block(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
-                                    ipmi_request_t request,
-                                    ipmi_response_t response,
-                                    ipmi_data_len_t data_len,
-                                    ipmi_context_t context)
-{
-    auto requestData =
-        reinterpret_cast<const MDRiiSendDataBlockRequest*>(request);
-    std::string resStatus;
-
-    if (*data_len != sizeof(MDRiiSendDataBlockRequest))
-    {
-        *data_len = 0;
-        return IPMI_CC_REQ_DATA_LEN_INVALID;
-    }
-
-    *data_len = 0;
-
-    std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
-    std::string service = ipmi::getService(*bus, MDRV2_INTERFACE, MDRV2_PATH);
-
-    int agentIndex = agentLookup(requestData->agentId, service);
-    if (agentIndex == -1)
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Unknown agent id",
-            phosphor::logging::entry("ID=%x", requestData->agentId));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    int idIndex = findLockHandle(requestData->lockHandle, service);
-
-    if ((idIndex < 0) || (idIndex >= maxDirEntries))
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "Invalid Data ID", phosphor::logging::entry("IDINDEX=%x", idIndex));
-        return IPMI_CC_PARM_OUT_OF_RANGE;
-    }
-
-    sdbusplus::message::message method = bus->new_method_call(
-        service.c_str(), MDRV2_PATH, MDRV2_INTERFACE, "SendDataBlock");
-    method.append((uint8_t)idIndex, requestData->xferOffset,
-                  requestData->xferLength, requestData->checksum);
-
-    sdbusplus::message::message reply = bus->call(method);
-    if (reply.is_method_error())
-    {
-        int errNumber = reply.get_errno();
-        if (errNumber == EINVAL)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data block Invalid checksum");
-            return IPMI_CC_OEM_INVALID_CHECKSUM;
-        }
-        else if (errNumber == ENOBUFS)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data block Invalid offset/length");
-            return IPMI_CC_REQUEST_DATA_FIELD_LENGTH_LIMIT_EXCEEDED;
-        }
-        else if (errNumber == EBUSY)
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Send data block failed, other data is updating");
-            return IPMI_CC_DESTINATION_UNAVAILABLE;
-        }
-        else
-        {
-            phosphor::logging::log<phosphor::logging::level::ERR>(
-                "Error Send data block",
-                phosphor::logging::entry("SERVICE=%s", service.c_str()),
-                phosphor::logging::entry("PATH=%s", MDRV2_PATH));
-            return IPMI_CC_UNSPECIFIED_ERROR;
-        }
-    }
-    reply.read(resStatus);
-
-    if (resStatus != "success")
-    {
-        phosphor::logging::log<phosphor::logging::level::ERR>(
-            "send data block failure.");
-        return IPMI_CC_DESTINATION_UNAVAILABLE;
-    }
-
-    return IPMI_CC_OK;
-}
-
-static void register_netfn_smbiosmdrv2_functions(void)
-{
-    // MDR V2 Command
-    // <Get MDRII Status Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_AGENT_STATUS,
-                           NULL, cmd_mdr2_agent_status, PRIVILEGE_OPERATOR);
-
-    // <Get MDRII Directory Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_GET_DIR, NULL,
-                           cmd_mdr2_get_dir, PRIVILEGE_OPERATOR);
-
-    // <Get MDRII Data Info Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_GET_DATA_INFO,
-                           NULL, cmd_mdr2_get_data_info, PRIVILEGE_OPERATOR);
-
-    // <Lock MDRII Data Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_LOCK_DATA, NULL,
-                           cmd_mdr2_lock_data, PRIVILEGE_OPERATOR);
-
-    // <Unlock MDRII Data Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_UNLOCK_DATA,
-                           NULL, cmd_mdr2_unlock_data, PRIVILEGE_OPERATOR);
-
-    // <Get MDRII Data Block Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_GET_DATA_BLOCK,
-                           NULL, cmd_mdr2_get_data_block, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Directory Command>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_SEND_DIR, NULL,
-                           cmd_mdr2_send_dir, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Info Offer>
-    ipmi_register_callback(
-        NETFUN_INTEL_APP_OEM,
-        IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_SEND_DATA_INFO_OFFER, NULL,
-        cmd_mdr2_data_info_offer, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Data Info>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_SEND_DATA_INFO,
-                           NULL, cmd_mdr2_send_data_info, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Data Start>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_DATA_START, NULL,
-                           cmd_mdr2_data_start, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Data Done>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_DATA_DONE, NULL,
-                           cmd_mdr2_data_done, PRIVILEGE_OPERATOR);
-
-    // <Send MDRII Data Block>
-    ipmi_register_callback(NETFUN_INTEL_APP_OEM,
-                           IPMI_NETFN_INTEL_OEM_APP_CMD::MDRII_SEND_DATA_BLOCK,
-                           NULL, cmd_mdr2_send_data_block, PRIVILEGE_OPERATOR);
-}
