Merge pull request #3477 from ethcore/sync-opt
Sync traffic optimization
This commit is contained in:
commit
2532b1527e
@ -33,6 +33,7 @@ const MAX_BODIES_TO_REQUEST: usize = 64;
|
|||||||
const MAX_RECEPITS_TO_REQUEST: usize = 128;
|
const MAX_RECEPITS_TO_REQUEST: usize = 128;
|
||||||
const SUBCHAIN_SIZE: u64 = 256;
|
const SUBCHAIN_SIZE: u64 = 256;
|
||||||
const MAX_ROUND_PARENTS: usize = 32;
|
const MAX_ROUND_PARENTS: usize = 32;
|
||||||
|
const MAX_PARALLEL_SUBCHAIN_DOWNLOAD: usize = 5;
|
||||||
|
|
||||||
#[derive(Copy, Clone, Eq, PartialEq, Debug)]
|
#[derive(Copy, Clone, Eq, PartialEq, Debug)]
|
||||||
/// Downloader state
|
/// Downloader state
|
||||||
@ -62,6 +63,14 @@ pub enum BlockRequest {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Indicates sync action
|
||||||
|
pub enum DownloadAction {
|
||||||
|
/// Do nothing
|
||||||
|
None,
|
||||||
|
/// Reset downloads for all peers
|
||||||
|
Reset
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Eq, PartialEq, Debug)]
|
#[derive(Eq, PartialEq, Debug)]
|
||||||
pub enum BlockDownloaderImportError {
|
pub enum BlockDownloaderImportError {
|
||||||
/// Imported data is rejected as invalid.
|
/// Imported data is rejected as invalid.
|
||||||
@ -175,11 +184,11 @@ impl BlockDownloader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Add new block headers.
|
/// Add new block headers.
|
||||||
pub fn import_headers(&mut self, io: &mut SyncIo, r: &UntrustedRlp, expected_hash: Option<H256>) -> Result<(), BlockDownloaderImportError> {
|
pub fn import_headers(&mut self, io: &mut SyncIo, r: &UntrustedRlp, expected_hash: Option<H256>) -> Result<DownloadAction, BlockDownloaderImportError> {
|
||||||
let item_count = r.item_count();
|
let item_count = r.item_count();
|
||||||
if self.state == State::Idle {
|
if self.state == State::Idle {
|
||||||
trace!(target: "sync", "Ignored unexpected block headers");
|
trace!(target: "sync", "Ignored unexpected block headers");
|
||||||
return Ok(())
|
return Ok(DownloadAction::None)
|
||||||
}
|
}
|
||||||
if item_count == 0 && (self.state == State::Blocks) {
|
if item_count == 0 && (self.state == State::Blocks) {
|
||||||
return Err(BlockDownloaderImportError::Invalid);
|
return Err(BlockDownloaderImportError::Invalid);
|
||||||
@ -188,6 +197,7 @@ impl BlockDownloader {
|
|||||||
let mut headers = Vec::new();
|
let mut headers = Vec::new();
|
||||||
let mut hashes = Vec::new();
|
let mut hashes = Vec::new();
|
||||||
let mut valid_response = item_count == 0; //empty response is valid
|
let mut valid_response = item_count == 0; //empty response is valid
|
||||||
|
let mut any_known = false;
|
||||||
for i in 0..item_count {
|
for i in 0..item_count {
|
||||||
let info: BlockHeader = try!(r.val_at(i).map_err(|e| {
|
let info: BlockHeader = try!(r.val_at(i).map_err(|e| {
|
||||||
trace!(target: "sync", "Error decoding block header RLP: {:?}", e);
|
trace!(target: "sync", "Error decoding block header RLP: {:?}", e);
|
||||||
@ -200,6 +210,7 @@ impl BlockDownloader {
|
|||||||
valid_response = expected == info.hash()
|
valid_response = expected == info.hash()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
any_known = any_known || self.blocks.contains_head(&info.hash());
|
||||||
if self.blocks.contains(&info.hash()) {
|
if self.blocks.contains(&info.hash()) {
|
||||||
trace!(target: "sync", "Skipping existing block header {} ({:?})", number, info.hash());
|
trace!(target: "sync", "Skipping existing block header {} ({:?})", number, info.hash());
|
||||||
continue;
|
continue;
|
||||||
@ -245,17 +256,22 @@ impl BlockDownloader {
|
|||||||
trace!(target: "sync", "Received {} subchain heads, proceeding to download", headers.len());
|
trace!(target: "sync", "Received {} subchain heads, proceeding to download", headers.len());
|
||||||
self.blocks.reset_to(hashes);
|
self.blocks.reset_to(hashes);
|
||||||
self.state = State::Blocks;
|
self.state = State::Blocks;
|
||||||
|
return Ok(DownloadAction::Reset);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
State::Blocks => {
|
State::Blocks => {
|
||||||
let count = headers.len();
|
let count = headers.len();
|
||||||
|
// At least one of the heades must advance the subchain. Otherwise they are all useless.
|
||||||
|
if !any_known {
|
||||||
|
return Err(BlockDownloaderImportError::Useless);
|
||||||
|
}
|
||||||
self.blocks.insert_headers(headers);
|
self.blocks.insert_headers(headers);
|
||||||
trace!(target: "sync", "Inserted {} headers", count);
|
trace!(target: "sync", "Inserted {} headers", count);
|
||||||
},
|
},
|
||||||
_ => trace!(target: "sync", "Unexpected headers({})", headers.len()),
|
_ => trace!(target: "sync", "Unexpected headers({})", headers.len()),
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(DownloadAction::None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Called by peer once it has new block bodies
|
/// Called by peer once it has new block bodies
|
||||||
@ -342,22 +358,24 @@ impl BlockDownloader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Find some headers or blocks to download for a peer.
|
/// Find some headers or blocks to download for a peer.
|
||||||
pub fn request_blocks(&mut self, io: &mut SyncIo) -> Option<BlockRequest> {
|
pub fn request_blocks(&mut self, io: &mut SyncIo, num_active_peers: usize) -> Option<BlockRequest> {
|
||||||
match self.state {
|
match self.state {
|
||||||
State::Idle => {
|
State::Idle => {
|
||||||
self.start_sync_round(io);
|
self.start_sync_round(io);
|
||||||
return self.request_blocks(io);
|
return self.request_blocks(io, num_active_peers);
|
||||||
},
|
},
|
||||||
State::ChainHead => {
|
State::ChainHead => {
|
||||||
// Request subchain headers
|
if num_active_peers < MAX_PARALLEL_SUBCHAIN_DOWNLOAD {
|
||||||
trace!(target: "sync", "Starting sync with better chain");
|
// Request subchain headers
|
||||||
// Request MAX_HEADERS_TO_REQUEST - 2 headers apart so that
|
trace!(target: "sync", "Starting sync with better chain");
|
||||||
// MAX_HEADERS_TO_REQUEST would include headers for neighbouring subchains
|
// Request MAX_HEADERS_TO_REQUEST - 2 headers apart so that
|
||||||
return Some(BlockRequest::Headers {
|
// MAX_HEADERS_TO_REQUEST would include headers for neighbouring subchains
|
||||||
start: self.last_imported_hash.clone(),
|
return Some(BlockRequest::Headers {
|
||||||
count: SUBCHAIN_SIZE,
|
start: self.last_imported_hash.clone(),
|
||||||
skip: (MAX_HEADERS_TO_REQUEST - 2) as u64,
|
count: SUBCHAIN_SIZE,
|
||||||
});
|
skip: (MAX_HEADERS_TO_REQUEST - 2) as u64,
|
||||||
|
});
|
||||||
|
}
|
||||||
},
|
},
|
||||||
State::Blocks => {
|
State::Blocks => {
|
||||||
// check to see if we need to download any block bodies first
|
// check to see if we need to download any block bodies first
|
||||||
|
@ -301,11 +301,16 @@ impl BlockCollection {
|
|||||||
self.heads.len() == 0 || (self.heads.len() == 1 && self.head.map_or(false, |h| h == self.heads[0]))
|
self.heads.len() == 0 || (self.heads.len() == 1 && self.head.map_or(false, |h| h == self.heads[0]))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Chech is collection contains a block header.
|
/// Check if collection contains a block header.
|
||||||
pub fn contains(&self, hash: &H256) -> bool {
|
pub fn contains(&self, hash: &H256) -> bool {
|
||||||
self.blocks.contains_key(hash)
|
self.blocks.contains_key(hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Check if collection contains a block header.
|
||||||
|
pub fn contains_head(&self, hash: &H256) -> bool {
|
||||||
|
self.heads.contains(hash)
|
||||||
|
}
|
||||||
|
|
||||||
/// Return used heap size.
|
/// Return used heap size.
|
||||||
pub fn heap_size(&self) -> usize {
|
pub fn heap_size(&self) -> usize {
|
||||||
self.heads.heap_size_of_children()
|
self.heads.heap_size_of_children()
|
||||||
|
@ -37,7 +37,7 @@
|
|||||||
/// Workflow for `ChainHead` state.
|
/// Workflow for `ChainHead` state.
|
||||||
/// In this state we try to get subchain headers with a single `GetBlockHeaders` request.
|
/// In this state we try to get subchain headers with a single `GetBlockHeaders` request.
|
||||||
/// On `NewPeer` / On `Restart`:
|
/// On `NewPeer` / On `Restart`:
|
||||||
/// If peer's total difficulty is higher, request N/M headers with interval M+1 starting from l
|
/// If peer's total difficulty is higher and there are less than 5 peers downloading, request N/M headers with interval M+1 starting from l
|
||||||
/// On `BlockHeaders(R)`:
|
/// On `BlockHeaders(R)`:
|
||||||
/// If R is empty:
|
/// If R is empty:
|
||||||
/// If l is equal to genesis block hash or l is more than 1000 blocks behind our best hash:
|
/// If l is equal to genesis block hash or l is more than 1000 blocks behind our best hash:
|
||||||
@ -49,8 +49,8 @@
|
|||||||
/// Else
|
/// Else
|
||||||
/// Set S to R, set s to `Blocks`.
|
/// Set S to R, set s to `Blocks`.
|
||||||
///
|
///
|
||||||
///
|
|
||||||
/// All other messages are ignored.
|
/// All other messages are ignored.
|
||||||
|
///
|
||||||
/// Workflow for `Blocks` state.
|
/// Workflow for `Blocks` state.
|
||||||
/// In this state we download block headers and bodies from multiple peers.
|
/// In this state we download block headers and bodies from multiple peers.
|
||||||
/// On `NewPeer` / On `Restart`:
|
/// On `NewPeer` / On `Restart`:
|
||||||
@ -62,7 +62,9 @@
|
|||||||
///
|
///
|
||||||
/// On `BlockHeaders(R)`:
|
/// On `BlockHeaders(R)`:
|
||||||
/// If R is empty remove current peer from P and restart.
|
/// If R is empty remove current peer from P and restart.
|
||||||
/// Validate received headers. For each header find a parent in H or R or the blockchain. Restart if there is a block with unknown parent.
|
/// Validate received headers:
|
||||||
|
/// For each header find a parent in H or R or the blockchain. Restart if there is a block with unknown parent.
|
||||||
|
/// Find at least one header from the received list in S. Restart if there is none.
|
||||||
/// Go to `CollectBlocks`.
|
/// Go to `CollectBlocks`.
|
||||||
///
|
///
|
||||||
/// On `BlockBodies(R)`:
|
/// On `BlockBodies(R)`:
|
||||||
@ -98,7 +100,7 @@ use ethcore::snapshot::{ManifestData, RestorationStatus};
|
|||||||
use sync_io::SyncIo;
|
use sync_io::SyncIo;
|
||||||
use time;
|
use time;
|
||||||
use super::SyncConfig;
|
use super::SyncConfig;
|
||||||
use block_sync::{BlockDownloader, BlockRequest, BlockDownloaderImportError as DownloaderImportError};
|
use block_sync::{BlockDownloader, BlockRequest, BlockDownloaderImportError as DownloaderImportError, DownloadAction};
|
||||||
use snapshot::{Snapshot, ChunkType};
|
use snapshot::{Snapshot, ChunkType};
|
||||||
use rand::{thread_rng, Rng};
|
use rand::{thread_rng, Rng};
|
||||||
use api::{PeerInfo as PeerInfoDigest, WARP_SYNC_PROTOCOL_ID};
|
use api::{PeerInfo as PeerInfoDigest, WARP_SYNC_PROTOCOL_ID};
|
||||||
@ -306,6 +308,15 @@ impl PeerInfo {
|
|||||||
fn is_allowed(&self) -> bool {
|
fn is_allowed(&self) -> bool {
|
||||||
self.confirmation != ForkConfirmation::Unconfirmed && !self.expired
|
self.confirmation != ForkConfirmation::Unconfirmed && !self.expired
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn reset_asking(&mut self) {
|
||||||
|
self.asking_blocks.clear();
|
||||||
|
self.asking_hash = None;
|
||||||
|
// mark any pending requests as expired
|
||||||
|
if self.asking != PeerAsking::Nothing && self.is_allowed() {
|
||||||
|
self.expired = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Blockchain sync handler.
|
/// Blockchain sync handler.
|
||||||
@ -425,12 +436,7 @@ impl ChainSync {
|
|||||||
}
|
}
|
||||||
for (_, ref mut p) in &mut self.peers {
|
for (_, ref mut p) in &mut self.peers {
|
||||||
if p.block_set != Some(BlockSet::OldBlocks) {
|
if p.block_set != Some(BlockSet::OldBlocks) {
|
||||||
p.asking_blocks.clear();
|
p.reset_asking();
|
||||||
p.asking_hash = None;
|
|
||||||
// mark any pending requests as expired
|
|
||||||
if p.asking != PeerAsking::Nothing && p.is_allowed() {
|
|
||||||
p.expired = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.state = SyncState::Idle;
|
self.state = SyncState::Idle;
|
||||||
@ -641,8 +647,9 @@ impl ChainSync {
|
|||||||
|
|
||||||
self.clear_peer_download(peer_id);
|
self.clear_peer_download(peer_id);
|
||||||
let expected_hash = self.peers.get(&peer_id).and_then(|p| p.asking_hash);
|
let expected_hash = self.peers.get(&peer_id).and_then(|p| p.asking_hash);
|
||||||
|
let allowed = self.peers.get(&peer_id).map(|p| p.is_allowed()).unwrap_or(false);
|
||||||
let block_set = self.peers.get(&peer_id).and_then(|p| p.block_set).unwrap_or(BlockSet::NewBlocks);
|
let block_set = self.peers.get(&peer_id).and_then(|p| p.block_set).unwrap_or(BlockSet::NewBlocks);
|
||||||
if !self.reset_peer_asking(peer_id, PeerAsking::BlockHeaders) || expected_hash.is_none() {
|
if !self.reset_peer_asking(peer_id, PeerAsking::BlockHeaders) || expected_hash.is_none() || !allowed {
|
||||||
trace!(target: "sync", "{}: Ignored unexpected headers, expected_hash = {:?}", peer_id, expected_hash);
|
trace!(target: "sync", "{}: Ignored unexpected headers, expected_hash = {:?}", peer_id, expected_hash);
|
||||||
self.continue_sync(io);
|
self.continue_sync(io);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
@ -687,7 +694,15 @@ impl ChainSync {
|
|||||||
self.continue_sync(io);
|
self.continue_sync(io);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
},
|
},
|
||||||
Ok(()) => (),
|
Ok(DownloadAction::Reset) => {
|
||||||
|
// mark all outstanding requests as expired
|
||||||
|
trace!("Resetting downloads for {:?}", block_set);
|
||||||
|
for (_, ref mut p) in self.peers.iter_mut().filter(|&(_, ref p)| p.block_set == Some(block_set)) {
|
||||||
|
p.reset_asking();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
Ok(DownloadAction::None) => {},
|
||||||
}
|
}
|
||||||
|
|
||||||
self.collect_blocks(io, block_set);
|
self.collect_blocks(io, block_set);
|
||||||
@ -979,7 +994,7 @@ impl ChainSync {
|
|||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
self.clear_peer_download(peer_id);
|
self.clear_peer_download(peer_id);
|
||||||
if !self.reset_peer_asking(peer_id, PeerAsking::SnapshotData) || self.state != SyncState::SnapshotData {
|
if !self.reset_peer_asking(peer_id, PeerAsking::SnapshotData) || (self.state != SyncState::SnapshotData && self.state != SyncState::SnapshotWaiting) {
|
||||||
trace!(target: "sync", "{}: Ignored unexpected snapshot data", peer_id);
|
trace!(target: "sync", "{}: Ignored unexpected snapshot data", peer_id);
|
||||||
self.continue_sync(io);
|
self.continue_sync(io);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
@ -1111,6 +1126,7 @@ impl ChainSync {
|
|||||||
};
|
};
|
||||||
let chain_info = io.chain().chain_info();
|
let chain_info = io.chain().chain_info();
|
||||||
let syncing_difficulty = chain_info.pending_total_difficulty;
|
let syncing_difficulty = chain_info.pending_total_difficulty;
|
||||||
|
let num_active_peers = self.peers.values().filter(|p| p.asking != PeerAsking::Nothing).count();
|
||||||
|
|
||||||
let higher_difficulty = peer_difficulty.map_or(true, |pd| pd > syncing_difficulty);
|
let higher_difficulty = peer_difficulty.map_or(true, |pd| pd > syncing_difficulty);
|
||||||
if force || self.state == SyncState::NewBlocks || higher_difficulty || self.old_blocks.is_some() {
|
if force || self.state == SyncState::NewBlocks || higher_difficulty || self.old_blocks.is_some() {
|
||||||
@ -1128,7 +1144,7 @@ impl ChainSync {
|
|||||||
let have_latest = io.chain().block_status(BlockID::Hash(peer_latest)) != BlockStatus::Unknown;
|
let have_latest = io.chain().block_status(BlockID::Hash(peer_latest)) != BlockStatus::Unknown;
|
||||||
if !have_latest && (higher_difficulty || force || self.state == SyncState::NewBlocks) {
|
if !have_latest && (higher_difficulty || force || self.state == SyncState::NewBlocks) {
|
||||||
// check if got new blocks to download
|
// check if got new blocks to download
|
||||||
if let Some(request) = self.new_blocks.request_blocks(io) {
|
if let Some(request) = self.new_blocks.request_blocks(io, num_active_peers) {
|
||||||
self.request_blocks(io, peer_id, request, BlockSet::NewBlocks);
|
self.request_blocks(io, peer_id, request, BlockSet::NewBlocks);
|
||||||
if self.state == SyncState::Idle {
|
if self.state == SyncState::Idle {
|
||||||
self.state = SyncState::Blocks;
|
self.state = SyncState::Blocks;
|
||||||
@ -1137,7 +1153,7 @@ impl ChainSync {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(request) = self.old_blocks.as_mut().and_then(|d| d.request_blocks(io)) {
|
if let Some(request) = self.old_blocks.as_mut().and_then(|d| d.request_blocks(io, num_active_peers)) {
|
||||||
self.request_blocks(io, peer_id, request, BlockSet::OldBlocks);
|
self.request_blocks(io, peer_id, request, BlockSet::OldBlocks);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user