Bitcoin ABC 0.30.5
P2P Digital Currency
|
#include <headerssync.h>
Classes | |
struct | ProcessingResult |
Result data structure for ProcessNextHeaders. More... | |
Public Types | |
enum class | State { PRESYNC , REDOWNLOAD , FINAL } |
Public Member Functions | |
~HeadersSyncState () | |
State | GetState () const |
Return the current state of our download. More... | |
int64_t | GetPresyncHeight () const |
Return the height reached during the PRESYNC phase. More... | |
uint32_t | GetPresyncTime () const |
Return the block timestamp of the last header received during the PRESYNC phase. More... | |
arith_uint256 | GetPresyncWork () const |
Return the amount of work in the chain received during the PRESYNC phase. More... | |
HeadersSyncState (NodeId id, const Consensus::Params &consensus_params, const CBlockIndex *chain_start, const arith_uint256 &minimum_required_work) | |
Construct a HeadersSyncState object representing a headers sync via this download-twice mechanism). More... | |
ProcessingResult | ProcessNextHeaders (const std::vector< CBlockHeader > &received_headers, bool full_headers_message) |
Process a batch of headers, once a sync via this mechanism has started. More... | |
CBlockLocator | NextHeadersRequestLocator () const |
Issue the next GETHEADERS message to our peer. More... | |
Private Member Functions | |
void | Finalize () |
Clear out all download state that might be in progress (freeing any used memory), and mark this object as no longer usable. More... | |
bool | ValidateAndStoreHeadersCommitments (const std::vector< CBlockHeader > &headers) |
Only called in PRESYNC. More... | |
bool | ValidateAndProcessSingleHeader (const CBlockHeader ¤t) |
In PRESYNC, process and update state for a single header. More... | |
bool | ValidateAndStoreRedownloadedHeader (const CBlockHeader &header) |
In REDOWNLOAD, check a header's commitment (if applicable) and add to buffer for later processing. More... | |
std::vector< CBlockHeader > | PopHeadersReadyForAcceptance () |
Return a set of headers that satisfy our proof-of-work threshold. More... | |
Private Attributes | |
const NodeId | m_id |
NodeId of the peer (used for log messages) More... | |
const Consensus::Params & | m_consensus_params |
We use the consensus params in our anti-DoS calculations. More... | |
const CBlockIndex * | m_chain_start {nullptr} |
Store the last block in our block index that the peer's chain builds from. More... | |
const arith_uint256 | m_minimum_required_work |
Minimum work that we're looking for on this chain. More... | |
arith_uint256 | m_current_chain_work |
Work that we've seen so far on the peer's chain. More... | |
const SaltedBlockHashHasher | m_hasher |
m_hasher is a salted hasher for making our 1-bit commitments to headers we've seen. More... | |
bitdeque | m_header_commitments |
A queue of commitment bits, created during the 1st phase, and verified during the 2nd. More... | |
const unsigned | m_commit_offset |
The (secret) offset on the heights for which to create commitments. More... | |
uint64_t | m_max_commitments {0} |
m_max_commitments is a bound we calculate on how long an honest peer's chain could be, given the MTP rule. More... | |
CBlockHeader | m_last_header_received |
Store the latest header received while in PRESYNC (initialized to m_chain_start) More... | |
int64_t | m_current_height {0} |
Height of m_last_header_received. More... | |
std::deque< CompressedHeader > | m_redownloaded_headers |
During phase 2 (REDOWNLOAD), we buffer redownloaded headers in memory until enough commitments have been verified; those are stored in m_redownloaded_headers. More... | |
int64_t | m_redownload_buffer_last_height {0} |
Height of last header in m_redownloaded_headers. More... | |
BlockHash | m_redownload_buffer_last_hash |
Hash of last header in m_redownloaded_headers (initialized to m_chain_start). More... | |
BlockHash | m_redownload_buffer_first_prev_hash |
The hashPrevBlock entry for the first header in m_redownloaded_headers We need this to reconstruct the full header when it's time for processing. More... | |
arith_uint256 | m_redownload_chain_work |
The accumulated work on the redownloaded chain. More... | |
bool | m_process_all_remaining_headers {false} |
Set this to true once we encounter the target blockheader during phase 2 (REDOWNLOAD). More... | |
State | m_download_state {State::PRESYNC} |
Current state of our headers sync. More... | |
We wish to download a peer's headers chain in a DoS-resistant way.
The Bitcoin protocol does not offer an easy way to determine the work on a peer's chain. Currently, we can query a peer's headers by using a GETHEADERS message, and our peer can return a set of up to 2000 headers that connect to something we know. If a peer's chain has more than 2000 blocks, then we need a way to verify that the chain actually has enough work on it to be useful to us – by being above our anti-DoS minimum-chain-work threshold – before we commit to storing those headers in memory. Otherwise, it would be cheap for an attacker to waste all our memory by serving us low-work headers (particularly for a new node coming online for the first time).
To prevent memory-DoS with low-work headers, while still always being able to reorg to whatever the most-work chain is, we require that a chain meet a work threshold before committing it to memory. We can do this by downloading a peer's headers twice, whenever we are not sure that the chain has sufficient work:
To prevent an attacker from using (eg) the honest chain to convince us that they have a high-work chain, but then feeding us an alternate set of low-difficulty headers in the second phase, we store commitments to the chain we see in the first download phase that we check in the second phase, as follows:
Definition at line 98 of file headerssync.h.
|
strong |
Definition at line 102 of file headerssync.h.
|
inline |
Definition at line 100 of file headerssync.h.
HeadersSyncState::HeadersSyncState | ( | NodeId | id, |
const Consensus::Params & | consensus_params, | ||
const CBlockIndex * | chain_start, | ||
const arith_uint256 & | minimum_required_work | ||
) |
Construct a HeadersSyncState object representing a headers sync via this download-twice mechanism).
id: node id (for logging) consensus_params: parameters needed for difficulty adjustment validation chain_start: best known fork point that the peer's headers branch from minimum_required_work: amount of chain work required to accept the chain
Definition at line 27 of file headerssync.cpp.
|
private |
Clear out all download state that might be in progress (freeing any used memory), and mark this object as no longer usable.
Free any memory in use, and mark this object as no longer usable.
This is required to guarantee that we won't reuse this object with the same SaltedBlockHashHasher for another sync.
Definition at line 66 of file headerssync.cpp.
|
inline |
Return the height reached during the PRESYNC phase.
Definition at line 126 of file headerssync.h.
|
inline |
Return the block timestamp of the last header received during the PRESYNC phase.
Definition at line 132 of file headerssync.h.
|
inline |
Return the amount of work in the chain received during the PRESYNC phase.
Definition at line 138 of file headerssync.h.
|
inline |
Return the current state of our download.
Definition at line 123 of file headerssync.h.
CBlockLocator HeadersSyncState::NextHeadersRequestLocator | ( | ) | const |
Issue the next GETHEADERS message to our peer.
This will return a locator appropriate for the current sync object, to continue the synchronization phase it is in.
Definition at line 365 of file headerssync.cpp.
|
private |
Return a set of headers that satisfy our proof-of-work threshold.
Definition at line 346 of file headerssync.cpp.
HeadersSyncState::ProcessingResult HeadersSyncState::ProcessNextHeaders | ( | const std::vector< CBlockHeader > & | received_headers, |
bool | full_headers_message | ||
) |
Process a batch of headers, once a sync via this mechanism has started.
Process the next batch of headers received from our peer.
received_headers: headers that were received over the network for processing. Assumes the caller has already verified the headers are continuous, and has checked that each header satisfies the proof-of-work target included in the header (but not necessarily verified that the proof-of-work target is correct and passes consensus rules). full_headers_message: true if the message was at max capacity, indicating more headers may be available ProcessingResult.pow_validated_headers: will be filled in with any headers that the caller can fully process and validate now (because these returned headers are on a chain with sufficient work) ProcessingResult.success: set to false if an error is detected and the sync is aborted; true otherwise. ProcessingResult.request_more: if true, the caller is suggested to call NextHeadersRequestLocator and send a getheaders message using it.
Validate and store commitments, and compare total chainwork to our target to see if we can switch to REDOWNLOAD mode.
Definition at line 84 of file headerssync.cpp.
|
private |
In PRESYNC, process and update state for a single header.
Definition at line 221 of file headerssync.cpp.
|
private |
Only called in PRESYNC.
Validate the work on the headers we received from the network, and store commitments for later. Update overall state with successfully processed headers. On failure, this invokes Finalize() and returns false.
Definition at line 173 of file headerssync.cpp.
|
private |
In REDOWNLOAD, check a header's commitment (if applicable) and add to buffer for later processing.
Definition at line 266 of file headerssync.cpp.
|
private |
Store the last block in our block index that the peer's chain builds from.
Definition at line 235 of file headerssync.h.
|
private |
The (secret) offset on the heights for which to create commitments.
m_header_commitments entries are created at any height h for which (h % HEADER_COMMITMENT_PERIOD) == m_commit_offset.
Definition at line 261 of file headerssync.h.
|
private |
We use the consensus params in our anti-DoS calculations.
Definition at line 229 of file headerssync.h.
|
private |
Work that we've seen so far on the peer's chain.
Definition at line 241 of file headerssync.h.
|
private |
Height of m_last_header_received.
Definition at line 279 of file headerssync.h.
|
private |
Current state of our headers sync.
Definition at line 316 of file headerssync.h.
|
private |
m_hasher is a salted hasher for making our 1-bit commitments to headers we've seen.
Definition at line 247 of file headerssync.h.
|
private |
A queue of commitment bits, created during the 1st phase, and verified during the 2nd.
Definition at line 253 of file headerssync.h.
|
private |
NodeId of the peer (used for log messages)
Definition at line 226 of file headerssync.h.
|
private |
Store the latest header received while in PRESYNC (initialized to m_chain_start)
Definition at line 276 of file headerssync.h.
|
private |
m_max_commitments is a bound we calculate on how long an honest peer's chain could be, given the MTP rule.
Any peer giving us more headers than this will have its sync aborted. This serves as a memory bound on m_header_commitments.
Definition at line 270 of file headerssync.h.
|
private |
Minimum work that we're looking for on this chain.
Definition at line 238 of file headerssync.h.
|
private |
Set this to true once we encounter the target blockheader during phase 2 (REDOWNLOAD).
At this point, we can process and store all remaining headers still in m_redownloaded_headers.
Definition at line 313 of file headerssync.h.
|
private |
The hashPrevBlock entry for the first header in m_redownloaded_headers We need this to reconstruct the full header when it's time for processing.
Definition at line 303 of file headerssync.h.
|
private |
Hash of last header in m_redownloaded_headers (initialized to m_chain_start).
We have to cache it because we don't have hashPrevBlock available in a CompressedHeader.
Definition at line 296 of file headerssync.h.
|
private |
Height of last header in m_redownloaded_headers.
Definition at line 289 of file headerssync.h.
|
private |
The accumulated work on the redownloaded chain.
Definition at line 306 of file headerssync.h.
|
private |
During phase 2 (REDOWNLOAD), we buffer redownloaded headers in memory until enough commitments have been verified; those are stored in m_redownloaded_headers.
Definition at line 286 of file headerssync.h.