xref: /openbmc/openpower-vpd-parser/vpd-manager/src/gpio_monitor.cpp (revision 7fea9f503fbdc0dce1ecf3bf684ac49ccf80ae89)
1fa5e4d32SSunny Srivastava #include "gpio_monitor.hpp"
2fa5e4d32SSunny Srivastava 
3fa5e4d32SSunny Srivastava #include "constants.hpp"
4fa5e4d32SSunny Srivastava #include "logger.hpp"
5fa5e4d32SSunny Srivastava #include "types.hpp"
6fa5e4d32SSunny Srivastava #include "utility/dbus_utility.hpp"
7fa5e4d32SSunny Srivastava #include "utility/json_utility.hpp"
8fa5e4d32SSunny Srivastava 
9fa5e4d32SSunny Srivastava #include <boost/asio.hpp>
10fa5e4d32SSunny Srivastava #include <boost/bind/bind.hpp>
11fa5e4d32SSunny Srivastava #include <gpiod.hpp>
12fa5e4d32SSunny Srivastava 
13fa5e4d32SSunny Srivastava namespace vpd
14fa5e4d32SSunny Srivastava {
handleChangeInGpioPin(const bool & i_isFruPresent)15fa5e4d32SSunny Srivastava void GpioEventHandler::handleChangeInGpioPin(const bool& i_isFruPresent)
16fa5e4d32SSunny Srivastava {
17fa5e4d32SSunny Srivastava     try
18fa5e4d32SSunny Srivastava     {
19fa5e4d32SSunny Srivastava         if (i_isFruPresent)
20fa5e4d32SSunny Srivastava         {
21fa5e4d32SSunny Srivastava             types::VPDMapVariant l_parsedVpd =
22fa5e4d32SSunny Srivastava                 m_worker->parseVpdFile(m_fruPath);
23fa5e4d32SSunny Srivastava 
24fa5e4d32SSunny Srivastava             if (std::holds_alternative<std::monostate>(l_parsedVpd))
25fa5e4d32SSunny Srivastava             {
26fa5e4d32SSunny Srivastava                 throw std::runtime_error(
27fa5e4d32SSunny Srivastava                     "VPD parsing failed for " + std::string(m_fruPath));
28fa5e4d32SSunny Srivastava             }
29fa5e4d32SSunny Srivastava 
30fa5e4d32SSunny Srivastava             types::ObjectMap l_dbusObjectMap;
31fa5e4d32SSunny Srivastava             m_worker->populateDbus(l_parsedVpd, l_dbusObjectMap, m_fruPath);
32fa5e4d32SSunny Srivastava 
33fa5e4d32SSunny Srivastava             if (l_dbusObjectMap.empty())
34fa5e4d32SSunny Srivastava             {
35fa5e4d32SSunny Srivastava                 throw std::runtime_error("Failed to create D-bus object map.");
36fa5e4d32SSunny Srivastava             }
37fa5e4d32SSunny Srivastava 
38fa5e4d32SSunny Srivastava             if (!dbusUtility::callPIM(move(l_dbusObjectMap)))
39fa5e4d32SSunny Srivastava             {
40fa5e4d32SSunny Srivastava                 throw std::runtime_error("call PIM failed");
41fa5e4d32SSunny Srivastava             }
42fa5e4d32SSunny Srivastava         }
43fa5e4d32SSunny Srivastava         else
44fa5e4d32SSunny Srivastava         {
45*bf257110SRekhaAparna01             m_worker->deleteFruVpd(jsonUtility::getInventoryObjPathFromJson(
46*bf257110SRekhaAparna01                 m_worker->getSysCfgJsonObj(), m_fruPath));
47fa5e4d32SSunny Srivastava         }
48fa5e4d32SSunny Srivastava     }
49fa5e4d32SSunny Srivastava     catch (std::exception& l_ex)
50fa5e4d32SSunny Srivastava     {
51fa5e4d32SSunny Srivastava         logging::logMessage(std::string(l_ex.what()));
52fa5e4d32SSunny Srivastava     }
53fa5e4d32SSunny Srivastava }
54fa5e4d32SSunny Srivastava 
handleTimerExpiry(const boost::system::error_code & i_errorCode,const std::shared_ptr<boost::asio::steady_timer> & i_timerObj)55fa5e4d32SSunny Srivastava void GpioEventHandler::handleTimerExpiry(
56fa5e4d32SSunny Srivastava     const boost::system::error_code& i_errorCode,
57fa5e4d32SSunny Srivastava     const std::shared_ptr<boost::asio::steady_timer>& i_timerObj)
58fa5e4d32SSunny Srivastava {
59fa5e4d32SSunny Srivastava     if (i_errorCode == boost::asio::error::operation_aborted)
60fa5e4d32SSunny Srivastava     {
61fa5e4d32SSunny Srivastava         logging::logMessage("Timer aborted for GPIO pin");
62fa5e4d32SSunny Srivastava         return;
63fa5e4d32SSunny Srivastava     }
64fa5e4d32SSunny Srivastava 
65fa5e4d32SSunny Srivastava     if (i_errorCode)
66fa5e4d32SSunny Srivastava     {
67fa5e4d32SSunny Srivastava         logging::logMessage("Timer wait failed for gpio pin" +
68fa5e4d32SSunny Srivastava                             std::string(i_errorCode.message()));
69fa5e4d32SSunny Srivastava         return;
70fa5e4d32SSunny Srivastava     }
71fa5e4d32SSunny Srivastava 
72fa5e4d32SSunny Srivastava     bool l_currentPresencePinValue = jsonUtility::processGpioPresenceTag(
73fa5e4d32SSunny Srivastava         m_worker->getSysCfgJsonObj(), m_fruPath, "pollingRequired",
74fa5e4d32SSunny Srivastava         "hotPlugging");
75fa5e4d32SSunny Srivastava 
76fa5e4d32SSunny Srivastava     if (m_prevPresencePinValue != l_currentPresencePinValue)
77fa5e4d32SSunny Srivastava     {
78fa5e4d32SSunny Srivastava         m_prevPresencePinValue = l_currentPresencePinValue;
79fa5e4d32SSunny Srivastava         handleChangeInGpioPin(l_currentPresencePinValue);
80fa5e4d32SSunny Srivastava     }
81fa5e4d32SSunny Srivastava 
82fa5e4d32SSunny Srivastava     i_timerObj->expires_at(std::chrono::steady_clock::now() +
83fa5e4d32SSunny Srivastava                            std::chrono::seconds(constants::VALUE_5));
84fa5e4d32SSunny Srivastava     i_timerObj->async_wait(
85fa5e4d32SSunny Srivastava         boost::bind(&GpioEventHandler::handleTimerExpiry, this,
86fa5e4d32SSunny Srivastava                     boost::asio::placeholders::error, i_timerObj));
87fa5e4d32SSunny Srivastava }
88fa5e4d32SSunny Srivastava 
setEventHandlerForGpioPresence(const std::shared_ptr<boost::asio::io_context> & i_ioContext)89fa5e4d32SSunny Srivastava void GpioEventHandler::setEventHandlerForGpioPresence(
90fa5e4d32SSunny Srivastava     const std::shared_ptr<boost::asio::io_context>& i_ioContext)
91fa5e4d32SSunny Srivastava {
92fa5e4d32SSunny Srivastava     m_prevPresencePinValue = jsonUtility::processGpioPresenceTag(
93fa5e4d32SSunny Srivastava         m_worker->getSysCfgJsonObj(), m_fruPath, "pollingRequired",
94fa5e4d32SSunny Srivastava         "hotPlugging");
95fa5e4d32SSunny Srivastava 
96fa5e4d32SSunny Srivastava     static std::vector<std::shared_ptr<boost::asio::steady_timer>> l_timers;
97fa5e4d32SSunny Srivastava 
98fa5e4d32SSunny Srivastava     auto l_timerObj = make_shared<boost::asio::steady_timer>(
99fa5e4d32SSunny Srivastava         *i_ioContext, std::chrono::seconds(constants::VALUE_5));
100fa5e4d32SSunny Srivastava 
101fa5e4d32SSunny Srivastava     l_timerObj->async_wait(
102fa5e4d32SSunny Srivastava         boost::bind(&GpioEventHandler::handleTimerExpiry, this,
103fa5e4d32SSunny Srivastava                     boost::asio::placeholders::error, l_timerObj));
104fa5e4d32SSunny Srivastava 
105fa5e4d32SSunny Srivastava     l_timers.push_back(l_timerObj);
106fa5e4d32SSunny Srivastava }
107fa5e4d32SSunny Srivastava 
initHandlerForGpio(const std::shared_ptr<boost::asio::io_context> & i_ioContext,const std::shared_ptr<Worker> & i_worker)108fa5e4d32SSunny Srivastava void GpioMonitor::initHandlerForGpio(
109fa5e4d32SSunny Srivastava     const std::shared_ptr<boost::asio::io_context>& i_ioContext,
110fa5e4d32SSunny Srivastava     const std::shared_ptr<Worker>& i_worker)
111fa5e4d32SSunny Srivastava {
112fa5e4d32SSunny Srivastava     std::vector<std::string> l_gpioPollingRequiredFrusList =
113fa5e4d32SSunny Srivastava         jsonUtility::getListOfGpioPollingFrus(m_sysCfgJsonObj);
114fa5e4d32SSunny Srivastava 
115fa5e4d32SSunny Srivastava     for (const auto& l_fruPath : l_gpioPollingRequiredFrusList)
116fa5e4d32SSunny Srivastava     {
117fa5e4d32SSunny Srivastava         std::shared_ptr<GpioEventHandler> l_gpioEventHandlerObj =
118fa5e4d32SSunny Srivastava             std::make_shared<GpioEventHandler>(l_fruPath, i_worker,
119fa5e4d32SSunny Srivastava                                                i_ioContext);
120fa5e4d32SSunny Srivastava 
121fa5e4d32SSunny Srivastava         m_gpioEventHandlerObjects.push_back(l_gpioEventHandlerObj);
122fa5e4d32SSunny Srivastava     }
123fa5e4d32SSunny Srivastava }
124fa5e4d32SSunny Srivastava } // namespace vpd
125