|
@ -300,7 +300,7 @@ void BlockChainSync::onPeerNewBlock(std::shared_ptr<EthereumPeer> _peer, RLP con |
|
|
u256 totalDifficulty = _r[1].toInt<u256>(); |
|
|
u256 totalDifficulty = _r[1].toInt<u256>(); |
|
|
if (totalDifficulty > _peer->m_totalDifficulty) |
|
|
if (totalDifficulty > _peer->m_totalDifficulty) |
|
|
{ |
|
|
{ |
|
|
clog(NetMessageDetail) << "Received block with no known parent. Resyncing..."; |
|
|
clog(NetMessageDetail) << "Received block with no known parent. Peer needs syncing..."; |
|
|
resetSyncFor(_peer, h, totalDifficulty); |
|
|
resetSyncFor(_peer, h, totalDifficulty); |
|
|
} |
|
|
} |
|
|
break; |
|
|
break; |
|
@ -648,12 +648,10 @@ void PV60Sync::noteDoneBlocks(std::shared_ptr<EthereumPeer> _peer, bool _clemenc |
|
|
clog(NetNote) << "Chain download failed. Aborted while incomplete."; |
|
|
clog(NetNote) << "Chain download failed. Aborted while incomplete."; |
|
|
else |
|
|
else |
|
|
{ |
|
|
{ |
|
|
// Done our chain-get.
|
|
|
// This can happen when the leading peer aborts and the one that is selected instead does not have all the blocks.
|
|
|
clog(NetWarn) << "Chain download failed. Peer with blocks didn't have them all. This peer is bad and should be punished."; |
|
|
// Just stop syncing to this peer. Sync will restart if there are no more peers to sync with.
|
|
|
clog(NetWarn) << downloadMan().remaining(); |
|
|
clog(NetNote) << "Peer does not have required blocks"; |
|
|
clog(NetWarn) << "WOULD BAN."; |
|
|
resetNeedsSyncing(_peer); |
|
|
// m_banned.insert(_peer->session()->id()); // We know who you are!
|
|
|
|
|
|
// _peer->disable("Peer sent hashes but was unable to provide the blocks.");
|
|
|
|
|
|
} |
|
|
} |
|
|
resetSync(); |
|
|
resetSync(); |
|
|
downloadMan().reset(); |
|
|
downloadMan().reset(); |
|
@ -747,7 +745,7 @@ void PV60Sync::onPeerNewHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& |
|
|
RecursiveGuard l(x_sync); |
|
|
RecursiveGuard l(x_sync); |
|
|
if (isSyncing() && (m_state != SyncState::NewBlocks || isSyncing(_peer))) |
|
|
if (isSyncing() && (m_state != SyncState::NewBlocks || isSyncing(_peer))) |
|
|
{ |
|
|
{ |
|
|
clog(NetMessageSummary) << "Ignoring since we're already downloading."; |
|
|
clog(NetMessageDetail) << "Ignoring new hashes since we're already downloading."; |
|
|
return; |
|
|
return; |
|
|
} |
|
|
} |
|
|
clog(NetMessageDetail) << "Not syncing and new block hash discovered: syncing without help."; |
|
|
clog(NetMessageDetail) << "Not syncing and new block hash discovered: syncing without help."; |
|
@ -838,7 +836,7 @@ void PV60Sync::onPeerAborting() |
|
|
// Can't check invariants here since the peers is already removed from the list and the state is not updated yet.
|
|
|
// Can't check invariants here since the peers is already removed from the list and the state is not updated yet.
|
|
|
if (m_syncer.expired() && m_state != SyncState::Idle) |
|
|
if (m_syncer.expired() && m_state != SyncState::Idle) |
|
|
{ |
|
|
{ |
|
|
clog(NetWarn) << "Syncing peer disconnected, restarting sync"; |
|
|
clog(NetNote) << "Syncing peer disconnected"; |
|
|
m_syncer.reset(); |
|
|
m_syncer.reset(); |
|
|
abortSync(); |
|
|
abortSync(); |
|
|
} |
|
|
} |
|
@ -969,11 +967,17 @@ void PV61Sync::completeSubchain(std::shared_ptr<EthereumPeer> _peer, unsigned _n |
|
|
{ |
|
|
{ |
|
|
//Done chain-get
|
|
|
//Done chain-get
|
|
|
m_syncingNeededBlocks.clear(); |
|
|
m_syncingNeededBlocks.clear(); |
|
|
|
|
|
// Add hashes to download skipping onces that are already downloaded
|
|
|
for (auto h = m_completeChainMap.rbegin(); h != m_completeChainMap.rend(); ++h) |
|
|
for (auto h = m_completeChainMap.rbegin(); h != m_completeChainMap.rend(); ++h) |
|
|
|
|
|
if (!host().chain().isKnown(h->second.hashes.front()) && !host().chain().isKnown(h->second.hashes.back())) |
|
|
|
|
|
{ |
|
|
|
|
|
if (host().bq().blockStatus(h->second.hashes.front()) == QueueStatus::Unknown || host().bq().blockStatus(h->second.hashes.back()) == QueueStatus::Unknown) |
|
|
m_syncingNeededBlocks.insert(m_syncingNeededBlocks.end(), h->second.hashes.begin(), h->second.hashes.end()); |
|
|
m_syncingNeededBlocks.insert(m_syncingNeededBlocks.end(), h->second.hashes.begin(), h->second.hashes.end()); |
|
|
m_completeChainMap.clear(); |
|
|
else |
|
|
m_knownHashes.clear(); |
|
|
for (h256 const& hash: h->second.hashes) |
|
|
m_syncingBlockNumber = 0; |
|
|
if (!host().chain().isKnown(hash) && host().bq().blockStatus(hash) == QueueStatus::Unknown) |
|
|
|
|
|
m_syncingNeededBlocks.insert(m_syncingNeededBlocks.end(), hash); |
|
|
|
|
|
} |
|
|
transition(syncer, SyncState::Blocks); |
|
|
transition(syncer, SyncState::Blocks); |
|
|
} |
|
|
} |
|
|
else |
|
|
else |
|
@ -1007,6 +1011,7 @@ void PV61Sync::onPeerHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _h |
|
|
{ |
|
|
{ |
|
|
// End of hash chain, add last chunk to download
|
|
|
// End of hash chain, add last chunk to download
|
|
|
m_readyChainMap.insert(make_pair(m_syncingBlockNumber, SubChain{ h256s{ _peer->m_latestHash }, _peer->m_latestHash })); |
|
|
m_readyChainMap.insert(make_pair(m_syncingBlockNumber, SubChain{ h256s{ _peer->m_latestHash }, _peer->m_latestHash })); |
|
|
|
|
|
m_knownHashes.insert(_peer->m_latestHash); |
|
|
m_hashScanComplete = true; |
|
|
m_hashScanComplete = true; |
|
|
_peer->m_syncHashNumber = 0; |
|
|
_peer->m_syncHashNumber = 0; |
|
|
requestSubchain(_peer); |
|
|
requestSubchain(_peer); |
|
@ -1032,7 +1037,13 @@ void PV61Sync::onPeerHashes(std::shared_ptr<EthereumPeer> _peer, h256s const& _h |
|
|
if (isSyncing(_peer) && _peer->m_syncHashNumber == m_syncingBlockNumber) |
|
|
if (isSyncing(_peer) && _peer->m_syncHashNumber == m_syncingBlockNumber) |
|
|
{ |
|
|
{ |
|
|
// Got new subchain marker
|
|
|
// Got new subchain marker
|
|
|
assert(_hashes.size() == 1); |
|
|
if (_hashes.size() != 1) |
|
|
|
|
|
{ |
|
|
|
|
|
clog(NetWarn) << "Peer sent too many hashes"; |
|
|
|
|
|
_peer->disable("Too many hashes"); |
|
|
|
|
|
restartSync(); |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
m_knownHashes.insert(_hashes[0]); |
|
|
m_knownHashes.insert(_hashes[0]); |
|
|
m_readyChainMap.insert(make_pair(m_syncingBlockNumber, SubChain{ h256s{ _hashes[0] }, _hashes[0] })); |
|
|
m_readyChainMap.insert(make_pair(m_syncingBlockNumber, SubChain{ h256s{ _hashes[0] }, _hashes[0] })); |
|
|
if ((m_readyChainMap.size() + m_downloadingChainMap.size() + m_completeChainMap.size()) * c_hashSubchainSize > _peer->m_expectedHashes) |
|
|
if ((m_readyChainMap.size() + m_downloadingChainMap.size() + m_completeChainMap.size()) * c_hashSubchainSize > _peer->m_expectedHashes) |
|
@ -1186,6 +1197,14 @@ bool PV61Sync::isPV61Syncing() const |
|
|
return m_syncingBlockNumber != 0; |
|
|
return m_syncingBlockNumber != 0; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void PV61Sync::completeSync() |
|
|
|
|
|
{ |
|
|
|
|
|
m_completeChainMap.clear(); |
|
|
|
|
|
m_knownHashes.clear(); |
|
|
|
|
|
m_syncingBlockNumber = 0; |
|
|
|
|
|
PV60Sync::completeSync(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
bool PV61Sync::invariants() const |
|
|
bool PV61Sync::invariants() const |
|
|
{ |
|
|
{ |
|
|
if (m_state == SyncState::Hashes) |
|
|
if (m_state == SyncState::Hashes) |
|
|