SNode.C
Loading...
Searching...
No Matches
DescriptorEventReceiver.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
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/DescriptorEventReceiver.h"
43
44#include "core/DescriptorEventPublisher.h"
45
46#ifndef DOXYGEN_SHOULD_SKIP_THIS
47
48#include "log/Logger.h"
49
50#include <climits>
51
52#endif /* DOXYGEN_SHOULD_SKIP_THIS */
53
54namespace core {
55
56 Observer::~Observer() {
57 }
58
59 void Observer::observed() {
60 observationCounter++;
61 }
62
63 void Observer::unObserved() {
64 observationCounter--;
65
66 if (observationCounter == 0) {
67 unobservedEvent();
68 }
69 }
70
71 const utils::Timeval DescriptorEventReceiver::TIMEOUT::DEFAULT = {-1, 0};
72 const utils::Timeval DescriptorEventReceiver::TIMEOUT::DISABLE = {0, 0};
73 const utils::Timeval DescriptorEventReceiver::TIMEOUT::MAX = {LONG_MAX, 0};
74
75 DescriptorEventReceiver::DescriptorEventReceiver(const std::string& name,
76 DescriptorEventPublisher& descriptorEventPublisher,
77 const utils::Timeval& timeout)
78 : EventReceiver(name)
79 , descriptorEventPublisher(descriptorEventPublisher)
80 , maxInactivity(timeout)
81 , initialTimeout(timeout) {
82 }
83
84 int DescriptorEventReceiver::getRegisteredFd() const {
85 return observedFd;
86 }
87
88 bool DescriptorEventReceiver::enable(int fd) {
89 if (!enabled) {
90 observedFd = fd;
91
92 enabled = true;
93 descriptorEventPublisher.enable(this);
94 LOG(TRACE) << getName() << ": Enabled";
95 } else {
96 LOG(WARNING) << getName() << ": Double enable";
97 }
98
99 return enabled;
100 }
101
102 void DescriptorEventReceiver::setEnabled(const utils::Timeval& currentTime) {
103 lastTriggered = currentTime;
104
105 observed();
106 }
107
108 void DescriptorEventReceiver::disable() {
109 if (enabled) {
110 enabled = false;
111 descriptorEventPublisher.disable(this);
112 LOG(TRACE) << getName() << ": Disabled";
113 } else {
114 LOG(WARNING) << getName() << ": Double disable";
115 }
116 }
117
118 void DescriptorEventReceiver::setDisabled() {
119 unObserved();
120 }
121
122 bool DescriptorEventReceiver::isEnabled() const {
123 return enabled;
124 }
125
126 void DescriptorEventReceiver::suspend() {
127 if (enabled) {
128 if (!suspended) {
129 suspended = true;
130 descriptorEventPublisher.suspend(this);
131 } else {
132 LOG(WARNING) << getName() << ": Double suspend";
133 }
134 } else {
135 LOG(WARNING) << getName() << ": Suspend while not enabled";
136 }
137 }
138
139 void DescriptorEventReceiver::resume() {
140 if (enabled) {
141 if (suspended) {
142 suspended = false;
143 lastTriggered = utils::Timeval::currentTime();
144 descriptorEventPublisher.resume(this);
145 } else {
146 LOG(WARNING) << getName() << ": Double resume";
147 }
148 } else {
149 LOG(WARNING) << getName() << ": Resume while not enabled";
150 }
151 }
152
153 bool DescriptorEventReceiver::isSuspended() const {
154 return suspended;
155 }
156
157 void DescriptorEventReceiver::setTimeout(const utils::Timeval& timeout) {
158 if (timeout == TIMEOUT::DEFAULT) {
159 this->maxInactivity = initialTimeout;
160 } else {
161 this->maxInactivity = timeout;
162 }
163
164 triggered(utils::Timeval::currentTime());
165 }
166
167 utils::Timeval DescriptorEventReceiver::getTimeout(const utils::Timeval& currentTime) const {
168 return maxInactivity > 0 ? currentTime > lastTriggered ? maxInactivity - (currentTime - lastTriggered) : 0 : TIMEOUT::MAX;
169 }
170
171 void DescriptorEventReceiver::onEvent(const utils::Timeval& currentTime) {
172 eventCounter++;
173 triggered(currentTime);
174
175 dispatchEvent();
176 }
177
178 void DescriptorEventReceiver::onSignal(int signum) {
179 signalEvent(signum);
180 }
181
182 void DescriptorEventReceiver::triggered(const utils::Timeval& currentTime) {
183 lastTriggered = currentTime;
184 }
185
186 void DescriptorEventReceiver::checkTimeout(const utils::Timeval& currentTime) {
187 if (maxInactivity > 0 && currentTime - lastTriggered >= maxInactivity) {
188 timeoutEvent();
189 }
190 }
191
192} // namespace core