From 33bfe76177df0b91a33b75bc33759977ba55ff5c Mon Sep 17 00:00:00 2001 From: Matthew Barth Date: Mon, 5 Nov 2018 11:13:25 -0600 Subject: Remove all signals when an event is removed Each event could have multiple signals associated and when the event is removed, all of those signals should also be removed. This ensures no dangling signals remain after an event is removed from be processed by fan control Tested: The total number of signals subscribed are reduced by the same number of signals of a removed event Change-Id: I651f70dbe6aed9c7eef06aff4eac9eae7ad89d77 Signed-off-by: Matthew Barth --- control/zone.cpp | 90 +++++++++++++++++++++++++++++++++----------------------- control/zone.hpp | 40 +++++++++++++++++++++++++ 2 files changed, 93 insertions(+), 37 deletions(-) (limited to 'control') diff --git a/control/zone.cpp b/control/zone.cpp index 17ec5fc..d37701e 100644 --- a/control/zone.cpp +++ b/control/zone.cpp @@ -353,50 +353,66 @@ void Zone::initEvent(const SetSpeedEvent& event) void Zone::removeEvent(const SetSpeedEvent& event) { - // Find the signal event to be removed - auto it = std::find_if( - _signalEvents.begin(), - _signalEvents.end(), - [&event](auto const& se) + // Remove signals of the event + for (auto& sig : std::get(event)) + { + auto it = findSignal(sig, + std::get(event), + std::get(event)); + if (it != std::end(getSignalEvents())) { - auto seEventData = *std::get(se); - if (std::get(seEventData).size() != - std::get(event).size()) - { - return false; - } - else - { - // TODO openbmc/openbmc#2328 - Use the action function target - // for comparison - auto actsEqual = [](auto const& a1, - auto const& a2) - { - return a1.target_type().name() == - a2.target_type().name(); - }; - return - ( - std::get(seEventData) == - std::get(event) && - std::equal(std::get(event).begin(), - std::get(event).end(), - std::get(seEventData).begin(), - actsEqual) - ); - } - }); - if (it != std::end(_signalEvents)) + removeSignal(it); + } + } + // Remove timers of the event + if (std::get(std::get(event)) != seconds(0)) { - std::get(*it).reset(); - if (std::get(*it) != nullptr) + auto it = findTimer(std::get(event), + std::get(event)); + if (it != std::end(getTimerEvents())) { - std::get(*it).reset(); + removeTimer(it); } - _signalEvents.erase(it); } } +std::vector::iterator Zone::findSignal( + const Signal& signal, + const Group& eGroup, + const std::vector& eActions) +{ + // Find the signal in the event to be removed + for (auto it = _signalEvents.begin(); it != _signalEvents.end(); ++ it) + { + const auto& seEventData = *std::get(*it); + if (eGroup == std::get(seEventData) && + std::get(signal) == + std::get(seEventData) && + std::get(signal).target_type().name() == + std::get(seEventData).target_type().name() && + eActions.size() == std::get(seEventData).size()) + { + // TODO openbmc/openbmc#2328 - Use the function target + // for comparison + auto actsEqual = [](auto const& a1, + auto const& a2) + { + return a1.target_type().name() == + a2.target_type().name(); + }; + if (std::equal(eActions.begin(), + eActions.end(), + std::get(seEventData).begin(), + actsEqual)) + { + return it; + } + } + } + + return _signalEvents.end(); +} + std::vector::iterator Zone::findTimer( const Group& eventGroup, const std::vector& eventActions) diff --git a/control/zone.hpp b/control/zone.hpp index 002feb4..11cfc65 100644 --- a/control/zone.hpp +++ b/control/zone.hpp @@ -342,6 +342,46 @@ class Zone return _eventLoop; } + /** + * @brief Get the list of signal events + * + * @return - List of signal events + */ + inline auto& getSignalEvents() + { + return _signalEvents; + } + + /** + * @brief Find the first instance of a signal event + * + * @param[in] signal - Event signal to find + * @param[in] eGroup - Group associated with the signal + * @param[in] eActions - List of actions associated with the signal + * + * @return - Iterator to the stored signal event + */ + std::vector::iterator findSignal( + const Signal& signal, + const Group& eGroup, + const std::vector& eActions); + + /** + * @brief Remove the given signal event + * + * @param[in] seIter - Iterator pointing to the signal event to remove + */ + inline void removeSignal(std::vector::iterator& seIter) + { + assert(seIter != std::end(_signalEvents)); + std::get(*seIter).reset(); + if (std::get(*seIter) != nullptr) + { + std::get(*seIter).reset(); + } + _signalEvents.erase(seIter); + } + /** * @brief Get the list of timer events * -- cgit v1.2.1