#include "gpioMonitor.hpp"

#include "common_utility.hpp"
#include "ibm_vpd_utils.hpp"

#include <boost/asio.hpp>
#include <boost/bind/bind.hpp>
#include <gpiod.hpp>

using namespace std;
using namespace openpower::vpd::constants;

namespace openpower
{
namespace vpd
{
namespace manager
{

bool GpioEventHandler::getPresencePinValue()
{
    Byte gpioData = 1;
    gpiod::line presenceLine = gpiod::find_line(presencePin);
    if (!presenceLine)
    {
        cerr << "Error getPresencePinValue: couldn't find presence line:"
             << presencePin << " on GPIO \n";
        // return previous state as we couldn't read current state
        return prevPresPinValue;
    }

    presenceLine.request(
        {"Op-panel presence line", gpiod::line_request::DIRECTION_INPUT, 0});

    gpioData = presenceLine.get_value();

    return gpioData;
}

void GpioMonitor::initGpioInfos(
    std::shared_ptr<boost::asio::io_context>& ioContext)
{
    Byte outputValue = 0;
    Byte presenceValue = 0;
    string presencePinName{}, outputPinName{};
    string devNameAddr{}, driverType{}, busType{}, objectPath{};

    for (const auto& eachFRU : jsonFile["frus"].items())
    {
        for (const auto& eachInventory : eachFRU.value())
        {
            objectPath = eachInventory["inventoryPath"];

            if ((eachInventory.find("presence") != eachInventory.end()) &&
                (eachInventory.find("preAction") != eachInventory.end()))
            {
                if (!eachInventory["presence"].value("pollingRequired", false))
                {
                    // Polling not required for this FRU , skip.
                    continue;
                }

                for (const auto& presStatus : eachInventory["presence"].items())
                {
                    if (presStatus.key() == "pin")
                    {
                        presencePinName = presStatus.value();
                    }
                    else if (presStatus.key() == "value")
                    {
                        presenceValue = presStatus.value();
                    }
                }

                // Based on presence pin value, preAction pin will be set/reset
                // This action will be taken before vpd collection.
                for (const auto& preAction : eachInventory["preAction"].items())
                {
                    if (preAction.key() == "pin")
                    {
                        outputPinName = preAction.value();
                    }
                    else if (preAction.key() == "value")
                    {
                        outputValue = preAction.value();
                    }
                }

                if ((eachInventory.find("devAddress") != eachInventory.end()) &&
                    (eachInventory.find("driverType") != eachInventory.end()) &&
                    (eachInventory.find("busType") != eachInventory.end()))
                {
                    devNameAddr = eachInventory["devAddress"];
                    driverType = eachInventory["driverType"];
                    busType = eachInventory["busType"];

                    // Init all Gpio info variables
                    std::shared_ptr<GpioEventHandler> gpioObj =
                        make_shared<GpioEventHandler>(
                            presencePinName, presenceValue, outputPinName,
                            outputValue, devNameAddr, driverType, busType,
                            objectPath, ioContext);

                    gpioObjects.push_back(gpioObj);
                }
            }
        }
    }
}

void GpioEventHandler::toggleGpio()
{
    bool presPinVal = getPresencePinValue();
    bool isPresent = false;

    // preserve the new value
    prevPresPinValue = presPinVal;

    if (presPinVal == presenceValue)
    {
        isPresent = true;
    }

    // if FRU went away set the present property to false
    if (!isPresent)
    {
        inventory::ObjectMap objects;
        inventory::InterfaceMap interfaces;
        inventory::PropertyMap presProp;

        presProp.emplace("Present", false);
        interfaces.emplace("xyz.openbmc_project.Inventory.Item", presProp);
        objects.emplace(objectPath, move(interfaces));

        common::utility::callPIM(move(objects));
    }

    gpiod::line outputLine = gpiod::find_line(outputPin);
    if (!outputLine)
    {
        cerr << "Error: toggleGpio: couldn't find output line:" << outputPin
             << ". Skipping update\n";

        return;
    }

    outputLine.request({"FRU presence: update the output GPIO pin",
                        gpiod::line_request::DIRECTION_OUTPUT, 0},
                       isPresent ? outputValue : (!outputValue));

    string cmnd = createBindUnbindDriverCmnd(devNameAddr, busType, driverType,
                                             isPresent ? "bind" : "unbind");

    cout << cmnd << endl;
    executeCmd(cmnd);
}

void GpioEventHandler::handleTimerExpiry(
    const boost::system::error_code& ec,
    std::shared_ptr<boost::asio::steady_timer>& timer)
{
    if (ec == boost::asio::error::operation_aborted)
    {
        return;
    }

    if (ec)
    {
        std::cerr << "Timer wait failed for gpio pin" << ec.message();
        return;
    }

    if (hasEventOccurred())
    {
        toggleGpio();
    }
    timer->expires_at(std::chrono::steady_clock::now() +
                      std::chrono::seconds(5));
    timer->async_wait(boost::bind(&GpioEventHandler::handleTimerExpiry, this,
                                  boost::asio::placeholders::error, timer));
}

void GpioEventHandler::doEventAndTimerSetup(
    std::shared_ptr<boost::asio::io_context>& ioContext)
{
    prevPresPinValue = getPresencePinValue();

    static vector<std::shared_ptr<boost::asio::steady_timer>> timers;

    auto timer = make_shared<boost::asio::steady_timer>(
        *ioContext, std::chrono::seconds(5));

    timer->async_wait(boost::bind(&GpioEventHandler::handleTimerExpiry, this,
                                  boost::asio::placeholders::error, timer));

    timers.push_back(timer);
}

} // namespace manager
} // namespace vpd
} // namespace openpower
