GCC Code Coverage Report


Directory: ./
File: pdcom5/src/msrproto/ProtocolHandler.cpp
Date: 2025-01-19 04:08:20
Exec Total Coverage
Lines: 904 1106 81.7%
Branches: 988 1982 49.8%

Line Branch Exec Source
1 /*****************************************************************************
2 *
3 * Copyright (C) 2015-2016 Richard Hacker (lerichi at gmx dot net)
4 *
5 * This file is part of the PdCom library.
6 *
7 * The PdCom library is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU Lesser General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
11 *
12 * The PdCom library is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
14 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
15 * License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public License
18 * along with the PdCom library. If not, see <http://www.gnu.org/licenses/>.
19 *
20 *****************************************************************************/
21
22 #include "ProtocolHandler.h"
23
24 #include "../Debug.h"
25 #include "../SpyLayer.h"
26 #include "Channel.h"
27 #include "DirNode.h"
28 #include "Parameter.h"
29 #include "Subscription.h"
30 #include "expat_wrapper_impl.h"
31
32 #include <algorithm>
33 #include <cstring>
34 #include <errno.h>
35 #include <mutex>
36 #include <ostream>
37 #include <pdcom5/Exception.h>
38 #include <pdcom5/Subscriber.h>
39 #include <sstream>
40 #include <stdexcept>
41 #include <typeinfo>
42
43 #if defined _WIN32 || defined __ANDROID__
44 namespace {
45 struct PthreadCancelDisabler
46 {};
47 } // namespace
48
49 #else
50
51 #include <pthread.h>
52 namespace {
53 class PthreadCancelDisabler
54 {
55 int cancel_state_ = 0;
56
57 public:
58 PthreadCancelDisabler()
59 {
60 pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cancel_state_);
61 }
62 ~PthreadCancelDisabler() { pthread_setcancelstate(cancel_state_, nullptr); }
63 };
64 } // namespace
65
66 #endif
67
68 #ifdef PDC_DEBUG
69 #include <stdlib.h> // getenv(), atoi()
70 #endif
71
72 using PdCom::impl::MsrProto::Channel;
73 using PdCom::impl::MsrProto::ChannelSubscriptionsWithGroup;
74 using PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup;
75 using PdCom::impl::MsrProto::EventSubscription;
76 using PdCom::impl::MsrProto::ExpatWrapper;
77 using PdCom::impl::MsrProto::Parameter;
78 using PdCom::impl::MsrProto::PeriodicSubscription;
79 using PdCom::impl::MsrProto::PollSubscription;
80 using PdCom::impl::MsrProto::ProtocolHandler;
81 using PdCom::impl::MsrProto::Variable;
82 using PdCom::impl::MsrProto::XmlStream;
83
84
85 namespace {
86 struct Attribute
87 { /*{{{*/
88 3820 Attribute(const char **atts) : atts(atts) {}
89
90 7696 const char *find(const char *const arg, const char *const else_return = "")
91 {
92
3/4
✓ Branch 0 taken 27531 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 25617 times.
✓ Branch 5 taken 1914 times.
27531 for (size_t i = 0; arg and atts[i]; i += 2) {
93
4/6
✗ Branch 2 not taken.
✓ Branch 3 taken 25617 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 25617 times.
✓ Branch 6 taken 5782 times.
✓ Branch 7 taken 19835 times.
25617 if (!strcmp(atts[i], arg))
94 5782 return atts[i + 1];
95 }
96
97 1914 return else_return;
98 }
99
100 333 bool getString(const char *arg, const char *&val)
101 {
102 333 const char *attr = find(arg, nullptr);
103
104
2/2
✓ Branch 0 taken 154 times.
✓ Branch 1 taken 179 times.
333 if (!attr)
105 154 return false;
106
107 179 val = attr;
108 179 return true;
109 };
110
111 2594 bool getUInt(const char *arg, unsigned int &val)
112 {
113 2594 const char *attr = find(arg);
114
115 // Only test for numbers, even leading '+' is ignored
116
3/4
✓ Branch 1 taken 2056 times.
✓ Branch 2 taken 538 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 2056 times.
2594 if (*attr < '0' or *attr > '9')
117 538 return false;
118
119
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2056 times.
2056 val = atol(attr);
120 2056 return true;
121 }
122
123 140 bool getInt(const char *arg, int &val)
124 {
125 140 const char *attr = find(arg);
126
127 // Only test for numbers, even leading '+' is ignored
128
4/6
✓ Branch 1 taken 59 times.
✓ Branch 2 taken 81 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 59 times.
✓ Branch 7 taken 81 times.
✗ Branch 8 not taken.
140 if ((*attr < '0' or *attr > '9') and *attr != '-')
129 81 return false;
130
131
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 59 times.
59 val = atoi(attr);
132 59 return true;
133 }
134
135 377 bool isTrue(const char *arg)
136 {
137 377 unsigned int val;
138
1/4
✗ Branch 2 not taken.
✓ Branch 3 taken 377 times.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
377 return getUInt(arg, val) and val;
139 }
140
141 208 bool getDouble(const char *arg, double &val)
142 {
143 208 const char *attr = find(arg);
144
145
1/2
✓ Branch 1 taken 208 times.
✗ Branch 2 not taken.
208 if (*attr)
146
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 208 times.
208 val = atof(attr);
147
148 208 return *attr;
149 }
150
151 539 std::chrono::nanoseconds getTime_ns(const char *arg)
152 {
153 539 const char *attr = find(arg);
154
155 539 int64_t time_ns = 0;
156 539 size_t i = 0;
157 6402 do {
158
6/6
✓ Branch 1 taken 5324 times.
✓ Branch 2 taken 1617 times.
✓ Branch 3 taken 2090 times.
✓ Branch 4 taken 3234 times.
✓ Branch 6 taken 539 times.
✓ Branch 7 taken 1551 times.
6941 if (!*attr or (!i and *attr == '.')) {
159
2/2
✓ Branch 0 taken 539 times.
✓ Branch 1 taken 1617 times.
4312 if (!i)
160 539 i = 10;
161 }
162
2/4
✓ Branch 1 taken 4785 times.
✗ Branch 2 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 4785 times.
4785 else if (*attr < '0' or *attr > '9')
163 return std::chrono::nanoseconds {0};
164
165
2/2
✓ Branch 1 taken 6402 times.
✓ Branch 2 taken 539 times.
6941 if (*attr != '.') {
166
3/4
✓ Branch 1 taken 1617 times.
✓ Branch 2 taken 4785 times.
✓ Branch 3 taken 1617 times.
✗ Branch 4 not taken.
6402 if (*attr or i != 10)
167 6402 time_ns *= 10;
168
169
2/2
✓ Branch 1 taken 4785 times.
✓ Branch 2 taken 1617 times.
6402 if (*attr)
170 4785 time_ns += *attr - '0';
171 }
172
173
2/2
✓ Branch 1 taken 5324 times.
✓ Branch 2 taken 1617 times.
6941 if (*attr)
174 5324 ++attr;
175
176
8/8
✓ Branch 1 taken 6402 times.
✓ Branch 2 taken 539 times.
✓ Branch 3 taken 5390 times.
✓ Branch 4 taken 1012 times.
✓ Branch 5 taken 4851 times.
✓ Branch 6 taken 539 times.
✓ Branch 7 taken 6402 times.
✓ Branch 8 taken 539 times.
6941 } while (*attr == '.' or !i or --i);
177
178 539 return std::chrono::nanoseconds {time_ns};
179 }
180
181 50 bool getMessage(const char *tag_name, PdCom::Message &msg, bool hasSeqNo)
182 {
183
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 50 times.
50 const bool is_reset = !strcmp(tag_name, "reset");
184 50 const char *c = nullptr;
185 50 unsigned int u = 0;
186
4/6
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 48 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 2 times.
✗ Branch 8 not taken.
✓ Branch 9 taken 50 times.
50 if (!getString("name", c) && !getString("path", c)) {
187 log_debug("Missing name in message");
188 return false;
189 }
190
1/2
✓ Branch 2 taken 50 times.
✗ Branch 3 not taken.
50 msg.path = c;
191
4/6
✓ Branch 0 taken 48 times.
✓ Branch 1 taken 2 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 48 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 50 times.
50 if (hasSeqNo && !getUInt("seq", u)) {
192 log_debug("Invalid or missing sequence number in message");
193 return false;
194 }
195 50 msg.seqNo = u;
196 // index is optional, set it to -1 if absent.
197 50 msg.index = -1;
198 50 getInt("index", msg.index);
199
1/2
✓ Branch 3 taken 50 times.
✗ Branch 4 not taken.
50 msg.time = getTime_ns("time");
200
2/2
✓ Branch 0 taken 19 times.
✓ Branch 1 taken 31 times.
50 if (is_reset) {
201 19 msg.level = PdCom::LogLevel::Reset;
202 19 return true;
203 }
204 // text is also optional
205
2/2
✓ Branch 2 taken 30 times.
✓ Branch 3 taken 1 times.
31 if (getString("text", c)) {
206
1/2
✓ Branch 2 taken 30 times.
✗ Branch 3 not taken.
30 msg.text = c;
207 }
208 else {
209 1 msg.text = std::string();
210 }
211
2/2
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 29 times.
31 if (getUInt("state", u)) {
212 //
213
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
2 if (!u) {
214 1 msg.level = PdCom::LogLevel::Reset;
215 }
216
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1 times.
1 else if (!strcmp(tag_name, "crit_error"))
217 msg.level = PdCom::LogLevel::Critical;
218
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 else if (!strcmp(tag_name, "error"))
219 1 msg.level = PdCom::LogLevel::Error;
220 else if (!strcmp(tag_name, "warn"))
221 msg.level = PdCom::LogLevel::Warn;
222 else if (!strcmp(tag_name, "info"))
223 msg.level = PdCom::LogLevel::Info;
224 else {
225 log_debug("Invalid event attribute name %s", tag_name);
226 return false;
227 }
228 }
229
3/6
✓ Branch 2 taken 29 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 29 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 29 times.
29 else if (!getUInt("prio", u) || u > 7) {
230 log_debug("Invalid priority in message");
231 return false;
232 }
233 else
234 29 msg.level = static_cast<PdCom::LogLevel>(u);
235 31 return true;
236 }
237
238 5 bool getClientStatistics(PdCom::ClientStatistics &stats)
239 {
240 5 const char *c = nullptr;
241 5 unsigned count = 0;
242 5 double d = 0.0;
243 5 stats = {};
244
245
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 if (getString("name", c)) {
246
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 stats.name_ = c;
247 }
248
249 // yes, this is a typo. It is what it is.
250
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 if (getString("apname", c)) {
251
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 stats.application_name_ = c;
252 }
253
254
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 if (getUInt("countin", count))
255 5 stats.received_bytes_ = count;
256
257
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 if (getUInt("countout", count))
258 5 stats.sent_bytes_ = count;
259
260
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 if (getDouble("connectedtime", d))
261 10 stats.connected_time_ =
262
1/2
✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
10 std::chrono::duration_cast<std::chrono::nanoseconds>(
263 10 std::chrono::duration<double>(d));
264
265 5 return true;
266 }
267
268 const char **atts;
269 }; /*}}}*/
270 } // namespace
271
272
273 157 XmlStream::XmlStream(PdCom::impl::Process *process, std::streambuf *sb) :
274 157 os(sb), process(process)
275 {
276
1/2
✓ Branch 4 taken 157 times.
✗ Branch 5 not taken.
157 os.exceptions(std::ios_base::badbit);
277 157 }
278 662 void XmlStream::lock()
279 {
280 662 process->transmitSemaphore(true);
281 662 }
282 662 void XmlStream::unlock()
283 {
284 662 process->transmitSemaphore(false);
285 662 }
286
287
288 namespace {
289 struct XmlCommand
290 { /*{{{*/
291 struct Attribute
292 {
293 // Create an attribute for a command.
294 Attribute(XmlCommand &command, const char *name) : cout(command.cout)
295 {
296 // send the preamble
297 cout << ' ' << name << "=\"";
298 }
299 ~Attribute() noexcept(false) { cout << '"'; }
300
301 XmlStream &cout;
302 };
303
304 662 XmlCommand(XmlStream &cout, const char *command) :
305 662 cout(cout), guard(cout), p_flush(true)
306 {
307
2/4
✓ Branch 3 taken 662 times.
✗ Branch 4 not taken.
✓ Branch 7 taken 662 times.
✗ Branch 8 not taken.
662 cout << '<' << command;
308 662 }
309
310 662 ~XmlCommand() noexcept(false)
311 662 {
312
1/2
✓ Branch 3 taken 662 times.
✗ Branch 4 not taken.
662 cout << ">\r\n";
313
3/4
✗ Branch 1 not taken.
✓ Branch 2 taken 662 times.
✓ Branch 3 taken 554 times.
✓ Branch 4 taken 108 times.
662 if (p_flush)
314
1/2
✓ Branch 3 taken 554 times.
✗ Branch 4 not taken.
554 cout.flush();
315 662 }
316
317 485 XmlCommand &addEscapedAttribute(const char *name, const std::string &str)
318 {
319
4/8
✓ Branch 4 taken 485 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 485 times.
✗ Branch 9 not taken.
✓ Branch 13 taken 485 times.
✗ Branch 14 not taken.
✓ Branch 18 taken 485 times.
✗ Branch 19 not taken.
485 cout << ' ' << name << '=' << '"';
320
321
2/2
✓ Branch 8 taken 5154 times.
✓ Branch 9 taken 476 times.
5630 for (const char c : str) {
322
5/5
✓ Branch 0 taken 18 times.
✓ Branch 1 taken 18 times.
✓ Branch 2 taken 18 times.
✓ Branch 3 taken 9 times.
✓ Branch 4 taken 5091 times.
5154 switch (c) {
323 18 case '"':
324
1/2
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
18 cout << "&quot;";
325 18 break;
326 18 case '<':
327
1/2
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
18 cout << "&lt;";
328 18 break;
329 18 case '&':
330
1/2
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
18 cout << "&amp;";
331 18 break;
332 9 case '\0':
333
1/2
✓ Branch 2 taken 9 times.
✗ Branch 3 not taken.
9 throw PdCom::InvalidArgument("Value must not contain NULL");
334 5091 default:
335
1/2
✓ Branch 3 taken 5091 times.
✗ Branch 4 not taken.
5091 cout << c;
336 }
337 }
338
339
1/2
✓ Branch 4 taken 476 times.
✗ Branch 5 not taken.
476 cout << '"';
340
341 476 return *this;
342 }
343
344 template <typename T>
345 791 XmlCommand &addAttribute(const char *name, T const &val, bool active = true)
346 {
347
8/16
✓ Branch 0 taken 48 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 52 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 66 times.
✓ Branch 5 taken 31 times.
✓ Branch 6 taken 243 times.
✗ Branch 7 not taken.
✗ Branch 8 not taken.
✗ Branch 9 not taken.
✓ Branch 10 taken 142 times.
✗ Branch 11 not taken.
✓ Branch 12 taken 59 times.
✗ Branch 13 not taken.
✓ Branch 14 taken 150 times.
✗ Branch 15 not taken.
791 if (active)
348
35/80
✓ Branch 4 taken 48 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 48 times.
✗ Branch 9 not taken.
✓ Branch 12 taken 48 times.
✗ Branch 13 not taken.
✓ Branch 16 taken 48 times.
✗ Branch 17 not taken.
✓ Branch 21 taken 48 times.
✗ Branch 22 not taken.
✓ Branch 31 taken 52 times.
✗ Branch 32 not taken.
✓ Branch 35 taken 52 times.
✗ Branch 36 not taken.
✓ Branch 39 taken 52 times.
✗ Branch 40 not taken.
✓ Branch 43 taken 52 times.
✗ Branch 44 not taken.
✓ Branch 48 taken 52 times.
✗ Branch 49 not taken.
✓ Branch 58 taken 66 times.
✗ Branch 59 not taken.
✓ Branch 62 taken 66 times.
✗ Branch 63 not taken.
✓ Branch 66 taken 66 times.
✗ Branch 67 not taken.
✓ Branch 70 taken 66 times.
✗ Branch 71 not taken.
✓ Branch 75 taken 66 times.
✗ Branch 76 not taken.
✓ Branch 85 taken 243 times.
✗ Branch 86 not taken.
✓ Branch 89 taken 243 times.
✗ Branch 90 not taken.
✓ Branch 93 taken 243 times.
✗ Branch 94 not taken.
✓ Branch 97 taken 243 times.
✗ Branch 98 not taken.
✓ Branch 102 taken 243 times.
✗ Branch 103 not taken.
✗ Branch 112 not taken.
✗ Branch 113 not taken.
✗ Branch 116 not taken.
✗ Branch 117 not taken.
✗ Branch 120 not taken.
✗ Branch 121 not taken.
✗ Branch 124 not taken.
✗ Branch 125 not taken.
✗ Branch 129 not taken.
✗ Branch 130 not taken.
✓ Branch 139 taken 142 times.
✗ Branch 140 not taken.
✓ Branch 143 taken 142 times.
✗ Branch 144 not taken.
✓ Branch 147 taken 142 times.
✗ Branch 148 not taken.
✓ Branch 151 taken 142 times.
✗ Branch 152 not taken.
✓ Branch 156 taken 142 times.
✗ Branch 157 not taken.
✓ Branch 166 taken 59 times.
✗ Branch 167 not taken.
✓ Branch 170 taken 59 times.
✗ Branch 171 not taken.
✓ Branch 174 taken 59 times.
✗ Branch 175 not taken.
✓ Branch 178 taken 59 times.
✗ Branch 179 not taken.
✓ Branch 183 taken 59 times.
✗ Branch 184 not taken.
✓ Branch 193 taken 150 times.
✗ Branch 194 not taken.
✓ Branch 197 taken 150 times.
✗ Branch 198 not taken.
✓ Branch 201 taken 150 times.
✗ Branch 202 not taken.
✓ Branch 205 taken 150 times.
✗ Branch 206 not taken.
✓ Branch 210 taken 150 times.
✗ Branch 211 not taken.
760 cout << ' ' << name << "=\"" << val << '"';
349 791 return *this;
350 }
351
352 template <typename T>
353 43 XmlCommand &addAttributeStream(const char *name, T &&callback)
354 {
355
3/6
✓ Branch 4 taken 43 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 43 times.
✗ Branch 9 not taken.
✓ Branch 12 taken 43 times.
✗ Branch 13 not taken.
43 cout << ' ' << name << "=\"";
356 43 callback(cout);
357
1/2
✓ Branch 4 taken 43 times.
✗ Branch 5 not taken.
43 cout << '"';
358 43 return *this;
359 }
360
361 108 XmlCommand &noFlush()
362 {
363 108 p_flush = false;
364 108 return *this;
365 }
366
367 445 XmlCommand &setId(const char *id)
368 {
369
2/4
✓ Branch 0 taken 445 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 445 times.
✗ Branch 4 not taken.
445 if (id and *id)
370
1/2
✓ Branch 8 taken 445 times.
✗ Branch 9 not taken.
445 cout << " id=\"" << id << '"';
371
372 445 return *this;
373 }
374
375 XmlStream &cout;
376 std::lock_guard<XmlStream> guard;
377 bool p_flush;
378 }; /*}}}*/
379 } // namespace
380
381 ///////////////////////////////////////////////////////////////////////////
382 157 ProtocolHandler::ProtocolHandler(PdCom::impl::Process *process, IOLayer *io) :
383 PdCom::impl::ProtocolHandler(process),
384 StreambufLayer(io),
385 ExpatWrapper<ProtocolHandler, true>(),
386 cout(process, this),
387 root(std::make_shared<DirNode>()),
388
11/22
✓ Branch 4 taken 157 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 157 times.
✗ Branch 9 not taken.
✓ Branch 28 taken 157 times.
✗ Branch 29 not taken.
✓ Branch 32 taken 157 times.
✗ Branch 33 not taken.
✓ Branch 36 taken 157 times.
✗ Branch 37 not taken.
✓ Branch 43 taken 157 times.
✗ Branch 44 not taken.
✓ Branch 56 taken 157 times.
✗ Branch 57 not taken.
✓ Branch 60 taken 157 times.
✗ Branch 61 not taken.
✓ Branch 79 taken 157 times.
✗ Branch 80 not taken.
✓ Branch 87 taken 157 times.
✗ Branch 88 not taken.
✓ Branch 94 taken 157 times.
✗ Branch 95 not taken.
157 channel_subscriptions_(new ChannelSubscriptionsWithGroup())
389 { /*{{{*/
390
391 #if PDC_DEBUG
392 {
393 char *spy = getenv("SPY");
394 if (spy and atoi(spy))
395 insert(new SpyLayer(io));
396 }
397 #endif
398
399 157 const char buf[] = "<xml>";
400
1/2
✓ Branch 3 taken 157 times.
✗ Branch 4 not taken.
157 ExpatBuffer<sizeof(buf)> inputBuf(*this);
401 157 std::memcpy(inputBuf.fill(), buf, sizeof(buf));
402
2/4
✓ Branch 6 taken 157 times.
✗ Branch 7 not taken.
✗ Branch 9 not taken.
✓ Branch 10 taken 157 times.
157 if (!parse(std::move(inputBuf), sizeof(buf) - 1, false))
403 throw Exception("setting up xml parser failed");
404 157 } /*}}}*/
405
406 ///////////////////////////////////////////////////////////////////////////
407 312 ProtocolHandler::~ProtocolHandler()
408 { /*{{{*/
409 log_debug("killed %p", static_cast<void *>(this));
410 312 } /*}}}*/
411
412 ///////////////////////////////////////////////////////////////////////////
413 void ProtocolHandler::xmlError(const char *err_msg)
414 { /*{{{*/
415 if (state != StartUp and state != WaitForConnected)
416 throw ProtocolError(std::string("Fatal XML parsing error: ") + err_msg);
417 } /*}}}*/
418
419 #if PDC_DEBUG
420 const char *statename[] = {
421 "StartUp",
422 "Idle",
423 "GetListing",
424 "WaitForConnected",
425 };
426 #endif
427
428 namespace {
429 constexpr const char *message_tags[] = {
430 "crit_error", "error", "warn", "info", "reset",
431 };
432
433 32 bool is_message_tag(const char *name)
434 {
435
1/2
✓ Branch 0 taken 82 times.
✗ Branch 1 not taken.
82 for (const auto t : message_tags) {
436
4/6
✗ Branch 0 not taken.
✓ Branch 1 taken 82 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 82 times.
✓ Branch 4 taken 32 times.
✓ Branch 5 taken 50 times.
82 if (!strcmp(name, t))
437 32 return true;
438 }
439 return false;
440 }
441 } // namespace
442
443 ///////////////////////////////////////////////////////////////////////////
444 2805 void ProtocolHandler::startElement(const char *name, const char **atts)
445 { /*{{{*/
446 2805 const char *id = nullptr;
447
448
11/13
✗ Branch 3 not taken.
✓ Branch 4 taken 2805 times.
✓ Branch 5 taken 157 times.
✓ Branch 6 taken 155 times.
✓ Branch 7 taken 85 times.
✓ Branch 8 taken 1204 times.
✓ Branch 9 taken 293 times.
✓ Branch 10 taken 842 times.
✓ Branch 11 taken 4 times.
✓ Branch 12 taken 42 times.
✓ Branch 13 taken 18 times.
✓ Branch 14 taken 5 times.
✗ Branch 15 not taken.
2805 switch (state) {
449 157 case StartUp:
450
4/8
✗ Branch 0 not taken.
✓ Branch 1 taken 157 times.
✓ Branch 2 taken 157 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 157 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 157 times.
✗ Branch 9 not taken.
157 if (!strcmp(name, "xml") and level() == 1)
451 157 state = WaitForConnected;
452 157 break;
453
454 155 case WaitForConnected:
455
4/8
✗ Branch 0 not taken.
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 155 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 155 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 155 times.
✗ Branch 9 not taken.
155 if (!strcmp(name, "connected") and level() == 2) {
456 155 Attribute a(atts);
457 155 feature = a.find("features");
458
1/2
✓ Branch 3 taken 155 times.
✗ Branch 4 not taken.
155 m_name = a.find("app");
459
1/2
✓ Branch 3 taken 155 times.
✗ Branch 4 not taken.
155 m_version = a.find("appversion");
460
461
3/4
✗ Branch 3 not taken.
✓ Branch 4 taken 155 times.
✓ Branch 5 taken 10 times.
✓ Branch 6 taken 145 times.
155 if (!feature.group)
462 20 channel_subscriptions_.reset(
463
2/4
✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 10 times.
✗ Branch 6 not taken.
10 new ChannelSubscriptionsWithoutGroup());
464
465 // if login is mandatory, <saslauth> is sent automatically by
466 // server
467
8/10
✗ Branch 3 not taken.
✓ Branch 4 taken 155 times.
✓ Branch 5 taken 145 times.
✓ Branch 6 taken 10 times.
✗ Branch 8 not taken.
✓ Branch 9 taken 145 times.
✓ Branch 10 taken 28 times.
✓ Branch 11 taken 117 times.
✓ Branch 12 taken 28 times.
✓ Branch 13 taken 127 times.
155 if (feature.login and !strcmp(a.find("login"), "mandatory")) {
468
2/2
✓ Branch 12 taken 27 times.
✓ Branch 13 taken 1 times.
28 if (process->auth_manager_) {
469 27 state = WaitForLogin;
470 27 break;
471 }
472 else
473
1/2
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 throw LoginRequired();
474 }
475
2/4
✓ Branch 2 taken 127 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 127 times.
✗ Branch 6 not taken.
254 XmlCommand(cout, "remote_host")
476
1/2
✓ Branch 3 taken 127 times.
✗ Branch 4 not taken.
254 .addAttribute("access", "allow")
477 .addEscapedAttribute(
478
2/4
✓ Branch 6 taken 127 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 127 times.
✗ Branch 11 not taken.
254 "applicationname", process->applicationName())
479
2/4
✓ Branch 6 taken 127 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 127 times.
✗ Branch 11 not taken.
254 .addEscapedAttribute("hostname", process->hostname())
480
1/2
✓ Branch 2 taken 127 times.
✗ Branch 3 not taken.
127 .setId("init");
481
482
2/2
✓ Branch 4 taken 127 times.
✓ Branch 5 taken 27 times.
127 state = Idle;
483 }
484 127 break;
485
486 85 case WaitForLogin:
487
4/6
✗ Branch 2 not taken.
✓ Branch 3 taken 85 times.
✓ Branch 4 taken 85 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 23 times.
✓ Branch 7 taken 62 times.
255 if (level() == 2 and !strcmp(name, "saslauth")
488
3/4
✓ Branch 0 taken 85 times.
✗ Branch 1 not taken.
✓ Branch 6 taken 23 times.
✓ Branch 7 taken 62 times.
170 and processLogin(atts)) {
489
1/2
✓ Branch 3 taken 23 times.
✗ Branch 4 not taken.
46 XmlCommand(cout, "remote_host")
490
1/2
✓ Branch 3 taken 23 times.
✗ Branch 4 not taken.
46 .addAttribute("access", "allow")
491 .addEscapedAttribute(
492
2/4
✓ Branch 6 taken 23 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 23 times.
✗ Branch 11 not taken.
46 "applicationname", process->applicationName())
493
2/4
✓ Branch 6 taken 23 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 23 times.
✗ Branch 11 not taken.
46 .addEscapedAttribute("hostname", process->hostname())
494
1/2
✓ Branch 2 taken 23 times.
✗ Branch 3 not taken.
23 .setId("init");
495
496 23 state = Idle;
497 }
498 85 break;
499
500 1204 case Idle:
501
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 1204 times.
1204 if (level() != 2) {
502 log_debug("Unknown XML tag in Idle: %s", name);
503 break;
504 }
505
506 // find() returns "" if id is not found
507 1204 id = Attribute(atts).find("id");
508
509
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 1204 times.
✓ Branch 2 taken 383 times.
✓ Branch 3 taken 821 times.
1204 if (!strcmp(name, "data")) {
510 383 Attribute a(atts);
511
512 383 unsigned int dataGroupId = 0;
513 383 a.getUInt("group", dataGroupId);
514
515
1/2
✓ Branch 2 taken 383 times.
✗ Branch 3 not taken.
383 dataTime = a.getTime_ns("time");
516
517
3/4
✗ Branch 3 not taken.
✓ Branch 4 taken 383 times.
✓ Branch 5 taken 42 times.
✓ Branch 6 taken 341 times.
383 if (!feature.group) {
518 42 state = ReadDataOrEvent;
519 }
520
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 339 times.
341 else if (!dataGroupId) {
521 // Attribute "group" not found. Event transmission
522 2 state = ReadEvent;
523 }
524 else {
525 339 periodic_receive_handle_ =
526
1/2
✓ Branch 4 taken 339 times.
✗ Branch 5 not taken.
1356 channel_subscriptions_->startNewDataRecieving(
527
1/2
✓ Branch 1 taken 339 times.
✗ Branch 2 not taken.
678 dataGroupId, variable_cache_.lock(),
528 1017 dataTime);
529 339 state = ReadData;
530 }
531 }
532
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 821 times.
✓ Branch 2 taken 133 times.
✓ Branch 3 taken 688 times.
821 else if (!strcmp(name, "pu")) {
533 133 unsigned int index;
534
535
5/10
✗ Branch 3 not taken.
✓ Branch 4 taken 133 times.
✓ Branch 5 taken 133 times.
✗ Branch 6 not taken.
✓ Branch 11 taken 133 times.
✗ Branch 12 not taken.
✓ Branch 13 taken 133 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 133 times.
✗ Branch 17 not taken.
133 if (!feature.xsap and Attribute(atts).getUInt("index", index)) {
536
3/4
✓ Branch 1 taken 133 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 33 times.
✓ Branch 6 taken 100 times.
399 if (parameter_event_.find(index)
537 399 != parameter_event_.end()) {
538
1/2
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
33 pollParameter(index, "pollParameterEvent");
539 }
540 }
541 }
542
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 688 times.
✓ Branch 2 taken 77 times.
✓ Branch 3 taken 611 times.
688 else if (!strcmp(name, "parameter")) {
543 77 Parameter *p = getParameter(atts);
544
1/2
✓ Branch 0 taken 77 times.
✗ Branch 1 not taken.
77 if (p) {
545 77 Attribute a(atts);
546
1/2
✓ Branch 2 taken 77 times.
✗ Branch 3 not taken.
77 const auto time_ns = a.getTime_ns("mtime");
547 77 unsigned int pm;
548 203 const auto getValue = [&a, p, this]() {
549 42 auto cache = variable_cache_.lock();
550 42 const char *const hexvalue = a.find("hexvalue");
551
2/4
✓ Branch 0 taken 42 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 42 times.
✗ Branch 4 not taken.
42 if (hexvalue && *hexvalue)
552
2/4
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
✓ Branch 6 taken 42 times.
✗ Branch 7 not taken.
84 cache[0].readFromHexDec(hexvalue, *p);
553 else
554 cache[0].readFromString(a.find("value"), *p);
555 42 return cache;
556 77 };
557
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 77 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 71 times.
154 if (!strcmp(id, "pollParameter")
558
3/4
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 71 times.
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
77 and !subscription_pending_parameter_polls_.empty()) {
559 6 auto sub_collection = std::move(
560 12 subscription_pending_parameter_polls_.front());
561 6 subscription_pending_parameter_polls_.pop_front();
562
1/2
✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
6 sub_collection.subscriptions_.readData(
563
2/4
✓ Branch 2 taken 6 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 6 times.
✗ Branch 7 not taken.
12 getValue()[0].data(), time_ns);
564 }
565
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 71 times.
✓ Branch 2 taken 3 times.
✓ Branch 3 taken 68 times.
71 else if (!strcmp(id, "VariablePollParameter")) {
566
1/2
✓ Branch 3 taken 3 times.
✗ Branch 4 not taken.
6 variable_pending_polls_.processResult(
567
2/4
✓ Branch 2 taken 3 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 3 times.
✗ Branch 7 not taken.
6 *p, getValue()[0].data(), time_ns);
568 }
569
1/2
✓ Branch 0 taken 68 times.
✗ Branch 1 not taken.
68 else if (
570
1/2
✗ Branch 3 not taken.
✓ Branch 4 taken 68 times.
68 !feature.xsap
571
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 68 times.
✓ Branch 2 taken 33 times.
✓ Branch 3 taken 35 times.
68 and !strcmp(id, "pollParameterEvent")) {
572
2/4
✓ Branch 3 taken 33 times.
✗ Branch 4 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
99 parameter_event_[p->index_].readData(
573
2/4
✓ Branch 2 taken 33 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 33 times.
✗ Branch 7 not taken.
66 getValue()[0].data(), time_ns);
574 }
575
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 35 times.
✓ Branch 2 taken 5 times.
✓ Branch 3 taken 30 times.
35 else if (!strcmp(id, "pendingParameter"))
576
1/2
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
5 pendingAck(p);
577
2/6
✗ Branch 1 not taken.
✓ Branch 2 taken 30 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
✓ Branch 6 taken 30 times.
30 else if (a.getUInt("pm", pm) && pm == 1) {
578 parameter_event_[p->index_].readData(
579 getValue()[0].data(), time_ns);
580 }
581 }
582 }
583
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 611 times.
✓ Branch 2 taken 61 times.
✓ Branch 3 taken 550 times.
611 else if (!strcmp(name, "channel")) {
584 61 Channel *c = getChannel(atts);
585
1/2
✓ Branch 0 taken 61 times.
✗ Branch 1 not taken.
61 if (c) {
586
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 61 times.
✓ Branch 2 taken 7 times.
✓ Branch 3 taken 54 times.
122 if (!strcmp(id, "pollChannel")
587
3/4
✓ Branch 0 taken 7 times.
✓ Branch 1 taken 54 times.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
61 and !subscription_pending_channel_polls_.empty()) {
588 // can be nullptr if subscription was deleted in the
589 // meantime
590 7 auto channel_poll = std::move(
591 14 subscription_pending_channel_polls_.front());
592 7 subscription_pending_channel_polls_.pop_front();
593 7 Attribute a(atts);
594
1/2
✓ Branch 2 taken 7 times.
✗ Branch 3 not taken.
14 auto cache = variable_cache_.lock();
595
1/2
✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
7 if (const auto s = a.find("hexvalue", nullptr))
596
2/4
✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 6 taken 7 times.
✗ Branch 7 not taken.
7 cache[0].readFromHexDec(s, *c);
597 else
598 cache[0].readFromString(a.find("value"), *c);
599
2/4
✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
14 channel_poll.subscriptions_.readData(
600
1/2
✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
7 cache[0].data(), a.getTime_ns("time"));
601 }
602
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 54 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 54 times.
54 else if (!strcmp(id, "VariablePollChannel")) {
603 Attribute a(atts);
604 auto cache = variable_cache_.lock();
605 if (const auto s = a.find("hexvalue", nullptr))
606 cache[0].readFromHexDec(s, *c);
607 else
608 cache[0].readFromString(a.find("value"), *c);
609 variable_pending_polls_.processResult(
610 *c, cache[0].data(), a.getTime_ns("time"));
611 }
612
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 54 times.
✓ Branch 2 taken 21 times.
✓ Branch 3 taken 33 times.
54 else if (!strcmp(id, "pendingChannel"))
613 21 pendingAck(c);
614 }
615 }
616
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 550 times.
✓ Branch 2 taken 435 times.
✓ Branch 3 taken 115 times.
550 else if (!strcmp(name, "ack")) {
617 // Check which ack was called
618
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 435 times.
✓ Branch 2 taken 48 times.
✓ Branch 3 taken 387 times.
435 if (!strcmp(id, ChannelSubscriptions::xsadId))
619 48 channel_subscriptions_->xsadAck(process);
620
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 387 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 387 times.
387 else if (!strcmp(id, "xsapQ"))
621 xsapAck();
622
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 387 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 387 times.
387 else if (!strcmp(id, ChannelSubscriptions::xsodId))
623 channel_subscriptions_->xsodAck();
624
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 387 times.
✓ Branch 2 taken 26 times.
✓ Branch 3 taken 361 times.
387 else if (!strcmp(id, "pendingChannel"))
625 26 pendingAck(nullptr);
626
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 361 times.
✓ Branch 2 taken 72 times.
✓ Branch 3 taken 289 times.
361 else if (!strcmp(id, "findQ"))
627 72 processFindRequest();
628
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 289 times.
✓ Branch 2 taken 8 times.
✓ Branch 3 taken 281 times.
289 else if (!strcmp(id, "listQ")) {
629 8 fullListing = Cached;
630
631 // Process all pending list and find requests
632
1/2
✗ Branch 4 not taken.
✓ Branch 5 taken 8 times.
8 while (processListRequest())
633 ;
634
1/2
✗ Branch 4 not taken.
✓ Branch 5 taken 8 times.
8 while (processFindRequest())
635 ;
636 }
637
6/8
✗ Branch 0 not taken.
✓ Branch 1 taken 281 times.
✓ Branch 2 taken 15 times.
✓ Branch 3 taken 266 times.
✓ Branch 6 taken 15 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 15 times.
✓ Branch 9 taken 266 times.
281 else if (!strcmp(id, "history") and !seqNoQ.empty()) {
638
2/2
✓ Branch 6 taken 9 times.
✓ Branch 7 taken 6 times.
15 if (messageSeqNo != seqNoQ.front()) {
639 18 PdCom::Message m = {};
640
1/2
✓ Branch 2 taken 9 times.
✗ Branch 3 not taken.
9 m.seqNo = seqNoQ.front();
641
2/4
✓ Branch 17 taken 9 times.
✗ Branch 18 not taken.
✓ Branch 20 taken 9 times.
✗ Branch 21 not taken.
9 process->getMessageReply(m);
642 }
643 15 seqNoQ.pop();
644 }
645
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 266 times.
✓ Branch 2 taken 149 times.
✓ Branch 3 taken 117 times.
266 else if (!strcmp(id, "init"))
646 149 process->connected();
647 }
648
2/2
✓ Branch 0 taken 105 times.
✓ Branch 1 taken 10 times.
115 else if (
649
4/6
✗ Branch 0 not taken.
✓ Branch 1 taken 115 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 105 times.
✓ Branch 4 taken 97 times.
✓ Branch 5 taken 8 times.
220 !strcmp(name, "listing") or !strcmp(name, "channels")
650
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 97 times.
✓ Branch 2 taken 8 times.
✓ Branch 3 taken 89 times.
97 or !strcmp(name, "parameters")) {
651 26 state = GetListing;
652 }
653
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 89 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 83 times.
89 else if (!strcmp(name, "saslauth"))
654 6 processLogin(atts);
655
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 83 times.
✓ Branch 2 taken 3 times.
✓ Branch 3 taken 80 times.
83 else if (!strcmp(name, "ping"))
656 3 process->pingReply();
657
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 80 times.
✓ Branch 2 taken 22 times.
✓ Branch 3 taken 58 times.
80 else if (!strcmp(name, "broadcast"))
658 22 processBroadcast(atts);
659
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 58 times.
✓ Branch 2 taken 21 times.
✓ Branch 3 taken 37 times.
58 else if (!strcmp(name, "message_history"))
660 21 state = GetActiveMessages;
661
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 37 times.
✓ Branch 2 taken 5 times.
✓ Branch 3 taken 32 times.
37 else if (!strcmp(name, "clients")) {
662 5 state = ReadClientStatistics;
663 5 client_statistics.clear();
664 }
665
3/6
✓ Branch 1 taken 32 times.
✗ Branch 2 not taken.
✓ Branch 15 taken 32 times.
✗ Branch 16 not taken.
✓ Branch 17 taken 32 times.
✗ Branch 18 not taken.
32 else if (is_message_tag(name) and process->message_manager_) {
666 64 PdCom::Message msg = {};
667 32 Attribute a(atts);
668
3/6
✗ Branch 3 not taken.
✓ Branch 4 taken 32 times.
✓ Branch 6 taken 32 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 32 times.
✗ Branch 9 not taken.
32 if (a.getMessage(name, msg, feature.history)) {
669
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 32 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 26 times.
32 if (!strcmp(id, "history")) {
670 6 messageSeqNo = msg.seqNo;
671
1/2
✓ Branch 20 taken 6 times.
✗ Branch 21 not taken.
6 process->getMessageReply(std::move(msg));
672 }
673 else {
674
1/2
✓ Branch 20 taken 26 times.
✗ Branch 21 not taken.
26 process->processMessage(std::move(msg));
675 }
676 }
677 else {
678 log_debug("Failed to parse message");
679 }
680 }
681 1204 break;
682
683 293 case GetListing:
684
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 293 times.
293 if (level() != 3) {
685 log_debug("Unknown XML tag in GetListing: %s", name);
686 break;
687 }
688
689
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 293 times.
✓ Branch 2 taken 5 times.
✓ Branch 3 taken 288 times.
293 if (!strcmp(name, "dir")) {
690 5 const char *path;
691
2/6
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 5 times.
✗ Branch 8 not taken.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
10 if (Attribute(atts).getString("path", path)
692
10/32
✓ Branch 0 taken 5 times.
✗ Branch 1 not taken.
✓ Branch 9 taken 5 times.
✗ Branch 10 not taken.
✓ Branch 13 taken 5 times.
✗ Branch 14 not taken.
✓ Branch 17 taken 5 times.
✗ Branch 18 not taken.
✓ Branch 19 taken 5 times.
✗ Branch 20 not taken.
✓ Branch 22 taken 5 times.
✗ Branch 23 not taken.
✓ Branch 25 taken 5 times.
✗ Branch 26 not taken.
✓ Branch 28 taken 5 times.
✗ Branch 29 not taken.
✓ Branch 31 taken 5 times.
✗ Branch 32 not taken.
✓ Branch 34 taken 5 times.
✗ Branch 35 not taken.
✗ Branch 37 not taken.
✗ Branch 38 not taken.
✗ Branch 40 not taken.
✗ Branch 41 not taken.
✗ Branch 43 not taken.
✗ Branch 44 not taken.
✗ Branch 46 not taken.
✗ Branch 47 not taken.
✗ Branch 49 not taken.
✗ Branch 50 not taken.
✗ Branch 52 not taken.
✗ Branch 53 not taken.
10 and !root->find(path)) {
693
2/4
✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 5 times.
✗ Branch 6 not taken.
5 DirNode *node = new DirNode;
694
2/4
✓ Branch 10 taken 5 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 5 times.
✗ Branch 14 not taken.
5 root->insert(std::unique_ptr<DirNode>(node), path);
695 log_debug("Create dir %s", node->path().c_str());
696 }
697 }
698
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 288 times.
✓ Branch 2 taken 149 times.
✓ Branch 3 taken 139 times.
288 else if (!strcmp(name, "channel"))
699 149 getChannel(atts);
700
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 139 times.
✓ Branch 2 taken 139 times.
✗ Branch 3 not taken.
139 else if (!strcmp(name, "parameter"))
701 139 getParameter(atts);
702 293 break;
703
704 842 case ReadData:
705
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 842 times.
842 if (level() != 3) {
706 log_debug("Unknown XML tag in ReadData: %s", name);
707 break;
708 }
709
710 842 readPeriodicData(name, atts);
711
712 842 break;
713
714 4 case ReadEvent:
715
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 4 times.
4 if (level() != 3) {
716 log_debug("Unknown XML tag in ReadEvent: %s", name);
717 break;
718 }
719
720 4 readEventData(name, atts);
721
722 4 break;
723
724 42 case ReadDataOrEvent:
725
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 42 times.
42 if (level() != 3) {
726 log_debug("Unknown XML tag in ReadDataOrEvent: %s", name);
727 break;
728 }
729
730
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
42 if (name[0] == 'F') {
731 42 state = ReadData;
732 42 periodic_receive_handle_ =
733
1/2
✓ Branch 4 taken 42 times.
✗ Branch 5 not taken.
168 channel_subscriptions_->startNewDataRecieving(
734 210 0, variable_cache_.lock(), dataTime);
735 42 readPeriodicData(name, atts);
736 }
737 else if (name[0] == 'E') {
738 state = ReadEvent;
739 readEventData(name, atts);
740 }
741
742 42 break;
743 18 case GetActiveMessages:
744
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 18 times.
18 if (level() != 3) {
745 log_debug("Unknown XML tag in GetActiveMessages: %s", name);
746 break;
747 }
748
749 {
750 36 Message msg;
751
2/4
✓ Branch 4 taken 18 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 18 times.
✗ Branch 8 not taken.
18 if (Attribute(atts).getMessage(name, msg, true)) {
752
1/2
✓ Branch 4 taken 18 times.
✗ Branch 5 not taken.
18 messageList.push_back(std::move(msg));
753 18 }
754 }
755 18 break;
756
757 5 case ReadClientStatistics:
758
4/8
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 5 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 5 times.
✗ Branch 8 not taken.
✓ Branch 9 taken 5 times.
5 if (level() != 3 or strcmp(name, "client")) {
759 log_debug("Unknown XML tag in ReadClientStatistics: %s", name);
760 break;
761 }
762
763 {
764 10 PdCom::ClientStatistics stats;
765
2/4
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 5 times.
✗ Branch 8 not taken.
5 if (Attribute(atts).getClientStatistics(stats)) {
766
1/2
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
5 client_statistics.push_back(std::move(stats));
767 }
768 else {
769 log_debug("Drop incomplete <client> tag");
770 5 }
771 }
772 5 break;
773
774 default:
775 break;
776 };
777 2804 } /*}}}*/
778
779 ///////////////////////////////////////////////////////////////////////////
780 2647 void ProtocolHandler::endElement(const char *name)
781 { /*{{{*/
782
7/8
✗ Branch 3 not taken.
✓ Branch 4 taken 2647 times.
✓ Branch 5 taken 319 times.
✓ Branch 6 taken 1265 times.
✓ Branch 7 taken 6 times.
✓ Branch 8 taken 39 times.
✓ Branch 9 taken 10 times.
✓ Branch 10 taken 1008 times.
2647 switch (state) {
783 319 case GetListing:
784
2/2
✓ Branch 2 taken 26 times.
✓ Branch 3 taken 293 times.
319 if (level() == 2) {
785 26 state = Idle;
786
787 // name is either
788 // - channels
789 // - parameters
790 // - listing
791
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 26 times.
✓ Branch 2 taken 10 times.
✓ Branch 3 taken 16 times.
26 if (!strcmp(name, "listing"))
792 10 processListRequest();
793 }
794
795 319 break;
796
797 1265 case ReadData:
798
6/8
✓ Branch 2 taken 381 times.
✓ Branch 3 taken 884 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 381 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 381 times.
✓ Branch 8 taken 884 times.
✓ Branch 9 taken 381 times.
1265 if (level() != 2 or strcmp(name, "data"))
799 884 break;
800
1/2
✓ Branch 2 taken 381 times.
✗ Branch 3 not taken.
381 if (periodic_receive_handle_) {
801 381 periodic_receive_handle_.endNewDataRecieving();
802 381 periodic_receive_handle_ = {};
803 }
804 381 state = Idle;
805 381 break;
806
807 6 case ReadEvent:
808 case ReadDataOrEvent:
809
6/8
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 4 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 2 times.
✓ Branch 6 taken 2 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 2 times.
✓ Branch 9 taken 4 times.
6 if (level() == 2 and !strcmp(name, "data")) {
810 2 state = Idle;
811 }
812 6 break;
813
814 39 case GetActiveMessages:
815
6/8
✓ Branch 2 taken 21 times.
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 21 times.
✓ Branch 6 taken 21 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 21 times.
✓ Branch 9 taken 18 times.
39 if (level() == 2 and !strcmp(name, "message_history")) {
816 21 state = Idle;
817
1/2
✓ Branch 21 taken 21 times.
✗ Branch 22 not taken.
21 process->activeMessagesReply(std::move(messageList));
818 21 messageList.clear();
819 }
820 39 break;
821
822 10 case ReadClientStatistics:
823
6/8
✓ Branch 2 taken 5 times.
✓ Branch 3 taken 5 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 5 times.
✓ Branch 6 taken 5 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 5 times.
✓ Branch 9 taken 5 times.
10 if (level() == 2 and !strcmp(name, "clients")) {
824 5 state = Idle;
825
1/2
✓ Branch 21 taken 5 times.
✗ Branch 22 not taken.
5 process->clientStatisticsReply(std::move(client_statistics));
826 5 client_statistics.clear();
827 }
828 10 break;
829
830 1008 default:
831 1008 break;
832 }
833 2647 } /*}}}*/
834
835 ///////////////////////////////////////////////////////////////////////////
836 // Arguments:
837 // var: if var == 0, this is a test to check whether path was found
838 // if var != 0, found the variable --> subscribe it
839 52 void ProtocolHandler::pendingAck(Variable *var)
840 { /*{{{*/
841
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 52 times.
52 if (pending_queue_.empty())
842 throw ProtocolError("Empty pending request queue");
843
844
2/2
✓ Branch 8 taken 52 times.
✓ Branch 9 taken 52 times.
104 for (const auto &weak_sub : pending_queue_.front().subscriptions_) {
845
2/2
✓ Branch 4 taken 26 times.
✓ Branch 5 taken 26 times.
78 auto ps = weak_sub.lock();
846
2/2
✓ Branch 1 taken 26 times.
✓ Branch 2 taken 26 times.
52 if (!ps)
847 26 continue;
848
849
2/2
✓ Branch 3 taken 26 times.
✓ Branch 4 taken 26 times.
52 std::shared_ptr<PdCom::impl::Subscription> new_sub;
850
1/2
✓ Branch 0 taken 26 times.
✗ Branch 1 not taken.
26 if (var) {
851
1/2
✓ Branch 11 taken 26 times.
✗ Branch 12 not taken.
182 new_sub = subscribe(
852 26 ps->This_,
853 78 std::static_pointer_cast<const MsrProto::Variable>(
854
1/2
✓ Branch 1 taken 26 times.
✗ Branch 2 not taken.
52 var->shared_from_this()),
855 130 ps->subscriber_, ps->getSelector(), false);
856 }
857
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 26 times.
26 if (!new_sub) {
858 new_sub = std::make_shared<InvalidSubscription>(
859 ps->This_, ps->subscriber_, process->shared_from_this());
860 process->pending_callbacks_.push_back(
861 {new_sub, PdCom::Subscription::State::Invalid});
862 }
863
1/2
✓ Branch 19 taken 26 times.
✗ Branch 20 not taken.
26 process->replace(ps, new_sub);
864
1/2
✓ Branch 9 taken 26 times.
✗ Branch 10 not taken.
26 ps->replaceImpl(std::move(new_sub));
865 }
866
867
2/2
✓ Branch 0 taken 26 times.
✓ Branch 1 taken 26 times.
52 if (!var) {
868 // End of processing; clean up structures
869 26 pending_queue_.pop_front();
870 }
871 52 } /*}}}*/
872
873 ///////////////////////////////////////////////////////////////////////////
874 void ProtocolHandler::xsapAck()
875 { /*{{{*/
876 if (xsapQ.empty())
877 throw ProtocolError("no pending xsad commands available");
878
879 auto map_it = parameter_event_.find(xsapQ.front());
880 if (map_it != parameter_event_.end()) {
881 map_it->second.broadcastState(
882 PdCom::Subscription::State::Active,
883 process->pending_callbacks_);
884 }
885
886 xsapQ.pop();
887 } /*}}}*/
888
889 ///////////////////////////////////////////////////////////////////////////
890 216 Parameter *ProtocolHandler::getParameter(const char **_atts)
891 { /*{{{*/
892 216 Attribute atts(_atts);
893 216 unsigned index;
894
895
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 216 times.
216 if (!atts.getUInt("index", index))
896 return 0;
897
898
1/2
✓ Branch 2 taken 216 times.
✗ Branch 3 not taken.
216 Parameter *&param = parameter[index];
899
2/2
✓ Branch 1 taken 174 times.
✓ Branch 2 taken 42 times.
216 if (!param) {
900 // index="0" name="/Constant/Value" datasize="8" typ="TDBL_LIST"
901 // anz="10" cnum="10" rnum="1" orientation="VECTOR" flags="7"
902 174 const char *name = atts.find("name");
903
904 // Check index and name
905
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 174 times.
174 if (!*name) {
906 log_debug("Missing index or name for variable. Ignoring");
907 return 0;
908 }
909
910 // Check whether variable exists
911
3/6
✓ Branch 8 taken 174 times.
✗ Branch 9 not taken.
✓ Branch 12 taken 174 times.
✗ Branch 13 not taken.
✗ Branch 26 not taken.
✓ Branch 27 taken 174 times.
174 if (std::dynamic_pointer_cast<Variable>(root->find(name))) {
912 log_debug("%s exists. Ignoring", name);
913 return 0;
914 }
915
916
1/2
✓ Branch 6 taken 174 times.
✗ Branch 7 not taken.
348 SizeInfo size_info;
917
1/2
✓ Branch 2 taken 174 times.
✗ Branch 3 not taken.
174 const auto type_info = getDataType(_atts, size_info);
918
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 174 times.
174 if (!type_info) {
919 log_debug("Datatype for %s unspecified. Ignoring", name);
920 return 0;
921 }
922
923 174 unsigned int flags;
924
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 174 times.
174 if (!atts.getUInt("flags", flags)) {
925 log_debug("Flags for %s unspecified. Ignoring", name);
926 return 0;
927 }
928
929 174 auto shared_param = std::make_shared<Parameter>(
930
1/2
✓ Branch 4 taken 174 times.
✗ Branch 5 not taken.
522 std::move(size_info), type_info, process->shared_from_this(),
931
2/4
✓ Branch 9 taken 174 times.
✗ Branch 10 not taken.
✓ Branch 16 taken 174 times.
✗ Branch 17 not taken.
870 index, flags, atts.find("alias"), atts.isTrue("dir"));
932 174 param = shared_param.get();
933
1/2
✓ Branch 10 taken 174 times.
✗ Branch 11 not taken.
174 root->insert(std::move(shared_param), name);
934 }
935
936 216 return param;
937 } /*}}}*/
938
939 ///////////////////////////////////////////////////////////////////////////
940 210 Channel *ProtocolHandler::getChannel(const char **_atts)
941 { /*{{{*/
942 210 Attribute atts(_atts);
943 210 unsigned index;
944
945
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 210 times.
210 if (!atts.getUInt("index", index))
946 return 0;
947
948
1/2
✓ Branch 2 taken 210 times.
✗ Branch 3 not taken.
210 Channel *&chan = channel[index];
949
2/2
✓ Branch 1 taken 203 times.
✓ Branch 2 taken 7 times.
210 if (!chan) {
950 // index="0" name="/Constant/Value" datasize="8" typ="TDBL_LIST"
951 // anz="10" cnum="10" rnum="1" orientation="VECTOR" flags="7"
952 203 const char *name = atts.find("name");
953
954 // Check index and name
955
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 203 times.
203 if (!*name) {
956 log_debug("Missing index or name for variable. Ignoring");
957 return 0;
958 }
959
960 // Check whether variable exists
961
3/6
✓ Branch 8 taken 203 times.
✗ Branch 9 not taken.
✓ Branch 12 taken 203 times.
✗ Branch 13 not taken.
✗ Branch 26 not taken.
✓ Branch 27 taken 203 times.
203 if (std::dynamic_pointer_cast<Variable>(root->find(name))) {
962 log_debug("%s exists. Ignoring", name);
963 return 0;
964 }
965
966
1/2
✓ Branch 8 taken 203 times.
✗ Branch 9 not taken.
406 SizeInfo size_info;
967
1/2
✓ Branch 2 taken 203 times.
✗ Branch 3 not taken.
203 const auto type_info = getDataType(_atts, size_info);
968
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 203 times.
203 if (!type_info) {
969 log_debug("Datatype for %s unspecified. Ignoring", name);
970 return 0;
971 }
972
973 203 unsigned int bufsize;
974
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 203 times.
203 if (!atts.getUInt("bufsize", bufsize)) {
975 log_debug("bufsize for %s unspecified. Ignoring", name);
976 return 0;
977 }
978
979 203 double freq;
980
3/6
✓ Branch 1 taken 203 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 203 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 203 times.
203 if (!atts.getDouble("HZ", freq) or !freq) {
981 log_debug("HZ for %s unspecified or zero. Ignoring", name);
982 return 0;
983 }
984
985 203 unsigned int taskId = 0;
986 203 atts.getUInt("task", taskId);
987
988 203 auto shared_chan = std::make_shared<Channel>(
989
1/2
✓ Branch 4 taken 203 times.
✗ Branch 5 not taken.
609 std::move(size_info), type_info, process->shared_from_this(),
990 609 index, taskId, 1.0 / freq, atts.find("alias"), bufsize,
991
2/4
✓ Branch 10 taken 203 times.
✗ Branch 11 not taken.
✓ Branch 16 taken 203 times.
✗ Branch 17 not taken.
1421 atts.isTrue("dir"));
992 203 chan = shared_chan.get();
993
1/2
✓ Branch 10 taken 203 times.
✗ Branch 11 not taken.
203 root->insert(std::move(shared_chan), name);
994 }
995
996 210 return chan;
997 } /*}}}*/
998
999 namespace {
1000 struct TypeMap
1001 {
1002 const char *msr_name;
1003 const PdCom::TypeInfo *type_info;
1004 size_t len;
1005 template <size_t N>
1006 constexpr TypeMap(const char (&name)[N], PdCom::TypeInfo const &ti) :
1007 msr_name(&name[0]), type_info(&ti), len(N - 1)
1008 {}
1009 };
1010 } // namespace
1011
1012 ///////////////////////////////////////////////////////////////////////////
1013 const PdCom::TypeInfo *
1014 377 ProtocolHandler::getDataType(const char **atts, PdCom::SizeInfo &size_info)
1015 { /*{{{*/
1016 377 unsigned int rnum = 0;
1017 377 unsigned int cnum = 0;
1018 377 unsigned int count = 0;
1019 377 const char *orientation = 0;
1020 using PdCom::SizeInfo;
1021 using PdCom::TypeInfo;
1022 377 const TypeInfo *ans = nullptr;
1023
1024 static constexpr TypeMap type_map[] = {
1025 {"TDBL", details::TypeInfoTraits<double>::type_info},
1026 {"TINT", details::TypeInfoTraits<int32_t>::type_info},
1027 {"TUINT", details::TypeInfoTraits<uint32_t>::type_info},
1028 {"TCHAR", details::TypeInfoTraits<int8_t>::type_info},
1029 {"TUCHAR", details::TypeInfoTraits<uint8_t>::type_info},
1030 {"TSHORT", details::TypeInfoTraits<int16_t>::type_info},
1031 {"TUSHORT", details::TypeInfoTraits<uint16_t>::type_info},
1032 {"TLINT", details::TypeInfoTraits<int64_t>::type_info},
1033 {"TULINT", details::TypeInfoTraits<uint64_t>::type_info},
1034 {"TFLT", details::TypeInfoTraits<float>::type_info},
1035 };
1036
1037 377 const char *const typeStr = Attribute(atts).find("typ");
1038
1039 // Try and find the name in msr_dtypemap
1040
2/2
✓ Branch 0 taken 3770 times.
✓ Branch 1 taken 377 times.
4147 for (const auto &t : type_map) {
1041
4/6
✗ Branch 2 not taken.
✓ Branch 3 taken 3770 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 3770 times.
✓ Branch 6 taken 377 times.
✓ Branch 7 taken 3393 times.
3770 if (!strncmp(typeStr, t.msr_name, t.len))
1042 377 ans = t.type_info;
1043 }
1044
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 377 times.
377 if (!ans)
1045 return nullptr;
1046
1047 // cache variable data
1048
2/2
✓ Branch 1 taken 2940 times.
✓ Branch 2 taken 377 times.
3317 for (int i = 0; atts[i]; i += 2) {
1049
3/4
✗ Branch 1 not taken.
✓ Branch 2 taken 2940 times.
✓ Branch 3 taken 49 times.
✓ Branch 4 taken 2891 times.
2940 if (!strcmp(atts[i], "rnum")) {
1050
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 49 times.
49 rnum = atoi(atts[i + 1]);
1051 }
1052
3/4
✗ Branch 1 not taken.
✓ Branch 2 taken 2891 times.
✓ Branch 3 taken 49 times.
✓ Branch 4 taken 2842 times.
2891 else if (!strcmp(atts[i], "cnum")) {
1053
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 49 times.
49 cnum = atoi(atts[i + 1]);
1054 }
1055
3/4
✗ Branch 1 not taken.
✓ Branch 2 taken 2842 times.
✓ Branch 3 taken 49 times.
✓ Branch 4 taken 2793 times.
2842 else if (!strcmp(atts[i], "anz")) {
1056
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 49 times.
49 count = atoi(atts[i + 1]);
1057 }
1058
3/4
✗ Branch 1 not taken.
✓ Branch 2 taken 2793 times.
✓ Branch 3 taken 49 times.
✓ Branch 4 taken 2744 times.
2793 else if (!strcmp(atts[i], "orientation")) {
1059 49 orientation = atts[i + 1];
1060 }
1061 }
1062
1063
2/2
✓ Branch 0 taken 49 times.
✓ Branch 1 taken 328 times.
377 if (orientation) {
1064
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 49 times.
✓ Branch 2 taken 32 times.
✓ Branch 3 taken 17 times.
49 if (!strcmp(orientation, "VECTOR")) {
1065
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 32 times.
32 if (!count)
1066 return nullptr;
1067
1068 32 size_info = SizeInfo::RowVector(count);
1069 }
1070
3/6
✓ Branch 0 taken 17 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 17 times.
✓ Branch 4 taken 17 times.
✗ Branch 5 not taken.
17 else if (orientation and !strncmp(orientation, "MATRIX", 6)) {
1071
2/4
✓ Branch 0 taken 17 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 17 times.
17 if (!(rnum and cnum))
1072 return nullptr;
1073
1074 17 size_info = SizeInfo::Matrix(rnum, cnum);
1075 }
1076 else
1077 return nullptr;
1078 }
1079 else {
1080
3/6
✓ Branch 0 taken 328 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 328 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 328 times.
328 if (rnum > 1 or cnum > 1 or count > 1)
1081 return nullptr;
1082
1083 328 size_info = SizeInfo::Scalar();
1084 }
1085
1086 377 return ans;
1087 } /*}}}*/
1088
1089 ///////////////////////////////////////////////////////////////////////////
1090 64 bool ProtocolHandler::login(const char *mech, const char *clientData)
1091 { /*{{{*/
1092
3/4
✗ Branch 3 not taken.
✓ Branch 4 taken 64 times.
✓ Branch 5 taken 1 times.
✓ Branch 6 taken 63 times.
64 if (!feature.login)
1093 1 return false;
1094
1095
2/4
✓ Branch 3 taken 63 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 63 times.
✗ Branch 7 not taken.
126 XmlCommand sasl(cout, "auth");
1096
3/4
✓ Branch 0 taken 31 times.
✓ Branch 1 taken 32 times.
✓ Branch 3 taken 31 times.
✗ Branch 4 not taken.
63 if (mech and *mech)
1097
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 sasl.addAttribute("mech", mech);
1098
1099
4/4
✓ Branch 0 taken 59 times.
✓ Branch 1 taken 4 times.
✓ Branch 3 taken 28 times.
✓ Branch 4 taken 31 times.
63 if (clientData and *clientData)
1100
1/2
✓ Branch 1 taken 28 times.
✗ Branch 2 not taken.
28 sasl.addAttribute("clientdata", clientData);
1101
1102 63 return true;
1103 } /*}}}*/
1104
1105 ///////////////////////////////////////////////////////////////////////////
1106 1 void ProtocolHandler::logout()
1107 { /*{{{*/
1108
2/4
✓ Branch 7 taken 1 times.
✗ Branch 8 not taken.
✓ Branch 11 taken 1 times.
✗ Branch 12 not taken.
1 XmlCommand(cout, "auth").addAttribute("logout", 1);
1109 1 } /*}}}*/
1110
1111 ///////////////////////////////////////////////////////////////////////////
1112 91 bool ProtocolHandler::processLogin(const char **_atts)
1113 { /*{{{*/
1114 // pdserv sends an empty saslauth tag after logout()
1115
2/2
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 90 times.
91 if (!*_atts)
1116 1 return false;
1117 90 Attribute atts(_atts);
1118 90 int finished = 0;
1119
6/6
✓ Branch 1 taken 28 times.
✓ Branch 2 taken 62 times.
✓ Branch 3 taken 4 times.
✓ Branch 4 taken 24 times.
✓ Branch 5 taken 4 times.
✓ Branch 6 taken 86 times.
90 if (atts.getInt("success", finished) and !finished)
1120 4 finished = -1;
1121
1122 90 const char *mechlist = 0;
1123 90 atts.getString("mechlist", mechlist);
1124
1125 90 const char *serverdata = 0;
1126 90 atts.getString("serverdata", serverdata);
1127
1128
1/2
✓ Branch 16 taken 90 times.
✗ Branch 17 not taken.
90 process->loginReply(mechlist, serverdata, finished);
1129 90 return finished > 0;
1130 } /*}}}*/
1131
1132
1133 ///////////////////////////////////////////////////////////////////////////
1134 28 void ProtocolHandler::setPolite(bool state)
1135 { /*{{{*/
1136
3/10
✗ Branch 3 not taken.
✓ Branch 4 taken 28 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 28 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 28 times.
✗ Branch 12 not taken.
✗ Branch 13 not taken.
✗ Branch 14 not taken.
✗ Branch 15 not taken.
28 if (state != polite and feature.polite) {
1137 polite = state;
1138
1139 XmlCommand(cout, "remote_host").addAttribute("polite", state);
1140 }
1141 28 } /*}}}*/
1142
1143 ///////////////////////////////////////////////////////////////////////////
1144 1140 bool ProtocolHandler::asyncData()
1145 { /*{{{*/
1146
1/2
✓ Branch 3 taken 1140 times.
✗ Branch 4 not taken.
1140 ExpatBuffer<ReadBufferSize> inputBuffer(*this);
1147
1148 const auto n =
1149
1/2
✓ Branch 6 taken 1140 times.
✗ Branch 7 not taken.
1140 StreambufLayer::read(inputBuffer.fill<char>(), inputBuffer.size());
1150
4/6
✗ Branch 3 not taken.
✓ Branch 4 taken 1140 times.
✓ Branch 5 taken 1140 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 1123 times.
✓ Branch 8 taken 17 times.
1140 if (!protocolError and n > 0)
1151
2/2
✓ Branch 6 taken 1122 times.
✓ Branch 7 taken 1 times.
1124 parse(std::move(inputBuffer), n);
1152 1139 return n != 0;
1153 } /*}}}*/
1154
1155 ///////////////////////////////////////////////////////////////////////////
1156
1157 2446 void ProtocolHandler::cancelSubscriptions()
1158 {
1159
2/2
✓ Branch 6 taken 52 times.
✓ Branch 7 taken 2446 times.
2498 for (auto &kv : cancelations_) {
1160 52 const auto tm = kv.first;
1161
1/2
✗ Branch 5 not taken.
✓ Branch 6 taken 52 times.
52 if (tm == PdCom::event_mode) {
1162 for (const auto &var : kv.second) {
1163 if (var->is_parameter_) {
1164 const bool unsubscribe =
1165 parameter_event_.at(var->index_).remove_expired();
1166 if (unsubscribe and feature.xsap) {
1167 XmlCommand(cout, "xsop")
1168 .addAttribute("parameters", var->index_);
1169 }
1170 }
1171 else {
1172 channel_subscriptions_->unsubscribeEvent(*var, cout);
1173 }
1174 }
1175 kv.second.clear();
1176 }
1177
1178
1/2
✓ Branch 1 taken 52 times.
✗ Branch 2 not taken.
52 else if (tm.getInterval() > 0) {
1179
2/2
✓ Branch 6 taken 3 times.
✓ Branch 7 taken 52 times.
55 for (const auto &var : kv.second) {
1180
1/2
✓ Branch 12 taken 3 times.
✗ Branch 13 not taken.
3 channel_subscriptions_->unsubscribePeriodic(tm, *var, cout);
1181 }
1182 52 kv.second.clear();
1183 }
1184 }
1185 2446 }
1186 ///////////////////////////////////////////////////////////////////////////
1187 84 bool ProtocolHandler::find(const std::string &path)
1188 { /*{{{*/
1189
1/2
✓ Branch 5 taken 84 times.
✗ Branch 6 not taken.
168 auto var = std::dynamic_pointer_cast<Variable>(root->find(path));
1190
5/6
✓ Branch 1 taken 73 times.
✓ Branch 2 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 73 times.
✓ Branch 7 taken 11 times.
✓ Branch 8 taken 73 times.
84 if (var or path.empty()) {
1191
1/2
✓ Branch 21 taken 11 times.
✗ Branch 22 not taken.
11 process->findReply(PdCom::impl::Variable::toUApi(var));
1192 11 return true;
1193 }
1194
1195 // we already have all variables cached, so we know the requested one does
1196 // not exist.
1197
2/4
✗ Branch 3 not taken.
✓ Branch 4 taken 73 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 73 times.
73 if (fullListing == Cached) {
1198 process->findReply({});
1199 return true;
1200 }
1201
1202
1/2
✓ Branch 3 taken 73 times.
✗ Branch 4 not taken.
73 findQ.push(path);
1203
1204
2/4
✗ Branch 3 not taken.
✓ Branch 4 taken 73 times.
✓ Branch 5 taken 73 times.
✗ Branch 6 not taken.
73 if (fullListing == Uncached) {
1205 // Do not flush next command
1206
3/6
✓ Branch 3 taken 73 times.
✗ Branch 4 not taken.
✓ Branch 7 taken 73 times.
✗ Branch 8 not taken.
✓ Branch 12 taken 73 times.
✗ Branch 13 not taken.
73 XmlCommand(cout, "rp").addEscapedAttribute("name", path).noFlush();
1207
1208
4/8
✓ Branch 3 taken 73 times.
✗ Branch 4 not taken.
✓ Branch 7 taken 73 times.
✗ Branch 8 not taken.
✓ Branch 11 taken 73 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 73 times.
✗ Branch 15 not taken.
73 XmlCommand(cout, "rk").addEscapedAttribute("name", path).setId("findQ");
1209 }
1210
1211 73 return false;
1212 } /*}}}*/
1213
1214 ///////////////////////////////////////////////////////////////////////////
1215 80 bool ProtocolHandler::processFindRequest()
1216 { /*{{{*/
1217
2/2
✓ Branch 2 taken 8 times.
✓ Branch 3 taken 72 times.
80 if (findQ.empty()) // required by listReply!
1218 8 return false;
1219
1220
2/4
✓ Branch 5 taken 72 times.
✗ Branch 6 not taken.
✓ Branch 9 taken 72 times.
✗ Branch 10 not taken.
144 std::shared_ptr<DirNode> node = root->find(findQ.front());
1221
1/2
✓ Branch 23 taken 72 times.
✗ Branch 24 not taken.
144 process->findReply(PdCom::impl::Variable::toUApi(
1222 144 std::dynamic_pointer_cast<PdCom::impl::Variable>(node)));
1223
1/2
✓ Branch 2 taken 72 times.
✗ Branch 3 not taken.
72 findQ.pop();
1224
1225 72 return !findQ.empty();
1226 } /*}}}*/
1227
1228 ///////////////////////////////////////////////////////////////////////////
1229 19 bool ProtocolHandler::list(const std::string &dir)
1230 { /*{{{*/
1231 19 listQ.push(dir);
1232
1233
2/4
✗ Branch 3 not taken.
✓ Branch 4 taken 19 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 19 times.
19 if (fullListing != Uncached) {
1234 if (fullListing == Cached)
1235 processListRequest();
1236 }
1237
7/8
✓ Branch 2 taken 12 times.
✓ Branch 3 taken 7 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 12 times.
✓ Branch 9 taken 2 times.
✓ Branch 10 taken 10 times.
✓ Branch 11 taken 9 times.
✓ Branch 12 taken 10 times.
19 else if (dir.empty() or !feature.list) {
1238 // List everything. First send a <rp> and the state machine
1239 // will send a <rp> thereafter
1240
1/2
✓ Branch 6 taken 9 times.
✗ Branch 7 not taken.
9 XmlCommand(cout, "rk").noFlush();
1241
2/4
✓ Branch 5 taken 9 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 9 times.
✗ Branch 9 not taken.
9 XmlCommand(cout, "rp").setId("listQ");
1242
1243 9 fullListing = InProgress;
1244 }
1245 else
1246
1/2
✓ Branch 3 taken 10 times.
✗ Branch 4 not taken.
20 XmlCommand(cout, "list")
1247
1/2
✓ Branch 3 taken 10 times.
✗ Branch 4 not taken.
20 .addEscapedAttribute("path", dir)
1248
1/2
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
10 .addAttribute("noderived", 1);
1249
1250
1/2
✗ Branch 3 not taken.
✓ Branch 4 taken 19 times.
19 return fullListing == Cached;
1251 } /*}}}*/
1252
1253 6 void PdCom::impl::MsrProto::ProtocolHandler::getClientStatistics()
1254 {
1255
1/2
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
6 XmlCommand(cout, "read_statistics");
1256 6 }
1257
1258 ///////////////////////////////////////////////////////////////////////////
1259 18 bool ProtocolHandler::processListRequest()
1260 { /*{{{*/
1261
2/4
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
✓ Branch 7 taken 18 times.
✗ Branch 8 not taken.
36 std::string path = listQ.front();
1262
1/2
✓ Branch 2 taken 18 times.
✗ Branch 3 not taken.
18 listQ.pop();
1263
1264 36 std::vector<std::string> dirList;
1265
1266 36 DirNode::List list;
1267 18 const bool listall = path.empty();
1268
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 12 times.
18 if (listall)
1269
1/2
✓ Branch 6 taken 6 times.
✗ Branch 7 not taken.
6 root->getAllChildren(list);
1270 else {
1271 // Directory list
1272
1/2
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
24 std::shared_ptr<DirNode> node = root->find(path);
1273
1/2
✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
12 if (node)
1274
1/2
✓ Branch 5 taken 12 times.
✗ Branch 6 not taken.
12 node->getChildren(list);
1275 }
1276
1277 36 std::vector<PdCom::Variable> varList;
1278
1/2
✓ Branch 2 taken 18 times.
✗ Branch 3 not taken.
18 varList.reserve(list.size());
1279
1280
2/2
✓ Branch 5 taken 276 times.
✓ Branch 6 taken 18 times.
294 for (const auto &node : list) {
1281
7/8
✓ Branch 0 taken 18 times.
✓ Branch 1 taken 258 times.
✓ Branch 8 taken 18 times.
✗ Branch 9 not taken.
✓ Branch 10 taken 6 times.
✓ Branch 11 taken 12 times.
✓ Branch 12 taken 6 times.
✓ Branch 13 taken 270 times.
276 if (!listall and node->hasChildren())
1282
2/4
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 6 times.
✗ Branch 9 not taken.
6 dirList.push_back(node->path());
1283
1284 552 auto v = std::dynamic_pointer_cast<PdCom::impl::Variable>(node);
1285
2/2
✓ Branch 1 taken 218 times.
✓ Branch 2 taken 58 times.
276 if (v)
1286
1/2
✓ Branch 6 taken 218 times.
✗ Branch 7 not taken.
218 varList.emplace_back(impl::Variable::toUApi(v));
1287 }
1288
1289
3/6
✓ Branch 17 taken 18 times.
✗ Branch 18 not taken.
✓ Branch 21 taken 18 times.
✗ Branch 22 not taken.
✓ Branch 24 taken 18 times.
✗ Branch 25 not taken.
18 process->listReply(varList, dirList);
1290
1291 36 return !listQ.empty();
1292 } /*}}}*/
1293
1294 ///////////////////////////////////////////////////////////////////////////
1295 7 void ProtocolHandler::pollChannel(PollSubscription &sub)
1296 { /*{{{*/
1297 14 const auto var_ptr = sub.variable_.lock();
1298
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 7 times.
7 if (!var_ptr)
1299 throw EmptyVariable();
1300 7 const auto index = static_cast<const Variable &>(*var_ptr).index_;
1301
2/4
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 7 times.
✗ Branch 7 not taken.
21 if (subscription_pending_channel_polls_.append(
1302
1/2
✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
14 index, sub.shared_from_this()))
1303
1/2
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
7 pollChannel(index, "pollChannel");
1304 7 }
1305
1306 7 void ProtocolHandler::pollChannel(size_t index, const char *id)
1307 {
1308
1/2
✓ Branch 3 taken 7 times.
✗ Branch 4 not taken.
14 XmlCommand(cout, "rk")
1309
1/2
✓ Branch 3 taken 7 times.
✗ Branch 4 not taken.
14 .addAttribute("index", index)
1310
1/2
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
14 .addAttribute("hex", 1)
1311
1/2
✓ Branch 2 taken 7 times.
✗ Branch 3 not taken.
7 .setId(id);
1312 7 } /*}}}*/
1313
1314 6 void ProtocolHandler::pollParameter(PollSubscription &sub)
1315 {
1316 12 const auto var_ptr = sub.variable_.lock();
1317
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6 times.
6 if (!var_ptr)
1318 throw EmptyVariable();
1319 6 const auto index = static_cast<const Variable &>(*var_ptr).index_;
1320
2/4
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 6 times.
✗ Branch 7 not taken.
18 if (subscription_pending_parameter_polls_.append(
1321
1/2
✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
12 index, sub.shared_from_this()))
1322
1/2
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
6 pollParameter(index, "pollParameter");
1323 6 }
1324
1325 ///////////////////////////////////////////////////////////////////////////
1326 47 void ProtocolHandler::pollParameter(size_t index, const char *id)
1327 { /*{{{*/
1328
1/2
✓ Branch 3 taken 47 times.
✗ Branch 4 not taken.
94 XmlCommand(cout, "rp")
1329
1/2
✓ Branch 3 taken 47 times.
✗ Branch 4 not taken.
94 .addAttribute("index", index)
1330
1/2
✓ Branch 4 taken 47 times.
✗ Branch 5 not taken.
94 .addAttribute("short", 1)
1331
1/2
✓ Branch 4 taken 47 times.
✗ Branch 5 not taken.
94 .addAttribute("hex", 1)
1332
1/2
✓ Branch 2 taken 47 times.
✗ Branch 3 not taken.
47 .setId(id);
1333 47 } /*}}}*/
1334
1335 8 void ProtocolHandler::pollVariable(
1336 const PdCom::impl::Variable &_var,
1337 VariablePollPromise &&promise)
1338 {
1339 8 const auto &var = static_cast<const MsrProto::Variable &>(_var);
1340
2/4
✓ Branch 7 taken 8 times.
✗ Branch 8 not taken.
✓ Branch 11 taken 8 times.
✗ Branch 12 not taken.
8 if (variable_pending_polls_.append(var, std::move(promise))) {
1341
2/4
✗ Branch 3 not taken.
✓ Branch 4 taken 8 times.
✓ Branch 5 taken 8 times.
✗ Branch 6 not taken.
8 if (var.is_parameter_)
1342 8 pollParameter(var.index_, "VariablePollParameter");
1343 else
1344 pollChannel(var.index_, "VariablePollChannel");
1345 }
1346 8 }
1347
1348 ///////////////////////////////////////////////////////////////////////////
1349 84 std::shared_ptr<PdCom::impl::Subscription> ProtocolHandler::subscribe(
1350 PdCom::Subscription *subscription,
1351 std::shared_ptr<const PdCom::impl::Variable> _var,
1352 PdCom::Subscriber &subscriber,
1353 const PdCom::Selector &selector,
1354 bool notify_pending)
1355 {
1356 168 auto var = std::dynamic_pointer_cast<const Variable>(_var);
1357
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 84 times.
84 if (!var)
1358 return nullptr;
1359
2/2
✓ Branch 10 taken 4 times.
✓ Branch 11 taken 80 times.
84 if (subscriber.getTransmission() == PdCom::poll_mode) {
1360 4 auto ans = std::make_shared<PollSubscription>(
1361
1/2
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
8 var, subscription, subscriber, process->shared_from_this(),
1362
1/2
✓ Branch 3 taken 4 times.
✗ Branch 4 not taken.
12 selector);
1363
1/2
✓ Branch 11 taken 4 times.
✗ Branch 12 not taken.
4 process->pending_callbacks_.push_back(
1364 {ans, PdCom::Subscription::State::Active});
1365 4 return ans;
1366 }
1367
2/2
✓ Branch 10 taken 34 times.
✓ Branch 11 taken 46 times.
80 else if (subscriber.getTransmission() == PdCom::event_mode) {
1368
3/4
✗ Branch 4 not taken.
✓ Branch 5 taken 34 times.
✓ Branch 6 taken 31 times.
✓ Branch 7 taken 3 times.
34 if (var->is_parameter_) {
1369 31 auto ans = std::make_shared<EventSubscription>(
1370
1/2
✓ Branch 4 taken 31 times.
✗ Branch 5 not taken.
62 var, subscription, subscriber, process->shared_from_this(),
1371
1/2
✓ Branch 3 taken 31 times.
✗ Branch 4 not taken.
93 selector);
1372
3/4
✓ Branch 4 taken 31 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 19 times.
✓ Branch 9 taken 12 times.
31 if (parameter_event_[var->index_].empty()) {
1373 // not yet subscribed
1374
2/4
✗ Branch 3 not taken.
✓ Branch 4 taken 19 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 19 times.
19 if (feature.xsap) {
1375 XmlCommand(cout, "xsap")
1376 .addAttribute("parameters", var->index_)
1377 .addAttribute("hex", 1)
1378 .setId("xsapQ");
1379 xsapQ.push(var->index_);
1380 if (notify_pending)
1381 process->pending_callbacks_.push_back(
1382 {ans, PdCom::Subscription::State::Pending});
1383 }
1384 else {
1385
1/2
✓ Branch 4 taken 19 times.
✗ Branch 5 not taken.
19 parameter_event_[var->index_].currentState =
1386 PdCom::Subscription::State::Active;
1387
1/2
✓ Branch 11 taken 19 times.
✗ Branch 12 not taken.
19 process->pending_callbacks_.push_back(
1388 {ans, PdCom::Subscription::State::Active});
1389 }
1390 }
1391 else {
1392
1/2
✓ Branch 0 taken 12 times.
✗ Branch 1 not taken.
12 if (notify_pending
1393
1/6
✗ Branch 0 not taken.
✓ Branch 1 taken 12 times.
✗ Branch 6 not taken.
✗ Branch 7 not taken.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
12 or parameter_event_[var->index_].currentState
1394 != PdCom::Subscription::State::Pending)
1395
1/2
✓ Branch 12 taken 12 times.
✗ Branch 13 not taken.
24 process->pending_callbacks_.push_back(
1396
1/2
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
12 {ans, parameter_event_[var->index_].currentState});
1397 }
1398
2/4
✓ Branch 5 taken 31 times.
✗ Branch 6 not taken.
✓ Branch 11 taken 31 times.
✗ Branch 12 not taken.
31 parameter_event_[var->index_].insert(ans);
1399 31 return ans;
1400 }
1401 else {
1402 3 return channel_subscriptions_->subscribeEvent(
1403 subscription, var, subscriber, selector, notify_pending,
1404
1/2
✓ Branch 7 taken 3 times.
✗ Branch 8 not taken.
3 cout);
1405 }
1406 }
1407 else {
1408
2/4
✗ Branch 4 not taken.
✓ Branch 5 taken 46 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 46 times.
46 if (var->is_parameter_)
1409 return {};
1410 46 return channel_subscriptions_->subscribePeriodic(
1411
1/2
✓ Branch 7 taken 46 times.
✗ Branch 8 not taken.
46 subscription, var, subscriber, selector, notify_pending, cout);
1412 }
1413 }
1414
1415
1416 ///////////////////////////////////////////////////////////////////////////
1417 28 std::shared_ptr<PdCom::impl::Subscription> ProtocolHandler::subscribe(
1418 PdCom::Subscription *subscription,
1419 const std::string &path,
1420 PdCom::Subscriber &subscriber,
1421 const PdCom::Selector &selector)
1422 { /*{{{*/
1423
1/2
✓ Branch 4 taken 28 times.
✗ Branch 5 not taken.
28 setPolite(false);
1424
1425
5/6
✓ Branch 5 taken 28 times.
✗ Branch 6 not taken.
✓ Branch 12 taken 2 times.
✓ Branch 13 taken 26 times.
✓ Branch 15 taken 26 times.
✓ Branch 16 taken 2 times.
54 if (auto var = std::dynamic_pointer_cast<Variable>(root->find(path))) {
1426
3/4
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
✓ Branch 9 taken 26 times.
✓ Branch 10 taken 2 times.
2 return subscribe(subscription, var, subscriber, selector, false);
1427 }
1428
1429 // Variable is unknown. Find it first. This requires interaction
1430 // with the server, which automatically postpones calls to
1431 // request->subscriber to a later point in time
1432 26 const auto it = std::find_if(
1433 78 pending_queue_.begin(), pending_queue_.end(),
1434
1/2
✓ Branch 3 taken 26 times.
✗ Branch 4 not taken.
104 [&path](const PendingRequest &pr) { return pr.path_ == path; });
1435
1436 26 auto ans = std::make_shared<PendingSubscription>(
1437
1/2
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
52 subscription, subscriber, process->shared_from_this(),
1438
1/2
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
78 selector.impl_);
1439
1440
1/2
✓ Branch 6 taken 26 times.
✗ Branch 7 not taken.
26 if (it == pending_queue_.end()) {
1441 // Discovery not yet active
1442
2/4
✓ Branch 2 taken 26 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 26 times.
✗ Branch 6 not taken.
52 XmlCommand(cout, "rp")
1443
1/2
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
52 .addAttribute("name", path)
1444
1/2
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
52 .addAttribute("hex", 1)
1445
1/2
✓ Branch 2 taken 26 times.
✗ Branch 3 not taken.
26 .setId("pendingParameter")
1446 26 .noFlush();
1447
2/4
✓ Branch 2 taken 26 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 26 times.
✗ Branch 6 not taken.
52 XmlCommand(cout, "rk")
1448
1/2
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
52 .addAttribute("name", path)
1449
1/2
✓ Branch 2 taken 26 times.
✗ Branch 3 not taken.
26 .setId("pendingChannel");
1450
5/10
✓ Branch 4 taken 26 times.
✗ Branch 5 not taken.
✓ Branch 12 taken 26 times.
✗ Branch 13 not taken.
✓ Branch 16 taken 26 times.
✗ Branch 17 not taken.
✓ Branch 23 taken 26 times.
✓ Branch 24 taken 26 times.
✗ Branch 33 not taken.
✗ Branch 34 not taken.
26 pending_queue_.push_back({path, {{ans}}});
1451 }
1452 else {
1453 it->subscriptions_.insert(ans);
1454 }
1455
1/2
✓ Branch 11 taken 26 times.
✗ Branch 12 not taken.
26 process->pending_callbacks_.push_back(
1456 {ans, PdCom::Subscription::State::Pending});
1457 26 return ans;
1458 } /*}}}*/
1459
1460 33 void ProtocolHandler::unsubscribe(EventSubscription &s) noexcept
1461 {
1462
1/2
✓ Branch 3 taken 33 times.
✗ Branch 4 not taken.
66 const auto var_ptr = s.variable_.lock();
1463
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 33 times.
33 if (!var_ptr)
1464 return;
1465 try {
1466
3/8
✓ Branch 6 taken 33 times.
✗ Branch 7 not taken.
✓ Branch 11 taken 33 times.
✗ Branch 12 not taken.
✓ Branch 17 taken 33 times.
✗ Branch 18 not taken.
✗ Branch 20 not taken.
✗ Branch 21 not taken.
99 cancelations_[PdCom::event_mode].insert(
1467 66 std::static_pointer_cast<const Variable>(var_ptr));
1468 }
1469 catch (std::bad_alloc const &) {
1470 // never going to happen
1471 }
1472 }
1473
1474 46 void ProtocolHandler::unsubscribe(PeriodicSubscription &s) noexcept
1475 {
1476
1/2
✓ Branch 3 taken 46 times.
✗ Branch 4 not taken.
92 const auto var_ptr = s.variable_.lock();
1477
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 46 times.
46 if (!var_ptr)
1478 return;
1479 try {
1480
2/4
✓ Branch 11 taken 46 times.
✗ Branch 12 not taken.
✓ Branch 16 taken 46 times.
✗ Branch 17 not taken.
92 cancelations_[s.subscriber_.getTransmission()].insert(
1481
1/4
✓ Branch 4 taken 46 times.
✗ Branch 5 not taken.
✗ Branch 8 not taken.
✗ Branch 9 not taken.
92 std::static_pointer_cast<const Variable>(var_ptr));
1482 }
1483 catch (std::bad_alloc const &) {
1484 // never going to happen
1485 }
1486 catch (...) {
1487 }
1488 }
1489
1490 ///////////////////////////////////////////////////////////////////////////
1491 4 void ProtocolHandler::ping()
1492 { /*{{{*/
1493
1/2
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
4 XmlCommand(cout, "ping");
1494 4 } /*}}}*/
1495
1496 ///////////////////////////////////////////////////////////////////////////
1497 14 std::string ProtocolHandler::name() const
1498 { /*{{{*/
1499 14 return m_name;
1500 } /*}}}*/
1501
1502 ///////////////////////////////////////////////////////////////////////////
1503 11 std::string ProtocolHandler::version() const
1504 { /*{{{*/
1505 11 return m_version;
1506 } /*}}}*/
1507
1508 ///////////////////////////////////////////////////////////////////////////
1509 43 PdCom::Variable::SetValueFuture ProtocolHandler::writeParameter(
1510 const Parameter &p,
1511 const void *src,
1512 PdCom::TypeInfo::DataType src_type,
1513 size_t idx,
1514 size_t n)
1515 { /*{{{*/
1516
1/2
✓ Branch 3 taken 43 times.
✗ Branch 4 not taken.
86 XmlCommand(cout, "wp")
1517
1/2
✓ Branch 4 taken 43 times.
✗ Branch 5 not taken.
86 .addAttribute("index", p.index_)
1518
1/2
✓ Branch 2 taken 43 times.
✗ Branch 3 not taken.
43 .addAttribute("startindex", idx, idx)
1519 .addAttributeStream(
1520 "hexvalue",
1521 43 [&, this](XmlStream &os) {
1522
1/2
✗ Branch 5 not taken.
✓ Branch 6 taken 43 times.
43 encoder_.toHex(*p.type_info, src, src_type, n, os);
1523
1/2
✓ Branch 4 taken 43 times.
✗ Branch 5 not taken.
129 })
1524
1/2
✓ Branch 2 taken 43 times.
✗ Branch 3 not taken.
43 .setId("wp");
1525 // TODO(igh-vh) implement parameter setValue notification in msr and
1526 // pdserv
1527 43 return {};
1528 } /*}}}*/
1529
1530 ///////////////////////////////////////////////////////////////////////////
1531 ///////////////////////////////////////////////////////////////////////////
1532 29625 bool fcmp(const char *p, const char *target, size_t len)
1533 {
1534
7/10
✗ Branch 0 not taken.
✓ Branch 1 taken 29625 times.
✓ Branch 2 taken 17345 times.
✓ Branch 3 taken 12280 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 17345 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 17345 times.
✓ Branch 8 taken 1820 times.
✓ Branch 9 taken 15525 times.
29625 return len >= ::strlen(target) && !strncmp(p, target, len);
1535 }
1536
1537 155 void ProtocolHandler::Feature::operator=(const char *str)
1538 { /*{{{*/
1539
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 155 times.
155 if (!str)
1540 return;
1541 155 *this = Feature {};
1542
1543 155 const char *p = str;
1544 size_t len;
1545
1546 1820 do {
1547 1975 p = str;
1548
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1975 times.
1975 str = strchr(str, ',');
1549
1550
3/4
✓ Branch 0 taken 1820 times.
✓ Branch 1 taken 155 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 155 times.
1975 len = str ? str++ - p : strlen(p);
1551
1552
1/2
✓ Branch 0 taken 1975 times.
✗ Branch 1 not taken.
1975 if (len) {
1553
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "pushparameters", len))
1554 155 pushparameters = true;
1555
1556
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "binparameters", len))
1557 155 binparameters = true;
1558
1559
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "eventchannels", len))
1560 155 eventchannels = true;
1561
1562
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "statistics", len))
1563 155 statistics = true;
1564
1565
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "pmtime", len))
1566 155 pmtime = true;
1567
1568
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "aic", len))
1569 155 aic = true;
1570
1571
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "messages", len))
1572 155 messages = true;
1573
1574
2/2
✓ Branch 1 taken 145 times.
✓ Branch 2 taken 1830 times.
1975 if (fcmp(p, "group", len))
1575 145 group = true;
1576
1577
2/2
✓ Branch 1 taken 145 times.
✓ Branch 2 taken 1830 times.
1975 if (fcmp(p, "history", len))
1578 145 history = true;
1579
1580
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1975 times.
1975 if (fcmp(p, "quiet", len))
1581 quiet = true;
1582
1583
2/2
✓ Branch 1 taken 145 times.
✓ Branch 2 taken 1830 times.
1975 if (fcmp(p, "list", len)) {
1584 145 list = true;
1585 145 group = true;
1586 }
1587
1588
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1975 times.
1975 if (fcmp(p, "exclusive", len))
1589 exclusive = true;
1590
1591
2/2
✓ Branch 1 taken 155 times.
✓ Branch 2 taken 1820 times.
1975 if (fcmp(p, "polite", len))
1592 155 polite = true;
1593
1594 /* feature disabled, because xsap does not work on all
1595 parameters in pdserv (those who are generated with
1596 msr:splitvectors), Hm
1597
1598 if (fcmp(p, "xsap", len))
1599 xsap = true;
1600 */
1601
1602
2/2
✓ Branch 1 taken 145 times.
✓ Branch 2 taken 1830 times.
1975 if (fcmp(p, "login", 5))
1603 145 login = true;
1604
1605
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1975 times.
1975 if (fcmp(p, "eventid=", 8))
1606 eventid = strtoul(p + 8, 0, 0);
1607 }
1608
1609
2/2
✓ Branch 0 taken 1820 times.
✓ Branch 1 taken 155 times.
1975 } while (str);
1610 } /*}}}*/
1611
1612
1613 16 void ProtocolHandler::getMessage(uint32_t seqNo)
1614 {
1615
3/4
✗ Branch 3 not taken.
✓ Branch 4 taken 16 times.
✓ Branch 5 taken 1 times.
✓ Branch 6 taken 15 times.
16 if (!feature.history) {
1616
1/2
✓ Branch 19 taken 1 times.
✗ Branch 20 not taken.
1 process->getMessageReply({});
1617 }
1618 else {
1619
1/2
✓ Branch 3 taken 15 times.
✗ Branch 4 not taken.
30 XmlCommand(cout, "message_history")
1620
1/2
✓ Branch 3 taken 15 times.
✗ Branch 4 not taken.
30 .addAttribute("seq", seqNo)
1621
1/2
✓ Branch 2 taken 15 times.
✗ Branch 3 not taken.
15 .setId("history");
1622 15 seqNoQ.push(seqNo);
1623 }
1624 16 }
1625
1626 22 void ProtocolHandler::getActiveMessages()
1627 {
1628
3/4
✗ Branch 3 not taken.
✓ Branch 4 taken 22 times.
✓ Branch 5 taken 1 times.
✓ Branch 6 taken 21 times.
22 if (!feature.history) {
1629
1/2
✓ Branch 18 taken 1 times.
✗ Branch 19 not taken.
1 process->activeMessagesReply({});
1630 }
1631 else
1632
1/2
✓ Branch 4 taken 21 times.
✗ Branch 5 not taken.
21 XmlCommand(cout, "message_history");
1633 22 }
1634
1635 29 void ProtocolHandler::broadcast(
1636 const std::string &message,
1637 const std::string &attr)
1638 {
1639
2/4
✓ Branch 3 taken 29 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 20 times.
✗ Branch 7 not taken.
58 XmlCommand msg(cout, "broadcast");
1640
2/2
✓ Branch 3 taken 20 times.
✓ Branch 4 taken 9 times.
29 msg.addEscapedAttribute(attr.c_str(), message);
1641 20 }
1642
1643
1644 22 void ProtocolHandler::processBroadcast(const char **atts)
1645 {
1646 22 Attribute a(atts);
1647 22 const char *try_keys[] = {"action", "text"};
1648 22 const char *value = nullptr, *user = nullptr;
1649
1/2
✓ Branch 1 taken 22 times.
✗ Branch 2 not taken.
22 if (!a.getString("user", user))
1650 22 user = "anonymous";
1651
2/4
✓ Branch 0 taken 33 times.
✗ Branch 1 not taken.
✗ Branch 7 not taken.
✓ Branch 8 taken 22 times.
33 for (const char *key : try_keys) {
1652
2/2
✓ Branch 1 taken 22 times.
✓ Branch 2 taken 11 times.
33 if (a.getString(key, value)) {
1653
5/10
✓ Branch 19 taken 22 times.
✗ Branch 20 not taken.
✓ Branch 23 taken 22 times.
✗ Branch 24 not taken.
✓ Branch 29 taken 22 times.
✗ Branch 30 not taken.
✓ Branch 36 taken 22 times.
✗ Branch 37 not taken.
✓ Branch 40 taken 22 times.
✗ Branch 41 not taken.
22 process->broadcastReply(value, key, a.getTime_ns("time"), user);
1654 22 return;
1655 }
1656 }
1657 }
1658
1659 4 void PdCom::impl::MsrProto::ProtocolHandler::readEventData(
1660 const char *name,
1661 const char **atts)
1662 {
1663
2/2
✓ Branch 1 taken 2 times.
✓ Branch 2 taken 2 times.
4 if (name[0] == 'E') {
1664 2 Attribute a(atts);
1665 2 unsigned int index;
1666
1/2
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
2 if (a.getUInt("c", index)) {
1667 2 const auto map_it =
1668
1/2
✓ Branch 7 taken 2 times.
✗ Branch 8 not taken.
4 channel_subscriptions_->getEventMap().find(index);
1669
3/10
✓ Branch 10 taken 2 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 2 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 2 times.
✗ Branch 17 not taken.
✗ Branch 18 not taken.
✗ Branch 19 not taken.
✗ Branch 21 not taken.
✗ Branch 22 not taken.
4 if (map_it != channel_subscriptions_->getEventMap().end()
1670
3/6
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 2 times.
✗ Branch 8 not taken.
4 and channel[index]) {
1671
1/2
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
4 auto cache = variable_cache_.lock();
1672
1/2
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
2 if (const char *d = a.find("d")) {
1673
3/6
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2 times.
✓ Branch 10 taken 2 times.
✗ Branch 11 not taken.
4 cache[0].readFromBase64(
1674
1/2
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
2 a.find("d"), strlen(d), *channel[index]);
1675
2/4
✓ Branch 5 taken 2 times.
✗ Branch 6 not taken.
✓ Branch 10 taken 2 times.
✗ Branch 11 not taken.
2 map_it->second.readData(cache[0].data(), dataTime);
1676 }
1677 }
1678 }
1679 }
1680 4 }
1681
1682 884 void PdCom::impl::MsrProto::ProtocolHandler::readPeriodicData(
1683 const char *name,
1684 const char **atts)
1685 {
1686
5/6
✓ Branch 1 taken 545 times.
✓ Branch 2 taken 339 times.
✓ Branch 5 taken 545 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 545 times.
✓ Branch 8 taken 339 times.
884 if (name[0] == 'F' and periodic_receive_handle_) {
1687 545 Attribute a(atts);
1688 545 unsigned int index = 0;
1689
4/8
✓ Branch 1 taken 545 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 545 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 545 times.
✗ Branch 9 not taken.
✓ Branch 10 taken 545 times.
✗ Branch 11 not taken.
545 if (a.getUInt("c", index) and channel[index]) {
1690
2/4
✓ Branch 4 taken 545 times.
✗ Branch 5 not taken.
✓ Branch 9 taken 545 times.
✗ Branch 10 not taken.
545 periodic_receive_handle_.readData(*channel[index], a.find("d"));
1691 }
1692 }
1693
4/8
✗ Branch 0 not taken.
✓ Branch 1 taken 339 times.
✓ Branch 2 taken 339 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 339 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 339 times.
✗ Branch 9 not taken.
339 else if (!strcmp(name, "time") and periodic_receive_handle_) {
1694
1/2
✓ Branch 6 taken 339 times.
✗ Branch 7 not taken.
339 periodic_receive_handle_.readTimestamps(Attribute(atts).find("d"));
1695 }
1696 884 }
1697
1698 template class PdCom::impl::MsrProto::ExpatWrapper<ProtocolHandler, true>;
1699
1700 13 bool ProtocolHandler::SubscriptionPendingPollList::append(
1701 unsigned int const id,
1702 const std::shared_ptr<PollSubscription> &sub)
1703 {
1704
4/8
✗ Branch 6 not taken.
✓ Branch 7 taken 13 times.
✓ Branch 12 taken 13 times.
✗ Branch 13 not taken.
✓ Branch 15 taken 13 times.
✗ Branch 16 not taken.
✓ Branch 19 taken 13 times.
✗ Branch 20 not taken.
13 for (auto &pp : *this) {
1705 if (pp.id_ == id) {
1706 pp.subscriptions_.insert(sub);
1707 pp.subscriptions_.remove_expired();
1708 return false;
1709 }
1710 }
1711
4/8
✓ Branch 8 taken 13 times.
✗ Branch 9 not taken.
✓ Branch 12 taken 13 times.
✗ Branch 13 not taken.
✓ Branch 19 taken 13 times.
✓ Branch 20 taken 13 times.
✗ Branch 29 not taken.
✗ Branch 30 not taken.
13 push_back({id, {{sub}}});
1712 13 return true;
1713 }
1714
1715 8 bool ProtocolHandler::VariablePendingPollList::append(
1716 const Variable &var,
1717 VariablePollPromise promise)
1718 {
1719
4/8
✗ Branch 6 not taken.
✓ Branch 7 taken 8 times.
✓ Branch 12 taken 8 times.
✗ Branch 13 not taken.
✓ Branch 15 taken 8 times.
✗ Branch 16 not taken.
✓ Branch 19 taken 8 times.
✗ Branch 20 not taken.
8 for (auto &pp : polls_) {
1720 if (pp.id_ == var.index_ and pp.is_parameter_ == var.is_parameter_) {
1721 pp.promises_.push_back(std::move(promise));
1722 return false;
1723 }
1724 }
1725 8 polls_.emplace_back(std::move(promise), var.index_, var.is_parameter_);
1726 8 return true;
1727 }
1728
1729
1730 3 void ProtocolHandler::VariablePendingPollList::processResult(
1731 Variable const &var,
1732 const char *data,
1733 std::chrono::nanoseconds ts)
1734 {
1735
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 3 times.
3 if (polls_.empty())
1736 return;
1737
1/2
✓ Branch 10 taken 3 times.
✗ Branch 11 not taken.
6 auto head = std::move(polls_.front());
1738 3 polls_.pop_front();
1739
3/6
✗ Branch 3 not taken.
✓ Branch 4 taken 3 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 3 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 3 times.
3 if (var.is_parameter_ != head.is_parameter_)
1740 throw ProtocolError("Channel/Parameter poll reply mismatch");
1741
1/2
✗ Branch 3 not taken.
✓ Branch 4 taken 3 times.
3 if (head.id_ != var.index_)
1742 throw ProtocolError("Got wrong parameter index reply");
1743
2/4
✓ Branch 3 taken 3 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 3 times.
✗ Branch 7 not taken.
6 auto ans = PdCom::VariablePollResult {var.toUApi()};
1744
1/2
✓ Branch 1 taken 3 times.
✗ Branch 2 not taken.
3 std::copy(
1745
1/2
✓ Branch 4 taken 3 times.
✗ Branch 5 not taken.
3 data, data + var.totalSizeInBytes(),
1746 3 reinterpret_cast<char *>(ans.getData()));
1747
2/2
✓ Branch 5 taken 3 times.
✓ Branch 6 taken 3 times.
6 for (auto &promise : head.promises_)
1748
1/2
✓ Branch 2 taken 3 times.
✗ Branch 3 not taken.
3 promise.resolve(ans, ts);
1749 }
1750
1751 312 ProtocolHandler::VariablePendingPollList::~VariablePendingPollList()
1752 {
1753 312 const PdCom::ProcessGoneAway ex {};
1754
2/2
✓ Branch 6 taken 5 times.
✓ Branch 7 taken 156 times.
161 for (auto &poll_list : polls_)
1755
2/2
✓ Branch 6 taken 5 times.
✓ Branch 7 taken 5 times.
10 for (auto &pending_poll : poll_list.promises_)
1756 try {
1757
1/2
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
5 pending_poll.reject(ex);
1758 }
1759 catch (...) {
1760 }
1761 156 }
1762
1763 PdCom::Variable::SetValueFuture ProtocolHandler::PendingSetValues::push()
1764 {
1765 auto ans = createFuture<PdCom::Exception const &>();
1766 queue_.push(std::move(ans.second));
1767 return std::move(ans.first);
1768 }
1769
1770 void ProtocolHandler::PendingSetValues::pop()
1771 {
1772 if (queue_.empty())
1773 return;
1774
1775 auto promise = std::move(queue_.front());
1776 queue_.pop();
1777 promise.resolve();
1778 }
1779
1780
1781 312 ProtocolHandler::PendingSetValues::~PendingSetValues()
1782 {
1783 312 auto q = std::move(queue_);
1784 312 const PdCom::ProcessGoneAway ex {};
1785
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 156 times.
156 while (!q.empty()) {
1786 try {
1787 q.front().reject(ex);
1788 }
1789 catch (...) {
1790 }
1791 q.pop();
1792 }
1793 156 }
1794
1795 std::vector<PdCom::Process::SubscriptionInfo>
1796 ProtocolHandler::getActiveSubscriptions() const
1797 {
1798 std::vector<PdCom::Process::SubscriptionInfo> ans;
1799 // parameter event
1800 for (const auto &map : parameter_event_) {
1801 PdCom::Variable v = impl::Variable::toUApi(
1802 std::static_pointer_cast<const MsrProto::Variable>(
1803 parameter.at(map.first)->shared_from_this()));
1804 for (const auto &weak_sub : map.second) {
1805 if (const auto sub = weak_sub.lock()) {
1806 ans.emplace_back(sub->This_, &sub->subscriber_, v);
1807 }
1808 }
1809 }
1810 // channels
1811 channel_subscriptions_->dump(ans, channel);
1812 return ans;
1813 }
1814
1815 std::shared_ptr<PdCom::impl::Subscription>
1816 2 ChannelSubscriptionsWithGroup::subscribeEvent(
1817 PdCom::Subscription *subscription,
1818 std::shared_ptr<const Variable> var,
1819 PdCom::Subscriber &subscriber,
1820 const PdCom::Selector &selector,
1821 bool notify_pending,
1822 XmlStream &cout)
1823 {
1824 4 auto process = var->process.lock();
1825 2 auto ans = std::make_shared<EventSubscription>(
1826
1/2
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
4 var, subscription, subscriber, process, selector);
1827
1828
2/4
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
✓ Branch 8 taken 2 times.
✗ Branch 9 not taken.
2 if (event_[var->index_].empty()) {
1829
2/4
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
4 XmlCommand(cout, "xsad")
1830
1/2
✓ Branch 5 taken 2 times.
✗ Branch 6 not taken.
4 .addAttribute("channels", var->index_)
1831
1/2
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
4 .addAttribute("group", 0)
1832
1/2
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
2 .addAttribute("coding", "Base64")
1833
1/2
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
4 .addAttribute("event", 1)
1834
1/2
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
2 .setId(xsadId);
1835
1/2
✓ Branch 8 taken 2 times.
✗ Branch 9 not taken.
2 xsadQ.push({0, var->index_});
1836
1/2
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
2 if (notify_pending)
1837
1/2
✓ Branch 9 taken 2 times.
✗ Branch 10 not taken.
2 process->pending_callbacks_.push_back(
1838 {ans, PdCom::Subscription::State::Pending});
1839 }
1840 else {
1841 if (notify_pending
1842 or event_[var->index_].currentState
1843 != PdCom::Subscription::State::Pending)
1844 process->pending_callbacks_.push_back(
1845 {ans, event_[var->index_].currentState});
1846 }
1847
2/4
✓ Branch 5 taken 2 times.
✗ Branch 6 not taken.
✓ Branch 11 taken 2 times.
✗ Branch 12 not taken.
2 event_[var->index_].insert(ans);
1848 4 return ans;
1849 }
1850
1851 std::shared_ptr<PdCom::impl::Subscription>
1852 42 PdCom::impl::MsrProto::ChannelSubscriptionsWithGroup::subscribePeriodic(
1853 PdCom::Subscription *subscription,
1854 std::shared_ptr<const Variable> var,
1855 PdCom::Subscriber &subscriber,
1856 const PdCom::Selector &selector,
1857 bool notify_pending,
1858 XmlStream &cout)
1859 {
1860 84 auto process = var->process.lock();
1861 42 auto ans = std::make_shared<PeriodicSubscription>(
1862
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
84 var, subscription, subscriber, process, selector);
1863 42 const auto si = periodic_subscriptions_with_group_.add(
1864
1/2
✓ Branch 4 taken 42 times.
✗ Branch 5 not taken.
42 process->pending_callbacks_, notify_pending, ans);
1865
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 42 times.
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 if (si.needs_server_action) {
1866
2/4
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 42 times.
✗ Branch 5 not taken.
84 XmlCommand(cout, "xsad")
1867
1/2
✓ Branch 5 taken 42 times.
✗ Branch 6 not taken.
84 .addAttribute("channels", var->index_)
1868
1/2
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 .addAttribute("group", si.group_id)
1869
1/2
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 .addAttribute("coding", "Base64")
1870
1/2
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 .addAttribute("reduction", si.decimation_)
1871
1/2
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 .addAttribute("blocksize", si.blocksize_)
1872
1/2
✓ Branch 2 taken 42 times.
✗ Branch 3 not taken.
42 .setId(xsadId);
1873
1/2
✓ Branch 8 taken 42 times.
✗ Branch 9 not taken.
42 xsadQ.push({si.group_id, var->index_});
1874 }
1875 84 return ans;
1876 }
1877
1878 void ChannelSubscriptionsWithGroup::unsubscribeEvent(
1879 const Variable &var,
1880 XmlStream &cout)
1881 {
1882 const bool unsubscribe = event_.at(var.index_).remove_expired();
1883 if (unsubscribe) {
1884 XmlCommand(cout, "xsod")
1885 .addAttribute("channels", var.index_)
1886 .addAttribute("group", 0);
1887 }
1888 }
1889
1890 1 void ChannelSubscriptionsWithGroup::unsubscribePeriodic(
1891 PdCom::Transmission tm,
1892 const Variable &var,
1893 XmlStream &cout)
1894 {
1895
1/2
✓ Branch 3 taken 1 times.
✗ Branch 4 not taken.
1 const auto si = periodic_subscriptions_with_group_.remove(var, tm);
1896
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 if (si.needs_server_action) {
1897
1/2
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 xsodQ.push(si.group_id);
1898
2/4
✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1 times.
✗ Branch 5 not taken.
2 XmlCommand(cout, "xsod")
1899
1/2
✓ Branch 4 taken 1 times.
✗ Branch 5 not taken.
2 .addAttribute("channels", var.index_)
1900
1/2
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 .addAttribute("group", si.group_id)
1901
1/2
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 .setId(xsodId);
1902 }
1903 1 }
1904
1905 44 void PdCom::impl::MsrProto::ChannelSubscriptionsWithGroup::xsadAck(
1906 Process *process)
1907 {
1908
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 44 times.
44 if (xsadQ.empty())
1909 throw ProtocolError("no pending xsad commands available");
1910
1/2
✓ Branch 3 taken 44 times.
✗ Branch 4 not taken.
44 const auto ids = xsadQ.front();
1911
1/2
✓ Branch 2 taken 44 times.
✗ Branch 3 not taken.
44 xsadQ.pop();
1912
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 42 times.
44 if (ids.group_id == 0) {
1913
2/4
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2 times.
✗ Branch 6 not taken.
4 const auto map_it = event_.find(ids.channel_id);
1914
1/2
✗ Branch 6 not taken.
✓ Branch 7 taken 2 times.
2 if (map_it == event_.end())
1915 return;
1916
2/4
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 2 times.
✗ Branch 8 not taken.
2 map_it->second.broadcastState(
1917 PdCom::Subscription::State::Active,
1918 process->pending_callbacks_);
1919 }
1920 else {
1921
2/4
✓ Branch 3 taken 42 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 44 times.
✗ Branch 7 not taken.
84 periodic_subscriptions_with_group_.subscribeWasConfirmed(
1922 42 process->pending_callbacks_, ids.group_id, ids.channel_id);
1923 }
1924 }
1925
1926 void PdCom::impl::MsrProto::ChannelSubscriptionsWithGroup::xsodAck()
1927 {
1928 if (xsodQ.empty())
1929 throw ProtocolError("no pending xsod commands available");
1930
1931 const auto id = xsodQ.front();
1932 xsodQ.pop();
1933
1934 periodic_subscriptions_with_group_.unsubscribeDone(id);
1935 }
1936
1937 void PdCom::impl::MsrProto::ChannelSubscriptionsWithGroup::dump(
1938 std::vector<PdCom::Process::SubscriptionInfo> &ans,
1939 const std::unordered_map<unsigned, Channel *> &channels) const
1940 {
1941 ChannelSubscriptions::dump(ans, channels);
1942 // channel periodic
1943 periodic_subscriptions_with_group_.dump(ans, channels);
1944 }
1945
1946 std::shared_ptr<PdCom::impl::Subscription>
1947 1 PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup::subscribeEvent(
1948 PdCom::Subscription *subscription,
1949 std::shared_ptr<const Variable> var,
1950 PdCom::Subscriber &subscriber,
1951 const PdCom::Selector &selector,
1952 bool notify_pending,
1953 XmlStream &cout)
1954 {
1955 // can't have periodic and event at the same time
1956
2/4
✓ Branch 6 taken 1 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 1 times.
✗ Branch 9 not taken.
1 if (periodic_subscriptions_without_group_.hasChannel(var->index_))
1957 1 return {};
1958
1959
1960 auto process = var->process.lock();
1961 auto ans = std::make_shared<EventSubscription>(
1962 var, subscription, subscriber, process, selector);
1963
1964 if (event_[var->index_].empty()) {
1965 event_[var->index_].currentState = PdCom::Subscription::State::Pending;
1966 XmlCommand(cout, "xsad")
1967 .addAttribute("channels", var->index_)
1968 .addAttribute("coding", "Base64")
1969 .addAttribute("event", 1)
1970 .setId(xsadId);
1971 xsadQ.push({true, var->index_});
1972 if (notify_pending)
1973 process->pending_callbacks_.push_back(
1974 {ans, PdCom::Subscription::State::Pending});
1975 }
1976 else {
1977 if (notify_pending
1978 or event_[var->index_].currentState
1979 != PdCom::Subscription::State::Pending)
1980 process->pending_callbacks_.push_back(
1981 {ans, event_[var->index_].currentState});
1982 }
1983 event_[var->index_].insert(ans);
1984 return ans;
1985 }
1986
1987 std::shared_ptr<PdCom::impl::Subscription>
1988 4 ChannelSubscriptionsWithoutGroup::subscribePeriodic(
1989 PdCom::Subscription *subscription,
1990 std::shared_ptr<const Variable> var,
1991 PdCom::Subscriber &subscriber,
1992 const PdCom::Selector &selector,
1993 bool notify_pending,
1994 XmlStream &cout)
1995 {
1996 {
1997
2/4
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 4 times.
✗ Branch 8 not taken.
8 const auto it = event_.find(var->index_);
1998
4/10
✗ Branch 4 not taken.
✓ Branch 5 taken 4 times.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
✓ Branch 11 taken 4 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 4 times.
✗ Branch 15 not taken.
✗ Branch 17 not taken.
✓ Branch 18 taken 4 times.
4 if (it != event_.end() && !it->second.empty())
1999 // periodic and event not allowed at the same time.
2000 return {};
2001 }
2002 8 auto process = var->process.lock();
2003 4 auto ans = std::make_shared<PeriodicSubscription>(
2004
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
8 var, subscription, subscriber, process, selector);
2005 4 const auto si = periodic_subscriptions_without_group_.add(
2006
1/2
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
4 process->pending_callbacks_, notify_pending, ans);
2007
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 if (si.needs_server_action) {
2008
2/4
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
8 XmlCommand(cout, "xsad")
2009
1/2
✓ Branch 5 taken 4 times.
✗ Branch 6 not taken.
8 .addAttribute("channels", var->index_)
2010
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 .addAttribute("coding", "Base64")
2011
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 .addAttribute("reduction", si.decimation_)
2012
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 .addAttribute("blocksize", si.blocksize_)
2013
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 .setId(xsadId);
2014
1/2
✓ Branch 8 taken 4 times.
✗ Branch 9 not taken.
4 xsadQ.push({false, var->index_});
2015 }
2016 4 return ans;
2017 }
2018
2019 void PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup::unsubscribeEvent(
2020 const Variable &var,
2021 XmlStream &cout)
2022 {
2023 const bool unsubscribe = event_.at(var.index_).remove_expired();
2024 if (unsubscribe) {
2025 XmlCommand(cout, "xsod").addAttribute("channels", var.index_);
2026 }
2027 }
2028
2029 2 void PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup::
2030 unsubscribePeriodic(
2031 PdCom::Transmission /* tm */,
2032 const Variable &var,
2033 XmlStream &cout)
2034 {
2035 2 const auto si = periodic_subscriptions_without_group_.remove(var);
2036
3/4
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
✓ Branch 2 taken 1 times.
✓ Branch 3 taken 1 times.
2 if (si.needs_server_action) {
2037
2/4
✓ Branch 5 taken 1 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 1 times.
✗ Branch 9 not taken.
1 XmlCommand(cout, "xsod").addAttribute("channels", var.index_);
2038 }
2039 2 }
2040
2041 4 void PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup::xsadAck(
2042 Process *process)
2043 {
2044
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 4 times.
4 if (xsadQ.empty())
2045 throw ProtocolError("no pending xsad commands available");
2046
1/2
✓ Branch 3 taken 4 times.
✗ Branch 4 not taken.
4 const auto ids = xsadQ.front();
2047
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 xsadQ.pop();
2048
2/4
✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 4 times.
4 if (ids.first) {
2049 const auto map_it = event_.find(ids.second);
2050 if (map_it == event_.end())
2051 return;
2052 map_it->second.broadcastState(
2053 PdCom::Subscription::State::Active,
2054 process->pending_callbacks_);
2055 }
2056 else {
2057
2/4
✓ Branch 3 taken 4 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 4 times.
✗ Branch 7 not taken.
8 periodic_subscriptions_without_group_.subscribeWasConfirmed(
2058 4 process->pending_callbacks_, ids.second);
2059 }
2060 }
2061
2062 void PdCom::impl::MsrProto::ChannelSubscriptionsWithoutGroup::dump(
2063 std::vector<PdCom::Process::SubscriptionInfo> &ans,
2064 const std::unordered_map<unsigned, Channel *> &channels) const
2065 {
2066 ChannelSubscriptions::dump(ans, channels);
2067 periodic_subscriptions_without_group_.dump(ans, channels);
2068 }
2069
2070 void PdCom::impl::MsrProto::ChannelSubscriptions::dump(
2071 std::vector<PdCom::Process::SubscriptionInfo> &ans,
2072 const std::unordered_map<unsigned, Channel *> &channels) const
2073 {
2074 // channel event
2075 for (const auto &map : event_) {
2076 PdCom::Variable v = impl::Variable::toUApi(
2077 std::static_pointer_cast<const MsrProto::Variable>(
2078 channels.at(map.first)->shared_from_this()));
2079 for (const auto &weak_sub : map.second) {
2080 if (const auto sub = weak_sub.lock()) {
2081 ans.emplace_back(sub->This_, &sub->subscriber_, v);
2082 }
2083 }
2084 }
2085 }
2086