Bond
 
All Classes Namespaces Files Functions Variables Typedefs Friends Pages
Loading...
Searching...
No Matches
compact_binary.h
1// Copyright (c) Microsoft. All rights reserved.
2// Licensed under the MIT license. See LICENSE file in the project root for full license information.
3
4#pragma once
5
6#include <bond/core/config.h>
7
8#include "detail/simple_array.h"
9#include "encoding.h"
10
11#include <bond/core/bond_version.h>
12#include <bond/core/detail/checked.h>
13#include <bond/core/detail/recursionguard.h>
14#include <bond/core/traits.h>
15#include <bond/stream/output_counter.h>
16
17#include <boost/call_traits.hpp>
18#include <boost/noncopyable.hpp>
19#include <boost/static_assert.hpp>
20
21#include <cstring>
22
23/*
24
25 .----------.--------------. .----------.---------.
26 struct (v1) | fields | BT_STOP_BASE |...| fields | BT_STOP |
27 '----------'--------------' '----------'---------'
28
29 .----------.----------.--------------. .----------.---------.
30 struct (v2) | length | fields | BT_STOP_BASE |...| fields | BT_STOP |
31 '----------'----------'--------------' '----------'---------'
32
33 length variable int encoded uint32 length of following fields, up to and
34 including BT_STOP but excluding length itself.
35
36 .----------.----------. .----------.
37 fields | field | field |...| field |
38 '----------'----------' '----------'
39
40 .----------.----------.
41 field | id+type | value |
42 '----------'----------'
43
44 .---.---.---.---.---.---.---.---. i - id bits (BE unsigned int)
45 id+type 0 <= id <= 5 | i | i | i | t | t | t | t | t | t - type bits
46 '---'---'---'---'---'---'---'---' v - value bits
47 2 0 4 0
48
49 .---.---.---.---.---.---.---.---.---. .---.
50 5 < id <= 0xff | 1 | 1 | 0 | t | t | t | t | t | i |...| i |
51 '---'---'---'---'---'---'---'---'---' '---'
52 4 0 7 0
53
54 .---.---.---.---.---.---.---.---.---. .---.---. .---.
55 0xff < id <= 0xffff | 1 | 1 | 1 | t | t | t | t | t | i |...| i | i |...| i |
56 '---'---'---'---'---'---'---'---'---' '---'---' '---'
57 4 0 7 0 15 8
58
59
60 .---.---.---.---.---.---.---.---.
61 value bool | | | | | | | | v |
62 '---'---'---'---'---'---'---'---'
63 0
64
65 .---.---.---.---.---.---.---.---.
66 int8, uint8 | v | v | v | v | v | v | v | v |
67 '---'---'---'---'---'---'---'---'
68 7 0
69
70 signed uses two's complement
71
72 .---.---. .---.---.---. .---.
73 uint16, uint32, | 1 | v |...| v | 0 | v |...| v | [...]
74 uint64 '---'---' '---'---'---' '---'
75 6 0 13 7
76
77 LEB128 variable encoding, high bit of every byte
78 indicates if there is another byte
79
80
81 int16, int32, zig zag encoded to unsigned integer:
82 int64
83 0 -> 0
84 -1 -> 1
85 1 -> 2
86 -2 -> 3
87 ...
88
89 and then encoded as unsigned integer
90
91
92 float, double 32-bit or 64-bit little endian IEEE 764
93
94
95 enum enum constants are encoded using int32
96
97 .-------.------------.
98 string, wstring | count | characters |
99 '-------'------------'
100
101 count variable encoded uint32 count of 1-byte (for
102 string) or 2-byte (for wstring) Unicode code
103 units
104
105 characters 1-byte UTF-8 code units (for string) or 2-byte
106 UTF-16LE code units (for wstring)
107
108
109 .-------.-------.-------.
110 blob, list, set, | type | count | items |
111 vector, nullable '-------'-------'-------'
112
113 .---.---.---.---.---.---.---.---.
114 type (v1) | | | | t | t | t | t | t |
115 '---'---'---'---'---'---'---'---'
116 4 0
117
118 .---.---.---.---.---.---.---.---.
119 type (v2) | c | c | c | t | t | t | t | t |
120 '---'---'---'---'---'---'---'---'
121 2 0 4 0
122
123 if count of items is < 7, 'c' are bit of (count + 1),
124 otherwise 'c' bits are 0.
125
126 count variable encoded uint32 count of items
127 omitted in v2 if 'c' bits within type byte are not 0
128
129 items each item encoded according to its type
130
131
132 .----------.------------.-------.-----.-------.
133 map | key type | value type | count | key | value |
134 '----------'------------'-------'-----'-------'
135
136 .---.---.---.---.---.---.---.---.
137 key type, | | | | t | t | t | t | t |
138 value type '---'---'---'---'---'---'---'---'
139 4 0
140
141 count variable encoded uint32 count of {key,mapped} pairs
142
143 key, mapped each item encoded according to its type
144
145*/
146
147namespace bond
148{
149
150
151template <typename BufferT>
153
155template <typename BufferT>
157{
158public:
159 typedef BufferT Buffer;
160 typedef DynamicParser<CompactBinaryReader&> Parser;
161 typedef CompactBinaryWriter<Buffer> Writer;
162
163 BOND_STATIC_CONSTEXPR uint16_t magic = COMPACT_PROTOCOL;
164 BOND_STATIC_CONSTEXPR uint16_t version = v2;
165
167 CompactBinaryReader(typename boost::call_traits<Buffer>::param_type input,
168 uint16_t version_value = default_version<CompactBinaryReader>::value)
169 : _input(input),
170 _version(version_value)
171 {
172 BOOST_ASSERT(protocol_has_multiple_versions<CompactBinaryReader>::value
173 ? _version <= CompactBinaryReader::version
174 : _version == default_version<CompactBinaryReader>::value);
175 }
176
177
178 // This identical to compiler generated ctor except for noexcept declaration.
179 // Copy ctor that is explicitly declared throw() is needed for boost::variant
180 // to use optimized code path.
182 CompactBinaryReader(const CompactBinaryReader& that) BOND_NOEXCEPT
183 : _input(that._input),
184 _version(that._version)
185 {}
186
187
189 bool operator==(const CompactBinaryReader& rhs) const
190 {
191 return _input == rhs._input;
192 }
193
194
196 typename boost::call_traits<Buffer>::const_reference
197 GetBuffer() const
198 {
199 return _input;
200 }
201
202
204 typename boost::call_traits<Buffer>::reference
206 {
207 return _input;
208 }
209
210
211 bool ReadVersion()
212 {
213 uint16_t magic_value;
214
215 _input.Read(magic_value);
216 _input.Read(_version);
217
218 return magic_value == CompactBinaryReader::magic
219 && (protocol_has_multiple_versions<CompactBinaryReader>::value
220 ? _version <= CompactBinaryReader::version
221 : _version == default_version<CompactBinaryReader>::value);
222 }
223
224
225 // ReadStructBegin
226 void ReadStructBegin(bool base = false)
227 {
228 if (!base && v2 == _version)
229 {
230 uint32_t length;
231 Read(length);
232 }
233 }
234
235 // ReadStructEnd
236 void ReadStructEnd(bool = false)
237 {}
238
239 // ReadFieldBegin
240 void ReadFieldBegin(BondDataType& type, uint16_t& id)
241 {
242 uint8_t raw;
243
244 _input.Read(raw);
245
246 type = static_cast<BondDataType>(raw & 0x1f);
247 id = static_cast<uint16_t>(raw & (0x07 << 5));
248
249 if (id == (0x07 << 5))
250 {
251 // ID is in (0xff, 0xffff] and is in the next two bytes
252 _input.Read(id);
253 }
254 else if (id == (0x06 << 5))
255 {
256 // ID is in (5, 0xff] and is in the next one byte
257 _input.Read(raw);
258 id = static_cast<uint16_t>(raw);
259 }
260 else
261 {
262 // ID is in [0, 5] and was in the byte we already read
263 id >>= 5;
264 }
265 }
266
267 // ReadFieldEnd
268 void ReadFieldEnd()
269 {}
270
271
272 // ReadContainerBegin
273 void ReadContainerBegin(uint32_t& size, BondDataType& type)
274 {
275 uint8_t raw;
276
277 _input.Read(raw);
278 type = static_cast<BondDataType>(raw & 0x1f);
279
280 if (v2 == _version && (raw & (0x07 << 5)))
281 size = (raw >> 5) - 1;
282 else
283 Read(size);
284 }
285
286
287 // container of 2-tuple (e.g. map)
288 void ReadContainerBegin(uint32_t& size, std::pair<BondDataType, BondDataType>& type)
289 {
290 uint8_t raw;
291
292 _input.Read(raw);
293 type.first = static_cast<BondDataType>(raw);
294
295 _input.Read(raw);
296 type.second = static_cast<BondDataType>(raw);
297
298 Read(size);
299 }
300
301
302 // ReadContainerEnd
303 void ReadContainerEnd()
304 {}
305
306
307 // Read for floating point
308 template <typename T>
309 typename boost::enable_if<std::is_floating_point<T> >::type
310 Read(T& value)
311 {
312 _input.Read(value);
313 }
314
315 // Read for unsigned integers
316 template <typename T>
317 typename boost::enable_if<std::is_unsigned<T> >::type
318 Read(T& value)
319 {
320 ReadVariableUnsigned(_input, value);
321 }
322
323 // Read for signed integers
324 template <typename T>
325 typename boost::enable_if<is_signed_int<T> >::type
326 Read(T& value)
327 {
328 typename std::make_unsigned<T>::type unsigned_value;
329
330 ReadVariableUnsigned(_input, unsigned_value);
331 value = DecodeZigZag(unsigned_value);
332 }
333
334
335 // Read for enums
336 template <typename T>
337 typename boost::enable_if<std::is_enum<T> >::type
338 Read(T& value)
339 {
340 BOOST_STATIC_ASSERT(sizeof(value) == sizeof(int32_t));
341 int32_t raw;
342 Read(raw);
343 std::memcpy(&value, &raw, sizeof(raw));
344 }
345
346
347 // Read for int8_t
348 void Read(int8_t& value)
349 {
350 _input.Read(value);
351 }
352
353
354 // Read for uint8_t
355 void Read(uint8_t& value)
356 {
357 _input.Read(value);
358 }
359
360
361 // Read for bool
362 void Read(bool& value)
363 {
364 _input.Read(value);
365 }
366
367
368 // Read for strings
369 template <typename T>
370 typename boost::enable_if<is_string_type<T> >::type
371 Read(T& value)
372 {
373 uint32_t length = 0;
374
375 Read(length);
376
377 constexpr uint8_t charSize = static_cast<uint8_t>(sizeof(typename detail::string_char_int_type<T>::type));
378 uint32_t numStringBytes = detail::checked_multiply(length, charSize);
379 if (!_input.CanRead(numStringBytes))
380 {
381 OutOfBoundStringSizeException();
382 }
383
384 detail::ReadStringData(_input, value, length);
385 }
386
387
388 // Read for blob
389 void Read(blob& value, uint32_t size)
390 {
391 _input.Read(value, size);
392 }
393
394 // Does the reader have enough input buffer left to read an array of T?
395 template<typename T>
396 bool CanReadArray(uint32_t num_elems)
397 {
398 // Non-float types have variable length encoding going down to 1 Byte.
399 // Strings need 1 Byte per charcter. Wide strings handled below.
400 return _input.CanRead(num_elems);
401 }
402
403 template<typename T>
404 typename boost::enable_if<typename std::is_floating_point<T>::value, bool>::type
405 CanReadArray(uint32_t num_elems)
406 {
407 // We will need to read num_elems instances of T. This will not overflow because
408 // num_elems < 2^32 and we call this only if std::is_floating_point<T>.
409 uint64_t num_bytes = static_cast<uint64_t>(num_elems) * sizeof(T);
410
411 // Check the upper half to ensure we don't try to read more than 4 GB, the
412 // Reader wouldn't be able to handle that. Then ask the Reader if it has enough
413 // data left for our vector.
414 return (num_bytes >> 32 == 0) && _input.CanRead(static_cast<uint32_t>(num_bytes & 0xffffffff));
415 }
416
417 template <typename T>
418 void Skip()
419 {
420 SkipType<get_type_id<T>::value>();
421 }
422
423 template <typename T>
424 void Skip(const bonded<T, CompactBinaryReader&>&)
425 {
426 SkipType<bond::BT_STRUCT>();
427 }
428
429 void Skip(BondDataType type)
430 {
431 SkipType(type);
432 }
433
434protected:
435 using BT = BondDataType;
436
437 template <BT T>
438 typename boost::enable_if_c<(T == BT_BOOL || T == BT_UINT8 || T == BT_INT8)>::type
439 SkipType(uint32_t size = 1)
440 {
441 _input.Skip(detail::checked_multiply(size, sizeof(uint8_t)));
442 }
443
444 template <BT T>
445 typename boost::enable_if_c<(T == BT_UINT16 || T == BT_UINT32 || T == BT_UINT64
446 || T == BT_INT16 || T == BT_INT32 || T == BT_INT64)>::type
447 SkipType()
448 {
449 uint64_t value;
450 Read(value);
451 }
452
453 template <BT T>
454 typename boost::enable_if_c<(T == BT_FLOAT)>::type
455 SkipType(uint32_t size = 1)
456 {
457 _input.Skip(detail::checked_multiply(size, sizeof(float)));
458 }
459
460 template <BT T>
461 typename boost::enable_if_c<(T == BT_DOUBLE)>::type
462 SkipType(uint32_t size = 1)
463 {
464 _input.Skip(detail::checked_multiply(size, sizeof(double)));
465 }
466
467 template <BT T>
468 typename boost::enable_if_c<(T == BT_STRING)>::type
469 SkipType()
470 {
471 uint32_t length;
472
473 Read(length);
474 _input.Skip(length);
475 }
476
477 template <BT T>
478 typename boost::enable_if_c<(T == BT_WSTRING)>::type
479 SkipType()
480 {
481 uint32_t length;
482
483 Read(length);
484 _input.Skip(detail::checked_multiply(length, sizeof(uint16_t)));
485 }
486
487 template <BT T>
488 typename boost::enable_if_c<(T == BT_SET || T == BT_LIST)>::type
489 SkipType()
490 {
491 BondDataType element_type;
492 uint32_t size;
493
494 bond::detail::RecursionGuard guard;
495
496 ReadContainerBegin(size, element_type);
497 SkipType(element_type, size);
498 ReadContainerEnd();
499 }
500
501 template <BT T>
502 typename boost::enable_if_c<(T == BT_MAP)>::type
503 SkipType()
504 {
505 std::pair<BondDataType, BondDataType> element_type;
506 uint32_t size;
507
508 bond::detail::RecursionGuard guard;
509
510 ReadContainerBegin(size, element_type);
511 for (int64_t i = 0; i < size; ++i)
512 {
513 SkipType(element_type.first);
514 SkipType(element_type.second);
515 }
516 ReadContainerEnd();
517 }
518
519 void SkipStructV1()
520 {
521 BOOST_ASSERT(v1 == _version);
522
523 bond::detail::RecursionGuard guard;
524
525 for (;;)
526 {
527 ReadStructBegin();
528
529 uint16_t id;
530 BondDataType field_type;
531
532 for (ReadFieldBegin(field_type, id);
533 field_type != bond::BT_STOP && field_type != bond::BT_STOP_BASE;
534 ReadFieldEnd(), ReadFieldBegin(field_type, id))
535 {
536 SkipType(field_type);
537 }
538
539 ReadStructEnd();
540
541 if (field_type == bond::BT_STOP)
542 break;
543 }
544 }
545
546 void SkipStructV2()
547 {
548 BOOST_ASSERT(v2 == _version);
549
550 uint32_t length;
551 Read(length);
552 _input.Skip(length);
553 }
554
555 template <BT T>
556 typename boost::enable_if_c<(T == BT_STRUCT)>::type
557 SkipType()
558 {
559 if (v2 == _version)
560 {
561 SkipStructV2();
562 }
563 else
564 {
565 SkipStructV1();
566 }
567 }
568
569 template <BT T>
570 typename boost::enable_if_c<(T == BT_STRUCT)>::type
571 SkipType(uint32_t size)
572 {
573 if (v2 == _version)
574 {
575 for (int64_t i = 0; i < size; ++i)
576 {
577 SkipStructV2();
578 }
579 }
580 else
581 {
582 for (int64_t i = 0; i < size; ++i)
583 {
584 SkipStructV1();
585 }
586 }
587 }
588
589 template <BT T>
590 typename boost::enable_if_c<(T == BT_UINT16 || T == BT_UINT32 || T == BT_UINT64
591 || T == BT_INT16 || T == BT_INT32 || T == BT_INT64
592 || T == BT_STRING || T == BT_WSTRING
593 || T == BT_SET || T == BT_LIST || T == BT_MAP)>::type
594 SkipType(uint32_t size)
595 {
596 for (int64_t i = 0; i < size; ++i)
597 {
598 SkipType<T>();
599 }
600 }
601
602 template <typename... Args>
603 void SkipType(BondDataType type, Args&&... args)
604 {
605 switch (type)
606 {
607 case BT_BOOL:
608 case BT_UINT8:
609 case BT_INT8:
610 SkipType<BT_BOOL>(std::forward<Args>(args)...);
611 break;
612
613 case BT_UINT64:
614 case BT_UINT32:
615 case BT_UINT16:
616 case BT_INT64:
617 case BT_INT32:
618 case BT_INT16:
619 SkipType<BT_UINT64>(std::forward<Args>(args)...);
620 break;
621
622 case BT_FLOAT:
623 SkipType<BT_FLOAT>(std::forward<Args>(args)...);
624 break;
625
626 case BT_DOUBLE:
627 SkipType<BT_DOUBLE>(std::forward<Args>(args)...);
628 break;
629
630 case BT_STRING:
631 SkipType<BT_STRING>(std::forward<Args>(args)...);
632 break;
633
634 case BT_WSTRING:
635 SkipType<BT_WSTRING>(std::forward<Args>(args)...);
636 break;
637
638 case BT_SET:
639 case BT_LIST:
640 SkipType<BT_SET>(std::forward<Args>(args)...);
641 break;
642
643 case BT_MAP:
644 SkipType<BT_MAP>(std::forward<Args>(args)...);
645 break;
646
647 case BT_STRUCT:
648 SkipType<BT_STRUCT>(std::forward<Args>(args)...);
649 break;
650
651 default:
652 bond::UnknownDataTypeException();
653 break;
654 }
655 }
656
657 Buffer _input;
658 uint16_t _version;
659
660 template <typename Input, typename Output>
661 friend
662 bool is_protocol_version_same(const CompactBinaryReader<Input>&,
663 const CompactBinaryWriter<Output>&);
664};
665
666template <typename BufferT>
667BOND_CONSTEXPR_OR_CONST uint16_t CompactBinaryReader<BufferT>::magic;
668
669
670class CompactBinaryCounter
671{
672 template <typename Buffer>
673 friend class CompactBinaryWriter;
674
675private:
676 struct type : OutputCounter // Must be a new type and not an alias.
677 {};
678};
679
680
682template <typename BufferT>
683class CompactBinaryWriter
684 : boost::noncopyable
685{
686 struct Pass1
687 {
688 Pass1(CompactBinaryWriter* writer)
689 : writer(writer)
690 {}
691
692 ~Pass1()
693 {
694 writer->_it = NULL;
695 }
696
697 CompactBinaryWriter* writer;
698 };
699
700 using Counter = CompactBinaryCounter::type;
701
702public:
703 typedef BufferT Buffer;
704 typedef CompactBinaryReader<Buffer> Reader;
705 typedef CompactBinaryWriter<Counter> Pass0;
706
707
709 CompactBinaryWriter(Buffer& output,
710 uint16_t version = default_version<Reader>::value)
711 : _output(output),
712 _it(NULL),
713 _version(version)
714 {
715 BOOST_ASSERT(protocol_has_multiple_versions<Reader>::value
716 ? _version <= Reader::version
717 : _version == default_version<Reader>::value);
718 }
719
720 template<typename T>
721 CompactBinaryWriter(Counter& output,
722 const CompactBinaryWriter<T>& pass1)
723 : _output(output),
724 _version(pass1._version)
725 {}
726
727
729 typename boost::call_traits<Buffer>::reference
731 {
732 return _output;
733 }
734
735
736 bool NeedPass0()
737 {
738 return v2 == _version && !_it;
739 }
740
741
742 Pass1 WithPass0(Pass0& pass0)
743 {
744 _it = pass0._lengths.begin();
745 return this;
746 }
747
748 void WriteVersion()
749 {
750 _output.Write(Reader::magic);
751 _output.Write(_version);
752 }
753
754
755 void WriteStructBegin(const Metadata& /*metadata*/, bool base)
756 {
757 if (!base)
758 {
759 LengthBegin(_output);
760 }
761 }
762
763 void WriteStructEnd(bool base = false)
764 {
765 if (base)
766 {
767 _output.Write(static_cast<uint8_t>(BT_STOP_BASE));
768 }
769 else
770 {
771 _output.Write(static_cast<uint8_t>(BT_STOP));
772 LengthEnd(_output);
773 }
774 }
775
776 // WriteField for basic types
777 template <typename T>
778 void WriteField(uint16_t id, const bond::Metadata& /*metadata*/, const T& value)
779 {
780 WriteFieldBegin(get_type_id<T>::value, id);
781 Write(value);
782 WriteFieldEnd();
783 }
784
785 // WriteFieldBegin
786 void WriteFieldBegin(BondDataType type, uint16_t id, const ::bond::Metadata& /*metadata*/)
787 {
788 WriteFieldBegin(type, id);
789 }
790
791 void WriteFieldBegin(BondDataType type, uint16_t id)
792 {
793 BOOST_ASSERT((type & 0x1f) == type);
794
795 if (id <= 5)
796 {
797 _output.Write(static_cast<uint8_t>(type | ((id) << 5)));
798 }
799 else if (id <= 0xff)
800 {
801 _output.Write(static_cast<uint8_t>(type | (0x06 << 5)));
802 _output.Write(static_cast<uint8_t>(id));
803 }
804 else
805 {
806 _output.Write(static_cast<uint8_t>(type | (0x07 << 5)));
807 _output.Write(id);
808 }
809 }
810
811 // WriteFieldEnd
812 void WriteFieldEnd()
813 {}
814
815 // WriteContainerBegin
816 void WriteContainerBegin(uint32_t size, BondDataType type)
817 {
818 BOOST_ASSERT((type & 0x1f) == type);
819
820 if (v2 == _version && size < 7)
821 {
822 Write(static_cast<uint8_t>(type | ((size + 1) << 5)));
823 }
824 else
825 {
826 Write(static_cast<uint8_t>(type));
827 Write(size);
828 }
829 }
830
831 // container of 2-tuples (e.g. map)
832 void WriteContainerBegin(uint32_t size, std::pair<BondDataType, BondDataType> type)
833 {
834 Write(static_cast<uint8_t>(type.first));
835 Write(static_cast<uint8_t>(type.second));
836 WriteVariableUnsigned(_output, size);
837 }
838
839 // WriteContainerEnd
840 void WriteContainerEnd()
841 {}
842
843 // Write for floating point
844 template <typename T>
845 typename boost::enable_if<std::is_floating_point<T> >::type
846 Write(const T& value)
847 {
848 _output.Write(value);
849 }
850
851 // Write for unsigned integers
852 template <typename T>
853 typename boost::enable_if<std::is_unsigned<T> >::type
854 Write(const T& value)
855 {
856 WriteVariableUnsigned(_output, value);
857 }
858
859 // Write for signed integers
860 template <typename T>
861 typename boost::enable_if<is_signed_int<T> >::type
862 Write(const T& value)
863 {
864 WriteVariableUnsigned(_output, EncodeZigZag(value));
865 }
866
867 // Write for enums
868 template <typename T>
869 typename boost::enable_if<std::is_enum<T> >::type
870 Write(const T& value)
871 {
872 BOOST_STATIC_ASSERT(sizeof(value) == sizeof(int32_t));
873 Write(static_cast<int32_t>(value));
874 }
875
876 // Write for int8_t
877 void Write(const int8_t& value)
878 {
879 _output.Write(value);
880 }
881
882 // Write for uint8_t
883 void Write(const uint8_t& value)
884 {
885 _output.Write(value);
886 }
887
888 // Write for bool
889 void Write(const bool& value)
890 {
891 _output.Write(value);
892 }
893
894 // Write for strings
895 template <typename T>
896 typename boost::enable_if<is_string_type<T> >::type
897 Write(const T& value)
898 {
899 uint32_t length = string_length(value);
900
901 Write(length);
902 detail::WriteStringData(_output, value, length);
903 }
904
905 // Write for blob
906 void Write(const blob& value)
907 {
908 _output.Write(value);
909 }
910
911protected:
912 template <typename Buffer>
913 friend class CompactBinaryWriter;
914
915 void LengthBegin(Counter& counter)
916 {
917 _stack.push(_lengths.size());
918 _lengths.push(counter.GetCount());
919 }
920
921 void LengthEnd(Counter& counter)
922 {
923 uint32_t& length = _lengths[_stack.pop()];
924
925 length = counter.GetCount() - length;
926 counter.WriteVariableUnsigned(length);
927 }
928
929 template<typename T>
930 void LengthBegin(T&)
931 {
932 if (v2 == _version)
933 {
934 Write(*_it++);
935 }
936 }
937
938 template<typename T>
939 void LengthEnd(T&)
940 {}
941
942protected:
943 Buffer& _output;
944 const uint32_t* _it;
945 uint16_t _version;
946 detail::SimpleArray<uint32_t> _stack;
947 detail::SimpleArray<uint32_t> _lengths;
948
949 template <typename Input, typename Output>
950 friend
951 bool is_protocol_version_same(const CompactBinaryReader<Input>&,
952 const CompactBinaryWriter<Output>&);
953};
954
955template <typename Input> struct
956protocol_has_multiple_versions<CompactBinaryReader<Input> >
957 : enable_protocol_versions<CompactBinaryReader<Input> > {};
958
959template <typename Input, typename Output>
960inline
961bool is_protocol_version_same(const CompactBinaryReader<Input>& reader,
962 const CompactBinaryWriter<Output>& writer)
963{
964 return reader._version == writer._version;
965}
966
967} // namespace bond
Reader for Compact Binary Protocol.
Definition compact_binary.h:157
CompactBinaryReader(typename boost::call_traits< Buffer >::param_type input, uint16_t version_value=default_version< CompactBinaryReader >::value)
Construct from input buffer/stream containing serialized data.
Definition compact_binary.h:167
boost::call_traits< Buffer >::reference GetBuffer()
Access to underlying buffer.
Definition compact_binary.h:205
bool operator==(const CompactBinaryReader &rhs) const
Comparison operator.
Definition compact_binary.h:189
CompactBinaryReader(const CompactBinaryReader &that) BOND_NOEXCEPT
Copy constructor.
Definition compact_binary.h:182
boost::call_traits< Buffer >::const_reference GetBuffer() const
Access to underlying buffer.
Definition compact_binary.h:197
Writer for Compact Binary Protocol.
Definition compact_binary.h:685
boost::call_traits< Buffer >::reference GetBuffer()
Access to underlying buffer.
Definition compact_binary.h:730
CompactBinaryWriter(Buffer &output, uint16_t version=default_version< Reader >::value)
Construct from output buffer/stream.
Definition compact_binary.h:709
namespace bond
Definition apply.h:17