1 #ifndef PROTOZERO_PBF_WRITER_HPP
2 #define PROTOZERO_PBF_WRITER_HPP
30 #if __BYTE_ORDER != __LITTLE_ENDIAN
31 # include <protozero/byteswap.hpp>
35 #ifndef protozero_assert
36 # define protozero_assert(x) assert(x)
53 inline void add_varint(uint64_t value) {
54 protozero_assert(m_pos == 0 &&
"you can't add fields to a parent pbf_writer if there is an existing pbf_writer for a submessage");
60 protozero_assert(((tag > 0 && tag < 19000) || (tag > 19999 && tag <= ((1 << 29) - 1))) &&
"tag out of range");
61 uint32_t b = (tag << 3) | uint32_t(type);
65 inline void add_tagged_varint(
pbf_tag_type tag, uint64_t value) {
66 add_field(tag, pbf_wire_type::varint);
71 inline void add_fixed(T value) {
72 protozero_assert(m_pos == 0 &&
"you can't add fields to a parent pbf_writer if there is an existing pbf_writer for a submessage");
74 #if __BYTE_ORDER == __LITTLE_ENDIAN
75 m_data->append(reinterpret_cast<const char*>(&value),
sizeof(T));
77 auto size = m_data->size();
78 m_data->resize(size +
sizeof(T));
79 byteswap<sizeof(T)>(
reinterpret_cast<const char*
>(&value), const_cast<char*>(m_data->data() + size));
83 template <
typename T,
typename It>
84 inline void add_packed_fixed(
pbf_tag_type tag, It first, It last, std::input_iterator_tag) {
91 while (first != last) {
92 sw.add_fixed<T>(*first++);
96 template <
typename T,
typename It>
97 inline void add_packed_fixed(
pbf_tag_type tag, It first, It last, std::forward_iterator_tag) {
102 add_length_varint(tag,
sizeof(T) *
pbf_length_type(std::distance(first, last)));
104 while (first != last) {
105 add_fixed<T>(*first++);
109 template <
typename It>
110 inline void add_packed_varint(
pbf_tag_type tag, It first, It last) {
117 while (first != last) {
118 sw.add_varint(uint64_t(*first++));
122 template <
typename It>
123 inline void add_packed_svarint(
pbf_tag_type tag, It first, It last) {
130 while (first != last) {
143 add_field(tag, pbf_wire_type::length_delimited);
144 m_data->append(
size_t(reserve_bytes),
'\0');
145 m_pos = m_data->size();
148 inline void close_submessage() {
154 auto n =
write_varint(m_data->begin() + long(m_pos) - reserve_bytes, length);
156 m_data->erase(m_data->begin() + long(m_pos) - reserve_bytes + n, m_data->begin() + long(m_pos));
161 add_field(tag, pbf_wire_type::length_delimited);
173 m_parent_writer(
nullptr),
183 m_parent_writer(
nullptr),
195 m_data(parent_writer.m_data),
196 m_parent_writer(&parent_writer),
198 m_parent_writer->open_submessage(tag);
214 if (m_parent_writer) {
215 m_parent_writer->close_submessage();
231 add_field(tag, pbf_wire_type::varint);
232 add_fixed<char>(value);
242 add_tagged_varint(tag, uint64_t(value));
252 add_tagged_varint(tag, uint64_t(value));
272 add_tagged_varint(tag, value);
282 add_tagged_varint(tag, uint64_t(value));
302 add_tagged_varint(tag, value);
312 add_field(tag, pbf_wire_type::fixed32);
313 add_fixed<uint32_t>(value);
323 add_field(tag, pbf_wire_type::fixed32);
324 add_fixed<int32_t>(value);
334 add_field(tag, pbf_wire_type::fixed64);
335 add_fixed<uint64_t>(value);
345 add_field(tag, pbf_wire_type::fixed64);
346 add_fixed<int64_t>(value);
356 add_field(tag, pbf_wire_type::fixed32);
357 add_fixed<float>(value);
367 add_field(tag, pbf_wire_type::fixed64);
368 add_fixed<double>(value);
379 protozero_assert(m_pos == 0 &&
"you can't add fields to a parent pbf_writer if there is an existing pbf_writer for a submessage");
381 assert(size <= std::numeric_limits<pbf_length_type>::max());
383 m_data->append(value, size);
393 add_bytes(tag, value.data(), value.size());
414 add_bytes(tag, value.data(), value.size());
425 add_bytes(tag, value, std::strlen(value));
446 add_bytes(tag, value.data(), value.size());
465 template <
typename InputIterator>
467 add_packed_varint(tag, first, last);
479 template <
typename InputIterator>
481 add_packed_varint(tag, first, last);
493 template <
typename InputIterator>
495 add_packed_varint(tag, first, last);
507 template <
typename InputIterator>
509 add_packed_svarint(tag, first, last);
521 template <
typename InputIterator>
523 add_packed_varint(tag, first, last);
535 template <
typename InputIterator>
537 add_packed_varint(tag, first, last);
549 template <
typename InputIterator>
551 add_packed_svarint(tag, first, last);
563 template <
typename InputIterator>
565 add_packed_varint(tag, first, last);
577 template <
typename InputIterator>
579 add_packed_fixed<uint32_t, InputIterator>(tag, first, last,
580 typename std::iterator_traits<InputIterator>::iterator_category());
592 template <
typename InputIterator>
594 add_packed_fixed<int32_t, InputIterator>(tag, first, last,
595 typename std::iterator_traits<InputIterator>::iterator_category());
607 template <
typename InputIterator>
609 add_packed_fixed<uint64_t, InputIterator>(tag, first, last,
610 typename std::iterator_traits<InputIterator>::iterator_category());
622 template <
typename InputIterator>
624 add_packed_fixed<int64_t, InputIterator>(tag, first, last,
625 typename std::iterator_traits<InputIterator>::iterator_category());
637 template <
typename InputIterator>
639 add_packed_fixed<float, InputIterator>(tag, first, last,
640 typename std::iterator_traits<InputIterator>::iterator_category());
652 template <
typename InputIterator>
654 add_packed_fixed<double, InputIterator>(tag, first, last,
655 typename std::iterator_traits<InputIterator>::iterator_category());
664 #endif // PROTOZERO_PBF_WRITER_HPP
void add_packed_fixed64(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:608
void add_packed_sint32(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:508
void add_packed_sint64(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:550
void add_string(pbf_tag_type tag, const char *value)
Definition: pbf_writer.hpp:424
void add_packed_sfixed64(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:623
void add_packed_sfixed32(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:593
uint32_t pbf_length_type
Definition: pbf_types.hpp:45
uint64_t encode_zigzag64(int64_t value) noexcept
Definition: varint.hpp:112
void add_sint64(pbf_tag_type tag, int64_t value)
Definition: pbf_writer.hpp:291
void add_sfixed64(pbf_tag_type tag, int64_t value)
Definition: pbf_writer.hpp:344
void add_uint32(pbf_tag_type tag, uint32_t value)
Definition: pbf_writer.hpp:271
void add_bytes(pbf_tag_type tag, const std::string &value)
Definition: pbf_writer.hpp:392
#define protozero_assert(x)
Wrapper for assert() used for testing.
Definition: pbf_writer.hpp:36
void add_packed_enum(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:480
Definition: pbf_writer.hpp:47
void add_int64(pbf_tag_type tag, int64_t value)
Definition: pbf_writer.hpp:281
void add_int32(pbf_tag_type tag, int32_t value)
Definition: pbf_writer.hpp:251
void add_string(pbf_tag_type tag, const std::string &value)
Definition: pbf_writer.hpp:413
void add_uint64(pbf_tag_type tag, uint64_t value)
Definition: pbf_writer.hpp:301
pbf_wire_type
Definition: pbf_types.hpp:33
Contains the declaration of low-level types used in the pbf format.
void add_message(pbf_tag_type tag, const std::string &value)
Definition: pbf_writer.hpp:445
void add_float(pbf_tag_type tag, float value)
Definition: pbf_writer.hpp:355
uint32_t pbf_tag_type
Definition: pbf_types.hpp:26
void add_enum(pbf_tag_type tag, int32_t value)
Definition: pbf_writer.hpp:241
void add_packed_uint64(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:564
pbf_writer() noexcept
Definition: pbf_writer.hpp:181
void add_bytes(pbf_tag_type tag, const char *value, size_t size)
Definition: pbf_writer.hpp:378
void add_packed_uint32(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:522
uint32_t encode_zigzag32(int32_t value) noexcept
Definition: varint.hpp:105
void add_packed_int64(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:536
void add_string(pbf_tag_type tag, const char *value, size_t size)
Definition: pbf_writer.hpp:403
void add_fixed64(pbf_tag_type tag, uint64_t value)
Definition: pbf_writer.hpp:333
void add_bool(pbf_tag_type tag, bool value)
Definition: pbf_writer.hpp:230
void add_packed_bool(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:466
void add_packed_double(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:653
int write_varint(OutputIterator data, uint64_t value)
Definition: varint.hpp:89
void add_packed_float(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:638
pbf_writer(pbf_writer &parent_writer, pbf_tag_type tag)
Definition: pbf_writer.hpp:194
void add_packed_int32(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:494
void add_sfixed32(pbf_tag_type tag, int32_t value)
Definition: pbf_writer.hpp:322
pbf_writer(std::string &data) noexcept
Definition: pbf_writer.hpp:171
Contains low-level varint and zigzag encoding and decoding functions.
void add_sint32(pbf_tag_type tag, int32_t value)
Definition: pbf_writer.hpp:261
void add_fixed32(pbf_tag_type tag, uint32_t value)
Definition: pbf_writer.hpp:311
void add_packed_fixed32(pbf_tag_type tag, InputIterator first, InputIterator last)
Definition: pbf_writer.hpp:578
void add_message(pbf_tag_type tag, const char *value, size_t size)
Definition: pbf_writer.hpp:435
void add_double(pbf_tag_type tag, double value)
Definition: pbf_writer.hpp:366
All parts of the protozero header-only library are in this namespace.
Definition: byteswap.hpp:15