2014-09-03 02:20:09 +02:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2018-07-26 18:36:45 -04:00
// Copyright (c) 2009-2018 The Bitcoin Core developers
2014-09-29 08:22:03 +02:00
// Distributed under the MIT software license, see the accompanying
2014-09-03 02:20:09 +02:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2014-11-03 16:16:40 +01:00
# ifndef BITCOIN_CHAIN_H
# define BITCOIN_CHAIN_H
2014-09-03 02:20:09 +02:00
2017-11-10 13:57:53 +13:00
# include <arith_uint256.h>
2018-05-13 23:39:53 -07:00
# include <consensus/params.h>
2019-01-06 11:43:38 -08:00
# include <flatfile.h>
2017-11-10 13:57:53 +13:00
# include <primitives/block.h>
# include <tinyformat.h>
# include <uint256.h>
2014-09-03 02:20:09 +02:00
# include <vector>
2017-03-02 12:20:34 -05:00
/**
* Maximum amount of time that a block timestamp is allowed to exceed the
* current network - adjusted time before the block will be accepted .
*/
2018-10-23 22:02:20 +03:00
static constexpr int64_t MAX_FUTURE_BLOCK_TIME = 2 * 60 * 60 ;
2017-03-02 12:20:34 -05:00
/**
* Timestamp window used as a grace period by code that compares external
* timestamps ( such as timestamps passed to RPCs , or wallet key creation times )
* to block timestamps . This should be set at least as high as
* MAX_FUTURE_BLOCK_TIME .
*/
2018-10-23 22:02:20 +03:00
static constexpr int64_t TIMESTAMP_WINDOW = MAX_FUTURE_BLOCK_TIME ;
/**
* Maximum gap between node time and block time used
* for the " Catching up... " mode in GUI .
*
* Ref : https : //github.com/bitcoin/bitcoin/pull/1026
*/
static constexpr int64_t MAX_BLOCK_TIME_GAP = 90 * 60 ;
2017-03-02 12:20:34 -05:00
2016-04-05 15:14:48 +02:00
class CBlockFileInfo
{
public :
unsigned int nBlocks ; //!< number of blocks stored in file
unsigned int nSize ; //!< number of used bytes of block file
unsigned int nUndoSize ; //!< number of used bytes in the undo file
unsigned int nHeightFirst ; //!< lowest height of block in file
unsigned int nHeightLast ; //!< highest height of block in file
uint64_t nTimeFirst ; //!< earliest time of block in file
uint64_t nTimeLast ; //!< latest time of block in file
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
2016-10-28 16:29:17 -07:00
inline void SerializationOp ( Stream & s , Operation ser_action ) {
2016-04-05 15:14:48 +02:00
READWRITE ( VARINT ( nBlocks ) ) ;
READWRITE ( VARINT ( nSize ) ) ;
READWRITE ( VARINT ( nUndoSize ) ) ;
READWRITE ( VARINT ( nHeightFirst ) ) ;
READWRITE ( VARINT ( nHeightLast ) ) ;
READWRITE ( VARINT ( nTimeFirst ) ) ;
READWRITE ( VARINT ( nTimeLast ) ) ;
}
void SetNull ( ) {
nBlocks = 0 ;
nSize = 0 ;
nUndoSize = 0 ;
nHeightFirst = 0 ;
nHeightLast = 0 ;
nTimeFirst = 0 ;
nTimeLast = 0 ;
}
CBlockFileInfo ( ) {
SetNull ( ) ;
}
std : : string ToString ( ) const ;
/** update statistics (does not update nSize) */
void AddBlock ( unsigned int nHeightIn , uint64_t nTimeIn ) {
if ( nBlocks = = 0 | | nHeightFirst > nHeightIn )
nHeightFirst = nHeightIn ;
if ( nBlocks = = 0 | | nTimeFirst > nTimeIn )
nTimeFirst = nTimeIn ;
nBlocks + + ;
if ( nHeightIn > nHeightLast )
nHeightLast = nHeightIn ;
if ( nTimeIn > nTimeLast )
nTimeLast = nTimeIn ;
}
} ;
2016-04-28 13:35:16 +02:00
enum BlockStatus : uint32_t {
2014-10-25 16:46:54 +08:00
//! Unused.
2014-09-03 02:20:09 +02:00
BLOCK_VALID_UNKNOWN = 0 ,
2014-07-12 00:02:35 +02:00
2018-12-03 16:50:08 -05:00
//! Reserved (was BLOCK_VALID_HEADER).
BLOCK_VALID_RESERVED = 1 ,
2014-07-12 00:02:35 +02:00
2014-10-25 16:46:54 +08:00
//! All parent headers found, difficulty matches, timestamp >= median previous, checkpoint. Implies all parents
//! are also at least TREE.
2014-07-12 00:02:35 +02:00
BLOCK_VALID_TREE = 2 ,
2014-10-25 16:46:54 +08:00
/**
* Only first tx is coinbase , 2 < = coinbase input script length < = 100 , transactions valid , no duplicate txids ,
* sigops , size , merkle root . Implies all parents are at least TREE but not necessarily TRANSACTIONS . When all
* parent blocks also have TRANSACTIONS , CBlockIndex : : nChainTx will be set .
*/
2014-07-12 00:02:35 +02:00
BLOCK_VALID_TRANSACTIONS = 3 ,
2015-04-28 14:47:17 +00:00
//! Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends, BIP30.
2014-10-25 16:46:54 +08:00
//! Implies all parents are also at least CHAIN.
2014-07-12 00:02:35 +02:00
BLOCK_VALID_CHAIN = 4 ,
2014-10-25 16:46:54 +08:00
//! Scripts & signatures ok. Implies all parents are also at least SCRIPTS.
2014-07-12 00:02:35 +02:00
BLOCK_VALID_SCRIPTS = 5 ,
2014-10-25 16:46:54 +08:00
//! All validity bits.
2018-12-03 16:50:08 -05:00
BLOCK_VALID_MASK = BLOCK_VALID_RESERVED | BLOCK_VALID_TREE | BLOCK_VALID_TRANSACTIONS |
2014-09-03 02:20:09 +02:00
BLOCK_VALID_CHAIN | BLOCK_VALID_SCRIPTS ,
2016-06-29 11:48:51 +02:00
BLOCK_HAVE_DATA = 8 , //!< full block available in blk*.dat
BLOCK_HAVE_UNDO = 16 , //!< undo data available in rev*.dat
2014-09-03 02:20:09 +02:00
BLOCK_HAVE_MASK = BLOCK_HAVE_DATA | BLOCK_HAVE_UNDO ,
2016-06-29 11:48:51 +02:00
BLOCK_FAILED_VALID = 32 , //!< stage after last reached validness failed
BLOCK_FAILED_CHILD = 64 , //!< descends from failed block
2014-09-03 02:20:09 +02:00
BLOCK_FAILED_MASK = BLOCK_FAILED_VALID | BLOCK_FAILED_CHILD ,
2016-03-18 17:20:12 +01:00
2016-06-29 11:48:51 +02:00
BLOCK_OPT_WITNESS = 128 , //!< block data in blk*.data was received with a witness-enforcing client
2014-09-03 02:20:09 +02:00
} ;
/** The block chain is a tree shaped structure starting with the
* genesis block at the root , with each block potentially having multiple
* candidates to be the next block . A blockindex may have multiple pprev pointing
* to it , but at most one of them can be part of the currently active branch .
*/
class CBlockIndex
{
public :
2014-11-25 18:54:36 +01:00
//! pointer to the hash of the block, if any. Memory is owned by this CBlockIndex
2014-09-03 02:20:09 +02:00
const uint256 * phashBlock ;
2014-10-25 16:46:54 +08:00
//! pointer to the index of the predecessor of this block
2014-09-03 02:20:09 +02:00
CBlockIndex * pprev ;
2014-10-25 16:46:54 +08:00
//! pointer to the index of some further predecessor of this block
2014-09-03 02:20:09 +02:00
CBlockIndex * pskip ;
2014-10-25 16:46:54 +08:00
//! height of the entry in the chain. The genesis block has height 0
2014-09-03 02:20:09 +02:00
int nHeight ;
2014-10-25 16:46:54 +08:00
//! Which # file this block is stored in (blk?????.dat)
2014-09-03 02:20:09 +02:00
int nFile ;
2014-10-25 16:46:54 +08:00
//! Byte offset within blk?????.dat where this block's data is stored
2014-09-03 02:20:09 +02:00
unsigned int nDataPos ;
2014-10-25 16:46:54 +08:00
//! Byte offset within rev?????.dat where this block's undo data is stored
2014-09-03 02:20:09 +02:00
unsigned int nUndoPos ;
2014-10-25 16:46:54 +08:00
//! (memory only) Total amount of work (expected number of hashes) in the chain up to and including this block
2014-12-16 15:43:03 +01:00
arith_uint256 nChainWork ;
2014-09-03 02:20:09 +02:00
2014-10-25 16:46:54 +08:00
//! Number of transactions in this block.
//! Note: in a potential headers-first mode, this number cannot be relied upon
2014-09-03 02:20:09 +02:00
unsigned int nTx ;
2014-10-25 16:46:54 +08:00
//! (memory only) Number of transactions in the chain up to and including this block.
//! This value will be non-zero only if and only if transactions for this block and all its parents are available.
//! Change to 64-bit type when necessary; won't happen before 2030
unsigned int nChainTx ;
2014-09-03 02:20:09 +02:00
2014-10-25 16:46:54 +08:00
//! Verification status of this block. See enum BlockStatus
2017-09-05 06:03:42 +08:00
uint32_t nStatus ;
2014-09-03 02:20:09 +02:00
2014-10-25 16:46:54 +08:00
//! block header
2017-09-05 06:03:42 +08:00
int32_t nVersion ;
2014-09-03 02:20:09 +02:00
uint256 hashMerkleRoot ;
2018-08-06 15:40:20 -04:00
uint256 hashClaimTrie ;
2017-09-05 06:03:42 +08:00
uint32_t nTime ;
uint32_t nBits ;
uint32_t nNonce ;
2014-09-03 02:20:09 +02:00
2014-10-25 16:46:54 +08:00
//! (memory only) Sequential id assigned to distinguish order in which blocks are received.
2016-08-26 23:05:09 +02:00
int32_t nSequenceId ;
2014-09-03 02:20:09 +02:00
2017-07-30 21:42:17 +02:00
//! (memory only) Maximum nTime in the chain up to and including this block.
2017-01-08 04:05:14 +00:00
unsigned int nTimeMax ;
2014-09-03 02:20:09 +02:00
void SetNull ( )
{
2017-08-07 07:36:37 +02:00
phashBlock = nullptr ;
pprev = nullptr ;
pskip = nullptr ;
2014-09-03 02:20:09 +02:00
nHeight = 0 ;
nFile = 0 ;
nDataPos = 0 ;
nUndoPos = 0 ;
2014-12-16 15:43:03 +01:00
nChainWork = arith_uint256 ( ) ;
2014-09-03 02:20:09 +02:00
nTx = 0 ;
nChainTx = 0 ;
nStatus = 0 ;
nSequenceId = 0 ;
2017-01-08 04:05:14 +00:00
nTimeMax = 0 ;
2014-09-03 02:20:09 +02:00
nVersion = 0 ;
2014-12-15 09:11:16 +01:00
hashMerkleRoot = uint256 ( ) ;
2018-08-06 15:40:20 -04:00
hashClaimTrie = uint256 ( ) ;
2014-09-03 02:20:09 +02:00
nTime = 0 ;
nBits = 0 ;
nNonce = 0 ;
}
CBlockIndex ( )
{
SetNull ( ) ;
}
2017-08-01 12:22:41 +02:00
explicit CBlockIndex ( const CBlockHeader & block )
2014-09-03 02:20:09 +02:00
{
SetNull ( ) ;
nVersion = block . nVersion ;
hashMerkleRoot = block . hashMerkleRoot ;
2018-08-06 15:40:20 -04:00
hashClaimTrie = block . hashClaimTrie ;
2014-09-03 02:20:09 +02:00
nTime = block . nTime ;
nBits = block . nBits ;
nNonce = block . nNonce ;
}
2019-01-06 11:46:30 -08:00
FlatFilePos GetBlockPos ( ) const {
FlatFilePos ret ;
2014-09-03 02:20:09 +02:00
if ( nStatus & BLOCK_HAVE_DATA ) {
ret . nFile = nFile ;
ret . nPos = nDataPos ;
}
return ret ;
}
2019-01-06 11:46:30 -08:00
FlatFilePos GetUndoPos ( ) const {
FlatFilePos ret ;
2014-09-03 02:20:09 +02:00
if ( nStatus & BLOCK_HAVE_UNDO ) {
ret . nFile = nFile ;
ret . nPos = nUndoPos ;
}
return ret ;
}
CBlockHeader GetBlockHeader ( ) const
{
CBlockHeader block ;
block . nVersion = nVersion ;
if ( pprev )
block . hashPrevBlock = pprev - > GetBlockHash ( ) ;
block . hashMerkleRoot = hashMerkleRoot ;
2018-08-06 15:40:20 -04:00
block . hashClaimTrie = hashClaimTrie ;
2014-09-03 02:20:09 +02:00
block . nTime = nTime ;
block . nBits = nBits ;
block . nNonce = nNonce ;
return block ;
}
uint256 GetBlockHash ( ) const
{
return * phashBlock ;
}
2018-08-06 15:40:20 -04:00
uint256 GetBlockPoWHash ( ) const
{
return GetBlockHeader ( ) . GetPoWHash ( ) ;
}
2018-12-03 18:14:08 -05:00
/**
* Check whether this block ' s and all previous blocks ' transactions have been
* downloaded ( and stored to disk ) at some point .
*
* Does not imply the transactions are consensus - valid ( ConnectTip might fail )
* Does not imply the transactions are still stored on disk . ( IsBlockPruned might return true )
*/
bool HaveTxsDownloaded ( ) const { return nChainTx ! = 0 ; }
2014-09-03 02:20:09 +02:00
int64_t GetBlockTime ( ) const
{
return ( int64_t ) nTime ;
}
2017-01-08 04:05:14 +00:00
int64_t GetBlockTimeMax ( ) const
{
return ( int64_t ) nTimeMax ;
}
2017-07-05 16:49:57 +02:00
static constexpr int nMedianTimeSpan = 11 ;
2014-09-03 02:20:09 +02:00
int64_t GetMedianTimePast ( ) const
{
int64_t pmedian [ nMedianTimeSpan ] ;
int64_t * pbegin = & pmedian [ nMedianTimeSpan ] ;
int64_t * pend = & pmedian [ nMedianTimeSpan ] ;
const CBlockIndex * pindex = this ;
for ( int i = 0 ; i < nMedianTimeSpan & & pindex ; i + + , pindex = pindex - > pprev )
* ( - - pbegin ) = pindex - > GetBlockTime ( ) ;
std : : sort ( pbegin , pend ) ;
return pbegin [ ( pend - pbegin ) / 2 ] ;
}
std : : string ToString ( ) const
{
2018-08-06 15:40:20 -04:00
return strprintf ( " CBlockIndex(pprev=%p, nHeight=%d, merkle=%s, claimtrie=%s, hashBlock=%s) " ,
2014-09-03 02:20:09 +02:00
pprev , nHeight ,
hashMerkleRoot . ToString ( ) ,
2018-08-06 15:40:20 -04:00
hashClaimTrie . ToString ( ) ,
2014-09-03 02:20:09 +02:00
GetBlockHash ( ) . ToString ( ) ) ;
}
2014-10-25 16:46:54 +08:00
//! Check whether this block index entry is valid up to the passed validity level.
2014-09-03 02:20:09 +02:00
bool IsValid ( enum BlockStatus nUpTo = BLOCK_VALID_TRANSACTIONS ) const
{
assert ( ! ( nUpTo & ~ BLOCK_VALID_MASK ) ) ; // Only validity flags allowed.
if ( nStatus & BLOCK_FAILED_MASK )
return false ;
return ( ( nStatus & BLOCK_VALID_MASK ) > = nUpTo ) ;
}
2014-10-25 16:46:54 +08:00
//! Raise the validity level of this block index entry.
//! Returns true if the validity was changed.
2014-09-03 02:20:09 +02:00
bool RaiseValidity ( enum BlockStatus nUpTo )
{
assert ( ! ( nUpTo & ~ BLOCK_VALID_MASK ) ) ; // Only validity flags allowed.
if ( nStatus & BLOCK_FAILED_MASK )
return false ;
if ( ( nStatus & BLOCK_VALID_MASK ) < nUpTo ) {
nStatus = ( nStatus & ~ BLOCK_VALID_MASK ) | nUpTo ;
return true ;
}
return false ;
}
2014-10-25 16:46:54 +08:00
//! Build the skiplist pointer for this entry.
2014-09-03 02:20:09 +02:00
void BuildSkip ( ) ;
2014-10-25 16:46:54 +08:00
//! Efficiently find an ancestor of this block.
2014-09-03 02:20:09 +02:00
CBlockIndex * GetAncestor ( int height ) ;
const CBlockIndex * GetAncestor ( int height ) const ;
} ;
2015-11-30 00:46:49 +01:00
arith_uint256 GetBlockProof ( const CBlockIndex & block ) ;
/** Return the time it would take to redo the work difference between from and to, assuming the current hashrate corresponds to the difficulty at tip, in seconds. */
int64_t GetBlockProofEquivalentTime ( const CBlockIndex & to , const CBlockIndex & from , const CBlockIndex & tip , const Consensus : : Params & ) ;
2017-04-16 06:57:11 -07:00
/** Find the forking point between two chain tips. */
const CBlockIndex * LastCommonAncestor ( const CBlockIndex * pa , const CBlockIndex * pb ) ;
2015-11-30 00:46:49 +01:00
2014-09-03 02:20:09 +02:00
/** Used to marshal pointers into hashes for db storage. */
class CDiskBlockIndex : public CBlockIndex
{
public :
uint256 hashPrev ;
CDiskBlockIndex ( ) {
2014-12-15 09:11:16 +01:00
hashPrev = uint256 ( ) ;
2014-09-03 02:20:09 +02:00
}
2014-11-25 16:26:20 +01:00
explicit CDiskBlockIndex ( const CBlockIndex * pindex ) : CBlockIndex ( * pindex ) {
2014-12-15 09:11:16 +01:00
hashPrev = ( pprev ? pprev - > GetBlockHash ( ) : uint256 ( ) ) ;
2014-09-03 02:20:09 +02:00
}
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
2016-10-28 16:29:17 -07:00
inline void SerializationOp ( Stream & s , Operation ser_action ) {
2016-11-10 08:00:05 +01:00
int _nVersion = s . GetVersion ( ) ;
2016-10-28 16:29:17 -07:00
if ( ! ( s . GetType ( ) & SER_GETHASH ) )
2017-02-13 13:41:02 -05:00
READWRITE ( VARINT ( _nVersion , VarIntMode : : NONNEGATIVE_SIGNED ) ) ;
2014-09-03 02:20:09 +02:00
2017-02-13 13:41:02 -05:00
READWRITE ( VARINT ( nHeight , VarIntMode : : NONNEGATIVE_SIGNED ) ) ;
2014-09-03 02:20:09 +02:00
READWRITE ( VARINT ( nStatus ) ) ;
READWRITE ( VARINT ( nTx ) ) ;
if ( nStatus & ( BLOCK_HAVE_DATA | BLOCK_HAVE_UNDO ) )
2017-02-13 13:41:02 -05:00
READWRITE ( VARINT ( nFile , VarIntMode : : NONNEGATIVE_SIGNED ) ) ;
2014-09-03 02:20:09 +02:00
if ( nStatus & BLOCK_HAVE_DATA )
READWRITE ( VARINT ( nDataPos ) ) ;
if ( nStatus & BLOCK_HAVE_UNDO )
READWRITE ( VARINT ( nUndoPos ) ) ;
// block header
READWRITE ( this - > nVersion ) ;
READWRITE ( hashPrev ) ;
READWRITE ( hashMerkleRoot ) ;
2018-08-06 15:40:20 -04:00
READWRITE ( hashClaimTrie ) ;
2014-09-03 02:20:09 +02:00
READWRITE ( nTime ) ;
READWRITE ( nBits ) ;
READWRITE ( nNonce ) ;
}
uint256 GetBlockHash ( ) const
{
CBlockHeader block ;
block . nVersion = nVersion ;
block . hashPrevBlock = hashPrev ;
block . hashMerkleRoot = hashMerkleRoot ;
2018-08-06 15:40:20 -04:00
block . hashClaimTrie = hashClaimTrie ;
2014-09-03 02:20:09 +02:00
block . nTime = nTime ;
block . nBits = nBits ;
block . nNonce = nNonce ;
return block . GetHash ( ) ;
}
std : : string ToString ( ) const
{
std : : string str = " CDiskBlockIndex( " ;
str + = CBlockIndex : : ToString ( ) ;
2018-08-06 15:40:20 -04:00
str + = strprintf ( " \n hashBlock=%s, hashClaimTrie=%s, hashPrev=%s) " ,
2014-09-03 02:20:09 +02:00
GetBlockHash ( ) . ToString ( ) ,
2018-08-06 15:40:20 -04:00
hashClaimTrie . ToString ( ) ,
2014-09-03 02:20:09 +02:00
hashPrev . ToString ( ) ) ;
return str ;
}
} ;
/** An in-memory indexed chain of blocks. */
class CChain {
private :
std : : vector < CBlockIndex * > vChain ;
public :
2017-08-07 07:36:37 +02:00
/** Returns the index entry for the genesis block of this chain, or nullptr if none. */
2014-09-03 02:20:09 +02:00
CBlockIndex * Genesis ( ) const {
2017-08-07 07:36:37 +02:00
return vChain . size ( ) > 0 ? vChain [ 0 ] : nullptr ;
2014-09-03 02:20:09 +02:00
}
2017-08-07 07:36:37 +02:00
/** Returns the index entry for the tip of this chain, or nullptr if none. */
2014-09-03 02:20:09 +02:00
CBlockIndex * Tip ( ) const {
2017-08-07 07:36:37 +02:00
return vChain . size ( ) > 0 ? vChain [ vChain . size ( ) - 1 ] : nullptr ;
2014-09-03 02:20:09 +02:00
}
2017-08-07 07:36:37 +02:00
/** Returns the index entry at a particular height in this chain, or nullptr if no such height exists. */
2014-09-03 02:20:09 +02:00
CBlockIndex * operator [ ] ( int nHeight ) const {
if ( nHeight < 0 | | nHeight > = ( int ) vChain . size ( ) )
2017-08-07 07:36:37 +02:00
return nullptr ;
2014-09-03 02:20:09 +02:00
return vChain [ nHeight ] ;
}
/** Compare two chains efficiently. */
friend bool operator = = ( const CChain & a , const CChain & b ) {
return a . vChain . size ( ) = = b . vChain . size ( ) & &
a . vChain [ a . vChain . size ( ) - 1 ] = = b . vChain [ b . vChain . size ( ) - 1 ] ;
}
/** Efficiently check whether a block is present in this chain. */
bool Contains ( const CBlockIndex * pindex ) const {
return ( * this ) [ pindex - > nHeight ] = = pindex ;
}
2017-08-07 07:36:37 +02:00
/** Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip. */
2014-09-03 02:20:09 +02:00
CBlockIndex * Next ( const CBlockIndex * pindex ) const {
if ( Contains ( pindex ) )
return ( * this ) [ pindex - > nHeight + 1 ] ;
else
2017-08-07 07:36:37 +02:00
return nullptr ;
2014-09-03 02:20:09 +02:00
}
/** Return the maximal height in the chain. Is equal to chain.Tip() ? chain.Tip()->nHeight : -1. */
int Height ( ) const {
return vChain . size ( ) - 1 ;
}
2014-10-19 21:41:37 -04:00
/** Set/initialize a chain with a given tip. */
void SetTip ( CBlockIndex * pindex ) ;
2014-09-03 02:20:09 +02:00
/** Return a CBlockLocator that refers to a block in this chain (by default the tip). */
2017-08-07 07:36:37 +02:00
CBlockLocator GetLocator ( const CBlockIndex * pindex = nullptr ) const ;
2014-09-03 02:20:09 +02:00
/** Find the last common block between this chain and a block index entry. */
const CBlockIndex * FindFork ( const CBlockIndex * pindex ) const ;
2016-06-16 15:57:48 +01:00
2019-03-26 16:46:22 +00:00
/** Find the earliest block with timestamp equal or greater than the given time and height equal or greater than the given height. */
CBlockIndex * FindEarliestAtLeast ( int64_t nTime , int height ) const ;
2014-09-03 02:20:09 +02:00
} ;
2014-11-03 16:16:40 +01:00
# endif // BITCOIN_CHAIN_H