/*
// 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);
}
