COMMS
Template library intended to help with implementation of communication protocols.
Loading...
Searching...
No Matches
Variant.h
Go to the documentation of this file.
1//
2// Copyright 2017 - 2025 (C). Alex Robenko. All rights reserved.
3//
4// This Source Code Form is subject to the terms of the Mozilla Public
5// License, v. 2.0. If a copy of the MPL was not distributed with this
6// file, You can obtain one at http://mozilla.org/MPL/2.0/.
7
10
11#pragma once
12
13#include "comms/ErrorStatus.h"
14#include "comms/options.h"
15#include "basic/Variant.h"
16#include "details/AdaptBasicField.h"
17#include "comms/details/macro_common.h"
18#include "comms/details/variant_access.h"
19
20namespace comms
21{
22
23namespace field
24{
25
72template <typename TFieldBase, typename TMembers, typename... TOptions>
73class Variant : public
74 details::AdaptBasicFieldT<
75 basic::Variant<
76 TFieldBase,
77 details::OptionsParser<TOptions...>::ForcedMembersVersionDependency,
78 TMembers
79 >,
80 TOptions...>
81{
82 using BaseImpl =
83 details::AdaptBasicFieldT<
84 basic::Variant<
85 TFieldBase,
86 details::OptionsParser<TOptions...>::ForcedMembersVersionDependency,
87 TMembers
88 >,
89 TOptions...>;
90
92 "TMembers is expected to be a tuple of std::tuple<...>");
93
94 static_assert(
95 1U <= std::tuple_size<TMembers>::value,
96 "Number of members is expected to be at least 1.");
97
98public:
100 using FieldBase = TFieldBase;
101
103 using Endian = typename BaseImpl::Endian;
104
106 using VersionType = typename BaseImpl::VersionType;
107
109 using ParsedOptions = details::OptionsParser<TOptions...>;
110
112 using CommsTag = typename BaseImpl::CommsTag;
113
117 using ValueType = typename BaseImpl::ValueType;
118
122 using Members = typename BaseImpl::Members;
123
127 using FieldType = typename ParsedOptions::FieldType;
128
131 Variant() = default;
132
134 explicit Variant(const ValueType& val)
135 : BaseImpl(val)
136 {
137 }
138
140 explicit Variant(ValueType&& val)
141 : BaseImpl(std::move(val))
142 {
143 }
144
147 static constexpr bool hasFailOnInvalid()
148 {
149 return ParsedOptions::HasFailOnInvalid;
150 }
151
154 static constexpr bool hasIgnoreInvalid()
155 {
156 return ParsedOptions::HasIgnoreInvalid;
157 }
158
161 static constexpr bool hasEmptySerialization()
162 {
163 return ParsedOptions::HasEmptySerialization;
164 }
165
168 static constexpr bool hasFieldType()
169 {
170 return ParsedOptions::HasFieldType;
171 }
172
175 static constexpr bool hasFixedValue()
176 {
177 return ParsedOptions::HasFixedValue;
178 }
179
182 static constexpr bool hasName()
183 {
184 return ParsedOptions::HasName;
185 }
186
190 {
191 return BaseImpl::value();
192 }
193
196 const ValueType& value() const
197 {
198 return BaseImpl::value();
199 }
200
203 const ValueType& getValue() const
204 {
205 return BaseImpl::getValue();
206 }
207
210 template <typename U>
211 void setValue(U&& val)
212 {
213 BaseImpl::setValue(std::forward<U>(val));
214 }
215
221 std::size_t length() const
222 {
223 return BaseImpl::length();
224 }
225
228 static constexpr std::size_t minLength()
229 {
230 return BaseImpl::minLength();
231 }
232
235 static constexpr std::size_t maxLength()
236 {
237 return BaseImpl::maxLength();
238 }
239
247 template <typename TIter>
248 ErrorStatus read(TIter& iter, std::size_t size)
249 {
250 return BaseImpl::read(iter, size);
251 }
252
255 static constexpr bool hasReadNoStatus()
256 {
257 return false;
258 }
259
261 template <typename TIter>
262 void readNoStatus(TIter& iter) = delete;
263
265 bool canWrite() const
266 {
267 return BaseImpl::canWrite();
268 }
269
278 template <typename TIter>
279 ErrorStatus write(TIter& iter, std::size_t size) const
280 {
281 return BaseImpl::write(iter, size);
282 }
283
286 static constexpr bool hasWriteNoStatus()
287 {
288 return BaseImpl::hasWriteNoStatus();
289 }
290
296 template <typename TIter>
297 void writeNoStatus(TIter& iter) const
298 {
299 BaseImpl::writeNoStatus(iter);
300 }
301
304 bool valid() const
305 {
306 return BaseImpl::valid();
307 }
308
313 bool refresh()
314 {
315 return BaseImpl::refresh();
316 }
317
321 std::size_t currentField() const
322 {
323 return BaseImpl::currentField();
324 }
325
333 void selectField(std::size_t idx)
334 {
335 BaseImpl::selectField(idx);
336 }
337
373 template <typename TFunc>
374 void currentFieldExec(TFunc&& func)
375 {
376 BaseImpl::currentFieldExec(std::forward<TFunc>(func));
377 }
378
396 template <typename TFunc>
397 void currentFieldExec(TFunc&& func) const
398 {
399 BaseImpl::currentFieldExec(std::forward<TFunc>(func));
400 }
401
415 template <std::size_t TIdx, typename... TArgs>
416 typename std::tuple_element<TIdx, Members>::type& initField(TArgs&&... args)
417 {
418 return BaseImpl::template initField<TIdx>(std::forward<TArgs>(args)...);
419 }
420
428 template <std::size_t TIdx>
430 {
431 BaseImpl::template deinitField<TIdx>();
432 }
433
439 template <std::size_t TIdx>
440 typename std::tuple_element<TIdx, Members>::type& accessField()
441 {
442 return BaseImpl::template accessField<TIdx>();
443 }
444
450 template <std::size_t TIdx>
451 const typename std::tuple_element<TIdx, Members>::type& accessField() const
452 {
453 return BaseImpl::template accessField<TIdx>();
454 }
455
459 bool currentFieldValid() const
460 {
461 return BaseImpl::currentFieldValid();
462 }
463
466 void reset()
467 {
468 BaseImpl::reset();
469 }
470
472 static constexpr bool isVersionDependent()
473 {
474 return ParsedOptions::HasCustomVersionUpdate || BaseImpl::isVersionDependent();
475 }
476
478 static constexpr bool hasNonDefaultRefresh()
479 {
480 return BaseImpl::hasNonDefaultRefresh();
481 }
482
487 {
488 return BaseImpl::getVersion();
489 }
490
494 {
495 return BaseImpl::setVersion(version);
496 }
497
498protected:
499 using BaseImpl::readData;
500 using BaseImpl::writeData;
501
502private:
503 static_assert(!ParsedOptions::HasSerOffset,
504 "comms::option::def::NumValueSerOffset option is not applicable to Variant field");
505 static_assert(!ParsedOptions::HasFixedLengthLimit,
506 "comms::option::def::FixedLength option is not applicable to Variant field");
507 static_assert(!ParsedOptions::HasFixedBitLengthLimit,
508 "comms::option::def::FixedBitLength option is not applicable to Variant field");
509 static_assert(!ParsedOptions::HasVarLengthLimits,
510 "comms::option::def::VarLength option is not applicable to Variant field");
511 static_assert(!ParsedOptions::HasAvailableLengthLimit,
512 "comms::option::def::AvailableLengthLimit option is not applicable to Variant field");
513 static_assert(!ParsedOptions::HasSequenceElemLengthForcing,
514 "comms::option::def::SequenceElemLengthForcingEnabled option is not applicable to Variant field");
515 static_assert(!ParsedOptions::HasSequenceSizeForcing,
516 "comms::option::def::SequenceSizeForcingEnabled option is not applicable to Variant field");
517 static_assert(!ParsedOptions::HasSequenceLengthForcing,
518 "comms::option::def::SequenceLengthForcingEnabled option is not applicable to Variant field");
519 static_assert(!ParsedOptions::HasSequenceFixedSize,
520 "comms::option::def::SequenceFixedSize option is not applicable to Variant field");
521 static_assert(!ParsedOptions::HasSequenceFixedSizeUseFixedSizeStorage,
522 "comms::option::app::SequenceFixedSizeUseFixedSizeStorage option is not applicable to Variant field");
523 static_assert(!ParsedOptions::HasSequenceSizeFieldPrefix,
524 "comms::option::def::SequenceSizeFieldPrefix option is not applicable to Variant field");
525 static_assert(!ParsedOptions::HasSequenceSerLengthFieldPrefix,
526 "comms::option::def::SequenceSerLengthFieldPrefix option is not applicable to Variant field");
527 static_assert(!ParsedOptions::HasSequenceElemSerLengthFieldPrefix,
528 "comms::option::def::SequenceElemSerLengthFieldPrefix option is not applicable to Variant field");
529 static_assert(!ParsedOptions::HasSequenceElemFixedSerLengthFieldPrefix,
530 "comms::option::def::SequenceElemFixedSerLengthFieldPrefix option is not applicable to Variant field");
531 static_assert(!ParsedOptions::HasSequenceTrailingFieldSuffix,
532 "comms::option::def::SequenceTrailingFieldSuffix option is not applicable to Variant field");
533 static_assert(!ParsedOptions::HasSequenceTerminationFieldSuffix,
534 "comms::option::def::SequenceTerminationFieldSuffix option is not applicable to Variant field");
535 static_assert(!ParsedOptions::HasFixedSizeStorage,
536 "comms::option::app::FixedSizeStorage option is not applicable to Variant field");
537 static_assert(!ParsedOptions::HasCustomStorageType,
538 "comms::option::app::CustomStorageType option is not applicable to Variant field");
539 static_assert(!ParsedOptions::HasScalingRatio,
540 "comms::option::def::ScalingRatio option is not applicable to Variant field");
541 static_assert(!ParsedOptions::HasUnits,
542 "comms::option::def::Units option is not applicable to Variant field");
543 static_assert(!ParsedOptions::HasOrigDataView,
544 "comms::option::app::OrigDataView option is not applicable to Variant field");
545 static_assert(!ParsedOptions::HasMultiRangeValidation,
546 "comms::option::def::ValidNumValueRange (or similar) option is not applicable to Variant field");
547 static_assert(!ParsedOptions::HasVersionsRange,
548 "comms::option::def::ExistsBetweenVersions (or similar) option is not applicable to Variant field");
549 static_assert(!ParsedOptions::HasInvalidByDefault,
550 "comms::option::def::InvalidByDefault option is not applicable to Variant field");
551 static_assert(!ParsedOptions::HasMissingOnReadFail,
552 "comms::option::def::MissingOnReadFail option is not applicable to Variant field");
553 static_assert(!ParsedOptions::HasMissingOnInvalid,
554 "comms::option::def::MissingOnInvalid option is not applicable to Variant field");
555};
556
557namespace details
558{
559
560template <typename TVar>
561class VariantEqualityCompHelper
562{
563public:
564 VariantEqualityCompHelper(const TVar& other, bool& result)
565 : other_(other),
566 result_(result)
567 {}
568
569 template <std::size_t TIdx, typename TField>
570 void operator()(const TField& field)
571 {
572 result_ = (field == other_.template accessField<TIdx>());
573 }
574
575private:
576 const TVar& other_;
577 bool& result_;
578
579};
580
581template <typename TVar>
582VariantEqualityCompHelper<TVar> makeVariantEqualityCompHelper(TVar& other, bool& result)
583{
584 return VariantEqualityCompHelper<TVar>(other, result);
585}
586
587template <typename TVar>
588class VariantLessCompHelper
589{
590public:
591 VariantLessCompHelper(const TVar& other, bool& result)
592 : other_(other),
593 result_(result)
594 {}
595
596 template <std::size_t TIdx, typename TField>
597 void operator()(const TField& field)
598 {
599 result_ = (field < other_.template accessField<TIdx>());
600 }
601
602private:
603 const TVar& other_;
604 bool& result_;
605};
606
607template <typename TVar>
608VariantLessCompHelper<TVar> makeVariantLessCompHelper(TVar& other, bool& result)
609{
610 return VariantLessCompHelper<TVar>(other, result);
611}
612
613} // namespace details
614
620template <typename TFieldBase, typename TMembers, typename... TOptions>
624{
625 if (&field1 == &field2) {
626 return true;
627 }
628
629 if (field1.currentFieldValid() != field2.currentFieldValid()) {
630 return false;
631 }
632
633 if (!field1.currentFieldValid()) {
634 return true;
635 }
636
637 if (field1.currentField() != field2.currentField()) {
638 return false;
639 }
640
641 bool result = false;
642 field1.currentFieldExec(details::makeVariantEqualityCompHelper(field2, result));
643 return result;
644}
645
651template <typename TFieldBase, typename TMembers, typename... TOptions>
655{
656 return !(field1 == field2);
657}
658
663template <typename TFieldBase, typename TMembers, typename... TOptions>
667{
668 if (!field1.currentFieldValid()) {
669 return (!field2.currentFieldValid());
670 }
671
672 if (!field2.currentFieldValid()) {
673 return false;
674 }
675
676 if (field1.currentField() < field2.currentField()) {
677 return true;
678 }
679
680 if (field1.currentField() != field2.currentField()) {
681 return false;
682 }
683
684 if (&field1 == &field2) {
685 return false;
686 }
687
688 bool result = false;
689 field1.currentFieldExec(details::makeVariantLessCompHelper(field2, result));
690 return result;
691}
692
698template <typename T>
699constexpr bool isVariant()
700{
701 return std::is_same<typename T::CommsTag, tag::Variant>::value;
702}
703
707template <typename TFieldBase, typename TMembers, typename... TOptions>
708inline
709Variant<TFieldBase, TMembers, TOptions...>&
711{
712 return field;
713}
714
718template <typename TFieldBase, typename TMembers, typename... TOptions>
719inline
720const Variant<TFieldBase, TMembers, TOptions...>&
722{
723 return field;
724}
725
762#define COMMS_VARIANT_MEMBERS_ACCESS(...) \
763 COMMS_EXPAND(COMMS_DEFINE_FIELD_ENUM(__VA_ARGS__)) \
764 COMMS_AS_VARIANT_FUNC { \
765 auto& var = comms::field::toFieldBase(*this); \
766 using Var = typename std::decay<decltype(var)>::type; \
767 static_assert(std::tuple_size<typename Var::Members>::value == FieldIdx_numOfValues, \
768 "Invalid number of names for variant field"); \
769 return var; \
770 }\
771 COMMS_AS_VARIANT_CONST_FUNC { \
772 auto& var = comms::field::toFieldBase(*this); \
773 using Var = typename std::decay<decltype(var)>::type; \
774 static_assert(std::tuple_size<typename Var::Members>::value == FieldIdx_numOfValues, \
775 "Invalid number of names for variant field"); \
776 return var; \
777 } \
778 COMMS_DO_VARIANT_MEM_ACC_FUNC(asVariant(), __VA_ARGS__)
779
780
791#define COMMS_VARIANT_MEMBERS_ACCESS_NOTEMPLATE(...) \
792 COMMS_EXPAND(COMMS_DEFINE_FIELD_ENUM(__VA_ARGS__)) \
793 COMMS_DO_VARIANT_MEM_ACC_FUNC_NOTEMPLATE(__VA_ARGS__)
794
945#define COMMS_VARIANT_MEMBERS_NAMES(...) \
946 COMMS_EXPAND(COMMS_VARIANT_MEMBERS_ACCESS(__VA_ARGS__)) \
947 COMMS_EXPAND(COMMS_DO_FIELD_TYPEDEF(typename Base::Members, Field_, FieldIdx_, __VA_ARGS__))
948
949
950} // namespace field
951
952} // namespace comms
953
954
This file contain definition of error statuses used by comms module.
Defines a "variant" field, that can contain any of the provided ones.
Definition Variant.h:81
bool setVersion(VersionType version)
Default implementation of version update.
Definition Variant.h:493
typename ParsedOptions::FieldType FieldType
Type of actual extending field specified via comms::option::def::FieldType.
Definition Variant.h:127
static constexpr bool hasNonDefaultRefresh()
Compile time check if this class has non-default refresh functionality.
Definition Variant.h:478
void deinitField()
Destruct previously initialised (via initField()) contained field.
Definition Variant.h:429
void currentFieldExec(TFunc &&func) const
Execute provided function object with current field as parameter (const variant).
Definition Variant.h:397
void readNoStatus(TIter &iter)=delete
Read operation without error check and status report is not supported.
const std::tuple_element< TIdx, Members >::type & accessField() const
Access already constructed field at specifed index (known at compile time).
Definition Variant.h:451
TFieldBase FieldBase
Base class provided in the first template parameter.
Definition Variant.h:100
const Variant< TFieldBase, TMembers, TOptions... > & toFieldBase(const Variant< TFieldBase, TMembers, TOptions... > &field)
Upcast type of the field definition to its parent comms::field::Variant type in order to have access ...
Definition Variant.h:721
std::tuple_element< TIdx, Members >::type & accessField()
Access already constructed field at specifed index (known at compile time).
Definition Variant.h:440
std::size_t currentField() const
Get index of the current field (within the Members tuple).
Definition Variant.h:321
bool canWrite() const
Check of whether the field has a consistent value for writing.
Definition Variant.h:265
void selectField(std::size_t idx)
Select type of the variant field.
Definition Variant.h:333
ErrorStatus write(TIter &iter, std::size_t size) const
Write current field value to output data sequence.
Definition Variant.h:279
bool refresh()
Refresh the field's value.
Definition Variant.h:313
bool valid() const
Check validity of all the contained field.
Definition Variant.h:304
static constexpr bool hasWriteNoStatus()
Compile time check of whether the field has proper writeNoStatus() member function.
Definition Variant.h:286
static constexpr bool isVersionDependent()
Compile time check if this class is version dependent.
Definition Variant.h:472
static constexpr bool hasFixedValue()
Compile time inquiry of whether comms::option::def::FixedValue option has been used.
Definition Variant.h:175
static constexpr bool hasFailOnInvalid()
Compile time inquiry of whether comms::option::def::FailOnInvalid option has been used.
Definition Variant.h:147
void currentFieldExec(TFunc &&func)
Execute provided function object with current field as parameter.
Definition Variant.h:374
typename BaseImpl::CommsTag CommsTag
Tag indicating type of the field.
Definition Variant.h:112
std::size_t length() const
Get length required to serialise contained fields.
Definition Variant.h:221
Variant(const ValueType &val)
Constructor.
Definition Variant.h:134
static constexpr bool hasEmptySerialization()
Compile time inquiry of whether comms::option::def::EmptySerialization option has been used.
Definition Variant.h:161
void setValue(U &&val)
Set value.
Definition Variant.h:211
typename BaseImpl::Endian Endian
Endian used for serialisation.
Definition Variant.h:103
Variant()=default
Default constructor.
static constexpr std::size_t maxLength()
Get maximal length that is required to serialise all possible contained fields.
Definition Variant.h:235
details::OptionsParser< TOptions... > ParsedOptions
All the options provided to this class bundled into struct.
Definition Variant.h:109
typename BaseImpl::VersionType VersionType
Version type.
Definition Variant.h:106
ValueType & value()
Get access to the internal storage buffer.
Definition Variant.h:189
static constexpr bool hasReadNoStatus()
Compile time check of whether the field has proper readNoStatus() member function.
Definition Variant.h:255
bool operator<(const Variant< TFieldBase, TMembers, TOptions... > &field1, const Variant< TFieldBase, TMembers, TOptions... > &field2)
Order comparison operator.
Definition Variant.h:664
void writeNoStatus(TIter &iter) const
Write current field value to output data sequence without error check and status report.
Definition Variant.h:297
std::tuple_element< TIdx, Members >::type & initField(TArgs &&... args)
Construct and initialise specified contained field in the internal buffer.
Definition Variant.h:416
constexpr bool isVariant()
Compile time check function of whether a provided type is any variant of comms::field::Variant.
Definition Variant.h:699
typename BaseImpl::Members Members
All the supported types.
Definition Variant.h:122
void reset()
Invalidate current state.
Definition Variant.h:466
const ValueType & getValue() const
Get value.
Definition Variant.h:203
Variant(ValueType &&val)
Constructor.
Definition Variant.h:140
const ValueType & value() const
Get access to the internal storage buffer.
Definition Variant.h:196
static constexpr bool hasFieldType()
Compile time inquiry of whether comms::option::def::FieldType option has been used.
Definition Variant.h:168
bool operator!=(const Variant< TFieldBase, TMembers, TOptions... > &field1, const Variant< TFieldBase, TMembers, TOptions... > &field2)
Non-equality comparison operator.
Definition Variant.h:652
static constexpr std::size_t minLength()
Get minimal length that is required to serialise all possible contained fields.
Definition Variant.h:228
bool operator==(const Variant< TFieldBase, TMembers, TOptions... > &field1, const Variant< TFieldBase, TMembers, TOptions... > &field2)
Equality comparison operator.
Definition Variant.h:621
static constexpr bool hasName()
Compile time inquiry of whether comms::option::def::HasName option has been used.
Definition Variant.h:182
typename BaseImpl::ValueType ValueType
Value type.
Definition Variant.h:117
VersionType getVersion() const
Get version of the field.
Definition Variant.h:486
bool currentFieldValid() const
Check whether the field contains a valid instance of other field.
Definition Variant.h:459
static constexpr bool hasIgnoreInvalid()
Compile time inquiry of whether comms::option::def::IgnoreInvalid option has been used.
Definition Variant.h:154
ErrorStatus read(TIter &iter, std::size_t size)
Read field value from input data sequence.
Definition Variant.h:248
Variant< TFieldBase, TMembers, TOptions... > & toFieldBase(Variant< TFieldBase, TMembers, TOptions... > &field)
Upcast type of the field definition to its parent comms::field::Variant type in order to have access ...
Definition Variant.h:710
Main namespace for all classes / functions of COMMS library.
ErrorStatus
Error statuses reported by the Communication module.
Definition ErrorStatus.h:17
constexpr unsigned version()
Version of the COMMS library as single numeric value.
Definition version.h:64
STL namespace.
Contains definition of all the options used by the COMMS library.
Check whether provided type is a variant of std::tuple.
Definition Tuple.h:36