1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
|
/************************************************************************
*
* Copyright (C) 2024-2025 IRCAD France
*
* This file is part of Sight.
*
* Sight is free software: you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Sight is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with Sight. If not, see <https://www.gnu.org/licenses/>.
*
***********************************************************************/
#include "update_sequence.hpp"
#include "detail/update_registry.hpp"
#include <future>
namespace sight::app
{
//------------------------------------------------------------------------------
void update_sequence::starting()
{
if(!m_parent.empty())
{
app::register_updater(m_parent, this->get_sptr());
}
if(m_loop)
{
this->updating();
}
}
//-----------------------------------------------------------------------------
void update_sequence::stopping()
{
if(!m_parent.empty())
{
app::unregister_updater(m_parent);
}
}
//-----------------------------------------------------------------------------
void update_sequence::updating()
{
std::vector<std::pair<sight::service::base::sptr, std::string> > services;
std::set<std::string> is_going_to_be_started;
std::set<std::string> is_going_to_be_stopped;
for(const auto& element : m_elements)
{
sight::service::base::sptr srv;
if(element.type == type_t::SERVICE)
{
srv = std::dynamic_pointer_cast<sight::service::base>(sight::core::id::get_object(element.uid));
}
else
{
srv = app::get_updater(element.uid);
}
if(srv != nullptr)
{
const auto is_slot_start = (element.slot == service::base::slots::START);
const auto is_slot_stop = (element.slot == service::base::slots::STOP);
if(is_slot_start)
{
// memorize that a start of this service is called.
is_going_to_be_started.insert(element.uid);
}
else if(is_slot_stop)
{
//memorize that a stop of this service is called.
is_going_to_be_stopped.insert(element.uid);
}
const auto is_started = srv->started();
const auto will_be_started = is_going_to_be_started.contains(element.uid);
const auto is_stopped = srv->stopped();
const auto will_be_stopped = is_going_to_be_stopped.contains(element.uid);
// Service is started, or will be started and current slot is not start = OK we add the slot.
if((is_started or will_be_started) and not is_slot_start)
{
services.emplace_back(srv, element.slot);
if(srv->is_auto_connected())
{
SIGHT_ERROR(
"[sight::app::update_sequence] Service " << std::quoted(element.uid)
<< " is auto-connected, this is likely to introduce timing issues."
);
}
}
// Service is stopped or will be stopped and current slot isn't start or stop = NOK we discard the sequence
else if((is_stopped or will_be_stopped) and not is_slot_start and not is_slot_stop)
{
if(not element.ignore_stopped)
{
SIGHT_WARN(
"[sight::app::update_sequence] Service " << std::quoted(element.uid)
<< " is stopped or will be stopped."
);
services.clear();
break;
}
}
// Service isn't stopped and slot isn't stop = OK we add slot
else if(not (is_stopped or will_be_stopped) or not is_slot_stop)
{
services.emplace_back(srv, element.slot);
if(srv->is_auto_connected())
{
SIGHT_ERROR(
"[sight::app::update_sequence] Service " << std::quoted(element.uid)
<< " is auto-connected, this is likely to introduce timing issues."
);
}
}
}
}
std::ranges::for_each(
services,
[this](const auto& _srv)
{
if(this->worker() == _srv.first->worker())
{
_srv.first->slot(_srv.second)->run();
}
else
{
_srv.first->slot(_srv.second)->async_run().wait();
}
});
if(m_loop)
{
this->slot(service::base::slots::UPDATE)->async_run();
}
}
//-----------------------------------------------------------------------------
} // namespace sight::app
|