f379578a3 Release 1.7.1 040947ba2 CMake: Add language C to project() c60e87879 Update include catch.hpp to 2.13.8 6599d4828 Update change log ebd2e4f40 Allow setting C++ version to compile with in CMake config bbb2a95d0 Github actions: Only install x64 version of vcpkgs in windows build 546edf929 Avoid narrowing conversion by being explicit 91adaecd6 Fix various issues reported by clang-tidy and disable some checks 68f30a1f7 Add Github actions CI build for Linux/macOS/Windows c13886b78 Update some links bd429c52f Include tools subdirectory *after* testing is enabled 3133dc52f Use std::memcpy instead of just memcpy 7ee29422a Merge pull request #106 from daniel-j-h/fix-byteswap-ub aba0800aa CMake config: clang-tidy target needs writer_tests which need protobuf 74516e8df Modernize CMake config a little bit b4486ca7a Disable some clang-tidy warnings 19f4b75f6 Fix appveyor build 58b1a19a4 Modernize Doxygen config file 046c07d0e Update included catch2 framework to current version v2.13.7 49acea746 fix some doxygen warnings by removing obsolete config entries 0c5426df3 fix cmake dep warning: 'Compatibility with CMake < 2.8.12 will be removed from a future version of CMake' dadf7bd51 Fixes float and double byteswap undefined behavior 85db94025 Merge pull request #105 from ffontaine/master d3a35791d Merge pull request #104 from joto/master 03daae49c CMakeLists.txt: respect BUILD_TESTING 67133e362 Add missing includes 9f85f3a5e Update README.md 010ffcf69 Release 1.7.0 6ad492994 Fixed docs adde4dedf Travis: Add non-Intel architectures 37c3d0e1d Add typedefs and functions to make buffer adaptor work as container 83563acdb Remove delegating constructor because clang-tidy doesn't like it b4afc06db Use #include "" for library-internal includes d1929788e Add missing includes fae5247f0 Update change log 2abb1b5cd Travis: Test with std::string_view, not std::experimental::string_view 697bd812d Use forwarding constructor in fixed_size_buffer_adaptor c8fd2e819 Move fixed_size_buffer_adaptor into buffer_fixed.hpp abb856ecc Remove semicolon where it doesn't belong 6243855bc Change the way the customization for special buffer classes work d6a8ed098 Remove useless post-increment 184046cb0 Remove need for push_back() on custom buffer types 0a974e067 Remove templated buffer adaptor wrappers ed6ba5097 Add buffer implementation based on std::vector<char> cec309c3c Use more descriptive names for buffer test types a7b99da6f Use TEMPLATE_TEST_CASE to test different buffer implementations a0abc493c Use explicit for constructor 72850abc9 Remove broken doxygen link 1e347c620 Add more convenient fixed_size_buffer_adaptor constructor fb575e0ea Make members private 50e953b71 Make older compilers happy c850ef150 Extend tests of static buffer use f6d8394c0 Rename fixed_size_buffer to fixed_size_buffer_adaptor 3b18162e3 Make the buffer backend used by the pbf writer configurable. 6fd19c58d "Modernize" travis config 981aba084 Use explicit cast to avoid undefined behaviour 550974d5d Travis: Do not test GCC 4.7 any more 866e024fc Revert "Workaround in catch code so it compiles with older compilers" 65dfad056 Disable a clang-tidy test triggered by Catch. 02bf73df5 Workaround in catch code so it compiles with older compilers f98792a15 Travis: Do not update homebrew for faster builds 2d87da7ec Switch tests to Catch2 5dc45ac3b Avoid signed/unsigned comparison 3a93f19ba Add missing includes b49c077ac Disable clang-tidy for files where we don't have a compile command 34396fc7d Travis: Fix gcc8 build c3060101c Handle clang-tidy warnings 64ef96ff0 Revert "Initialize test messages" a0828d538 Travis: Also build with GCC 8 171c5c446 Update travis xcode versions f5a223aa7 Use "auto*" instead of just "auto" for pointer types e4fa23616 Initialize test messages e3a59454a Simpler code and avoid shadowing of external function 830f049b4 Use STL algorithms insted of raw loops faa7e6e8a Disable config settings not used in newer Doxygen versions 3b2e11438 Remove unnecessary enum name 7487f8109 Release 1.6.8 6dcaf8fde Travis config: Use "official" way to pull homebrew package. c61eb29c3 Revert "Disable warnings from clang-tidy about a missing file." 28d05a0a8 Disable warnings from clang-tidy about a missing file. 329920a3c Pesky aliases of clang-tidy warnings strike again. 79fd87922 User plain assert() instead of our own so compare() can be noexcept. 473e6ec13 Update change log. 393e279b7 Make pbf_writer destructor noexcept. 48a38b3f2 Disable clang-tidy misc-non-private-member-variables-in-classes. e9c148c8a Use no-argument version of main(). 29ba04123 Disable clang tidy checks for C arrays. 2fcfb56e2 More places to use std::array instead of a C style array. 7321761a3 Disable a clang-tidy warning. 4d9d8fff4 Make data_view::compare() noexcept. 3325364cf User uppercase integer literal suffix. df0a23c5e Use std::array instead of C arrays in some places. 8247ed76b Make clang-tidy include order check happy. f1b504e16 Update travis config to user newer compilers and operating systems. ccf692d47 Disable some clang-tidy warnings. 095abd259 CMake config: Also look for newer clang-tidy versions. 2c1f6f9c8 Use uppercase integer literal suffixes. fadd024d4 Release 1.6.7 8c6acbff7 Fix signed-unsigned comparison. b36774ccb Release 1.6.6 5a92b744f Remove useless asserts, simplify condition. 06bafb56c Fix several possible UBs. b7b290b1a Release 1.6.5 51753d514 Merge pull request #95 from tomhughes/subscript b90faaf03 Avoid out of bounds array subscript 7d418492e Merge pull request #94 from nigels-com/proto2 015f9cc5e Specify proto2 syntax to appease protoc 23d48fd2a Use universal initialization syntax in constructors. 0f610fad5 Update travis config: Use xenial for most builds. d71da0b04 Update appveyor config: Simpler builds, current MSVC, 32bit build 3ef46ba78 Release 1.6.4 3a1ef0138 Tighten some tests. 18eebb8c3 Remove unused code from tests. 29ef3e4e7 More casts to remove undefined behaviour. 6108e6480 No more bitwise operations on signed integers in zigzag encoder/decoder. 6e0d34985 Remove bitwise operations on signed integers in varint decoder. 4af65f262 Update change log. 2f82182fe Add some tips to test/create_pbf_test_data.sh. c55f4ed55 Fix some doxygen warnings. afa362a03 Add static_asserts to check movability of some classes. efeb45e0c Disable readability-implicit-bool-conversion clang-tidy warning. 78febda5b Explicit conversion and tests for new pbf_reader::data() function. 0d5492c9c Revert "Explicit conversion and tests for new pbf_reader::data() function." 43cf8fa5a Fix travis config. bd2ae4682 Explicit conversion and tests for new pbf_reader::data() function. 28cd406bd Update travis with newer compiler versions. 0555e6a1f Add function to get the not yet read data from a pbf_reader. bf4284bee Disable docker builds on travis. They are being phased out by travis. 5ffe45b71 New add_packed_fixed template function. e54cd858d Add helper function that computes the length a varint would have. 72d7e143a More consistent implementation of operators. 3a41880c2 Do not download protobuf library, it isn't found by cmake anyway. 3c662ce3c Remove comment that doesn't apply (any more). 45da6dd4d Update zigzag tests. 4ad573dbf Extra cast so we do the xor with unsigned ints. 509aec5ab Update appveyor build to current Visual Studio compiler. 67b24e1a3 Remove unnecessary workaround in Appveyor config. c559af682 Remove xcode6.4 build soon to be removed from travis. 0662dcecc Release 1.6.3 da5bfc019 Move byteswap_inplace functions from detail into protozero namespace. a44efc34e Travis: Ignore install problems on OSX. 5775b2b23 Travis update to newer OSX image. 032aa037c Special case the distance between default initialized iterators. 0ca02161e Make dereferencing operator of fixed_iterator noexcept. a0095f603 Test code must call functions that it wants to test. 6791b0bc3 Add unit tests. 191eb4004 Add some paranoia asserts. 99ca512f5 Use TEST_CASEs instead of SECTIOs in some tests. 040e2bc14 Add some asserts and tests. git-subtree-dir: third_party/protozero git-subtree-split: f379578a3f7c8162aac0ac31c2696de09a5b5f93
1055 lines
36 KiB
C++
1055 lines
36 KiB
C++
#ifndef PROTOZERO_BASIC_PBF_WRITER_HPP
|
|
#define PROTOZERO_BASIC_PBF_WRITER_HPP
|
|
|
|
/*****************************************************************************
|
|
|
|
protozero - Minimalistic protocol buffer decoder and encoder in C++.
|
|
|
|
This file is from https://github.com/mapbox/protozero where you can find more
|
|
documentation.
|
|
|
|
*****************************************************************************/
|
|
|
|
/**
|
|
* @file basic_pbf_writer.hpp
|
|
*
|
|
* @brief Contains the basic_pbf_writer template class.
|
|
*/
|
|
|
|
#include "buffer_tmpl.hpp"
|
|
#include "config.hpp"
|
|
#include "data_view.hpp"
|
|
#include "types.hpp"
|
|
#include "varint.hpp"
|
|
|
|
#if PROTOZERO_BYTE_ORDER != PROTOZERO_LITTLE_ENDIAN
|
|
# include <protozero/byteswap.hpp>
|
|
#endif
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <cstring>
|
|
#include <initializer_list>
|
|
#include <iterator>
|
|
#include <limits>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
namespace protozero {
|
|
|
|
namespace detail {
|
|
|
|
template <typename B, typename T> class packed_field_varint;
|
|
template <typename B, typename T> class packed_field_svarint;
|
|
template <typename B, typename T> class packed_field_fixed;
|
|
|
|
} // end namespace detail
|
|
|
|
/**
|
|
* The basic_pbf_writer is used to write PBF formatted messages into a buffer.
|
|
*
|
|
* This uses TBuffer as the type for the underlaying buffer. In typical uses
|
|
* this is std::string, but you can use a different type that must support
|
|
* the right interface. Please see the documentation for details.
|
|
*
|
|
* Almost all methods in this class can throw an std::bad_alloc exception if
|
|
* the underlying buffer class wants to resize.
|
|
*/
|
|
template <typename TBuffer>
|
|
class basic_pbf_writer {
|
|
|
|
// A pointer to a buffer holding the data already written to the PBF
|
|
// message. For default constructed writers or writers that have been
|
|
// rolled back, this is a nullptr.
|
|
TBuffer* m_data = nullptr;
|
|
|
|
// A pointer to a parent writer object if this is a submessage. If this
|
|
// is a top-level writer, it is a nullptr.
|
|
basic_pbf_writer* m_parent_writer = nullptr;
|
|
|
|
// This is usually 0. If there is an open submessage, this is set in the
|
|
// parent to the rollback position, ie. the last position before the
|
|
// submessage was started. This is the position where the header of the
|
|
// submessage starts.
|
|
std::size_t m_rollback_pos = 0;
|
|
|
|
// This is usually 0. If there is an open submessage, this is set in the
|
|
// parent to the position where the data of the submessage is written to.
|
|
std::size_t m_pos = 0;
|
|
|
|
void add_varint(uint64_t value) {
|
|
protozero_assert(m_pos == 0 && "you can't add fields to a parent basic_pbf_writer if there is an existing basic_pbf_writer for a submessage");
|
|
protozero_assert(m_data);
|
|
add_varint_to_buffer(m_data, value);
|
|
}
|
|
|
|
void add_field(pbf_tag_type tag, pbf_wire_type type) {
|
|
protozero_assert(((tag > 0 && tag < 19000) || (tag > 19999 && tag <= ((1U << 29U) - 1))) && "tag out of range");
|
|
const uint32_t b = (tag << 3U) | uint32_t(type);
|
|
add_varint(b);
|
|
}
|
|
|
|
void add_tagged_varint(pbf_tag_type tag, uint64_t value) {
|
|
add_field(tag, pbf_wire_type::varint);
|
|
add_varint(value);
|
|
}
|
|
|
|
template <typename T>
|
|
void add_fixed(T value) {
|
|
protozero_assert(m_pos == 0 && "you can't add fields to a parent basic_pbf_writer if there is an existing basic_pbf_writer for a submessage");
|
|
protozero_assert(m_data);
|
|
#if PROTOZERO_BYTE_ORDER != PROTOZERO_LITTLE_ENDIAN
|
|
byteswap_inplace(&value);
|
|
#endif
|
|
buffer_customization<TBuffer>::append(m_data, reinterpret_cast<const char*>(&value), sizeof(T));
|
|
}
|
|
|
|
template <typename T, typename It>
|
|
void add_packed_fixed(pbf_tag_type tag, It first, It last, std::input_iterator_tag /*unused*/) {
|
|
if (first == last) {
|
|
return;
|
|
}
|
|
|
|
basic_pbf_writer sw{*this, tag};
|
|
|
|
while (first != last) {
|
|
sw.add_fixed<T>(*first++);
|
|
}
|
|
}
|
|
|
|
template <typename T, typename It>
|
|
void add_packed_fixed(pbf_tag_type tag, It first, It last, std::forward_iterator_tag /*unused*/) {
|
|
if (first == last) {
|
|
return;
|
|
}
|
|
|
|
const auto length = std::distance(first, last);
|
|
add_length_varint(tag, sizeof(T) * pbf_length_type(length));
|
|
reserve(sizeof(T) * std::size_t(length));
|
|
|
|
while (first != last) {
|
|
add_fixed<T>(*first++);
|
|
}
|
|
}
|
|
|
|
template <typename It>
|
|
void add_packed_varint(pbf_tag_type tag, It first, It last) {
|
|
if (first == last) {
|
|
return;
|
|
}
|
|
|
|
basic_pbf_writer sw{*this, tag};
|
|
|
|
while (first != last) {
|
|
sw.add_varint(uint64_t(*first++));
|
|
}
|
|
}
|
|
|
|
template <typename It>
|
|
void add_packed_svarint(pbf_tag_type tag, It first, It last) {
|
|
if (first == last) {
|
|
return;
|
|
}
|
|
|
|
basic_pbf_writer sw{*this, tag};
|
|
|
|
while (first != last) {
|
|
sw.add_varint(encode_zigzag64(*first++));
|
|
}
|
|
}
|
|
|
|
// The number of bytes to reserve for the varint holding the length of
|
|
// a length-delimited field. The length has to fit into pbf_length_type,
|
|
// and a varint needs 8 bit for every 7 bit.
|
|
enum : int {
|
|
reserve_bytes = sizeof(pbf_length_type) * 8 / 7 + 1
|
|
};
|
|
|
|
// If m_rollpack_pos is set to this special value, it means that when
|
|
// the submessage is closed, nothing needs to be done, because the length
|
|
// of the submessage has already been written correctly.
|
|
enum : std::size_t {
|
|
size_is_known = std::numeric_limits<std::size_t>::max()
|
|
};
|
|
|
|
void open_submessage(pbf_tag_type tag, std::size_t size) {
|
|
protozero_assert(m_pos == 0);
|
|
protozero_assert(m_data);
|
|
if (size == 0) {
|
|
m_rollback_pos = buffer_customization<TBuffer>::size(m_data);
|
|
add_field(tag, pbf_wire_type::length_delimited);
|
|
buffer_customization<TBuffer>::append_zeros(m_data, std::size_t(reserve_bytes));
|
|
} else {
|
|
m_rollback_pos = size_is_known;
|
|
add_length_varint(tag, pbf_length_type(size));
|
|
reserve(size);
|
|
}
|
|
m_pos = buffer_customization<TBuffer>::size(m_data);
|
|
}
|
|
|
|
void rollback_submessage() {
|
|
protozero_assert(m_pos != 0);
|
|
protozero_assert(m_rollback_pos != size_is_known);
|
|
protozero_assert(m_data);
|
|
buffer_customization<TBuffer>::resize(m_data, m_rollback_pos);
|
|
m_pos = 0;
|
|
}
|
|
|
|
void commit_submessage() {
|
|
protozero_assert(m_pos != 0);
|
|
protozero_assert(m_rollback_pos != size_is_known);
|
|
protozero_assert(m_data);
|
|
const auto length = pbf_length_type(buffer_customization<TBuffer>::size(m_data) - m_pos);
|
|
|
|
protozero_assert(buffer_customization<TBuffer>::size(m_data) >= m_pos - reserve_bytes);
|
|
const auto n = add_varint_to_buffer(buffer_customization<TBuffer>::at_pos(m_data, m_pos - reserve_bytes), length);
|
|
|
|
buffer_customization<TBuffer>::erase_range(m_data, m_pos - reserve_bytes + n, m_pos);
|
|
m_pos = 0;
|
|
}
|
|
|
|
void close_submessage() {
|
|
protozero_assert(m_data);
|
|
if (m_pos == 0 || m_rollback_pos == size_is_known) {
|
|
return;
|
|
}
|
|
if (buffer_customization<TBuffer>::size(m_data) - m_pos == 0) {
|
|
rollback_submessage();
|
|
} else {
|
|
commit_submessage();
|
|
}
|
|
}
|
|
|
|
void add_length_varint(pbf_tag_type tag, pbf_length_type length) {
|
|
add_field(tag, pbf_wire_type::length_delimited);
|
|
add_varint(length);
|
|
}
|
|
|
|
public:
|
|
|
|
/**
|
|
* Create a writer using the specified buffer as a data store. The
|
|
* basic_pbf_writer stores a pointer to that buffer and adds all data to
|
|
* it. The buffer doesn't have to be empty. The basic_pbf_writer will just
|
|
* append data.
|
|
*/
|
|
explicit basic_pbf_writer(TBuffer& buffer) noexcept :
|
|
m_data{&buffer} {
|
|
}
|
|
|
|
/**
|
|
* Create a writer without a data store. In this form the writer can not
|
|
* be used!
|
|
*/
|
|
basic_pbf_writer() noexcept = default;
|
|
|
|
/**
|
|
* Construct a basic_pbf_writer for a submessage from the basic_pbf_writer
|
|
* of the parent message.
|
|
*
|
|
* @param parent_writer The basic_pbf_writer
|
|
* @param tag Tag (field number) of the field that will be written
|
|
* @param size Optional size of the submessage in bytes (use 0 for unknown).
|
|
* Setting this allows some optimizations but is only possible in
|
|
* a few very specific cases.
|
|
*/
|
|
basic_pbf_writer(basic_pbf_writer& parent_writer, pbf_tag_type tag, std::size_t size = 0) :
|
|
m_data{parent_writer.m_data},
|
|
m_parent_writer{&parent_writer} {
|
|
m_parent_writer->open_submessage(tag, size);
|
|
}
|
|
|
|
/// A basic_pbf_writer object can not be copied
|
|
basic_pbf_writer(const basic_pbf_writer&) = delete;
|
|
|
|
/// A basic_pbf_writer object can not be copied
|
|
basic_pbf_writer& operator=(const basic_pbf_writer&) = delete;
|
|
|
|
/**
|
|
* A basic_pbf_writer object can be moved. After this the other
|
|
* basic_pbf_writer will be invalid.
|
|
*/
|
|
basic_pbf_writer(basic_pbf_writer&& other) noexcept :
|
|
m_data{other.m_data},
|
|
m_parent_writer{other.m_parent_writer},
|
|
m_rollback_pos{other.m_rollback_pos},
|
|
m_pos{other.m_pos} {
|
|
other.m_data = nullptr;
|
|
other.m_parent_writer = nullptr;
|
|
other.m_rollback_pos = 0;
|
|
other.m_pos = 0;
|
|
}
|
|
|
|
/**
|
|
* A basic_pbf_writer object can be moved. After this the other
|
|
* basic_pbf_writer will be invalid.
|
|
*/
|
|
basic_pbf_writer& operator=(basic_pbf_writer&& other) noexcept {
|
|
m_data = other.m_data;
|
|
m_parent_writer = other.m_parent_writer;
|
|
m_rollback_pos = other.m_rollback_pos;
|
|
m_pos = other.m_pos;
|
|
other.m_data = nullptr;
|
|
other.m_parent_writer = nullptr;
|
|
other.m_rollback_pos = 0;
|
|
other.m_pos = 0;
|
|
return *this;
|
|
}
|
|
|
|
~basic_pbf_writer() noexcept {
|
|
try {
|
|
if (m_parent_writer != nullptr) {
|
|
m_parent_writer->close_submessage();
|
|
}
|
|
} catch (...) {
|
|
// This try/catch is used to make the destructor formally noexcept.
|
|
// close_submessage() is not noexcept, but will not throw the way
|
|
// it is called here, so we are good. But to be paranoid, call...
|
|
std::terminate();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check if this writer is valid. A writer is invalid if it was default
|
|
* constructed, moved from, or if commit() has been called on it.
|
|
* Otherwise it is valid.
|
|
*/
|
|
bool valid() const noexcept {
|
|
return m_data != nullptr;
|
|
}
|
|
|
|
/**
|
|
* Swap the contents of this object with the other.
|
|
*
|
|
* @param other Other object to swap data with.
|
|
*/
|
|
void swap(basic_pbf_writer& other) noexcept {
|
|
using std::swap;
|
|
swap(m_data, other.m_data);
|
|
swap(m_parent_writer, other.m_parent_writer);
|
|
swap(m_rollback_pos, other.m_rollback_pos);
|
|
swap(m_pos, other.m_pos);
|
|
}
|
|
|
|
/**
|
|
* Reserve size bytes in the underlying message store in addition to
|
|
* whatever the message store already holds. So unlike
|
|
* the `std::string::reserve()` method this is not an absolute size,
|
|
* but additional memory that should be reserved.
|
|
*
|
|
* @param size Number of bytes to reserve in underlying message store.
|
|
*/
|
|
void reserve(std::size_t size) {
|
|
protozero_assert(m_data);
|
|
buffer_customization<TBuffer>::reserve_additional(m_data, size);
|
|
}
|
|
|
|
/**
|
|
* Commit this submessage. This does the same as when the basic_pbf_writer
|
|
* goes out of scope and is destructed.
|
|
*
|
|
* @pre Must be a basic_pbf_writer of a submessage, ie one opened with the
|
|
* basic_pbf_writer constructor taking a parent message.
|
|
* @post The basic_pbf_writer is invalid and can't be used any more.
|
|
*/
|
|
void commit() {
|
|
protozero_assert(m_parent_writer && "you can't call commit() on a basic_pbf_writer without a parent");
|
|
protozero_assert(m_pos == 0 && "you can't call commit() on a basic_pbf_writer that has an open nested submessage");
|
|
m_parent_writer->close_submessage();
|
|
m_parent_writer = nullptr;
|
|
m_data = nullptr;
|
|
}
|
|
|
|
/**
|
|
* Cancel writing of this submessage. The complete submessage will be
|
|
* removed as if it was never created and no fields were added.
|
|
*
|
|
* @pre Must be a basic_pbf_writer of a submessage, ie one opened with the
|
|
* basic_pbf_writer constructor taking a parent message.
|
|
* @post The basic_pbf_writer is invalid and can't be used any more.
|
|
*/
|
|
void rollback() {
|
|
protozero_assert(m_parent_writer && "you can't call rollback() on a basic_pbf_writer without a parent");
|
|
protozero_assert(m_pos == 0 && "you can't call rollback() on a basic_pbf_writer that has an open nested submessage");
|
|
m_parent_writer->rollback_submessage();
|
|
m_parent_writer = nullptr;
|
|
m_data = nullptr;
|
|
}
|
|
|
|
///@{
|
|
/**
|
|
* @name Scalar field writer functions
|
|
*/
|
|
|
|
/**
|
|
* Add "bool" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_bool(pbf_tag_type tag, bool value) {
|
|
add_field(tag, pbf_wire_type::varint);
|
|
protozero_assert(m_pos == 0 && "you can't add fields to a parent basic_pbf_writer if there is an existing basic_pbf_writer for a submessage");
|
|
protozero_assert(m_data);
|
|
m_data->push_back(char(value));
|
|
}
|
|
|
|
/**
|
|
* Add "enum" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_enum(pbf_tag_type tag, int32_t value) {
|
|
add_tagged_varint(tag, uint64_t(value));
|
|
}
|
|
|
|
/**
|
|
* Add "int32" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_int32(pbf_tag_type tag, int32_t value) {
|
|
add_tagged_varint(tag, uint64_t(value));
|
|
}
|
|
|
|
/**
|
|
* Add "sint32" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_sint32(pbf_tag_type tag, int32_t value) {
|
|
add_tagged_varint(tag, encode_zigzag32(value));
|
|
}
|
|
|
|
/**
|
|
* Add "uint32" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_uint32(pbf_tag_type tag, uint32_t value) {
|
|
add_tagged_varint(tag, value);
|
|
}
|
|
|
|
/**
|
|
* Add "int64" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_int64(pbf_tag_type tag, int64_t value) {
|
|
add_tagged_varint(tag, uint64_t(value));
|
|
}
|
|
|
|
/**
|
|
* Add "sint64" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_sint64(pbf_tag_type tag, int64_t value) {
|
|
add_tagged_varint(tag, encode_zigzag64(value));
|
|
}
|
|
|
|
/**
|
|
* Add "uint64" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_uint64(pbf_tag_type tag, uint64_t value) {
|
|
add_tagged_varint(tag, value);
|
|
}
|
|
|
|
/**
|
|
* Add "fixed32" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_fixed32(pbf_tag_type tag, uint32_t value) {
|
|
add_field(tag, pbf_wire_type::fixed32);
|
|
add_fixed<uint32_t>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "sfixed32" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_sfixed32(pbf_tag_type tag, int32_t value) {
|
|
add_field(tag, pbf_wire_type::fixed32);
|
|
add_fixed<int32_t>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "fixed64" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_fixed64(pbf_tag_type tag, uint64_t value) {
|
|
add_field(tag, pbf_wire_type::fixed64);
|
|
add_fixed<uint64_t>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "sfixed64" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_sfixed64(pbf_tag_type tag, int64_t value) {
|
|
add_field(tag, pbf_wire_type::fixed64);
|
|
add_fixed<int64_t>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "float" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_float(pbf_tag_type tag, float value) {
|
|
add_field(tag, pbf_wire_type::fixed32);
|
|
add_fixed<float>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "double" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_double(pbf_tag_type tag, double value) {
|
|
add_field(tag, pbf_wire_type::fixed64);
|
|
add_fixed<double>(value);
|
|
}
|
|
|
|
/**
|
|
* Add "bytes" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Pointer to value to be written
|
|
* @param size Number of bytes to be written
|
|
*/
|
|
void add_bytes(pbf_tag_type tag, const char* value, std::size_t size) {
|
|
protozero_assert(m_pos == 0 && "you can't add fields to a parent basic_pbf_writer if there is an existing basic_pbf_writer for a submessage");
|
|
protozero_assert(m_data);
|
|
protozero_assert(size <= std::numeric_limits<pbf_length_type>::max());
|
|
add_length_varint(tag, pbf_length_type(size));
|
|
buffer_customization<TBuffer>::append(m_data, value, size);
|
|
}
|
|
|
|
/**
|
|
* Add "bytes" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_bytes(pbf_tag_type tag, const data_view& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
/**
|
|
* Add "bytes" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_bytes(pbf_tag_type tag, const std::string& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
/**
|
|
* Add "bytes" field to data. Bytes from the value are written until
|
|
* a null byte is encountered. The null byte is not added.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Pointer to zero-delimited value to be written
|
|
*/
|
|
void add_bytes(pbf_tag_type tag, const char* value) {
|
|
add_bytes(tag, value, std::strlen(value));
|
|
}
|
|
|
|
/**
|
|
* Add "bytes" field to data using vectored input. All the data in the
|
|
* 2nd and further arguments is "concatenated" with only a single copy
|
|
* into the final buffer.
|
|
*
|
|
* This will work with objects of any type supporting the data() and
|
|
* size() methods like std::string or protozero::data_view.
|
|
*
|
|
* Example:
|
|
* @code
|
|
* std::string data1 = "abc";
|
|
* std::string data2 = "xyz";
|
|
* writer.add_bytes_vectored(1, data1, data2);
|
|
* @endcode
|
|
*
|
|
* @tparam Ts List of types supporting data() and size() methods.
|
|
* @param tag Tag (field number) of the field
|
|
* @param values List of objects of types Ts with data to be appended.
|
|
*/
|
|
template <typename... Ts>
|
|
void add_bytes_vectored(pbf_tag_type tag, Ts&&... values) {
|
|
protozero_assert(m_pos == 0 && "you can't add fields to a parent basic_pbf_writer if there is an existing basic_pbf_writer for a submessage");
|
|
protozero_assert(m_data);
|
|
size_t sum_size = 0;
|
|
(void)std::initializer_list<size_t>{sum_size += values.size()...};
|
|
protozero_assert(sum_size <= std::numeric_limits<pbf_length_type>::max());
|
|
add_length_varint(tag, pbf_length_type(sum_size));
|
|
buffer_customization<TBuffer>::reserve_additional(m_data, sum_size);
|
|
(void)std::initializer_list<int>{(buffer_customization<TBuffer>::append(m_data, values.data(), values.size()), 0)...};
|
|
}
|
|
|
|
/**
|
|
* Add "string" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Pointer to value to be written
|
|
* @param size Number of bytes to be written
|
|
*/
|
|
void add_string(pbf_tag_type tag, const char* value, std::size_t size) {
|
|
add_bytes(tag, value, size);
|
|
}
|
|
|
|
/**
|
|
* Add "string" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_string(pbf_tag_type tag, const data_view& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
/**
|
|
* Add "string" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written
|
|
*/
|
|
void add_string(pbf_tag_type tag, const std::string& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
/**
|
|
* Add "string" field to data. Bytes from the value are written until
|
|
* a null byte is encountered. The null byte is not added.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Pointer to value to be written
|
|
*/
|
|
void add_string(pbf_tag_type tag, const char* value) {
|
|
add_bytes(tag, value, std::strlen(value));
|
|
}
|
|
|
|
/**
|
|
* Add "message" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Pointer to message to be written
|
|
* @param size Length of the message
|
|
*/
|
|
void add_message(pbf_tag_type tag, const char* value, std::size_t size) {
|
|
add_bytes(tag, value, size);
|
|
}
|
|
|
|
/**
|
|
* Add "message" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written. The value must be a complete message.
|
|
*/
|
|
void add_message(pbf_tag_type tag, const data_view& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
/**
|
|
* Add "message" field to data.
|
|
*
|
|
* @param tag Tag (field number) of the field
|
|
* @param value Value to be written. The value must be a complete message.
|
|
*/
|
|
void add_message(pbf_tag_type tag, const std::string& value) {
|
|
add_bytes(tag, value.data(), value.size());
|
|
}
|
|
|
|
///@}
|
|
|
|
///@{
|
|
/**
|
|
* @name Repeated packed field writer functions
|
|
*/
|
|
|
|
/**
|
|
* Add "repeated packed bool" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to bool.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_bool(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed enum" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_enum(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed int32" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_int32(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed sint32" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_sint32(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_svarint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed uint32" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to uint32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_uint32(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed int64" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int64_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_int64(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed sint64" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int64_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_sint64(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_svarint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed uint64" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to uint64_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_uint64(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_varint(tag, first, last);
|
|
}
|
|
|
|
/**
|
|
* Add a "repeated packed" fixed-size field to data. The following
|
|
* fixed-size fields are available:
|
|
*
|
|
* uint32_t -> repeated packed fixed32
|
|
* int32_t -> repeated packed sfixed32
|
|
* uint64_t -> repeated packed fixed64
|
|
* int64_t -> repeated packed sfixed64
|
|
* double -> repeated packed double
|
|
* float -> repeated packed float
|
|
*
|
|
* @tparam ValueType One of the following types: (u)int32/64_t, double, float.
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename ValueType, typename InputIterator>
|
|
void add_packed_fixed(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
static_assert(std::is_same<ValueType, uint32_t>::value ||
|
|
std::is_same<ValueType, int32_t>::value ||
|
|
std::is_same<ValueType, int64_t>::value ||
|
|
std::is_same<ValueType, uint64_t>::value ||
|
|
std::is_same<ValueType, double>::value ||
|
|
std::is_same<ValueType, float>::value, "Only some types are allowed");
|
|
add_packed_fixed<ValueType, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed fixed32" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to uint32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_fixed32(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<uint32_t, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed sfixed32" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int32_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_sfixed32(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<int32_t, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed fixed64" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to uint64_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_fixed64(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<uint64_t, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed sfixed64" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to int64_t.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_sfixed64(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<int64_t, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed float" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to float.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_float(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<float, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
/**
|
|
* Add "repeated packed double" field to data.
|
|
*
|
|
* @tparam InputIterator A type satisfying the InputIterator concept.
|
|
* Dereferencing the iterator must yield a type assignable to double.
|
|
* @param tag Tag (field number) of the field
|
|
* @param first Iterator pointing to the beginning of the data
|
|
* @param last Iterator pointing one past the end of data
|
|
*/
|
|
template <typename InputIterator>
|
|
void add_packed_double(pbf_tag_type tag, InputIterator first, InputIterator last) {
|
|
add_packed_fixed<double, InputIterator>(tag, first, last,
|
|
typename std::iterator_traits<InputIterator>::iterator_category{});
|
|
}
|
|
|
|
///@}
|
|
|
|
template <typename B, typename T> friend class detail::packed_field_varint;
|
|
template <typename B, typename T> friend class detail::packed_field_svarint;
|
|
template <typename B, typename T> friend class detail::packed_field_fixed;
|
|
|
|
}; // class basic_pbf_writer
|
|
|
|
/**
|
|
* Swap two basic_pbf_writer objects.
|
|
*
|
|
* @param lhs First object.
|
|
* @param rhs Second object.
|
|
*/
|
|
template <typename TBuffer>
|
|
inline void swap(basic_pbf_writer<TBuffer>& lhs, basic_pbf_writer<TBuffer>& rhs) noexcept {
|
|
lhs.swap(rhs);
|
|
}
|
|
|
|
namespace detail {
|
|
|
|
template <typename TBuffer>
|
|
class packed_field {
|
|
|
|
basic_pbf_writer<TBuffer> m_writer{};
|
|
|
|
public:
|
|
|
|
packed_field(const packed_field&) = delete;
|
|
packed_field& operator=(const packed_field&) = delete;
|
|
|
|
packed_field(packed_field&&) noexcept = default;
|
|
packed_field& operator=(packed_field&&) noexcept = default;
|
|
|
|
packed_field() = default;
|
|
|
|
packed_field(basic_pbf_writer<TBuffer>& parent_writer, pbf_tag_type tag) :
|
|
m_writer{parent_writer, tag} {
|
|
}
|
|
|
|
packed_field(basic_pbf_writer<TBuffer>& parent_writer, pbf_tag_type tag, std::size_t size) :
|
|
m_writer{parent_writer, tag, size} {
|
|
}
|
|
|
|
~packed_field() noexcept = default;
|
|
|
|
bool valid() const noexcept {
|
|
return m_writer.valid();
|
|
}
|
|
|
|
void commit() {
|
|
m_writer.commit();
|
|
}
|
|
|
|
void rollback() {
|
|
m_writer.rollback();
|
|
}
|
|
|
|
basic_pbf_writer<TBuffer>& writer() noexcept {
|
|
return m_writer;
|
|
}
|
|
|
|
}; // class packed_field
|
|
|
|
template <typename TBuffer, typename T>
|
|
class packed_field_fixed : public packed_field<TBuffer> {
|
|
|
|
public:
|
|
|
|
packed_field_fixed() :
|
|
packed_field<TBuffer>{} {
|
|
}
|
|
|
|
template <typename P>
|
|
packed_field_fixed(basic_pbf_writer<TBuffer>& parent_writer, P tag) :
|
|
packed_field<TBuffer>{parent_writer, static_cast<pbf_tag_type>(tag)} {
|
|
}
|
|
|
|
template <typename P>
|
|
packed_field_fixed(basic_pbf_writer<TBuffer>& parent_writer, P tag, std::size_t size) :
|
|
packed_field<TBuffer>{parent_writer, static_cast<pbf_tag_type>(tag), size * sizeof(T)} {
|
|
}
|
|
|
|
void add_element(T value) {
|
|
this->writer().template add_fixed<T>(value);
|
|
}
|
|
|
|
}; // class packed_field_fixed
|
|
|
|
template <typename TBuffer, typename T>
|
|
class packed_field_varint : public packed_field<TBuffer> {
|
|
|
|
public:
|
|
|
|
packed_field_varint() :
|
|
packed_field<TBuffer>{} {
|
|
}
|
|
|
|
template <typename P>
|
|
packed_field_varint(basic_pbf_writer<TBuffer>& parent_writer, P tag) :
|
|
packed_field<TBuffer>{parent_writer, static_cast<pbf_tag_type>(tag)} {
|
|
}
|
|
|
|
void add_element(T value) {
|
|
this->writer().add_varint(uint64_t(value));
|
|
}
|
|
|
|
}; // class packed_field_varint
|
|
|
|
template <typename TBuffer, typename T>
|
|
class packed_field_svarint : public packed_field<TBuffer> {
|
|
|
|
public:
|
|
|
|
packed_field_svarint() :
|
|
packed_field<TBuffer>{} {
|
|
}
|
|
|
|
template <typename P>
|
|
packed_field_svarint(basic_pbf_writer<TBuffer>& parent_writer, P tag) :
|
|
packed_field<TBuffer>{parent_writer, static_cast<pbf_tag_type>(tag)} {
|
|
}
|
|
|
|
void add_element(T value) {
|
|
this->writer().add_varint(encode_zigzag64(value));
|
|
}
|
|
|
|
}; // class packed_field_svarint
|
|
|
|
} // end namespace detail
|
|
|
|
} // end namespace protozero
|
|
|
|
#endif // PROTOZERO_BASIC_PBF_WRITER_HPP
|