1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2016 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
6 #ifndef BITCOIN_SERIALIZE_H
7 #define BITCOIN_SERIALIZE_H
9 #include "compat/endian.h"
24 #include "prevector.h"
26 static const unsigned int MAX_SIZE
= 0x02000000;
29 * Dummy data type to identify deserializing constructors.
31 * By convention, a constructor of a type T with signature
33 * template <typename Stream> T::T(deserialize_type, Stream& s)
35 * is a deserializing constructor, which builds the type by
36 * deserializing it from s. If T contains const fields, this
37 * is likely the only way to do so.
39 struct deserialize_type
{};
40 constexpr deserialize_type deserialize
{};
43 * Used to bypass the rule against non-const reference to temporary
44 * where it makes sense with wrappers such as CFlatData or CTxDB
47 inline T
& REF(const T
& val
)
49 return const_cast<T
&>(val
);
53 * Used to acquire a non-const pointer "this" to generate bodies
54 * of const serialization operations from a template
57 inline T
* NCONST_PTR(const T
* val
)
59 return const_cast<T
*>(val
);
63 * Lowest-level serialization and conversion.
64 * @note Sizes of these types are verified in the tests
66 template<typename Stream
> inline void ser_writedata8(Stream
&s
, uint8_t obj
)
68 s
.write((char*)&obj
, 1);
70 template<typename Stream
> inline void ser_writedata16(Stream
&s
, uint16_t obj
)
73 s
.write((char*)&obj
, 2);
75 template<typename Stream
> inline void ser_writedata32(Stream
&s
, uint32_t obj
)
78 s
.write((char*)&obj
, 4);
80 template<typename Stream
> inline void ser_writedata64(Stream
&s
, uint64_t obj
)
83 s
.write((char*)&obj
, 8);
85 template<typename Stream
> inline uint8_t ser_readdata8(Stream
&s
)
88 s
.read((char*)&obj
, 1);
91 template<typename Stream
> inline uint16_t ser_readdata16(Stream
&s
)
94 s
.read((char*)&obj
, 2);
97 template<typename Stream
> inline uint32_t ser_readdata32(Stream
&s
)
100 s
.read((char*)&obj
, 4);
103 template<typename Stream
> inline uint64_t ser_readdata64(Stream
&s
)
106 s
.read((char*)&obj
, 8);
109 inline uint64_t ser_double_to_uint64(double x
)
111 union { double x
; uint64_t y
; } tmp
;
115 inline uint32_t ser_float_to_uint32(float x
)
117 union { float x
; uint32_t y
; } tmp
;
121 inline double ser_uint64_to_double(uint64_t y
)
123 union { double x
; uint64_t y
; } tmp
;
127 inline float ser_uint32_to_float(uint32_t y
)
129 union { float x
; uint32_t y
; } tmp
;
135 /////////////////////////////////////////////////////////////////
137 // Templates for serializing to anything that looks like a stream,
138 // i.e. anything that supports .read(char*, size_t) and .write(char*, size_t)
146 SER_NETWORK
= (1 << 0),
148 SER_GETHASH
= (1 << 2),
151 #define READWRITE(obj) (::SerReadWrite(s, (obj), ser_action))
152 #define READWRITEMANY(...) (::SerReadWriteMany(s, ser_action, __VA_ARGS__))
155 * Implement three methods for serializable objects. These are actually wrappers over
156 * "SerializationOp" template, which implements the body of each class' serialization
157 * code. Adding "ADD_SERIALIZE_METHODS" in the body of the class causes these wrappers to be
160 #define ADD_SERIALIZE_METHODS \
161 template<typename Stream> \
162 void Serialize(Stream& s) const { \
163 NCONST_PTR(this)->SerializationOp(s, CSerActionSerialize()); \
165 template<typename Stream> \
166 void Unserialize(Stream& s) { \
167 SerializationOp(s, CSerActionUnserialize()); \
170 template<typename Stream
> inline void Serialize(Stream
& s
, char a
) { ser_writedata8(s
, a
); } // TODO Get rid of bare char
171 template<typename Stream
> inline void Serialize(Stream
& s
, int8_t a
) { ser_writedata8(s
, a
); }
172 template<typename Stream
> inline void Serialize(Stream
& s
, uint8_t a
) { ser_writedata8(s
, a
); }
173 template<typename Stream
> inline void Serialize(Stream
& s
, int16_t a
) { ser_writedata16(s
, a
); }
174 template<typename Stream
> inline void Serialize(Stream
& s
, uint16_t a
) { ser_writedata16(s
, a
); }
175 template<typename Stream
> inline void Serialize(Stream
& s
, int32_t a
) { ser_writedata32(s
, a
); }
176 template<typename Stream
> inline void Serialize(Stream
& s
, uint32_t a
) { ser_writedata32(s
, a
); }
177 template<typename Stream
> inline void Serialize(Stream
& s
, int64_t a
) { ser_writedata64(s
, a
); }
178 template<typename Stream
> inline void Serialize(Stream
& s
, uint64_t a
) { ser_writedata64(s
, a
); }
179 template<typename Stream
> inline void Serialize(Stream
& s
, float a
) { ser_writedata32(s
, ser_float_to_uint32(a
)); }
180 template<typename Stream
> inline void Serialize(Stream
& s
, double a
) { ser_writedata64(s
, ser_double_to_uint64(a
)); }
182 template<typename Stream
> inline void Unserialize(Stream
& s
, char& a
) { a
= ser_readdata8(s
); } // TODO Get rid of bare char
183 template<typename Stream
> inline void Unserialize(Stream
& s
, int8_t& a
) { a
= ser_readdata8(s
); }
184 template<typename Stream
> inline void Unserialize(Stream
& s
, uint8_t& a
) { a
= ser_readdata8(s
); }
185 template<typename Stream
> inline void Unserialize(Stream
& s
, int16_t& a
) { a
= ser_readdata16(s
); }
186 template<typename Stream
> inline void Unserialize(Stream
& s
, uint16_t& a
) { a
= ser_readdata16(s
); }
187 template<typename Stream
> inline void Unserialize(Stream
& s
, int32_t& a
) { a
= ser_readdata32(s
); }
188 template<typename Stream
> inline void Unserialize(Stream
& s
, uint32_t& a
) { a
= ser_readdata32(s
); }
189 template<typename Stream
> inline void Unserialize(Stream
& s
, int64_t& a
) { a
= ser_readdata64(s
); }
190 template<typename Stream
> inline void Unserialize(Stream
& s
, uint64_t& a
) { a
= ser_readdata64(s
); }
191 template<typename Stream
> inline void Unserialize(Stream
& s
, float& a
) { a
= ser_uint32_to_float(ser_readdata32(s
)); }
192 template<typename Stream
> inline void Unserialize(Stream
& s
, double& a
) { a
= ser_uint64_to_double(ser_readdata64(s
)); }
194 template<typename Stream
> inline void Serialize(Stream
& s
, bool a
) { char f
=a
; ser_writedata8(s
, f
); }
195 template<typename Stream
> inline void Unserialize(Stream
& s
, bool& a
) { char f
=ser_readdata8(s
); a
=f
; }
204 * size < 253 -- 1 byte
205 * size <= USHRT_MAX -- 3 bytes (253 + 2 bytes)
206 * size <= UINT_MAX -- 5 bytes (254 + 4 bytes)
207 * size > UINT_MAX -- 9 bytes (255 + 8 bytes)
209 inline unsigned int GetSizeOfCompactSize(uint64_t nSize
)
211 if (nSize
< 253) return sizeof(unsigned char);
212 else if (nSize
<= std::numeric_limits
<unsigned short>::max()) return sizeof(unsigned char) + sizeof(unsigned short);
213 else if (nSize
<= std::numeric_limits
<unsigned int>::max()) return sizeof(unsigned char) + sizeof(unsigned int);
214 else return sizeof(unsigned char) + sizeof(uint64_t);
217 inline void WriteCompactSize(CSizeComputer
& os
, uint64_t nSize
);
219 template<typename Stream
>
220 void WriteCompactSize(Stream
& os
, uint64_t nSize
)
224 ser_writedata8(os
, nSize
);
226 else if (nSize
<= std::numeric_limits
<unsigned short>::max())
228 ser_writedata8(os
, 253);
229 ser_writedata16(os
, nSize
);
231 else if (nSize
<= std::numeric_limits
<unsigned int>::max())
233 ser_writedata8(os
, 254);
234 ser_writedata32(os
, nSize
);
238 ser_writedata8(os
, 255);
239 ser_writedata64(os
, nSize
);
244 template<typename Stream
>
245 uint64_t ReadCompactSize(Stream
& is
)
247 uint8_t chSize
= ser_readdata8(is
);
248 uint64_t nSizeRet
= 0;
253 else if (chSize
== 253)
255 nSizeRet
= ser_readdata16(is
);
257 throw std::ios_base::failure("non-canonical ReadCompactSize()");
259 else if (chSize
== 254)
261 nSizeRet
= ser_readdata32(is
);
262 if (nSizeRet
< 0x10000u
)
263 throw std::ios_base::failure("non-canonical ReadCompactSize()");
267 nSizeRet
= ser_readdata64(is
);
268 if (nSizeRet
< 0x100000000ULL
)
269 throw std::ios_base::failure("non-canonical ReadCompactSize()");
271 if (nSizeRet
> (uint64_t)MAX_SIZE
)
272 throw std::ios_base::failure("ReadCompactSize(): size too large");
277 * Variable-length integers: bytes are a MSB base-128 encoding of the number.
278 * The high bit in each byte signifies whether another digit follows. To make
279 * sure the encoding is one-to-one, one is subtracted from all but the last digit.
280 * Thus, the byte sequence a[] with length len, where all but the last byte
281 * has bit 128 set, encodes the number:
283 * (a[len-1] & 0x7F) + sum(i=1..len-1, 128^i*((a[len-i-1] & 0x7F)+1))
286 * * Very small (0-127: 1 byte, 128-16511: 2 bytes, 16512-2113663: 3 bytes)
287 * * Every integer has exactly one encoding
288 * * Encoding does not depend on size of original integer type
289 * * No redundancy: every (infinite) byte sequence corresponds to a list
290 * of encoded integers.
292 * 0: [0x00] 256: [0x81 0x00]
293 * 1: [0x01] 16383: [0xFE 0x7F]
294 * 127: [0x7F] 16384: [0xFF 0x00]
295 * 128: [0x80 0x00] 16511: [0xFF 0x7F]
296 * 255: [0x80 0x7F] 65535: [0x82 0xFE 0x7F]
297 * 2^32: [0x8E 0xFE 0xFE 0xFF 0x00]
301 inline unsigned int GetSizeOfVarInt(I n
)
314 inline void WriteVarInt(CSizeComputer
& os
, I n
);
316 template<typename Stream
, typename I
>
317 void WriteVarInt(Stream
& os
, I n
)
319 unsigned char tmp
[(sizeof(n
)*8+6)/7];
322 tmp
[len
] = (n
& 0x7F) | (len
? 0x80 : 0x00);
329 ser_writedata8(os
, tmp
[len
]);
333 template<typename Stream
, typename I
>
334 I
ReadVarInt(Stream
& is
)
338 unsigned char chData
= ser_readdata8(is
);
339 if (n
> (std::numeric_limits
<I
>::max() >> 7)) {
340 throw std::ios_base::failure("ReadVarInt(): size too large");
342 n
= (n
<< 7) | (chData
& 0x7F);
344 if (n
== std::numeric_limits
<I
>::max()) {
345 throw std::ios_base::failure("ReadVarInt(): size too large");
354 #define FLATDATA(obj) REF(CFlatData((char*)&(obj), (char*)&(obj) + sizeof(obj)))
355 #define VARINT(obj) REF(WrapVarInt(REF(obj)))
356 #define COMPACTSIZE(obj) REF(CCompactSize(REF(obj)))
357 #define LIMITED_STRING(obj,n) REF(LimitedString< n >(REF(obj)))
360 * Wrapper for serializing arrays and POD.
368 CFlatData(void* pbeginIn
, void* pendIn
) : pbegin((char*)pbeginIn
), pend((char*)pendIn
) { }
369 template <class T
, class TAl
>
370 explicit CFlatData(std::vector
<T
,TAl
> &v
)
372 pbegin
= (char*)v
.data();
373 pend
= (char*)(v
.data() + v
.size());
375 template <unsigned int N
, typename T
, typename S
, typename D
>
376 explicit CFlatData(prevector
<N
, T
, S
, D
> &v
)
378 pbegin
= (char*)v
.data();
379 pend
= (char*)(v
.data() + v
.size());
381 char* begin() { return pbegin
; }
382 const char* begin() const { return pbegin
; }
383 char* end() { return pend
; }
384 const char* end() const { return pend
; }
386 template<typename Stream
>
387 void Serialize(Stream
& s
) const
389 s
.write(pbegin
, pend
- pbegin
);
392 template<typename Stream
>
393 void Unserialize(Stream
& s
)
395 s
.read(pbegin
, pend
- pbegin
);
405 explicit CVarInt(I
& nIn
) : n(nIn
) { }
407 template<typename Stream
>
408 void Serialize(Stream
&s
) const {
409 WriteVarInt
<Stream
,I
>(s
, n
);
412 template<typename Stream
>
413 void Unserialize(Stream
& s
) {
414 n
= ReadVarInt
<Stream
,I
>(s
);
423 explicit CCompactSize(uint64_t& nIn
) : n(nIn
) { }
425 template<typename Stream
>
426 void Serialize(Stream
&s
) const {
427 WriteCompactSize
<Stream
>(s
, n
);
430 template<typename Stream
>
431 void Unserialize(Stream
& s
) {
432 n
= ReadCompactSize
<Stream
>(s
);
436 template<size_t Limit
>
442 explicit LimitedString(std::string
& _string
) : string(_string
) {}
444 template<typename Stream
>
445 void Unserialize(Stream
& s
)
447 size_t size
= ReadCompactSize(s
);
449 throw std::ios_base::failure("String length limit exceeded");
453 s
.read((char*)string
.data(), size
);
456 template<typename Stream
>
457 void Serialize(Stream
& s
) const
459 WriteCompactSize(s
, string
.size());
461 s
.write((char*)string
.data(), string
.size());
466 CVarInt
<I
> WrapVarInt(I
& n
) { return CVarInt
<I
>(n
); }
469 * Forward declarations
475 template<typename Stream
, typename C
> void Serialize(Stream
& os
, const std::basic_string
<C
>& str
);
476 template<typename Stream
, typename C
> void Unserialize(Stream
& is
, std::basic_string
<C
>& str
);
480 * prevectors of unsigned char are a special case and are intended to be serialized as a single opaque blob.
482 template<typename Stream
, unsigned int N
, typename T
> void Serialize_impl(Stream
& os
, const prevector
<N
, T
>& v
, const unsigned char&);
483 template<typename Stream
, unsigned int N
, typename T
, typename V
> void Serialize_impl(Stream
& os
, const prevector
<N
, T
>& v
, const V
&);
484 template<typename Stream
, unsigned int N
, typename T
> inline void Serialize(Stream
& os
, const prevector
<N
, T
>& v
);
485 template<typename Stream
, unsigned int N
, typename T
> void Unserialize_impl(Stream
& is
, prevector
<N
, T
>& v
, const unsigned char&);
486 template<typename Stream
, unsigned int N
, typename T
, typename V
> void Unserialize_impl(Stream
& is
, prevector
<N
, T
>& v
, const V
&);
487 template<typename Stream
, unsigned int N
, typename T
> inline void Unserialize(Stream
& is
, prevector
<N
, T
>& v
);
491 * vectors of unsigned char are a special case and are intended to be serialized as a single opaque blob.
493 template<typename Stream
, typename T
, typename A
> void Serialize_impl(Stream
& os
, const std::vector
<T
, A
>& v
, const unsigned char&);
494 template<typename Stream
, typename T
, typename A
, typename V
> void Serialize_impl(Stream
& os
, const std::vector
<T
, A
>& v
, const V
&);
495 template<typename Stream
, typename T
, typename A
> inline void Serialize(Stream
& os
, const std::vector
<T
, A
>& v
);
496 template<typename Stream
, typename T
, typename A
> void Unserialize_impl(Stream
& is
, std::vector
<T
, A
>& v
, const unsigned char&);
497 template<typename Stream
, typename T
, typename A
, typename V
> void Unserialize_impl(Stream
& is
, std::vector
<T
, A
>& v
, const V
&);
498 template<typename Stream
, typename T
, typename A
> inline void Unserialize(Stream
& is
, std::vector
<T
, A
>& v
);
503 template<typename Stream
, typename K
, typename T
> void Serialize(Stream
& os
, const std::pair
<K
, T
>& item
);
504 template<typename Stream
, typename K
, typename T
> void Unserialize(Stream
& is
, std::pair
<K
, T
>& item
);
509 template<typename Stream
, typename K
, typename T
, typename Pred
, typename A
> void Serialize(Stream
& os
, const std::map
<K
, T
, Pred
, A
>& m
);
510 template<typename Stream
, typename K
, typename T
, typename Pred
, typename A
> void Unserialize(Stream
& is
, std::map
<K
, T
, Pred
, A
>& m
);
515 template<typename Stream
, typename K
, typename Pred
, typename A
> void Serialize(Stream
& os
, const std::set
<K
, Pred
, A
>& m
);
516 template<typename Stream
, typename K
, typename Pred
, typename A
> void Unserialize(Stream
& is
, std::set
<K
, Pred
, A
>& m
);
521 template<typename Stream
, typename T
> void Serialize(Stream
& os
, const std::shared_ptr
<const T
>& p
);
522 template<typename Stream
, typename T
> void Unserialize(Stream
& os
, std::shared_ptr
<const T
>& p
);
527 template<typename Stream
, typename T
> void Serialize(Stream
& os
, const std::unique_ptr
<const T
>& p
);
528 template<typename Stream
, typename T
> void Unserialize(Stream
& os
, std::unique_ptr
<const T
>& p
);
533 * If none of the specialized versions above matched, default to calling member function.
535 template<typename Stream
, typename T
>
536 inline void Serialize(Stream
& os
, const T
& a
)
541 template<typename Stream
, typename T
>
542 inline void Unserialize(Stream
& is
, T
& a
)
554 template<typename Stream
, typename C
>
555 void Serialize(Stream
& os
, const std::basic_string
<C
>& str
)
557 WriteCompactSize(os
, str
.size());
559 os
.write((char*)str
.data(), str
.size() * sizeof(C
));
562 template<typename Stream
, typename C
>
563 void Unserialize(Stream
& is
, std::basic_string
<C
>& str
)
565 unsigned int nSize
= ReadCompactSize(is
);
568 is
.read((char*)str
.data(), nSize
* sizeof(C
));
576 template<typename Stream
, unsigned int N
, typename T
>
577 void Serialize_impl(Stream
& os
, const prevector
<N
, T
>& v
, const unsigned char&)
579 WriteCompactSize(os
, v
.size());
581 os
.write((char*)v
.data(), v
.size() * sizeof(T
));
584 template<typename Stream
, unsigned int N
, typename T
, typename V
>
585 void Serialize_impl(Stream
& os
, const prevector
<N
, T
>& v
, const V
&)
587 WriteCompactSize(os
, v
.size());
588 for (typename prevector
<N
, T
>::const_iterator vi
= v
.begin(); vi
!= v
.end(); ++vi
)
589 ::Serialize(os
, (*vi
));
592 template<typename Stream
, unsigned int N
, typename T
>
593 inline void Serialize(Stream
& os
, const prevector
<N
, T
>& v
)
595 Serialize_impl(os
, v
, T());
599 template<typename Stream
, unsigned int N
, typename T
>
600 void Unserialize_impl(Stream
& is
, prevector
<N
, T
>& v
, const unsigned char&)
602 // Limit size per read so bogus size value won't cause out of memory
604 unsigned int nSize
= ReadCompactSize(is
);
608 unsigned int blk
= std::min(nSize
- i
, (unsigned int)(1 + 4999999 / sizeof(T
)));
610 is
.read((char*)&v
[i
], blk
* sizeof(T
));
615 template<typename Stream
, unsigned int N
, typename T
, typename V
>
616 void Unserialize_impl(Stream
& is
, prevector
<N
, T
>& v
, const V
&)
619 unsigned int nSize
= ReadCompactSize(is
);
621 unsigned int nMid
= 0;
624 nMid
+= 5000000 / sizeof(T
);
628 for (; i
< nMid
; i
++)
629 Unserialize(is
, v
[i
]);
633 template<typename Stream
, unsigned int N
, typename T
>
634 inline void Unserialize(Stream
& is
, prevector
<N
, T
>& v
)
636 Unserialize_impl(is
, v
, T());
644 template<typename Stream
, typename T
, typename A
>
645 void Serialize_impl(Stream
& os
, const std::vector
<T
, A
>& v
, const unsigned char&)
647 WriteCompactSize(os
, v
.size());
649 os
.write((char*)v
.data(), v
.size() * sizeof(T
));
652 template<typename Stream
, typename T
, typename A
, typename V
>
653 void Serialize_impl(Stream
& os
, const std::vector
<T
, A
>& v
, const V
&)
655 WriteCompactSize(os
, v
.size());
656 for (typename
std::vector
<T
, A
>::const_iterator vi
= v
.begin(); vi
!= v
.end(); ++vi
)
657 ::Serialize(os
, (*vi
));
660 template<typename Stream
, typename T
, typename A
>
661 inline void Serialize(Stream
& os
, const std::vector
<T
, A
>& v
)
663 Serialize_impl(os
, v
, T());
667 template<typename Stream
, typename T
, typename A
>
668 void Unserialize_impl(Stream
& is
, std::vector
<T
, A
>& v
, const unsigned char&)
670 // Limit size per read so bogus size value won't cause out of memory
672 unsigned int nSize
= ReadCompactSize(is
);
676 unsigned int blk
= std::min(nSize
- i
, (unsigned int)(1 + 4999999 / sizeof(T
)));
678 is
.read((char*)&v
[i
], blk
* sizeof(T
));
683 template<typename Stream
, typename T
, typename A
, typename V
>
684 void Unserialize_impl(Stream
& is
, std::vector
<T
, A
>& v
, const V
&)
687 unsigned int nSize
= ReadCompactSize(is
);
689 unsigned int nMid
= 0;
692 nMid
+= 5000000 / sizeof(T
);
696 for (; i
< nMid
; i
++)
697 Unserialize(is
, v
[i
]);
701 template<typename Stream
, typename T
, typename A
>
702 inline void Unserialize(Stream
& is
, std::vector
<T
, A
>& v
)
704 Unserialize_impl(is
, v
, T());
712 template<typename Stream
, typename K
, typename T
>
713 void Serialize(Stream
& os
, const std::pair
<K
, T
>& item
)
715 Serialize(os
, item
.first
);
716 Serialize(os
, item
.second
);
719 template<typename Stream
, typename K
, typename T
>
720 void Unserialize(Stream
& is
, std::pair
<K
, T
>& item
)
722 Unserialize(is
, item
.first
);
723 Unserialize(is
, item
.second
);
731 template<typename Stream
, typename K
, typename T
, typename Pred
, typename A
>
732 void Serialize(Stream
& os
, const std::map
<K
, T
, Pred
, A
>& m
)
734 WriteCompactSize(os
, m
.size());
735 for (typename
std::map
<K
, T
, Pred
, A
>::const_iterator mi
= m
.begin(); mi
!= m
.end(); ++mi
)
736 Serialize(os
, (*mi
));
739 template<typename Stream
, typename K
, typename T
, typename Pred
, typename A
>
740 void Unserialize(Stream
& is
, std::map
<K
, T
, Pred
, A
>& m
)
743 unsigned int nSize
= ReadCompactSize(is
);
744 typename
std::map
<K
, T
, Pred
, A
>::iterator mi
= m
.begin();
745 for (unsigned int i
= 0; i
< nSize
; i
++)
747 std::pair
<K
, T
> item
;
748 Unserialize(is
, item
);
749 mi
= m
.insert(mi
, item
);
758 template<typename Stream
, typename K
, typename Pred
, typename A
>
759 void Serialize(Stream
& os
, const std::set
<K
, Pred
, A
>& m
)
761 WriteCompactSize(os
, m
.size());
762 for (typename
std::set
<K
, Pred
, A
>::const_iterator it
= m
.begin(); it
!= m
.end(); ++it
)
763 Serialize(os
, (*it
));
766 template<typename Stream
, typename K
, typename Pred
, typename A
>
767 void Unserialize(Stream
& is
, std::set
<K
, Pred
, A
>& m
)
770 unsigned int nSize
= ReadCompactSize(is
);
771 typename
std::set
<K
, Pred
, A
>::iterator it
= m
.begin();
772 for (unsigned int i
= 0; i
< nSize
; i
++)
775 Unserialize(is
, key
);
776 it
= m
.insert(it
, key
);
785 template<typename Stream
, typename T
> void
786 Serialize(Stream
& os
, const std::unique_ptr
<const T
>& p
)
791 template<typename Stream
, typename T
>
792 void Unserialize(Stream
& is
, std::unique_ptr
<const T
>& p
)
794 p
.reset(new T(deserialize
, is
));
802 template<typename Stream
, typename T
> void
803 Serialize(Stream
& os
, const std::shared_ptr
<const T
>& p
)
808 template<typename Stream
, typename T
>
809 void Unserialize(Stream
& is
, std::shared_ptr
<const T
>& p
)
811 p
= std::make_shared
<const T
>(deserialize
, is
);
817 * Support for ADD_SERIALIZE_METHODS and READWRITE macro
819 struct CSerActionSerialize
821 constexpr bool ForRead() const { return false; }
823 struct CSerActionUnserialize
825 constexpr bool ForRead() const { return true; }
828 template<typename Stream
, typename T
>
829 inline void SerReadWrite(Stream
& s
, const T
& obj
, CSerActionSerialize ser_action
)
834 template<typename Stream
, typename T
>
835 inline void SerReadWrite(Stream
& s
, T
& obj
, CSerActionUnserialize ser_action
)
837 ::Unserialize(s
, obj
);
848 /* ::GetSerializeSize implementations
850 * Computing the serialized size of objects is done through a special stream
851 * object of type CSizeComputer, which only records the number of bytes written
854 * If your Serialize or SerializationOp method has non-trivial overhead for
855 * serialization, it may be worthwhile to implement a specialized version for
856 * CSizeComputer, which uses the s.seek() method to record bytes that would
857 * be written instead.
867 CSizeComputer(int nTypeIn
, int nVersionIn
) : nSize(0), nType(nTypeIn
), nVersion(nVersionIn
) {}
869 void write(const char *psz
, size_t _nSize
)
871 this->nSize
+= _nSize
;
874 /** Pretend _nSize bytes are written, without specifying them. */
875 void seek(size_t _nSize
)
877 this->nSize
+= _nSize
;
881 CSizeComputer
& operator<<(const T
& obj
)
883 ::Serialize(*this, obj
);
887 size_t size() const {
891 int GetVersion() const { return nVersion
; }
892 int GetType() const { return nType
; }
895 template<typename Stream
>
896 void SerializeMany(Stream
& s
)
900 template<typename Stream
, typename Arg
>
901 void SerializeMany(Stream
& s
, Arg
&& arg
)
903 ::Serialize(s
, std::forward
<Arg
>(arg
));
906 template<typename Stream
, typename Arg
, typename
... Args
>
907 void SerializeMany(Stream
& s
, Arg
&& arg
, Args
&&... args
)
909 ::Serialize(s
, std::forward
<Arg
>(arg
));
910 ::SerializeMany(s
, std::forward
<Args
>(args
)...);
913 template<typename Stream
>
914 inline void UnserializeMany(Stream
& s
)
918 template<typename Stream
, typename Arg
>
919 inline void UnserializeMany(Stream
& s
, Arg
& arg
)
921 ::Unserialize(s
, arg
);
924 template<typename Stream
, typename Arg
, typename
... Args
>
925 inline void UnserializeMany(Stream
& s
, Arg
& arg
, Args
&... args
)
927 ::Unserialize(s
, arg
);
928 ::UnserializeMany(s
, args
...);
931 template<typename Stream
, typename
... Args
>
932 inline void SerReadWriteMany(Stream
& s
, CSerActionSerialize ser_action
, Args
&&... args
)
934 ::SerializeMany(s
, std::forward
<Args
>(args
)...);
937 template<typename Stream
, typename
... Args
>
938 inline void SerReadWriteMany(Stream
& s
, CSerActionUnserialize ser_action
, Args
&... args
)
940 ::UnserializeMany(s
, args
...);
944 inline void WriteVarInt(CSizeComputer
&s
, I n
)
946 s
.seek(GetSizeOfVarInt
<I
>(n
));
949 inline void WriteCompactSize(CSizeComputer
&s
, uint64_t nSize
)
951 s
.seek(GetSizeOfCompactSize(nSize
));
954 template <typename T
>
955 size_t GetSerializeSize(const T
& t
, int nType
, int nVersion
= 0)
957 return (CSizeComputer(nType
, nVersion
) << t
).size();
960 template <typename S
, typename T
>
961 size_t GetSerializeSize(const S
& s
, const T
& t
)
963 return (CSizeComputer(s
.GetType(), s
.GetVersion()) << t
).size();
966 #endif // BITCOIN_SERIALIZE_H