1 // Copyright (c) 2016 The Bitcoin Core developers
2 // Distributed under the MIT software license, see the accompanying
3 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 #ifndef BITCOIN_BLOCK_ENCODINGS_H
6 #define BITCOIN_BLOCK_ENCODINGS_H
8 #include "primitives/block.h"
14 // Dumb helper to handle CTransaction compression at serialize-time
15 struct TransactionCompressor
{
19 TransactionCompressor(CTransaction
& txIn
) : tx(txIn
) {}
21 ADD_SERIALIZE_METHODS
;
23 template <typename Stream
, typename Operation
>
24 inline void SerializationOp(Stream
& s
, Operation ser_action
) {
25 READWRITE(tx
); //TODO: Compress tx encoding
29 class BlockTransactionsRequest
{
31 // A BlockTransactionsRequest message
33 std::vector
<uint16_t> indexes
;
35 ADD_SERIALIZE_METHODS
;
37 template <typename Stream
, typename Operation
>
38 inline void SerializationOp(Stream
& s
, Operation ser_action
) {
40 uint64_t indexes_size
= (uint64_t)indexes
.size();
41 READWRITE(COMPACTSIZE(indexes_size
));
42 if (ser_action
.ForRead()) {
44 while (indexes
.size() < indexes_size
) {
45 indexes
.resize(std::min((uint64_t)(1000 + indexes
.size()), indexes_size
));
46 for (; i
< indexes
.size(); i
++) {
48 READWRITE(COMPACTSIZE(index
));
49 if (index
> std::numeric_limits
<uint16_t>::max())
50 throw std::ios_base::failure("index overflowed 16 bits");
56 for (size_t j
= 0; j
< indexes
.size(); j
++) {
57 if (uint64_t(indexes
[j
]) + uint64_t(offset
) > std::numeric_limits
<uint16_t>::max())
58 throw std::ios_base::failure("indexes overflowed 16 bits");
59 indexes
[j
] = indexes
[j
] + offset
;
60 offset
= indexes
[j
] + 1;
63 for (size_t i
= 0; i
< indexes
.size(); i
++) {
64 uint64_t index
= indexes
[i
] - (i
== 0 ? 0 : (indexes
[i
- 1] + 1));
65 READWRITE(COMPACTSIZE(index
));
71 class BlockTransactions
{
73 // A BlockTransactions message
75 std::vector
<CTransaction
> txn
;
77 BlockTransactions() {}
78 BlockTransactions(const BlockTransactionsRequest
& req
) :
79 blockhash(req
.blockhash
), txn(req
.indexes
.size()) {}
81 ADD_SERIALIZE_METHODS
;
83 template <typename Stream
, typename Operation
>
84 inline void SerializationOp(Stream
& s
, Operation ser_action
) {
86 uint64_t txn_size
= (uint64_t)txn
.size();
87 READWRITE(COMPACTSIZE(txn_size
));
88 if (ser_action
.ForRead()) {
90 while (txn
.size() < txn_size
) {
91 txn
.resize(std::min((uint64_t)(1000 + txn
.size()), txn_size
));
92 for (; i
< txn
.size(); i
++)
93 READWRITE(REF(TransactionCompressor(txn
[i
])));
96 for (size_t i
= 0; i
< txn
.size(); i
++)
97 READWRITE(REF(TransactionCompressor(txn
[i
])));
102 // Dumb serialization/storage-helper for CBlockHeaderAndShortTxIDs and PartiallyDownlaodedBlock
103 struct PrefilledTransaction
{
104 // Used as an offset since last prefilled tx in CBlockHeaderAndShortTxIDs,
105 // as a proper transaction-in-block-index in PartiallyDownloadedBlock
109 ADD_SERIALIZE_METHODS
;
111 template <typename Stream
, typename Operation
>
112 inline void SerializationOp(Stream
& s
, Operation ser_action
) {
113 uint64_t idx
= index
;
114 READWRITE(COMPACTSIZE(idx
));
115 if (idx
> std::numeric_limits
<uint16_t>::max())
116 throw std::ios_base::failure("index overflowed 16-bits");
118 READWRITE(REF(TransactionCompressor(tx
)));
122 typedef enum ReadStatus_t
125 READ_STATUS_INVALID
, // Invalid object, peer is sending bogus crap
126 READ_STATUS_FAILED
, // Failed to process object
129 class CBlockHeaderAndShortTxIDs
{
131 mutable uint64_t shorttxidk0
, shorttxidk1
;
134 void FillShortTxIDSelector() const;
136 friend class PartiallyDownloadedBlock
;
138 static const int SHORTTXIDS_LENGTH
= 6;
140 std::vector
<uint64_t> shorttxids
;
141 std::vector
<PrefilledTransaction
> prefilledtxn
;
146 // Dummy for deserialization
147 CBlockHeaderAndShortTxIDs() {}
149 CBlockHeaderAndShortTxIDs(const CBlock
& block
, bool fUseWTXID
);
151 uint64_t GetShortID(const uint256
& txhash
) const;
153 size_t BlockTxCount() const { return shorttxids
.size() + prefilledtxn
.size(); }
155 ADD_SERIALIZE_METHODS
;
157 template <typename Stream
, typename Operation
>
158 inline void SerializationOp(Stream
& s
, Operation ser_action
) {
162 uint64_t shorttxids_size
= (uint64_t)shorttxids
.size();
163 READWRITE(COMPACTSIZE(shorttxids_size
));
164 if (ser_action
.ForRead()) {
166 while (shorttxids
.size() < shorttxids_size
) {
167 shorttxids
.resize(std::min((uint64_t)(1000 + shorttxids
.size()), shorttxids_size
));
168 for (; i
< shorttxids
.size(); i
++) {
169 uint32_t lsb
= 0; uint16_t msb
= 0;
172 shorttxids
[i
] = (uint64_t(msb
) << 32) | uint64_t(lsb
);
173 static_assert(SHORTTXIDS_LENGTH
== 6, "shorttxids serialization assumes 6-byte shorttxids");
177 for (size_t i
= 0; i
< shorttxids
.size(); i
++) {
178 uint32_t lsb
= shorttxids
[i
] & 0xffffffff;
179 uint16_t msb
= (shorttxids
[i
] >> 32) & 0xffff;
185 READWRITE(prefilledtxn
);
187 if (ser_action
.ForRead())
188 FillShortTxIDSelector();
192 class PartiallyDownloadedBlock
{
194 std::vector
<std::shared_ptr
<const CTransaction
> > txn_available
;
195 size_t prefilled_count
= 0, mempool_count
= 0;
199 PartiallyDownloadedBlock(CTxMemPool
* poolIn
) : pool(poolIn
) {}
201 ReadStatus
InitData(const CBlockHeaderAndShortTxIDs
& cmpctblock
);
202 bool IsTxAvailable(size_t index
) const;
203 ReadStatus
FillBlock(CBlock
& block
, const std::vector
<CTransaction
>& vtx_missing
) const;