2019-12-30 22:39:22 +13:00
// Copyright (c) 2016-2019 The Bitcoin Core developers
2016-04-15 12:23:57 -07:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2018-03-22 15:19:44 +01:00
# ifndef BITCOIN_BLOCKENCODINGS_H
# define BITCOIN_BLOCKENCODINGS_H
2016-04-15 12:23:57 -07:00
2017-11-10 13:57:53 +13:00
# include <primitives/block.h>
2016-04-15 12:23:57 -07:00
class CTxMemPool ;
2020-02-03 19:41:42 -08:00
// Transaction compression schemes for compact block relay can be introduced by writing
// an actual formatter here.
using TransactionCompression = DefaultFormatter ;
2016-04-15 12:23:57 -07:00
2020-02-15 19:05:11 -08:00
class DifferenceFormatter
{
uint64_t m_shift = 0 ;
public :
template < typename Stream , typename I >
void Ser ( Stream & s , I v )
{
if ( v < m_shift | | v > = std : : numeric_limits < uint64_t > : : max ( ) ) throw std : : ios_base : : failure ( " differential value overflow " ) ;
WriteCompactSize ( s , v - m_shift ) ;
m_shift = uint64_t ( v ) + 1 ;
}
template < typename Stream , typename I >
void Unser ( Stream & s , I & v )
{
uint64_t n = ReadCompactSize ( s ) ;
m_shift + = n ;
if ( m_shift < n | | m_shift > = std : : numeric_limits < uint64_t > : : max ( ) | | m_shift < std : : numeric_limits < I > : : min ( ) | | m_shift > std : : numeric_limits < I > : : max ( ) ) throw std : : ios_base : : failure ( " differential value overflow " ) ;
v = I ( m_shift + + ) ;
}
} ;
2016-04-15 12:23:57 -07:00
class BlockTransactionsRequest {
public :
// A BlockTransactionsRequest message
uint256 blockhash ;
std : : vector < uint16_t > indexes ;
2020-02-03 19:41:42 -08:00
SERIALIZE_METHODS ( BlockTransactionsRequest , obj )
{
READWRITE ( obj . blockhash , Using < VectorFormatter < DifferenceFormatter > > ( obj . indexes ) ) ;
2016-04-15 12:23:57 -07:00
}
} ;
class BlockTransactions {
public :
// A BlockTransactions message
uint256 blockhash ;
2016-11-10 17:34:17 -08:00
std : : vector < CTransactionRef > txn ;
2016-04-15 12:23:57 -07:00
BlockTransactions ( ) { }
2017-08-01 12:22:41 +02:00
explicit BlockTransactions ( const BlockTransactionsRequest & req ) :
2016-04-15 12:23:57 -07:00
blockhash ( req . blockhash ) , txn ( req . indexes . size ( ) ) { }
2020-02-03 19:41:42 -08:00
SERIALIZE_METHODS ( BlockTransactions , obj )
{
READWRITE ( obj . blockhash , Using < VectorFormatter < TransactionCompression > > ( obj . txn ) ) ;
2016-04-15 12:23:57 -07:00
}
} ;
2017-01-18 16:15:37 +01:00
// Dumb serialization/storage-helper for CBlockHeaderAndShortTxIDs and PartiallyDownloadedBlock
2016-04-15 12:23:57 -07:00
struct PrefilledTransaction {
// Used as an offset since last prefilled tx in CBlockHeaderAndShortTxIDs,
// as a proper transaction-in-block-index in PartiallyDownloadedBlock
uint16_t index ;
2016-11-10 17:34:17 -08:00
CTransactionRef tx ;
2016-04-15 12:23:57 -07:00
2020-02-03 19:41:42 -08:00
SERIALIZE_METHODS ( PrefilledTransaction , obj ) { READWRITE ( COMPACTSIZE ( obj . index ) , Using < TransactionCompression > ( obj . tx ) ) ; }
2016-04-15 12:23:57 -07:00
} ;
typedef enum ReadStatus_t
{
READ_STATUS_OK ,
READ_STATUS_INVALID , // Invalid object, peer is sending bogus crap
READ_STATUS_FAILED , // Failed to process object
2016-10-31 10:03:49 -04:00
READ_STATUS_CHECKBLOCK_FAILED , // Used only by FillBlock to indicate a
// failure in CheckBlock.
2016-04-15 12:23:57 -07:00
} ReadStatus ;
class CBlockHeaderAndShortTxIDs {
private :
mutable uint64_t shorttxidk0 , shorttxidk1 ;
uint64_t nonce ;
void FillShortTxIDSelector ( ) const ;
friend class PartiallyDownloadedBlock ;
static const int SHORTTXIDS_LENGTH = 6 ;
protected :
std : : vector < uint64_t > shorttxids ;
std : : vector < PrefilledTransaction > prefilledtxn ;
public :
CBlockHeader header ;
// Dummy for deserialization
CBlockHeaderAndShortTxIDs ( ) { }
2016-06-25 19:17:45 +02:00
CBlockHeaderAndShortTxIDs ( const CBlock & block , bool fUseWTXID ) ;
2016-04-15 12:23:57 -07:00
uint64_t GetShortID ( const uint256 & txhash ) const ;
size_t BlockTxCount ( ) const { return shorttxids . size ( ) + prefilledtxn . size ( ) ; }
2020-02-03 19:41:42 -08:00
SERIALIZE_METHODS ( CBlockHeaderAndShortTxIDs , obj )
{
READWRITE ( obj . header , obj . nonce , Using < VectorFormatter < CustomUintFormatter < SHORTTXIDS_LENGTH > > > ( obj . shorttxids ) , obj . prefilledtxn ) ;
2016-04-15 12:23:57 -07:00
if ( ser_action . ForRead ( ) ) {
2020-02-03 19:41:42 -08:00
if ( obj . BlockTxCount ( ) > std : : numeric_limits < uint16_t > : : max ( ) ) {
throw std : : ios_base : : failure ( " indexes overflowed 16 bits " ) ;
2016-04-15 12:23:57 -07:00
}
2020-02-03 19:41:42 -08:00
obj . FillShortTxIDSelector ( ) ;
2016-04-15 12:23:57 -07:00
}
}
} ;
class PartiallyDownloadedBlock {
protected :
2016-11-10 17:34:17 -08:00
std : : vector < CTransactionRef > txn_available ;
2017-01-12 12:19:14 -08:00
size_t prefilled_count = 0 , mempool_count = 0 , extra_count = 0 ;
2016-04-15 12:23:57 -07:00
CTxMemPool * pool ;
public :
CBlockHeader header ;
2017-08-01 12:22:41 +02:00
explicit PartiallyDownloadedBlock ( CTxMemPool * poolIn ) : pool ( poolIn ) { }
2016-04-15 12:23:57 -07:00
2016-12-04 20:44:37 -08:00
// extra_txn is a list of extra transactions to look at, in <witness hash, reference> form
2017-01-12 12:20:11 -08:00
ReadStatus InitData ( const CBlockHeaderAndShortTxIDs & cmpctblock , const std : : vector < std : : pair < uint256 , CTransactionRef > > & extra_txn ) ;
2016-04-15 12:23:57 -07:00
bool IsTxAvailable ( size_t index ) const ;
2016-11-11 13:01:27 -08:00
ReadStatus FillBlock ( CBlock & block , const std : : vector < CTransactionRef > & vtx_missing ) ;
2016-04-15 12:23:57 -07:00
} ;
2018-03-22 15:19:44 +01:00
# endif // BITCOIN_BLOCKENCODINGS_H