SNode.C
Loading...
Searching...
No Matches
EventMultiplexer.cpp
Go to the documentation of this file.
1/*
2 * SNode.C - A Slim Toolkit for Network Communication
3 * Copyright (C) Volker Christian <me@vchrist.at>
4 * 2020, 2021, 2022, 2023, 2024, 2025, 2026
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as published
8 * by the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20/*
21 * MIT License
22 *
23 * Permission is hereby granted, free of charge, to any person obtaining a copy
24 * of this software and associated documentation files (the "Software"), to deal
25 * in the Software without restriction, including without limitation the rights
26 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27 * copies of the Software, and to permit persons to whom the Software is
28 * furnished to do so, subject to the following conditions:
29 *
30 * The above copyright notice and this permission notice shall be included in
31 * all copies or substantial portions of the Software.
32 *
33 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
39 * THE SOFTWARE.
40 */
41
42#include "core/EventMultiplexer.h"
43
44#include "core/DescriptorEventPublisher.h"
45#include "core/DescriptorEventReceiver.h"
46#include "core/DynamicLoader.h"
47#include "core/TimerEventPublisher.h"
48
49#ifndef DOXYGEN_SHOULD_SKIP_THIS
50
51#include "utils/Timeval.h"
52
53#include <algorithm>
54#include <cerrno>
55#include <numeric>
56#include <utility>
57
58#endif /* DOXYGEN_SHOULD_SKIP_THIS */
59
60namespace core {
61
62 core::EventMultiplexer::EventMultiplexer(DescriptorEventPublisher* readDescriptorEventPublisher,
63 DescriptorEventPublisher* writeDescriptorEventPublisher,
64 DescriptorEventPublisher* exceptionDescriptorEventPublisher)
65 : descriptorEventPublishers{readDescriptorEventPublisher, writeDescriptorEventPublisher, exceptionDescriptorEventPublisher}
67 }
68
70 for (const DescriptorEventPublisher* descriptorEventPublisher : descriptorEventPublishers) {
71 delete descriptorEventPublisher;
72 }
74 }
75
79
83
84 void EventMultiplexer::span(Event* event) {
86 }
87
90 }
91
92 TickStatus EventMultiplexer::tick(const utils::Timeval& tickTimeOut, const sigset_t& sigMask) {
93 const utils::Timeval currentTime = utils::Timeval::currentTime();
94
95 int activeDescriptorCount = 0;
96
97 const TickStatus tickStatus = waitForEvents(tickTimeOut, currentTime, sigMask, activeDescriptorCount);
98
99 if (tickStatus == TickStatus::SUCCESS) {
100 spanActiveEvents(currentTime, activeDescriptorCount);
101 executeEventQueue(currentTime);
102 checkTimedOutEvents(currentTime);
103 releaseExpiredResources(currentTime);
104 }
105
106 return tickStatus;
107 }
108
109 void EventMultiplexer::signal(int sigNum) {
110 for (DescriptorEventPublisher* const descriptorEventPublisher : descriptorEventPublishers) {
111 descriptorEventPublisher->signal(sigNum);
112 }
114
116 }
117
119 for (DescriptorEventPublisher* const descriptorEventPublisher : descriptorEventPublishers) {
120 descriptorEventPublisher->disable();
121 }
123
125 }
126
130
132 const utils::Timeval& currentTime,
133 const sigset_t& sigMask,
134 int& activeDescriptorCount) {
135 TickStatus tickStatus = TickStatus::SUCCESS;
136
138 utils::Timeval nextTimeout = std::min(getNextTimeout(currentTime), tickTimeOut);
139
140 activeDescriptorCount = monitorDescriptors(nextTimeout, sigMask);
141
142 if (activeDescriptorCount < 0) {
143 if (errno == EINTR) {
144 tickStatus = TickStatus::INTERRUPTED;
145 } else {
146 tickStatus = TickStatus::TRACE;
147 }
148 }
149 } else {
150 tickStatus = TickStatus::NOOBSERVER;
151 }
152
153 return tickStatus;
154 }
155
156 void EventMultiplexer::spanActiveEvents(const utils::Timeval& currentTime, int activeDescriptorCount) {
158 spanActiveEvents(activeDescriptorCount);
159 }
160
161 void EventMultiplexer::executeEventQueue(const utils::Timeval& currentTime) {
162 eventQueue.execute(currentTime);
163 }
164
165 void EventMultiplexer::checkTimedOutEvents(const utils::Timeval& currentTime) {
166 for (DescriptorEventPublisher* const descriptorEventPublisher : descriptorEventPublishers) {
167 descriptorEventPublisher->checkTimedOutEvents(currentTime);
168 }
169 }
170
171 void EventMultiplexer::releaseExpiredResources(const utils::Timeval& currentTime) {
172 for (DescriptorEventPublisher* const descriptorEventPublisher : descriptorEventPublishers) {
173 descriptorEventPublisher->releaseDisabledEvents(currentTime);
174 }
177 }
178
179 utils::Timeval EventMultiplexer::getNextTimeout(const utils::Timeval& currentTime) {
181
182 if (eventQueue.empty()) {
183 for (const DescriptorEventPublisher* const descriptorEventPublisher : descriptorEventPublishers) {
184 nextTimeout = std::min(descriptorEventPublisher->getNextTimeout(currentTime), nextTimeout);
185 }
186 nextTimeout = std::min(timerEventPublisher->getNextTimeout(currentTime), nextTimeout);
187 nextTimeout = std::max(nextTimeout, utils::Timeval()); // In case nextTimeout is negative
188 } else {
189 nextTimeout = 0;
190 }
191
192 return nextTimeout;
193 }
194
196 return std::accumulate(descriptorEventPublishers.begin(),
198 0,
199 [](int count, const DescriptorEventPublisher* descriptorEventPublisher) -> int {
200 return count + descriptorEventPublisher->getObservedEventReceiverCount();
201 });
202 }
203
205 return std::accumulate(descriptorEventPublishers.begin(),
207 -1,
208 [](int maxFd, const DescriptorEventPublisher* descriptorEventPublisher) -> int {
209 return std::max(descriptorEventPublisher->maxFd(), maxFd);
210 });
211 }
212
214 : executeQueue(new std::list<Event*>())
215 , publishQueue(new std::list<Event*>()) {
216 }
217
219 delete executeQueue;
220 delete publishQueue;
221 }
222
224 publishQueue->push_back(event); // do not allow two or more same events in one tick
225 }
226
228 publishQueue->remove(event); // in case of erase remove the event from the published queue
229 }
230
231 void EventMultiplexer::EventQueue::execute(const utils::Timeval& currentTime) {
232 std::swap(executeQueue, publishQueue);
233
234 for (Event* event : *executeQueue) {
235 event->dispatch(currentTime);
236 }
237
238 executeQueue->clear();
239 }
240
242 return publishQueue->empty();
243 }
244
246 std::swap(executeQueue, publishQueue);
247
248 for (const Event* event : *executeQueue) {
250 }
251
252 executeQueue->clear();
253 }
254
255} // namespace core
utils::Timeval getNextTimeout(const utils::Timeval &currentTime) const
void checkTimedOutEvents(const utils::Timeval &currentTime)
void releaseDisabledEvents(const utils::Timeval &currentTime)
static void execDlCloseDeleyed()
std::list< Event * > * publishQueue
std::list< Event * > * executeQueue
void execute(const utils::Timeval &currentTime)
virtual int monitorDescriptors(utils::Timeval &tickTimeOut, const sigset_t &sigMask)=0
void releaseExpiredResources(const utils::Timeval &currentTime)
utils::Timeval getNextTimeout(const utils::Timeval &currentTime)
void checkTimedOutEvents(const utils::Timeval &currentTime)
TickStatus waitForEvents(const utils::Timeval &tickTimeOut, const utils::Timeval &currentTime, const sigset_t &sigMask, int &activeDescriptorCount)
TimerEventPublisher & getTimerEventPublisher()
TickStatus tick(const utils::Timeval &tickTimeOut, const sigset_t &sigMask)
void spanActiveEvents(const utils::Timeval &currentTime, int activeDescriptorCount)
void span(core::Event *event)
std::array< DescriptorEventPublisher *, 3 > descriptorEventPublishers
virtual void spanActiveEvents(int activeDescriptorCount)=0
DescriptorEventPublisher & getDescriptorEventPublisher(DISP_TYPE dispType)
void relax(core::Event *event)
EventMultiplexer(DescriptorEventPublisher *readDescriptorEventPublisher, DescriptorEventPublisher *writeDescriptorEventPublisher, DescriptorEventPublisher *exceptionDescriptorEventPublisher)
void executeEventQueue(const utils::Timeval &currentTime)
core::TimerEventPublisher *const timerEventPublisher
virtual void destruct()
void dispatch(const utils::Timeval &currentTime)
Definition Event.cpp:82
EventReceiver * getEventReceiver() const
Definition Event.cpp:87
utils::Timeval getNextTimeout(const utils::Timeval &currentTime)
void spanActiveEvents(const utils::Timeval &currentTime)
static Timeval currentTime()
Definition Timeval.cpp:76
Timeval() noexcept
Definition Timeval.cpp:55
Timeval & operator=(const Timeval &timeVal)
Definition Timeval.cpp:83
TickStatus
Definition TickStatus.h:51