You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
320 lines
11 KiB
320 lines
11 KiB
/*
|
|
This file is part of cpp-ethereum.
|
|
|
|
cpp-ethereum is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
cpp-ethereum is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/** @file BlockChainSync.h
|
|
* @author Gav Wood <i@gavwood.com>
|
|
* @date 2014
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <mutex>
|
|
|
|
#include <libdevcore/Guards.h>
|
|
#include <libethcore/Common.h>
|
|
#include <libp2p/Common.h>
|
|
#include "CommonNet.h"
|
|
#include "DownloadMan.h"
|
|
|
|
namespace dev
|
|
{
|
|
|
|
class RLPStream;
|
|
|
|
namespace eth
|
|
{
|
|
|
|
class EthereumHost;
|
|
class BlockQueue;
|
|
class EthereumPeer;
|
|
|
|
/**
|
|
* @brief Base BlockChain synchronization strategy class.
|
|
* Syncs to peers and keeps up to date. Base class handles blocks downloading but does not contain any details on state transfer logic.
|
|
*/
|
|
class BlockChainSync: public HasInvariants
|
|
{
|
|
public:
|
|
BlockChainSync(EthereumHost& _host);
|
|
virtual ~BlockChainSync();
|
|
void abortSync(); ///< Abort all sync activity
|
|
|
|
DownloadMan const& downloadMan() const;
|
|
DownloadMan& downloadMan();
|
|
|
|
/// @returns true is Sync is in progress
|
|
virtual bool isSyncing() const = 0;
|
|
|
|
/// Called by peer to report status
|
|
virtual void onPeerStatus(std::shared_ptr<EthereumPeer> _peer);
|
|
|
|
/// Called by peer once it has new blocks during syn
|
|
virtual void onPeerBlocks(std::shared_ptr<EthereumPeer> _peer, RLP const& _r);
|
|
|
|
/// Called by peer once it has new blocks
|
|
virtual void onPeerNewBlock(std::shared_ptr<EthereumPeer> _peer, RLP const& _r);
|
|
|
|
/// Called by peer once it has new hashes
|
|
virtual void onPeerNewHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _hashes) = 0;
|
|
|
|
/// Called by peer once it has another sequential block of hashes during sync
|
|
virtual void onPeerHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _hashes) = 0;
|
|
|
|
/// Called by peer when it is disconnecting
|
|
virtual void onPeerAborting() = 0;
|
|
|
|
/// @returns Synchonization status
|
|
virtual SyncStatus status() const = 0;
|
|
|
|
static char const* stateName(SyncState _s) { return s_stateNames[static_cast<int>(_s)]; }
|
|
|
|
protected:
|
|
//To be implemented in derived classes:
|
|
/// New valid peer appears
|
|
virtual void onNewPeer(std::shared_ptr<EthereumPeer> _peer) = 0;
|
|
|
|
/// Peer done downloading blocks
|
|
virtual void peerDoneBlocks(std::shared_ptr<EthereumPeer> _peer) = 0;
|
|
|
|
/// Resume downloading after witing state
|
|
virtual void continueSync() = 0;
|
|
|
|
/// Restart sync
|
|
virtual void restartSync() = 0;
|
|
|
|
/// Called after all blocks have been donloaded
|
|
virtual void completeSync() = 0;
|
|
|
|
/// Enter waiting state
|
|
virtual void pauseSync() = 0;
|
|
|
|
/// Restart sync for given peer
|
|
virtual void resetSyncFor(std::shared_ptr<EthereumPeer> _peer, h256 const& _latestHash, u256 const& _td) = 0;
|
|
|
|
EthereumHost& host() { return m_host; }
|
|
EthereumHost const& host() const { return m_host; }
|
|
|
|
/// Estimates max number of hashes peers can give us.
|
|
unsigned estimatedHashes() const;
|
|
|
|
/// Request blocks from peer if needed
|
|
void requestBlocks(std::shared_ptr<EthereumPeer> _peer);
|
|
|
|
protected:
|
|
Handler<> m_bqRoomAvailable; ///< Triggered once block queue
|
|
mutable RecursiveMutex x_sync;
|
|
SyncState m_state = SyncState::Idle; ///< Current sync state
|
|
unsigned m_estimatedHashes = 0; ///< Number of estimated hashes for the last peer over PV60. Used for status reporting only.
|
|
h256Hash m_knownNewHashes; ///< New hashes we know about use for logging only
|
|
|
|
private:
|
|
static char const* const s_stateNames[static_cast<int>(SyncState::Size)];
|
|
bool invariants() const override = 0;
|
|
void logNewBlock(h256 const& _h);
|
|
|
|
EthereumHost& m_host;
|
|
HashDownloadMan m_hashMan;
|
|
};
|
|
|
|
|
|
/**
|
|
* @brief Syncrhonization over PV60. Selects a single peer and tries to downloading hashes from it. After hash download is complete
|
|
* syncs to peers and keeps up to date
|
|
*/
|
|
|
|
/**
|
|
* Transitions:
|
|
*
|
|
* Idle->Hashes
|
|
* Triggered when:
|
|
* * A new peer appears that we can sync to
|
|
* * Transtition to Idle, there are peers we can sync to
|
|
* Effects:
|
|
* * Set chain sync (m_syncingTotalDifficulty, m_syncingLatestHash, m_syncer)
|
|
* * Requests hashes from m_syncer
|
|
*
|
|
* Hashes->Idle
|
|
* Triggered when:
|
|
* * Received too many hashes
|
|
* * Received 0 total hashes from m_syncer
|
|
* * m_syncer aborts
|
|
* Effects:
|
|
* In case of too many hashes sync is reset
|
|
*
|
|
* Hashes->Blocks
|
|
* Triggered when:
|
|
* * Received known hash from m_syncer
|
|
* * Received 0 hashes from m_syncer and m_syncingTotalBlocks not empty
|
|
* Effects:
|
|
* * Set up download manager, clear m_syncingTotalBlocks. Set all peers to help with downloading if they can
|
|
*
|
|
* Blocks->Idle
|
|
* Triggered when:
|
|
* * m_syncer aborts
|
|
* * m_syncer does not have required block
|
|
* * All blocks downloaded
|
|
* * Block qeueue is full with unknown blocks
|
|
* Effects:
|
|
* * Download manager is reset
|
|
*
|
|
* Blocks->Waiting
|
|
* Triggered when:
|
|
* * Block queue is full with known blocks
|
|
* Effects:
|
|
* * Stop requesting blocks from peers
|
|
*
|
|
* Waiting->Blocks
|
|
* Triggered when:
|
|
* * Block queue has space for new blocks
|
|
* Effects:
|
|
* * Continue requesting blocks from peers
|
|
*
|
|
* Idle->NewBlocks
|
|
* Triggered when:
|
|
* * New block hashes arrive
|
|
* Effects:
|
|
* * Set up download manager, clear m_syncingTotalBlocks. Download blocks from a single peer. If downloaded blocks have unknown parents, set the peer to sync
|
|
*
|
|
* NewBlocks->Idle
|
|
* Triggered when:
|
|
* * m_syncer aborts
|
|
* * m_syncer does not have required block
|
|
* * All new blocks downloaded
|
|
* * Block qeueue is full with unknown blocks
|
|
* Effects:
|
|
* * Download manager is reset
|
|
*
|
|
*/
|
|
class PV60Sync: public BlockChainSync
|
|
{
|
|
public:
|
|
PV60Sync(EthereumHost& _host);
|
|
|
|
/// @returns true is Sync is in progress
|
|
bool isSyncing() const override { return !!m_syncer.lock(); }
|
|
|
|
/// Called by peer once it has new hashes
|
|
void onPeerNewHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _hashes) override;
|
|
|
|
/// Called by peer once it has another sequential block of hashes during sync
|
|
void onPeerHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _hashes) override;
|
|
|
|
/// Called by peer when it is disconnecting
|
|
void onPeerAborting() override;
|
|
|
|
/// @returns Sync status
|
|
SyncStatus status() const override;
|
|
|
|
protected:
|
|
void onNewPeer(std::shared_ptr<EthereumPeer> _peer) override;
|
|
void continueSync() override;
|
|
void peerDoneBlocks(std::shared_ptr<EthereumPeer> _peer) override;
|
|
void restartSync() override;
|
|
void completeSync() override;
|
|
void pauseSync() override;
|
|
void resetSyncFor(std::shared_ptr<EthereumPeer> _peer, h256 const& _latestHash, u256 const& _td) override;
|
|
|
|
protected:
|
|
/// Transition sync state in a particular direction. @param _peer Peer that is responsible for state tranfer
|
|
void transition(std::shared_ptr<EthereumPeer> _peer, SyncState _s, bool _force = false, bool _needHelp = true);
|
|
|
|
/// Reset peer syncing requirements state.
|
|
void resetNeedsSyncing(std::shared_ptr<EthereumPeer> _peer) { setNeedsSyncing(_peer, h256(), 0); }
|
|
|
|
/// Update peer syncing requirements state.
|
|
void setNeedsSyncing(std::shared_ptr<EthereumPeer> _peer, h256 const& _latestHash, u256 const& _td);
|
|
|
|
/// Do we presently need syncing with this peer?
|
|
bool needsSyncing(std::shared_ptr<EthereumPeer> _peer) const;
|
|
|
|
/// Check whether the session should bother grabbing blocks from a peer.
|
|
bool shouldGrabBlocks(std::shared_ptr<EthereumPeer> _peer) const;
|
|
|
|
/// Attempt to begin syncing with the peer; first check the peer has a more difficlult chain to download, then start asking for hashes, then move to blocks
|
|
void attemptSync(std::shared_ptr<EthereumPeer> _peer);
|
|
|
|
/// Update our syncing state
|
|
void setState(std::shared_ptr<EthereumPeer> _peer, SyncState _s, bool _isSyncing = false, bool _needHelp = false);
|
|
|
|
/// Check if peer is main syncer
|
|
bool isSyncing(std::shared_ptr<EthereumPeer> _peer) const;
|
|
|
|
/// Check if we need (re-)syncing with the peer.
|
|
void noteNeedsSyncing(std::shared_ptr<EthereumPeer> _who);
|
|
|
|
/// Set main syncing peer
|
|
void changeSyncer(std::shared_ptr<EthereumPeer> _syncer, bool _needHelp);
|
|
|
|
/// Called when peer done downloading blocks
|
|
void noteDoneBlocks(std::shared_ptr<EthereumPeer> _who, bool _clemency);
|
|
|
|
/// Start chainhash sync
|
|
virtual void syncHashes(std::shared_ptr<EthereumPeer> _peer);
|
|
|
|
/// Request subchain, no-op for pv60
|
|
virtual void requestSubchain(std::shared_ptr<EthereumPeer> /*_peer*/) {}
|
|
|
|
/// Abort syncing
|
|
void abortSync();
|
|
|
|
/// Reset hash chain syncing
|
|
void resetSync();
|
|
|
|
bool invariants() const override;
|
|
|
|
h256s m_syncingNeededBlocks; ///< The blocks that we should download from this peer.
|
|
h256 m_syncingLastReceivedHash; ///< Hash most recently received from peer.
|
|
h256 m_syncingLatestHash; ///< Latest block's hash of the peer we are syncing to, as of the current sync.
|
|
u256 m_syncingTotalDifficulty; ///< Latest block's total difficulty of the peer we aresyncing to, as of the current sync.
|
|
std::weak_ptr<EthereumPeer> m_syncer; ///< Peer we are currently syncing with
|
|
};
|
|
|
|
/**
|
|
* @brief Syncrhonization over PV61. Selects a single peer and requests every c_hashSubchainSize hash, splitting the hashchain into subchains and downloading each subchain in parallel.
|
|
* Syncs to peers and keeps up to date
|
|
*/
|
|
class PV61Sync: public PV60Sync
|
|
{
|
|
public:
|
|
PV61Sync(EthereumHost& _host);
|
|
|
|
protected:
|
|
void restartSync() override;
|
|
void requestSubchain(std::shared_ptr<EthereumPeer> _peer) override;
|
|
void syncHashes(std::shared_ptr<EthereumPeer> _peer) override;
|
|
void onPeerHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _hashes) override;
|
|
void onPeerAborting() override;
|
|
SyncStatus status() const override;
|
|
bool invariants() const override;
|
|
|
|
private:
|
|
/// Called when subchain is complete. Check if if hashchain is fully downloaded and proceed to downloading blocks
|
|
void completeSubchain(std::shared_ptr<EthereumPeer> _peer, unsigned _n);
|
|
/// Find a subchain for peers to downloading
|
|
void requestSubchains();
|
|
/// Check if downloading hashes in parallel
|
|
bool isPV61Syncing() const;
|
|
|
|
std::map<unsigned, h256s> m_completeChainMap; ///< Fully downloaded subchains
|
|
std::map<unsigned, h256s> m_readyChainMap; ///< Subchains ready for download
|
|
std::map<unsigned, h256s> m_downloadingChainMap; ///< Subchains currently being downloading. In sync with m_chainSyncPeers
|
|
std::map<std::weak_ptr<EthereumPeer>, unsigned, std::owner_less<std::weak_ptr<EthereumPeer>>> m_chainSyncPeers; ///< Peers to m_downloadingSubchain number map
|
|
h256Hash m_knownHashes; ///< Subchain start markers. Used to track suchain completion
|
|
unsigned m_syncingBlockNumber = 0; ///< Current subchain marker
|
|
bool m_hashScanComplete = false; ///< True if leading peer completed hashchain scan and we have a list of subchains ready
|
|
};
|
|
}
|
|
}
|
|
|