43template<
typename Message>
inline
44void checkBinaryLength(
const Message&, MessageSize length, MessageSize minimalRequiredLength)
46 if ONIXS_B3_BOE_UNLIKELY(length < minimalRequiredLength)
47 throwBinaryBlockIsTooSmall(length, minimalRequiredLength, Message::className());
62template < class Container, class BlockLength >
68 return *
static_cast <const Container*
> (
this);
81 assert(container().blockLength() >= (offset + size<Value>()) &&
82 "The requested field exceeds provided block boundaries.");
84 const void*
const location = advanceByBytes(container().block(), offset);
85 return getValue<Value>(location);
92 assert(container().blockLength() >= (offset + size<Value>()) &&
93 "The requested field exceeds provided block boundaries.");
95 const void* location = advanceByBytes(container().block(), offset);
96 const Value* valuePtr =
static_cast<const Value*
>(location);
116 return (null != value);
126 return (since <= container().version() &&
ordinary (value, offset, null ) );
133 typedef typename Enumeration::Base Base;
134 typedef typename Enumeration::Enum Enum;
143 typedef typename Enumeration::Base Base;
144 typedef typename Enumeration::Enum Enum;
147 return null != value;
172 return null != value;
182 return (since <= container().version() &&
decimal(value, offset, null));
189 assert(container().blockLength() >= (offset + Length) &&
"The requested field exceeds provided block boundaries.");
191 const Char*
const text =
reinterpret_cast <const Char*
> (advanceByBytes(container().block(), offset));
193 return StrRef(text, strnlen(text, Length));
204 return !value.empty();
222template <
class Container,
class BlockLength >
229 return *
static_cast<Container*
>(
this);
236 const BlockLength encodedBlockLength =
container().blockLength();
238 assert(encodedBlockLength >= offset);
240 const size_t paddingLength = encodedBlockLength - offset;
241 std::memset(advanceByBytes(
container().block(), offset), 0, paddingLength);
248 assert(
container().blockLength() >= (offset + size<FieldValue>()) &&
"The requested field exceeds provided block boundaries.");
250 void*
const fieldPos = advanceByBytes(
container().block(), offset);
251 setValue(fieldPos, value);
258 if ONIXS_B3_BOE_UNLIKELY(since >
container().version())
259 throwDisallowedField();
268 typedef typename Enumeration::Base Base;
276 typedef typename Enumeration::Base Base;
277 setOrdinary(offset,
static_cast<Base
>(value), since);
284 assert(
container().blockLength() >= (offset + Size) &&
"The requested field exceeds provided block boundaries.");
285 assert(value.size() <= Size &&
"The string is truncated.");
287 void*
const fieldPos = advanceByBytes(
container().block(), offset);
288 const size_t sizeToCopy = (std::min)(Size,
static_cast<BlockLength
>(value.size()));
291 std::memcpy(fieldPos, value.data(), sizeToCopy);
293 std::memset(advanceByBytes(fieldPos, sizeToCopy), 0, Size - sizeToCopy);
300 if ONIXS_B3_BOE_UNLIKELY(since >
container().version())
301 throwDisallowedField();
314template <class BodySizeType>
396template <
class EntryType,
class BlockLength,
class NumInGroup,
class Length >
453 return Entry(entry_, size_, version_);
465 return entry_ == other.entry_;
471 return entry_ != other.entry_;
477 return entry_ < other.entry_;
483 return entry_ > other.entry_;
491 entry_ = advanceByBytes(entry_, size_);
501 entry_ = advanceBackByBytes(entry_, size_);
512 return Iterator(advanceByBytes(entry_, distance * size_), size_, version_);
521 return Iterator(advanceBackByBytes(entry_, distance * size_), size_, version_);
542 , blockLength_(blockLength)
547 assert(blockLength > 0);
548 assert(version != 0);
572 return Iterator(
encoded(), blockLength_, version_);
586 assert(index < size_);
589 return Entry(advanceByBytes(encoded_,
static_cast<ptrdiff_t
>(index) * blockLength_), blockLength_, version_);
605 template<
class OtherEntry,
class OtherBlockLength,
class OtherNumInGroup,
class OtherLength >
607 : encoded_(other.encoded_)
608 , blockLength_(other.blockLength_)
610 , version_(other.version_)
615 assert(blockLength_ == other.blockLength_);
616 assert(size_ == other.size_);
619 template <
class OtherEntry,
class OtherBlockLength,
class OtherNumInGroup,
class OtherLength>
622 encoded_ = other.encoded_;
624 blockLength_ = other.blockLength_;
626 assert(blockLength_ == other.blockLength_);
630 assert(size_ == other.size_);
632 version_ = other.version_;
639 template <
class OtherEntry,
class OtherBlockLength,
class OtherNumInGroup,
class OtherLength>
friend class SbeGroupEntries;
642 BlockLength blockLength_;
648template <
class EntryType,
class DimensionType,
class GroupSizeType >
665 typedef SbeGroupEntries <EntryType, typename Dimension::BlockLength, typename Dimension::NumInGroup, GroupSizeType >
Entries;
688 : header_(
static_cast <Dimension*
>(data))
750 assert(index <
size());
762 return Entries (entries_, header_->blockLength(), header_->numInGroup(), version_);
809 Dimension*
const group =
static_cast<Dimension*
>(header_);
810 group->setBlockLength(entrySize);
811 group->setNumInGroup(0);
815 Size allocate(Size entryCount,
const void* messageTail,
const void* blockEnd)
821 Dimension*
const group =
static_cast<Dimension*
>(header_);
823 const EntrySize entrySize = group->blockLength();
825 if ONIXS_B3_BOE_UNLIKELY(
826 entrySize < EntryType::blockLength(version_))
828 throwBadBinaryBlock();
831 const Size oldEntryCount = group->numInGroup();
833 if(oldEntryCount == entryCount)
836 const ptrdiff_t memShift =
837 (entryCount - oldEntryCount) *
static_cast<ptrdiff_t
>(entrySize);
839 const void*
const newMessageTail =
840 advanceByBytes(messageTail, memShift);
842 if ONIXS_B3_BOE_UNLIKELY(byteDistance(blockEnd, newMessageTail) < 0)
843 throwNotEnoughSpace();
845 const
void* const oldEndOfGroup =
846 advanceByBytes(entries_, static_cast<ptrdiff_t>(entrySize) * oldEntryCount);
848 void* const newEndGroup =
849 advanceByBytes(entries_, static_cast<ptrdiff_t>(entrySize) * entryCount);
854 byteDistance(messageTail, oldEndOfGroup));
856 group->setNumInGroup(entryCount);
858 return oldEntryCount;
863 void setup(Size entryCount, const
void* messageTail, const
void* blockEnd)
869 const Size oldEntryCount = allocate(entryCount, messageTail, blockEnd);
871 for(Size index = oldEntryCount; index < entryCount; ++index)
872 zeroPaddingBytes((*
this)[index].resetVariableFields());
877 void construct(Size entryCount,
const void* messageTail,
const void* blockEnd)
883 const Size oldEntryCount = allocate(entryCount, messageTail, blockEnd);
885 for(Size index = oldEntryCount; index < entryCount; ++index)
886 zeroPaddingBytes((*
this)[index].reset());
890 static void zeroPaddingBytes(Entry& entry)
892 assert(entry.valid());
893 entry.zeroPaddingBytes(EntryType::minimalBlockLength(entry.version()));
905template <
class BinarySize >
925 template<
class BinaryVariableLengthFieldType>
928 return *
static_cast<BinaryVariableLengthFieldType*
>(binary_);
939 assert(headSize <= size_);
950 if ONIXS_B3_BOE_UNLIKELY(
empty() || (size_ < BinaryVariableLengthFieldType::Size))
952 throwBadBinaryBlock();
957 if ONIXS_B3_BOE_UNLIKELY(headSize > size_)
959 throwBadBinaryBlock();
973template <
class BinarySize>
998 return Group(binary_, size_, version_);
1007 const BinarySize headSize =
head<Group>().binarySize();
1009 assert(headSize <= size_);
1011 return SbeGroupList(advanceByBytes(binary_, headSize), size_ - headSize, version_);
1020 const BinarySize headSize =
head<Group>().binarySize();
1022 assert(headSize <= size_);
1033 const BinarySize headSize = checkHead<Group>();
1035 return SbeGroupList(advanceByBytes(binary_, headSize), size_ - headSize, version_);
1044 const BinarySize headSize = checkHead<Group>();
1050 template<
class Group>
1051 BinarySize checkHead()
const
1053 if ONIXS_B3_BOE_UNLIKELY(size_ < Group::Dimension::Size)
1055 throwBadBinaryBlock();
1058 const Group group = head<Group>();
1060 const BinarySize headSize = group.binarySize();
1062 if ONIXS_B3_BOE_UNLIKELY(headSize > size_)
1064 throwBadBinaryBlock();
1069 const BinarySize entrySize = group.entrySize();
1070 const BinarySize expectedEntrySize = Group::Entry::minimalBlockLength(version_);
1072 if ONIXS_B3_BOE_UNLIKELY(entrySize < expectedEntrySize)
1074 throwBadBinaryBlock();
1087template<
typename Traits>
1090 if ONIXS_B3_BOE_UNLIKELY(version < Traits::MinimalVersion)
1092 throwBadMessageVersion(version, Traits::MinimalVersion);
1097template<
typename Traits>
1102 if ONIXS_B3_BOE_UNLIKELY(version < since)
1104 throwBadMessageVersion(version, since);
1109template<
typename Traits>
1112 if ONIXS_B3_BOE_UNLIKELY(
id != Traits::Id)
1114 throwBadSchemaId(Traits::Id,
id);
1119template<
typename Traits>
1213 return header_->templateId();
1221 return header_->version();
1229 return header_->schemaId();
1253 return advanceByBytes(header_, size_);
1261 return advanceByBytes(header_, size_);
1283 return header_->blockLength();
1326 return GroupList(list, listSize, header_->version());
1339 return GroupList(
const_cast<void*
>(list), listSize, header_->version());
1343 template<
typename Group>
1347 assert(group.valid());
1348 group.init(entrySize);
1352 template<
typename Group>
1354 void setupGroup(Group& group,
typename Group::Size entryCount,
const void* messageTail)
1356 assert(messageTail);
1357 assert(group.valid());
1358 group.setup(entryCount, messageTail,
blockEnd());
1362 template<
typename Group>
1364 void constructGroup(Group& group,
typename Group::Size entryCount,
const void* messageTail)
1366 assert(messageTail);
1367 assert(group.valid());
1368 group.construct(entryCount, messageTail,
blockEnd());
1372 template<
typename DATA>
1375 assert(oldMessageTail);
1377 const ptrdiff_t lengthChange =
static_cast<ptrdiff_t
>(value.length() - data.length());
1379 const void*
const newMessageTail = advanceByBytes(oldMessageTail, lengthChange);
1381 if ONIXS_B3_BOE_UNLIKELY(byteDistance(
blockEnd(), newMessageTail) < 0)
1382 throwNotEnoughSpace();
1384 const void*
const oldEndOfData = advanceByBytes(data.varData().data(), data.varData().size());
1386 void*
const newEndOfData = toOpaquePtr(advanceByBytes(&data, value.length() + DATA::Size));
1388 std::memmove(newEndOfData, oldEndOfData, byteDistance(oldMessageTail, oldEndOfData));
1390 data.varData(value);
1432 header_->setTemplateId(value);
1434 header_->setSchemaId(
id);
1444 const ptrdiff_t distance = byteDistance(tail,
binary());
1446 assert(distance > 0);
1448 assert(distance <= (std::numeric_limits<MessageSize>::max)());
1452 assert(size <= size_);
1458 template<
class Callable,
class Owner>
1465 template<
class Callable,
class Owner>
1468 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1469 throwDisallowedField();
1475 template<
class Callable,
class Owner>
1478 ONIXS_B3_BOE_CHECK_NOTHROW(callable(owner));
1479 return callable(owner).varData();
1483 template<
class Callable,
class Owner>
1486 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1493 template<
class Callable,
class Owner>
1496 ONIXS_B3_BOE_CHECK_NOTHROW(callable(owner));
1497 callable(owner).length(0);
1501 template<
class Group,
class Callable,
class Owner>
1504 const typename Group::EntrySize entrySize = Group::Entry::blockLength(
version());
1506 Group grp = callable(owner);
1512 template<
class Callable,
class Owner>
1515 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1522 template<
class Group,
class Callable,
class Owner>
1525 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1532 template<
class Group,
class Callable,
class Owner>
1535 ONIXS_B3_BOE_CHECK_NOTHROW(callable(owner));
1536 return callable(owner);
1540 template<
class Group,
class Callable,
class Owner>
1543 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1550 template<
class Group,
class Callable,
class Owner>
1553 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1560 template<
class Group,
class Callable,
class Owner>
1563 Group group = callable(owner);
1571 template<
class Group,
class Callable,
class Owner>
1574 if ONIXS_B3_BOE_UNLIKELY(since >
version())
1581 template<
class Group,
class Callable,
class Owner>
1582 Group
setupGroup(Callable callable,
typename Group::Size length, Owner& owner)
1584 Group group = callable(owner);
#define ONIXS_B3_BOE_LTWT_EXPORTED
#define ONIXS_B3_BOE_MESSAGING_NAMESPACE_END
#define ONIXS_B3_BOE_MESSAGING_NAMESPACE_BEGIN
#define ONIXS_B3_BOE_LTWT_CLASS
#define ONIXS_B3_BOE_UNUSED
#define ONIXS_B3_BOE_CONSTEXPR
#define ONIXS_B3_BOE_NODISCARD
#define ONIXS_B3_BOE_DEFAULT
#define ONIXS_B3_BOE_NULLPTR
#define ONIXS_B3_BOE_NOTHROW
#define ONIXS_B3_BOE_HOTPATH
bool decimal(Value &value, BlockLength offset, NullValue null) const noexcept
bool fixedStr(StrRef &value, BlockLength offset) const noexcept
Provides access to an optional string field value.
Value ordinary(BlockLength offset) const noexcept
Value & accessOrdinary(BlockLength offset) noexcept
const Value & accessOrdinary(BlockLength offset) const noexcept
bool fixedStr(StrRef &value, BlockLength offset, SchemaVersion since) const noexcept
Provides access to an optional string field value.
bool ordinary(Value &value, BlockLength offset, NullValue null) const noexcept
Provides access to an optional field value.
bool decimal(Value &value, BlockLength offset, NullValue null, SchemaVersion since) const noexcept
Value decimal(BlockLength offset) const noexcept
Enumeration::Enum enumeration(BlockLength offset) const noexcept
bool enumeration(typename Enumeration::Enum &value, BlockLength offset, NullValue null) const noexcept
Provides access to an optional field value.
BinaryBlock()=default
Initializes a blank instance.
bool ordinary(Value &value, BlockLength offset, NullValue null, SchemaVersion since) const noexcept
Provides access to an optional field value.
StrRef fixedStr(BlockLength offset) const noexcept
Provides access to a string field value.
bool enumeration(typename Enumeration::Enum &value, BlockLength offset, NullValue null, SchemaVersion since) const noexcept
Provides access to an optional field value.
NumInGroup numInGroup() const noexcept
Counter representing the number of entries in a repeating group.
BlockLength blockLength() const noexcept
Root block length.
SbeFields()=default
Initializes a blank instance.
Container & container() noexcept
void zeroPaddingBytes(BlockLength offset) noexcept
If specified, the extra space is padded at the end of each entry and should be set to zeroes by encod...
void setFixedStr(BlockLength offset, StrRef value) noexcept
Sets the field value.
void setOrdinary(BlockLength offset, FieldValue value, SchemaVersion since)
Sets the field value.
void setEnumeration(BlockLength offset, typename Enumeration::Enum value, SchemaVersion since)
Sets the field value.
void setEnumeration(BlockLength offset, typename Enumeration::Enum value) noexcept
Sets the field value.
void setOrdinary(BlockLength offset, FieldValue value) noexcept
Sets the field value.
void setFixedStr(BlockLength offset, StrRef value, SchemaVersion since)
Sets the field value.
std::random_access_iterator_tag iterator_category
Iterator() noexcept
Initializes the instance that refers to nothing.
bool valid() const noexcept
ptrdiff_t difference_type
Iterator(void *entry, EncodedLength size, SchemaVersion version) noexcept
Initializes the instance to the given repeating group.
SbeGroupEntries() noexcept
Initializes a blank instance referencing to nothing.
bool empty() const noexcept
SbeGroupEntries(void *encoded, BlockLength blockLength, Size groupSize, SchemaVersion version) noexcept
Initializes the instance referencing to data.
EntryType Entry
The type of the repeating group entry.
Iterator end() const
Returns the iterator pointing to the entry behind the end of the group.
Size size() const noexcept
EncodedLength encodedLength() const noexcept
SbeGroupEntries(const SbeGroupEntries< OtherEntry, OtherBlockLength, OtherNumInGroup, OtherLength > &other) noexcept
Copy constructor.
bool valid() const noexcept
friend class SbeGroupEntries
Length EncodedLength
The length of the binary data occupied by the group entries.
NumInGroup Size
Number of entries in the collection.
void * encoded() const noexcept
SbeGroupEntry()
Initializes a blank instance.
const void * block() const noexcept
SbeGroupEntry(void *encoded, BlockLength size, SchemaVersion version)
Initializes the instance from the memory block of the encoded message.
SchemaVersion version() const noexcept
bool valid() const noexcept
const void * encoded() const noexcept
BlockLength blockLength() const noexcept
void * encoded() noexcept
BodySizeType BlockLength
Type to present the length of binary data of the repeating group entry.
SbeVariableLengthFieldList< BinarySize > checkVariableLengthFields() const
Checks the variable length fields list consistency.
bool empty() const noexcept
SbeGroupList tail() const noexcept
SbeGroupList checkTail() const
Checks the list consistency.
Group head() const noexcept
SbeVariableLengthFieldList< BinarySize > variableLengthFields() const noexcept
SbeGroupList(void *binary, BinarySize size, SchemaVersion version) noexcept
Initializes the list over the memory block.
const void * tail() const noexcept
SbeGroup() noexcept
Initializes a blank instance referencing to nothing.
Entries::Size Size
Number of entries in the group.
GroupSizeType BinarySize
Length of group data.
EntrySize entrySize() const noexcept
BinarySize binarySize() const noexcept
bool empty() const noexcept
DimensionType::BlockLength EntrySize
Length of group entry data.
SbeGroup(void *data, BinarySize size, SchemaVersion version) noexcept
Initializes an instance referencing to a valid group of a given message.
Entries::Iterator Iterator
The iterator type for group entries.
Entries::Entry Entry
Group entry type.
Entries entries() const noexcept
void * binary() const noexcept
Size size() const noexcept
Iterator end() const noexcept
bool valid() const noexcept
const void * encoded() const noexcept
SbeGroupEntries< EntryType, typename Dimension::BlockLength, typename Dimension::NumInGroup, GroupSizeType > Entries
Binary group blocks.
DimensionType Dimension
Repeating group dimension type.
Iterator begin() const noexcept
void clear() noexcept
Blank the instance.
const void * binary() const noexcept
MessageTemplateId templateId() const noexcept
SchemaId schemaId() const noexcept
Group setupGroup(Callable callable, typename Group::Size length, SchemaVersion since, Owner &owner)
Setups the repeating group with the given number of entries.
const void * blockEnd() noexcept
GroupList groups() const noexcept
Group setupGroup(Callable callable, typename Group::Size length, Owner &owner)
Setups the repeating group with the given number of entries.
StrRef getVariableLengthField(Callable callable, SchemaVersion since, Owner &owner) const noexcept
const void * blockEnd() const noexcept
void setVariableLengthField(Callable callable, StrRef value, SchemaVersion since, Owner &owner)
Sets the value of the variable length field.
SbeMessage(void *data, MessageSize size)
Initializes the instance over the given memory block.
SbeVariableLengthFieldList< MessageSize > VariableLengthFieldList
Binary group list instantiation.
GroupList groups() noexcept
SbeMessage & version(SchemaVersion version) noexcept
Sets the SBE Schema version.
Group getGroup(Callable callable, Owner &owner) const noexcept
Group getGroup(Callable callable, SchemaVersion since, Owner &owner) const noexcept
MessageSize EncodedLength
Length of the message binary data.
const void * block() const noexcept
void setVarDataField(DATA &data, StrRef value, const void *oldMessageTail)
Sets the variable length field value.
SbeMessage(void *data, MessageSize size, NoCheck) noexcept
Initializes the instance over the given memory block.
void initGroup(Group &group, typename Group::EntrySize entrySize) noexcept
Resets the group to the initial state.
void setupGroup(Group &group, typename Group::Size entryCount, const void *messageTail)
Initializes the group header.
VariableLengthFieldList variableLengthFields() const noexcept
SchemaVersion version() const noexcept
SbeGroupList< MessageSize > GroupList
Binary group list instantiation.
SbeMessage(void *data, MessageSize size, SchemaVersion version)
Initializes the instance over the given memory block.
MessageSize bufferSize() const noexcept
bool valid() const noexcept
static constexpr MessageSize getMaxMessageSize() noexcept
Maximal message size.
Group constructGroup(Callable callable, typename Group::Size length, Owner &owner)
Creates a repeating group with the given number of entries, sets all optional fields of the group ent...
Group constructGroup(Callable callable, typename Group::Size length, SchemaVersion since, Owner &owner)
Creates a repeating group with the given number of entries, sets all optional fields of the group ent...
void setVariableLengthFieldToNull(Callable callable, SchemaVersion since, Owner &owner) noexcept
Sets the variable length field to null.
BlockLength blockLength() const noexcept
void setVariableLengthFieldToNull(Callable callable, Owner &owner) noexcept
Resets the variable length field.
void resetGroup(Callable callable, SchemaVersion since, Owner &owner)
Resets the repeating group.
SbeMessage() noexcept
Initializes a blank instance.
void setVariableLengthField(Callable callable, StrRef value, Owner &owner)
Sets the value of the variable length field.
void init(MessageHeader::TemplateId value, MessageHeader::BlockLength minimalBlockLength, MessageHeader::BlockLength blockLength, SchemaId id) noexcept
VariableLengthFieldList variableLengthFields() noexcept
MessageSize BlockLength
Length of the message body representing a block of fixed-length fields.
StrRef getVariableLengthField(Callable callable, const Owner &owner) const noexcept
void resetGroup(Callable callable, Owner &owner) noexcept
Sets the group to the initial state.
MessageSize calculateBinarySize(const void *tail) const noexcept
void constructGroup(Group &group, typename Group::Size entryCount, const void *messageTail)
Initializes the group header, sets all optional fields to null.
Variable-length fields list.
bool empty() const noexcept
SbeVariableLengthFieldList(void *binary, BinarySize size, SchemaVersion version) noexcept
Initializes the list over the given memory block.
SbeVariableLengthFieldList tail() const noexcept
BinaryVariableLengthFieldType & head() const noexcept
SbeVariableLengthFieldList checkTail() const
Checks the variable-length field list consistency.
constexpr UInt16 MaxB3BOEMessageSize
Maximum supported message size.
void checkSchemaId(SchemaId id)
Checks the compatibility with the provided SBE Schema ID.
MessageHeader::Version SchemaVersion
SBE-encoded data version type.
char Char
Character type alias.
UInt16 MessageSize
Message length type.
std::basic_string_view< Char > StrRef
void checkVersion(SchemaVersion version)
Checks the compatibility with the provided SBE Schema version.
MessageHeader::SchemaId SchemaId
void checkSchema(SchemaId id, SchemaVersion version)
Checks the compatibility with the provided SBE Schema version.
MessageHeader::TemplateId MessageTemplateId
Message type (template) identification.