Project
Loading...
Searching...
No Matches
DataDescriptorMatcher.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
18#include "Headers/DataHeader.h"
19#include "Headers/Stack.h"
20#include <iostream>
21
23{
24
26{
27 // First we check if there is any pending update
28 for (size_t i = 0; i < mPerformedUpdates; ++i) {
29 if (mUpdates[i].position == pos) {
30 return mUpdates[i].newValue;
31 }
32 }
33 // Otherwise we return the element.
34 return mElements.at(pos).value;
35}
36
37void VariableContext::publish(void (*callback)(VariableContext const&, TimesliceSlot slot, void*), void* context, TimesliceSlot slot)
38{
39 bool anyPublish = false;
40 for (size_t i = 0; i < MAX_MATCHING_VARIABLE; i++) {
41 auto& element = mElements[i];
42
43 if (element.commitVersion == element.publishVersion) {
44 continue;
45 }
46 element.publishVersion = element.commitVersion;
47 anyPublish = true;
48 }
49 if (anyPublish) {
50 callback(*this, slot, context);
51 }
52}
53
55{
56 for (size_t i = 0; i < mPerformedUpdates; ++i) {
57 auto& element = mElements[mUpdates[i].position];
58 element.value = mUpdates[i].newValue;
59 element.commitVersion++;
60 }
61 mPerformedUpdates = 0;
62}
63
65{
66 mPerformedUpdates = 0;
67 for (auto& element : mElements) {
68 element.value = None{};
69 }
70}
71
73{
74 if (auto ref = std::get_if<ContextRef>(&mValue)) {
75 auto& variable = context.get(ref->index);
76 if (auto value = std::get_if<std::string>(&variable)) {
77 return strncmp(header.dataOrigin.str, value->c_str(), header::DataOrigin::size) == 0;
78 }
79 auto maxSize = strnlen(header.dataOrigin.str, header::DataOrigin::size);
80 context.put({ref->index, std::string(header.dataOrigin.str, maxSize)});
81 return true;
82 } else if (auto s = std::get_if<std::string>(&mValue)) {
83 return strncmp(header.dataOrigin.str, s->c_str(), header::DataOrigin::size) == 0;
84 }
85 throw runtime_error("Mismatching type for variable");
86}
87
89{
90 if (auto ref = std::get_if<ContextRef>(&mValue)) {
91 auto& variable = context.get(ref->index);
92 if (auto value = std::get_if<std::string>(&variable)) {
93 return strncmp(header.dataDescription.str, value->c_str(), header::DataDescription::size) == 0;
94 }
95 auto maxSize = strnlen(header.dataDescription.str, header::DataDescription::size);
96 context.put({ref->index, std::string(header.dataDescription.str, maxSize)});
97 return true;
98 } else if (auto s = std::get_if<std::string>(&this->mValue)) {
99 return strncmp(header.dataDescription.str, s->c_str(), header::DataDescription::size) == 0;
100 }
101 throw runtime_error("Mismatching type for variable");
102}
103
105{
106 if (auto ref = std::get_if<ContextRef>(&mValue)) {
107 auto& variable = context.get(ref->index);
108 if (auto value = std::get_if<header::DataHeader::SubSpecificationType>(&variable)) {
109 return header.subSpecification == *value;
110 }
111 context.put({ref->index, header.subSpecification});
112 return true;
113 } else if (auto v = std::get_if<header::DataHeader::SubSpecificationType>(&mValue)) {
114 return header.subSpecification == *v;
115 }
116 throw runtime_error("Mismatching type for variable");
117}
118
123{
124 if (auto ref = std::get_if<ContextRef>(&mValue)) {
125 auto& variable = context.get(ref->index);
126 if (auto value = std::get_if<uint64_t>(&variable)) {
127 return (dph.startTime / mScale) == *value;
128 }
129 context.put({ref->index, dph.startTime / mScale});
130 // We always put in 12 the creation time
131 context.put({CREATIONTIME_POS, dph.creation});
132 // We always put in 13 the runNumber
133 context.put({RUNNUMBER_POS, dh.runNumber});
134 // We always put in 14 the tfCounter
135 context.put({TFCOUNTER_POS, dh.tfCounter});
136 // We always put in 15 the firstTForbit
137 context.put({FIRSTTFORBIT_POS, dh.firstTForbit});
138 return true;
139 } else if (auto v = std::get_if<uint64_t>(&mValue)) {
140 return (dph.startTime / mScale) == *v;
141 }
142 throw runtime_error("Mismatching type for variable");
143}
144
146 : mOp{other.mOp},
147 mLeft{ConstantValueMatcher{false}},
148 mRight{ConstantValueMatcher{false}}
149{
150 if (auto pval0 = std::get_if<OriginValueMatcher>(&other.mLeft)) {
151 mLeft = *pval0;
152 } else if (auto pval1 = std::get_if<DescriptionValueMatcher>(&other.mLeft)) {
153 mLeft = *pval1;
154 } else if (auto pval2 = std::get_if<SubSpecificationTypeValueMatcher>(&other.mLeft)) {
155 mLeft = *pval2;
156 } else if (auto pval3 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&other.mLeft)) {
157 mLeft = std::move(std::make_unique<DataDescriptorMatcher>(*pval3->get()));
158 } else if (auto pval4 = std::get_if<ConstantValueMatcher>(&other.mLeft)) {
159 mLeft = *pval4;
160 } else if (auto pval5 = std::get_if<StartTimeValueMatcher>(&other.mLeft)) {
161 mLeft = *pval5;
162 } else {
163 std::cerr << (other.mLeft.index() == std::variant_npos) << std::endl;
165 }
166
167 if (auto pval0 = std::get_if<OriginValueMatcher>(&other.mRight)) {
168 mRight = *pval0;
169 } else if (auto pval1 = std::get_if<DescriptionValueMatcher>(&other.mRight)) {
170 mRight = *pval1;
171 } else if (auto pval2 = std::get_if<SubSpecificationTypeValueMatcher>(&other.mRight)) {
172 mRight = *pval2;
173 } else if (auto pval3 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&other.mRight)) {
174 mRight = std::move(std::make_unique<DataDescriptorMatcher>(*pval3->get()));
175 } else if (auto pval4 = std::get_if<ConstantValueMatcher>(&other.mRight)) {
176 mRight = *pval4;
177 } else if (auto pval5 = std::get_if<StartTimeValueMatcher>(&other.mRight)) {
178 mRight = *pval5;
179 } else {
181 }
182}
183
188
191 : mOp{op},
192 mLeft{std::move(lhs)},
193 mRight{std::move(rhs)}
194{
195}
196
200{
202 dh.dataOrigin = matcher.origin;
203 dh.dataDescription = matcher.description;
204 dh.subSpecification = matcher.subSpec;
206 dph.startTime = 0;
207 header::Stack s{dh, dph};
208
209 return this->match(reinterpret_cast<char const*>(s.data()), context);
210}
211
215{
217 dh.dataOrigin = matcher.origin;
218 dh.dataDescription = matcher.description;
219 dh.subSpecification = 0;
221 dph.startTime = 0;
222 header::Stack s{dh, dph};
223
224 return this->match(reinterpret_cast<char const*>(s.data()), context);
225}
226
228{
229 return this->match(reinterpret_cast<char const*>(&header), context);
230}
231
233{
234 return this->match(reinterpret_cast<char const*>(stack.data()), context);
235}
236
237// actual polymorphic matcher which is able to cast the pointer to the correct
238// kind of header.
239bool DataDescriptorMatcher::match(char const* d, VariableContext& context) const
240{
241 bool leftValue = false, rightValue = false;
242
243 // FIXME: Using std::visit is not API compatible due to a new
244 // exception being thrown. This is the ABI compatible version.
245 // Replace with:
246 //
247 // auto eval = [&d](auto&& arg) -> bool {
248 // using T = std::decay_t<decltype(arg)>;
249 // if constexpr (std::is_same_v<T, std::unique_ptr<DataDescriptorMatcher>>) {
250 // return arg->match(d, context);
251 // if constexpr (std::is_same_v<T, ConstantValueMatcher>) {
252 // return arg->match(d);
253 // } else {
254 // return arg.match(d, context);
255 // }
256 // };
257 // switch (mOp) {
258 // case Op::Or:
259 // return std::visit(eval, mLeft) || std::visit(eval, mRight);
260 // case Op::And:
261 // return std::visit(eval, mLeft) && std::visit(eval, mRight);
262 // case Op::Xor:
263 // return std::visit(eval, mLeft) ^ std::visit(eval, mRight);
264 // case Op::Just:
265 // return std::visit(eval, mLeft);
266 // case Op::Not:
267 // return !std::visit(eval, mLeft);
268 // }
269 // When we drop support for macOS 10.13
270 if (auto pval0 = std::get_if<OriginValueMatcher>(&mLeft)) {
271 auto dh = o2::header::get<header::DataHeader*>(d);
272 if (dh == nullptr) {
273 throw runtime_error("Cannot find DataHeader");
274 }
275 leftValue = pval0->match(*dh, context);
276 } else if (auto pval1 = std::get_if<DescriptionValueMatcher>(&mLeft)) {
277 auto dh = o2::header::get<header::DataHeader*>(d);
278 if (dh == nullptr) {
279 throw runtime_error("Cannot find DataHeader");
280 }
281 leftValue = pval1->match(*dh, context);
282 } else if (auto pval2 = std::get_if<SubSpecificationTypeValueMatcher>(&mLeft)) {
283 auto dh = o2::header::get<header::DataHeader*>(d);
284 if (dh == nullptr) {
285 throw runtime_error("Cannot find DataHeader");
286 }
287 leftValue = pval2->match(*dh, context);
288 } else if (auto pval3 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&mLeft)) {
289 leftValue = (*pval3)->match(d, context);
290 } else if (auto pval4 = std::get_if<ConstantValueMatcher>(&mLeft)) {
291 leftValue = pval4->match();
292 } else if (auto pval5 = std::get_if<StartTimeValueMatcher>(&mLeft)) {
293 auto dh = o2::header::get<header::DataHeader*>(d);
294 auto dph = o2::header::get<DataProcessingHeader*>(d);
295 if (dph == nullptr) {
296 throw runtime_error("Cannot find DataProcessingHeader");
297 }
298 leftValue = pval5->match(*dh, *dph, context);
299 } else {
300 throw runtime_error("Bad parsing tree");
301 }
302 // Common speedup.
303 if (mOp == Op::And && leftValue == false) {
304 return false;
305 }
306 if (mOp == Op::Or && leftValue == true) {
307 return true;
308 }
309 if (mOp == Op::Just) {
310 return leftValue;
311 }
312 if (mOp == Op::Not) {
313 return !leftValue;
314 }
315
316 if (auto pval0 = std::get_if<OriginValueMatcher>(&mRight)) {
317 auto dh = o2::header::get<header::DataHeader*>(d);
318 rightValue = pval0->match(*dh, context);
319 } else if (auto pval1 = std::get_if<DescriptionValueMatcher>(&mRight)) {
320 auto dh = o2::header::get<header::DataHeader*>(d);
321 rightValue = pval1->match(*dh, context);
322 } else if (auto pval2 = std::get_if<SubSpecificationTypeValueMatcher>(&mRight)) {
323 auto dh = o2::header::get<header::DataHeader*>(d);
324 rightValue = pval2->match(*dh, context);
325 } else if (auto pval3 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&mRight)) {
326 rightValue = (*pval3)->match(d, context);
327 } else if (auto pval4 = std::get_if<ConstantValueMatcher>(&mRight)) {
328 rightValue = pval4->match();
329 } else if (auto pval5 = std::get_if<StartTimeValueMatcher>(&mRight)) {
330 auto dh = o2::header::get<header::DataHeader*>(d);
331 auto dph = o2::header::get<DataProcessingHeader*>(d);
332 rightValue = pval5->match(*dh, *dph, context);
333 }
334 // There are cases in which not having a rightValue might be legitimate,
335 // so we do not throw an exception.
336 switch (mOp) {
337 case Op::Or:
338 return leftValue || rightValue;
339 case Op::And:
340 return leftValue && rightValue;
341 case Op::Xor:
342 return leftValue ^ rightValue;
343 case Op::Just:
344 return leftValue;
345 case Op::Not:
346 return !leftValue;
347 }
348 throw runtime_error("Bad parsing tree");
349};
350
352{
353 if (other.mOp != this->mOp) {
354 return false;
355 }
356
357 bool leftValue = false;
358
359 {
360 auto v1 = std::get_if<OriginValueMatcher>(&this->mLeft);
361 auto v2 = std::get_if<OriginValueMatcher>(&other.mLeft);
362 if (v1 && v2 && *v1 == *v2) {
363 leftValue = true;
364 }
365 }
366
367 {
368 auto v1 = std::get_if<DescriptionValueMatcher>(&this->mLeft);
369 auto v2 = std::get_if<DescriptionValueMatcher>(&other.mLeft);
370 if (v1 && v2 && *v1 == *v2) {
371 leftValue = true;
372 }
373 }
374
375 {
376 auto v1 = std::get_if<SubSpecificationTypeValueMatcher>(&this->mLeft);
377 auto v2 = std::get_if<SubSpecificationTypeValueMatcher>(&other.mLeft);
378 if (v1 && v2 && *v1 == *v2) {
379 leftValue = true;
380 }
381 }
382
383 {
384 auto v1 = std::get_if<ConstantValueMatcher>(&this->mLeft);
385 auto v2 = std::get_if<ConstantValueMatcher>(&other.mLeft);
386 if (v1 && v2 && *v1 == *v2) {
387 leftValue = true;
388 }
389 }
390
391 {
392 auto v1 = std::get_if<StartTimeValueMatcher>(&this->mLeft);
393 auto v2 = std::get_if<StartTimeValueMatcher>(&other.mLeft);
394 if (v1 && v2 && *v1 == *v2) {
395 leftValue = true;
396 }
397 }
398
399 {
400 auto v1 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&this->mLeft);
401 auto v2 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&other.mLeft);
402 if (v1 && v2 && v1->get() && v2->get() && (**v1 == **v2)) {
403 leftValue = true;
404 }
405 }
406
407 // Shortcut the fact that the left side is different.
408 if (leftValue == false) {
409 return false;
410 }
411
412 if (mOp == Op::Just) {
413 return leftValue;
414 }
415
416 if (mOp == Op::Not) {
417 return leftValue;
418 }
419
420 {
421 auto v1 = std::get_if<OriginValueMatcher>(&this->mRight);
422 auto v2 = std::get_if<OriginValueMatcher>(&other.mRight);
423 if (v1 && v2 && *v1 == *v2) {
424 return true;
425 }
426 }
427
428 {
429 auto v1 = std::get_if<DescriptionValueMatcher>(&this->mRight);
430 auto v2 = std::get_if<DescriptionValueMatcher>(&other.mRight);
431 if (v1 && v2 && *v1 == *v2) {
432 return true;
433 }
434 }
435
436 {
437 auto v1 = std::get_if<SubSpecificationTypeValueMatcher>(&this->mRight);
438 auto v2 = std::get_if<SubSpecificationTypeValueMatcher>(&other.mRight);
439 if (v1 && v2 && *v1 == *v2) {
440 return true;
441 }
442 }
443
444 {
445 auto v1 = std::get_if<ConstantValueMatcher>(&this->mRight);
446 auto v2 = std::get_if<ConstantValueMatcher>(&other.mRight);
447 if (v1 && v2 && *v1 == *v2) {
448 return true;
449 }
450 }
451
452 {
453 auto v1 = std::get_if<StartTimeValueMatcher>(&this->mRight);
454 auto v2 = std::get_if<StartTimeValueMatcher>(&other.mRight);
455 if (v1 && v2 && *v1 == *v2) {
456 return true;
457 }
458 }
459
460 {
461 auto v1 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&this->mRight);
462 auto v2 = std::get_if<std::unique_ptr<DataDescriptorMatcher>>(&other.mRight);
463 if (v1 && v2 && v1->get() && v2->get() && (**v1 == **v2)) {
464 return true;
465 }
466 }
467 // We alredy know the left side is true.
468 return false;
469}
470
471std::ostream& operator<<(std::ostream& os, DataDescriptorMatcher const& matcher)
472{
473 auto edgeWalker = overloaded{
474 [&os](EdgeActions::EnterNode action) {
475 os << "(" << action.node->mOp;
476 if (action.node->mOp == DataDescriptorMatcher::Op::Just ||
477 action.node->mOp == DataDescriptorMatcher::Op::Not) {
479 }
481 },
482 [&os](EdgeActions::EnterLeft) { os << " "; },
483 [&os](EdgeActions::ExitLeft) { os << " "; },
484 [&os](EdgeActions::EnterRight) { os << " "; },
485 [&os](EdgeActions::ExitRight) { os << " "; },
486 [&os](EdgeActions::ExitNode) { os << ")"; },
487 [&os](auto) {}};
488 auto leafWalker = overloaded{
489 [&os](OriginValueMatcher const& origin) { os << "origin:" << origin; },
490 [&os](DescriptionValueMatcher const& description) { os << "description:" << description; },
491 [&os](SubSpecificationTypeValueMatcher const& subSpec) { os << "subSpec:" << subSpec; },
492 [&os](StartTimeValueMatcher const& startTime) { os << "startTime:" << startTime; },
493 [&os](ConstantValueMatcher const& constant) {},
494 [&os](auto t) { os << "not implemented " << typeid(decltype(t)).name(); }};
496 edgeWalker,
497 leafWalker);
498
499 return os;
500}
501
502std::ostream& operator<<(std::ostream& os, DataDescriptorMatcher::Op const& op)
503{
504 switch (op) {
506 os << "and";
507 break;
509 os << "or";
510 break;
512 os << "just";
513 break;
515 os << "not";
516 break;
518 os << "xor";
519 break;
520 }
521 return os;
522}
523
524} // namespace o2::framework::data_matcher
#define O2_BUILTIN_UNREACHABLE
int32_t i
uint32_t op
uint16_t pos
Definition RawData.h:3
uint32_t stack
Definition RawData.h:1
bool match(ConcreteDataMatcher const &matcher, VariableContext &context) const
bool operator==(DataDescriptorMatcher const &other) const
DataDescriptorMatcher & operator=(DataDescriptorMatcher const &other)
DataDescriptorMatcher(DataDescriptorMatcher const &other)
Something which can be matched against a header::DataDescription.
bool match(header::DataHeader const &header, VariableContext &context) const
Something which can be matched against a header::DataOrigin.
bool match(header::DataHeader const &header, VariableContext &context) const
Matcher on actual time, as reported in the DataProcessingHeader.
bool match(header::DataHeader const &dh, DataProcessingHeader const &dph, VariableContext &context) const
Something which can be matched against a header::SubSpecificationType.
bool match(header::DataHeader const &header, VariableContext &context) const
void publish(void(*callback)(VariableContext const &, TimesliceSlot slot, void *context), void *context, TimesliceSlot slot)
ContextElement::Value const & get(size_t pos) const
const GLdouble * v
Definition glcorearb.h:832
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLfloat GLfloat v1
Definition glcorearb.h:812
GLfloat GLfloat GLfloat v2
Definition glcorearb.h:813
std::variant< OriginValueMatcher, DescriptionValueMatcher, SubSpecificationTypeValueMatcher, std::unique_ptr< DataDescriptorMatcher >, ConstantValueMatcher, StartTimeValueMatcher > Node
std::ostream & operator<<(std::ostream &os, DataDescriptorMatcher const &matcher)
@ CREATIONTIME_POS
The DataHeader::runNumber associated to the timeslice.
@ FIRSTTFORBIT_POS
The DataHeader::tfCounter associated to the timeslice.
@ TFCOUNTER_POS
The DataProcessingHeader::startTime associated to the timeslice.
@ RUNNUMBER_POS
The DataHeader::firstTForbit associated to the timeslice.
RuntimeErrorRef runtime_error(const char *)
Defining DataPointCompositeObject explicitly as copiable.
header::DataHeader::SubSpecificationType subSpec
std::variant< uint32_t, uint64_t, std::string, None > Value
static void walk(DataDescriptorMatcher const &top, EDGEWALKER edgeWalker, LEAFWALKER leafWalker)
Marks an empty item in the context.
From https://en.cppreference.com/w/cpp/utility/variant/visit.
the main header struct
Definition DataHeader.h:618
TFCounterType tfCounter
Definition DataHeader.h:679
TForbitType firstTForbit
Definition DataHeader.h:674
DataDescription dataDescription
Definition DataHeader.h:636
SubSpecificationType subSpecification
Definition DataHeader.h:656
RunNumberType runNumber
Definition DataHeader.h:684
a move-only header stack with serialized headers This is the flat buffer where all the headers in a m...
Definition Stack.h:36
VectorOfTObjectPtrs other