veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
serialize.hpp
1// Copyright (c) 2009-2010 Satoshi Nakamoto
2// Copyright (c) 2009-2018 The Bitcoin Core developers
3// Copyright (c) 2019-2022 Xenios SEZC
4// https://www.veriblock.org
5// Distributed under the MIT software license, see the accompanying
6// file COPYING or http://www.opensource.org/licenses/mit-license.php.
7
8#ifndef BFI_BITCOIN_SERIALIZE_HPP
9#define BFI_BITCOIN_SERIALIZE_HPP
10
11#include <string.h>
12
13#include <algorithm>
14#include <cstdint>
15#include <ios>
16#include <limits>
17#include <map>
18#include <memory>
19#include <set>
20#include <string>
21#include <utility>
22#include <vector>
23#include <veriblock/pop/blob.hpp>
24#include <veriblock/pop/entities/popdata.hpp>
25#include <veriblock/pop/read_stream.hpp>
26#include <veriblock/pop/slice.hpp>
27#include <veriblock/pop/write_stream.hpp>
28
29namespace altintegration {
30
31namespace btc {
32
33static const uint64_t MAX_SIZE = 0x02000000;
34
45constexpr deserialize_type deserialize{};
46
51template <typename T>
52inline T& REF(const T& val) {
53 return const_cast<T&>(val);
54}
55
60template <typename T>
61inline T* NCONST_PTR(const T* val) {
62 return const_cast<T*>(val);
63}
64
66inline char* CharCast(char* c) { return c; }
67inline char* CharCast(unsigned char* c) { return (char*)c; }
68inline const char* CharCast(const char* c) { return c; }
69inline const char* CharCast(const unsigned char* c) { return (const char*)c; }
70
71/*
72 * Lowest-level serialization and conversion.
73 * @note Sizes of these types are verified in the tests
74 */
75template <typename Stream>
76inline void ser_writedata8(Stream& s, uint8_t obj) {
77 s.template writeLE<uint8_t>(obj);
78}
79template <typename Stream>
80inline void ser_writedata16(Stream& s, uint16_t obj) {
81 s.template writeLE<uint16_t>(obj);
82}
83template <typename Stream>
84inline void ser_writedata16be(Stream& s, uint16_t obj) {
85 s.template writeBE<uint16_t>(obj);
86}
87template <typename Stream>
88inline void ser_writedata32(Stream& s, uint32_t obj) {
89 s.template writeLE<uint32_t>(obj);
90}
91template <typename Stream>
92inline void ser_writedata32be(Stream& s, uint32_t obj) {
93 s.template writeBE<uint32_t>(obj);
94}
95template <typename Stream>
96inline void ser_writedata64(Stream& s, uint64_t obj) {
97 s.template writeLE<uint64_t>(obj);
98}
99
100template <typename Stream, typename T>
101T ReadLE(Stream& s) {
102 T out = 0;
104 if (!s.template readLE<T>(out, state)) {
105 throw std::ios_base::failure(state.toString());
106 }
107 return out;
108}
109
110template <typename Stream, typename T>
111T ReadBE(Stream& s) {
112 T out = 0;
114 if (!s.template readBE<T>(out, state)) {
115 throw std::ios_base::failure(state.toString());
116 }
117 return out;
118}
119
120template <typename Stream>
121inline uint8_t ser_readdata8(Stream& s) {
122 return ReadLE<Stream, uint8_t>(s);
123}
124template <typename Stream>
125inline uint16_t ser_readdata16(Stream& s) {
126 return ReadLE<Stream, uint16_t>(s);
127}
128template <typename Stream>
129inline uint16_t ser_readdata16be(Stream& s) {
130 return ReadBE<Stream, uint16_t>(s);
131}
132template <typename Stream>
133inline uint32_t ser_readdata32(Stream& s) {
134 return ReadLE<Stream, uint32_t>(s);
135}
136template <typename Stream>
137inline uint32_t ser_readdata32be(Stream& s) {
138 return ReadBE<Stream, uint32_t>(s);
139}
140template <typename Stream>
141inline uint64_t ser_readdata64(Stream& s) {
142 return ReadLE<Stream, uint64_t>(s);
143}
144inline uint64_t ser_double_to_uint64(double x) {
145 union {
146 double x;
147 uint64_t y;
148 } tmp;
149 tmp.x = x;
150 return tmp.y;
151}
152inline uint32_t ser_float_to_uint32(float x) {
153 union {
154 float x;
155 uint32_t y;
156 } tmp;
157 tmp.x = x;
158 return tmp.y;
159}
160inline double ser_uint64_to_double(uint64_t y) {
161 union {
162 double x;
163 uint64_t y;
164 } tmp;
165 tmp.y = y;
166 return tmp.x;
167}
168inline float ser_uint32_to_float(uint32_t y) {
169 union {
170 float x;
171 uint32_t y;
172 } tmp;
173 tmp.y = y;
174 return tmp.x;
175}
176
178//
179// Templates for serializing to anything that looks like a stream,
180// i.e. anything that supports .read(char*, size_t) and .write(char*, size_t)
181//
182
183class CSizeComputer;
184
185enum {
186 // primary actions
187 SER_NETWORK = (1 << 0),
188 SER_DISK = (1 << 1),
189 SER_GETHASH = (1 << 2),
190};
191
194template <typename X>
195X& ReadWriteAsHelper(X& x) {
196 return x;
197}
198template <typename X>
199const X& ReadWriteAsHelper(const X& x) {
200 return x;
201}
202
203#define READWRITE(...) (SerReadWriteMany(s, ser_action, __VA_ARGS__))
204#define READWRITEAS(type, obj) \
205 (SerReadWriteMany(s, ser_action, ReadWriteAsHelper<type>(obj)))
206
213#define ADD_SERIALIZE_METHODS \
214 template <typename Stream> \
215 void Serialize(Stream& s) const { \
216 NCONST_PTR(this)->SerializationOp(s, CSerActionSerialize()); \
217 } \
218 template <typename Stream> \
219 void Unserialize(Stream& s) { \
220 SerializationOp(s, CSerActionUnserialize()); \
221 }
222
223#ifndef CHAR_EQUALS_INT8
224template <typename Stream>
225inline void Serialize(Stream& s, char a) {
226 ser_writedata8(s, a);
227} // TODO Get rid of bare char
228#endif
229template <typename Stream>
230inline void Serialize(Stream& s, int8_t a) {
231 ser_writedata8(s, a);
232}
233template <typename Stream>
234inline void Serialize(Stream& s, uint8_t a) {
235 ser_writedata8(s, a);
236}
237template <typename Stream>
238inline void Serialize(Stream& s, int16_t a) {
239 ser_writedata16(s, a);
240}
241template <typename Stream>
242inline void Serialize(Stream& s, uint16_t a) {
243 ser_writedata16(s, a);
244}
245template <typename Stream>
246inline void Serialize(Stream& s, int32_t a) {
247 ser_writedata32(s, a);
248}
249template <typename Stream>
250inline void Serialize(Stream& s, uint32_t a) {
251 ser_writedata32(s, a);
252}
253template <typename Stream>
254inline void Serialize(Stream& s, int64_t a) {
255 ser_writedata64(s, a);
256}
257template <typename Stream>
258inline void Serialize(Stream& s, uint64_t a) {
259 ser_writedata64(s, a);
260}
261template <typename Stream>
262inline void Serialize(Stream& s, float a) {
263 ser_writedata32(s, ser_float_to_uint32(a));
264}
265template <typename Stream>
266inline void Serialize(Stream& s, double a) {
267 ser_writedata64(s, ser_double_to_uint64(a));
268}
269template <typename Stream, int N>
270inline void Serialize(Stream& s, const char (&a)[N]) {
271 s.write(a, N);
272}
273template <typename Stream, int N>
274inline void Serialize(Stream& s, const unsigned char (&a)[N]) {
275 s.write(CharCast(a), N);
276}
277template <typename Stream>
278inline void Serialize(Stream& s,
280 s.write(CharCast(slice.data()), slice.size());
281}
282template <typename Stream, size_t N>
283inline void Serialize(Stream& s, const altintegration::Blob<N>& blob) {
284 s.write(CharCast(blob.data()), blob.size());
285}
286
287#ifndef CHAR_EQUALS_INT8
288template <typename Stream>
289inline void Unserialize(Stream& s, char& a) {
290 a = ser_readdata8(s);
291} // TODO Get rid of bare char
292#endif
293template <typename Stream>
294inline void Unserialize(Stream& s, int8_t& a) {
295 a = ser_readdata8(s);
296}
297template <typename Stream>
298inline void Unserialize(Stream& s, uint8_t& a) {
299 a = ser_readdata8(s);
300}
301template <typename Stream>
302inline void Unserialize(Stream& s, int16_t& a) {
303 a = ser_readdata16(s);
304}
305template <typename Stream>
306inline void Unserialize(Stream& s, uint16_t& a) {
307 a = ser_readdata16(s);
308}
309template <typename Stream>
310inline void Unserialize(Stream& s, int32_t& a) {
311 a = ser_readdata32(s);
312}
313template <typename Stream>
314inline void Unserialize(Stream& s, uint32_t& a) {
315 a = ser_readdata32(s);
316}
317template <typename Stream>
318inline void Unserialize(Stream& s, int64_t& a) {
319 a = ser_readdata64(s);
320}
321template <typename Stream>
322inline void Unserialize(Stream& s, uint64_t& a) {
323 a = ser_readdata64(s);
324}
325template <typename Stream>
326inline void Unserialize(Stream& s, float& a) {
327 a = ser_uint32_to_float(ser_readdata32(s));
328}
329template <typename Stream>
330inline void Unserialize(Stream& s, double& a) {
331 a = ser_uint64_to_double(ser_readdata64(s));
332}
333template <typename Stream, int N>
334inline void Unserialize(Stream& s, char (&a)[N]) {
336 if (!s.read(N, (uint8_t*)a, state)) {
337 throw std::ios_base::failure(state.toString());
338 }
339}
340template <typename Stream, int N>
341inline void Unserialize(Stream& s, unsigned char (&a)[N]) {
343 if (!s.read(N, a, state)) {
344 throw std::ios_base::failure(state.toString());
345 }
346}
347template <typename Stream>
348inline void Unserialize(Stream& s,
351 if (!s.read(slice.size(), slice.data(), state)) {
352 throw std::ios_base::failure(state.toString());
353 }
354}
355template <typename Stream, size_t N>
356inline void Unserialize(Stream& s, altintegration::Blob<N>& blob) {
358 if (!s.read(blob.size(), blob.data(), state)) {
359 throw std::ios_base::failure(state.toString());
360 }
361}
362
363template <typename Stream>
364inline void Serialize(Stream& s, bool a) {
365 char f = a;
366 ser_writedata8(s, f);
367}
368template <typename Stream>
369inline void Unserialize(Stream& s, bool& a) {
370 char f = ser_readdata8(s);
371 a = f;
372}
373
381inline unsigned int GetSizeOfCompactSize(uint64_t nSize) {
382 if (nSize < 253)
383 return sizeof(unsigned char);
384 else if (nSize <= std::numeric_limits<unsigned short>::max())
385 return sizeof(unsigned char) + sizeof(unsigned short);
386 else if (nSize <= std::numeric_limits<unsigned int>::max())
387 return sizeof(unsigned char) + sizeof(unsigned int);
388 else
389 return sizeof(unsigned char) + sizeof(uint64_t);
390}
391
392inline void WriteCompactSize(CSizeComputer& os, uint64_t nSize);
393
394template <typename Stream>
395void WriteCompactSize(Stream& os, uint64_t nSize) {
396 if (nSize < 253) {
397 ser_writedata8(os, (uint8_t)nSize);
398 } else if (nSize <= std::numeric_limits<unsigned short>::max()) {
399 ser_writedata8(os, 253);
400 ser_writedata16(os, (uint16_t)nSize);
401 } else if (nSize <= std::numeric_limits<unsigned int>::max()) {
402 ser_writedata8(os, 254);
403 ser_writedata32(os, (uint32_t)nSize);
404 } else {
405 ser_writedata8(os, 255);
406 ser_writedata64(os, nSize);
407 }
408 return;
409}
410
411template <typename Stream>
412uint64_t ReadCompactSize(Stream& is) {
413 uint8_t chSize = ser_readdata8(is);
414 uint64_t nSizeRet = 0;
415 if (chSize < 253) {
416 nSizeRet = chSize;
417 } else if (chSize == 253) {
418 nSizeRet = ser_readdata16(is);
419 if (nSizeRet < 253)
420 throw std::ios_base::failure("non-canonical ReadCompactSize()");
421 } else if (chSize == 254) {
422 nSizeRet = ser_readdata32(is);
423 if (nSizeRet < 0x10000u)
424 throw std::ios_base::failure("non-canonical ReadCompactSize()");
425 } else {
426 nSizeRet = ser_readdata64(is);
427 if (nSizeRet < 0x100000000ULL)
428 throw std::ios_base::failure("non-canonical ReadCompactSize()");
429 }
430 if (nSizeRet > MAX_SIZE)
431 throw std::ios_base::failure("ReadCompactSize(): size too large");
432 return nSizeRet;
433}
434
469enum class VarIntMode { DEFAULT, NONNEGATIVE_SIGNED };
470
471template <VarIntMode Mode, typename I>
473 constexpr CheckVarIntMode() {
474 static_assert(Mode != VarIntMode::DEFAULT || std::is_unsigned<I>::value,
475 "Unsigned type required with mode DEFAULT.");
476 static_assert(
477 Mode != VarIntMode::NONNEGATIVE_SIGNED || std::is_signed<I>::value,
478 "Signed type required with mode NONNEGATIVE_SIGNED.");
479 }
480};
481
482template <VarIntMode Mode, typename I>
483inline unsigned int GetSizeOfVarInt(I n) {
485 int nRet = 0;
486 while (true) {
487 nRet++;
488 if (n <= 0x7F) break;
489 n = (n >> 7) - 1;
490 }
491 return nRet;
492}
493
494template <typename I>
495inline void WriteVarInt(CSizeComputer& os, I n);
496
497template <typename Stream, VarIntMode Mode, typename I>
498void WriteVarInt(Stream& os, I n) {
499 CheckVarIntMode<Mode, I>();
500 unsigned char tmp[(sizeof(n) * 8 + 6) / 7];
501 int len = 0;
502 while (true) {
503 tmp[len] = (n & 0x7F) | (len ? 0x80 : 0x00);
504 if (n <= 0x7F) break;
505 n = (n >> 7) - 1;
506 len++;
507 }
508 do {
509 ser_writedata8(os, tmp[len]);
510 } while (len--);
511}
512
513template <typename Stream, VarIntMode Mode, typename I>
514I ReadVarInt(Stream& is) {
515 CheckVarIntMode<Mode, I>();
516 I n = 0;
517 while (true) {
518 unsigned char chData = ser_readdata8(is);
519 if (n > (std::numeric_limits<I>::max() >> 7)) {
520 throw std::ios_base::failure("ReadVarInt(): size too large");
521 }
522 n = (n << 7) | (chData & 0x7F);
523 if (chData & 0x80) {
524 if (n == std::numeric_limits<I>::max()) {
525 throw std::ios_base::failure("ReadVarInt(): size too large");
526 }
527 n++;
528 } else {
529 return n;
530 }
531 }
532}
533
534#define VARINT(obj, ...) WrapVarInt<__VA_ARGS__>(REF(obj))
535#define COMPACTSIZE(obj) CCompactSize(REF(obj))
536#define LIMITED_STRING(obj, n) LimitedString<n>(REF(obj))
537
538template <VarIntMode Mode, typename I>
539class CVarInt {
540 protected:
541 I& n;
542
543 public:
544 explicit CVarInt(I& nIn) : n(nIn) {}
545
546 template <typename Stream>
547 void Serialize(Stream& s) const {
548 WriteVarInt<Stream, Mode, I>(s, n);
549 }
550
551 template <typename Stream>
552 void Unserialize(Stream& s) {
553 n = ReadVarInt<Stream, Mode, I>(s);
554 }
555};
556
566template <typename I>
568 protected:
569 I& m_val;
570
571 public:
572 explicit BigEndian(I& val) : m_val(val) {
573 static_assert(std::is_unsigned<I>::value,
574 "BigEndian type must be unsigned integer");
575 static_assert(sizeof(I) == 2 && std::numeric_limits<I>::min() == 0 &&
576 std::numeric_limits<I>::max() ==
577 std::numeric_limits<uint16_t>::max(),
578 "Unsupported BigEndian size");
579 }
580
581 template <typename Stream>
582 void Serialize(Stream& s) const {
583 ser_writedata16be(s, m_val);
584 }
585
586 template <typename Stream>
587 void Unserialize(Stream& s) {
588 m_val = ser_readdata16be(s);
589 }
590};
591
593 protected:
594 uint64_t& n;
595
596 public:
597 explicit CCompactSize(uint64_t& nIn) : n(nIn) {}
598
599 template <typename Stream>
600 void Serialize(Stream& s) const {
601 WriteCompactSize<Stream>(s, n);
602 }
603
604 template <typename Stream>
605 void Unserialize(Stream& s) {
606 n = ReadCompactSize<Stream>(s);
607 }
608};
609
610template <size_t Limit>
612 protected:
613 std::string& string;
614
615 public:
616 explicit LimitedString(std::string& _string) : string(_string) {}
617
618 template <typename Stream>
619 void Unserialize(Stream& s) {
620 size_t size = ReadCompactSize(s);
621 if (size > Limit) {
622 throw std::ios_base::failure("String length limit exceeded");
623 }
624 string.resize(size);
626 if (size != 0) {
627 if (!s.read(size, (uint8_t*)string.data(), state)) {
628 throw std::ios_base::failure(state.toString());
629 }
630 }
631 }
632
633 template <typename Stream>
634 void Serialize(Stream& s) const {
635 WriteCompactSize(s, string.size());
636 if (!string.empty()) s.write((char*)string.data(), string.size());
637 }
638};
639
640template <VarIntMode Mode = VarIntMode::DEFAULT, typename I>
641CVarInt<Mode, I> WrapVarInt(I& n) {
642 return CVarInt<Mode, I>{n};
643}
644
645template <typename I>
646BigEndian<I> WrapBigEndian(I& n) {
647 return BigEndian<I>(n);
648}
649
657template <typename Stream, typename C>
658void Serialize(Stream& os, const std::basic_string<C>& str);
659template <typename Stream, typename C>
660void Unserialize(Stream& is, std::basic_string<C>& str);
661
667template <typename Stream, typename T, typename A>
668void Serialize_impl(Stream& os,
669 const std::vector<T, A>& v,
670 const unsigned char&);
671template <typename Stream, typename T, typename A>
672void Serialize_impl(Stream& os, const std::vector<T, A>& v, const bool&);
673template <typename Stream, typename T, typename A, typename V>
674void Serialize_impl(Stream& os, const std::vector<T, A>& v, const V&);
675template <typename Stream, typename T, typename A>
676inline void Serialize(Stream& os, const std::vector<T, A>& v);
677template <typename Stream, typename T, typename A>
678void Unserialize_impl(Stream& is, std::vector<T, A>& v, const unsigned char&);
679template <typename Stream, typename T, typename A, typename V>
680void Unserialize_impl(Stream& is, std::vector<T, A>& v, const V&);
681template <typename Stream, typename T, typename A>
682inline void Unserialize(Stream& is, std::vector<T, A>& v);
683
687template <typename Stream, typename K, typename T>
688void Serialize(Stream& os, const std::pair<K, T>& item);
689template <typename Stream, typename K, typename T>
690void Unserialize(Stream& is, std::pair<K, T>& item);
691
695template <typename Stream, typename K, typename T, typename Pred, typename A>
696void Serialize(Stream& os, const std::map<K, T, Pred, A>& m);
697template <typename Stream, typename K, typename T, typename Pred, typename A>
698void Unserialize(Stream& is, std::map<K, T, Pred, A>& m);
699
703template <typename Stream, typename K, typename Pred, typename A>
704void Serialize(Stream& os, const std::set<K, Pred, A>& m);
705template <typename Stream, typename K, typename Pred, typename A>
706void Unserialize(Stream& is, std::set<K, Pred, A>& m);
707
711template <typename Stream, typename T>
712void Serialize(Stream& os, const std::shared_ptr<const T>& p);
713template <typename Stream, typename T>
714void Unserialize(Stream& os, std::shared_ptr<const T>& p);
715
719template <typename Stream, typename T>
720void Serialize(Stream& os, const std::unique_ptr<const T>& p);
721template <typename Stream, typename T>
722void Unserialize(Stream& os, std::unique_ptr<const T>& p);
723
728template <typename Stream, typename T>
729inline void Serialize(Stream& os, const T& a) {
730 a.Serialize(os);
731}
732
733template <typename Stream, typename T>
734inline void Unserialize(Stream& is, T&& a) {
735 a.Unserialize(is);
736}
737
741template <typename Stream, typename C>
742void Serialize(Stream& os, const std::basic_string<C>& str) {
743 WriteCompactSize(os, str.size());
744 if (!str.empty()) os.write((char*)str.data(), str.size() * sizeof(C));
745}
746
747template <typename Stream, typename C>
748void Unserialize(Stream& is, std::basic_string<C>& str) {
749 uint32_t nSize = (uint32_t)ReadCompactSize(is);
750 str.resize(nSize);
752 if (nSize != 0) {
753 if (!is.read(nSize * sizeof(C), (uint8_t*)str.data(), state)) {
754 throw std::ios_base::failure(state.toString());
755 }
756 }
757}
758
762template <typename Stream, typename T, typename A>
763void Serialize_impl(Stream& os,
764 const std::vector<T, A>& v,
765 const unsigned char&) {
766 WriteCompactSize(os, v.size());
767 if (!v.empty()) os.write((char*)v.data(), v.size() * sizeof(T));
768}
769
770template <typename Stream, typename T, typename A>
771void Serialize_impl(Stream& os, const std::vector<T, A>& v, const bool&) {
772 // A special case for std::vector<bool>, as dereferencing
773 // std::vector<bool>::const_iterator does not result in a const bool&
774 // due to std::vector's special casing for bool arguments.
775 WriteCompactSize(os, v.size());
776 for (bool elem : v) {
777 Serialize(os, elem);
778 }
779}
780
781template <typename Stream, typename T, typename A, typename V>
782void Serialize_impl(Stream& os, const std::vector<T, A>& v, const V&) {
783 WriteCompactSize(os, v.size());
784 for (typename std::vector<T, A>::const_iterator vi = v.begin(); vi != v.end();
785 ++vi) {
786 Serialize(os, (*vi));
787 }
788}
789
790template <typename Stream, typename T, typename A>
791inline void Serialize(Stream& os, const std::vector<T, A>& v) {
792 Serialize_impl(os, v, T());
793}
794
795template <typename Stream, typename T, typename A>
796void Unserialize_impl(Stream& is, std::vector<T, A>& v, const unsigned char&) {
797 // Limit size per read so bogus size value won't cause out of memory
798 v.clear();
799 uint64_t nSize = ReadCompactSize(is);
800 uint64_t i = 0;
802 while (i < nSize) {
803 uint64_t blk = std::min(nSize - i, (uint64_t)(1 + 4999999 / sizeof(T)));
804 v.resize(i + blk);
805 if (!is.read(blk * sizeof(T), (uint8_t*)&v[i], state)) {
806 throw std::ios_base::failure(state.toString());
807 }
808 i += blk;
809 }
810}
811
812template <typename Stream, typename T, typename A, typename V>
813void Unserialize_impl(Stream& is, std::vector<T, A>& v, const V&) {
814 v.clear();
815 uint64_t nSize = ReadCompactSize(is);
816 uint64_t i = 0;
817 uint64_t nMid = 0;
818 while (nMid < nSize) {
819 nMid += 5000000 / sizeof(T);
820 if (nMid > nSize) nMid = nSize;
821 v.resize(nMid);
822 for (; i < nMid; i++) Unserialize(is, v[i]);
823 }
824}
825
826template <typename Stream, typename T, typename A>
827inline void Unserialize(Stream& is, std::vector<T, A>& v) {
828 Unserialize_impl(is, v, T());
829}
830
834template <typename Stream, typename K, typename T>
835void Serialize(Stream& os, const std::pair<K, T>& item) {
836 Serialize(os, item.first);
837 Serialize(os, item.second);
838}
839
840template <typename Stream, typename K, typename T>
841void Unserialize(Stream& is, std::pair<K, T>& item) {
842 Unserialize(is, item.first);
843 Unserialize(is, item.second);
844}
845
849template <typename Stream, typename K, typename T, typename Pred, typename A>
850void Serialize(Stream& os, const std::map<K, T, Pred, A>& m) {
851 WriteCompactSize(os, m.size());
852 for (const auto& entry : m) Serialize(os, entry);
853}
854
855template <typename Stream, typename K, typename T, typename Pred, typename A>
856void Unserialize(Stream& is, std::map<K, T, Pred, A>& m) {
857 m.clear();
858 uint64_t nSize = ReadCompactSize(is);
859 typename std::map<K, T, Pred, A>::iterator mi = m.begin();
860 for (uint64_t i = 0; i < nSize; i++) {
861 std::pair<K, T> item;
862 Unserialize(is, item);
863 mi = m.insert(mi, item);
864 }
865}
866
870template <typename Stream, typename K, typename Pred, typename A>
871void Serialize(Stream& os, const std::set<K, Pred, A>& m) {
872 WriteCompactSize(os, m.size());
873 for (typename std::set<K, Pred, A>::const_iterator it = m.begin();
874 it != m.end();
875 ++it)
876 Serialize(os, (*it));
877}
878
879template <typename Stream, typename K, typename Pred, typename A>
880void Unserialize(Stream& is, std::set<K, Pred, A>& m) {
881 m.clear();
882 uint64_t nSize = ReadCompactSize(is);
883 typename std::set<K, Pred, A>::iterator it = m.begin();
884 for (uint64_t i = 0; i < nSize; i++) {
885 K key;
886 Unserialize(is, key);
887 it = m.insert(it, key);
888 }
889}
890
894template <typename Stream, typename T>
895void Serialize(Stream& os, const std::unique_ptr<const T>& p) {
896 Serialize(os, *p);
897}
898
899template <typename Stream, typename T>
900void Unserialize(Stream& is, std::unique_ptr<const T>& p) {
901 p.reset(new T(deserialize, is));
902}
903
907template <typename Stream, typename T>
908void Serialize(Stream& os, const std::shared_ptr<const T>& p) {
909 Serialize(os, *p);
910}
911
912template <typename Stream, typename T>
913void Unserialize(Stream& is, std::shared_ptr<const T>& p) {
914 p = std::make_shared<const T>(deserialize, is);
915}
916
921template <typename T>
922void UnserializeOrThrow(const std::vector<uint8_t>& in, T& out) {
923 ValidationState state;
924 ReadStream stream(in);
925 if (!DeserializeFromVbkEncoding(stream, out, state)) {
926 throw std::ios_base::failure(state.toString());
927 }
928}
929
930template <typename Stream>
931inline void Serialize(Stream& s, const PopData& popData) {
932 std::vector<uint8_t> bytes_data = popData.toVbkEncoding();
933 Serialize(s, bytes_data);
934}
935template <typename Stream>
936inline void Unserialize(Stream& s, PopData& popData) {
937 std::vector<uint8_t> bytes_data;
938 Unserialize(s, bytes_data);
939 UnserializeOrThrow(bytes_data, popData);
940}
941
946 constexpr bool ForRead() const { return false; }
947};
949 constexpr bool ForRead() const { return true; }
950};
951
952/* GetSerializeSize implementations
953 *
954 * Computing the serialized size of objects is done through a special stream
955 * object of type CSizeComputer, which only records the number of bytes written
956 * to it.
957 *
958 * If your Serialize or SerializationOp method has non-trivial overhead for
959 * serialization, it may be worthwhile to implement a specialized version for
960 * CSizeComputer, which uses the s.seek() method to record bytes that would
961 * be written instead.
962 */
964 protected:
965 size_t nSize;
966
967 const int nVersion;
968
969 public:
970 explicit CSizeComputer(int nVersionIn) : nSize(0), nVersion(nVersionIn) {}
971
972 void write(const char*, size_t _nSize) { this->nSize += _nSize; }
973
974 template <
975 typename T,
976 typename = typename std::enable_if<std::is_integral<T>::value>::type>
977 void writeLE(T) {
978 this->nSize += sizeof(T);
979 }
980
981 template <
982 typename T,
983 typename = typename std::enable_if<std::is_integral<T>::value>::type>
984 void writeBE(T, size_t) {
985 this->nSize += sizeof(T);
986 }
987
989 void seek(size_t _nSize) { this->nSize += _nSize; }
990
991 template <typename T>
992 CSizeComputer& operator<<(const T& obj) {
993 Serialize(*this, obj);
994 return (*this);
995 }
996
997 size_t size() const { return nSize; }
998
999 int GetVersion() const { return nVersion; }
1000};
1001
1002template <typename Stream>
1003void SerializeMany(Stream&) {}
1004
1005template <typename Stream, typename Arg, typename... Args>
1006void SerializeMany(Stream& s, const Arg& arg, const Args&... args) {
1007 Serialize(s, arg);
1008 SerializeMany(s, args...);
1009}
1010
1011template <typename Stream>
1012inline void UnserializeMany(Stream&) {}
1013
1014template <typename Stream, typename Arg, typename... Args>
1015inline void UnserializeMany(Stream& s, Arg&& arg, Args&&... args) {
1016 Unserialize(s, arg);
1017 UnserializeMany(s, args...);
1018}
1019
1020template <typename Stream, typename... Args>
1021inline void SerReadWriteMany(Stream& s,
1022 CSerActionSerialize,
1023 const Args&... args) {
1024 SerializeMany(s, args...);
1025}
1026
1027template <typename Stream, typename... Args>
1028inline void SerReadWriteMany(Stream& s, CSerActionUnserialize, Args&&... args) {
1029 UnserializeMany(s, args...);
1030}
1031
1032template <typename I>
1033inline void WriteVarInt(CSizeComputer& s, I n) {
1034 s.seek(GetSizeOfVarInt<I>(n));
1035}
1036
1037inline void WriteCompactSize(CSizeComputer& s, uint64_t nSize) {
1038 s.seek(GetSizeOfCompactSize(nSize));
1039}
1040
1041template <typename T>
1042size_t GetSerializeSize(const T& t, int nVersion = 0) {
1043 return (CSizeComputer(nVersion) << t).size();
1044}
1045
1046template <typename... T>
1047size_t GetSerializeSizeMany(int nVersion, const T&... t) {
1048 CSizeComputer sc(nVersion);
1049 SerializeMany(sc, t...);
1050 return sc.size();
1051}
1052
1053} // namespace btc
1054
1055} // namespace altintegration
1056
1057#endif // BITCOIN_SERIALIZE_H
Class that is used for storing validation state.
Serialization wrapper class for big-endian integers.
Definition: serialize.hpp:567
void seek(size_t _nSize)
Pretend _nSize bytes are written, without specifying them.
Definition: serialize.hpp:989
Defines logging helpers.
Definition: block.hpp:14
bool DeserializeFromVbkEncoding(ReadStream &stream, AltBlockAddon &out, ValidationState &state)
This is an overloaded member function, provided for convenience. It differs from the above function o...
Contiguous byte array of fixed size.
Definition: blob.hpp:25
Non-owning contiguous array.
Definition: slice.hpp:22
Support for ADD_SERIALIZE_METHODS and READWRITE macro.
Definition: serialize.hpp:945
Dummy data type to identify deserializing constructors.
Definition: serialize.hpp:44