Skip to content

Commit

Permalink
Optimize pruning during batch finalization of blocks
Browse files Browse the repository at this point in the history
  • Loading branch information
nazar-pc committed Jul 2, 2024
1 parent 227fb4e commit c452058
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 40 deletions.
3 changes: 2 additions & 1 deletion substrate/client/api/src/backend.rs
Original file line number Diff line number Diff line change
Expand Up @@ -217,7 +217,8 @@ pub trait BlockImportOperation<Block: BlockT> {
where
I: IntoIterator<Item = (Vec<u8>, Option<Vec<u8>>)>;

/// Mark a block as finalized.
/// Mark a block as finalized, if multiple blocks are finalized in the same operation then they
/// must be marked in ascending order.
fn mark_finalized(
&mut self,
hash: Block::Hash,
Expand Down
84 changes: 45 additions & 39 deletions substrate/client/db/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1357,6 +1357,8 @@ impl<Block: BlockT> Backend<Block> {
Ok(())
}

/// `remove_displaced` can be set to `false` if this is not the last of many subsequent calls
/// for performance reasons.
fn finalize_block_with_transaction(
&self,
transaction: &mut Transaction<DbHash>,
Expand All @@ -1365,6 +1367,7 @@ impl<Block: BlockT> Backend<Block> {
last_finalized: Option<Block::Hash>,
justification: Option<Justification>,
current_transaction_justifications: &mut HashMap<Block::Hash, Justification>,
remove_displaced: bool,
) -> ClientResult<MetaUpdate<Block>> {
// TODO: ensure best chain contains this block.
let number = *header.number();
Expand All @@ -1377,6 +1380,7 @@ impl<Block: BlockT> Backend<Block> {
hash,
with_state,
current_transaction_justifications,
remove_displaced,
)?;

if let Some(justification) = justification {
Expand Down Expand Up @@ -1454,7 +1458,8 @@ impl<Block: BlockT> Backend<Block> {

let mut current_transaction_justifications: HashMap<Block::Hash, Justification> =
HashMap::new();
for (block_hash, justification) in operation.finalized_blocks {
let mut finalized_blocks = operation.finalized_blocks.into_iter().peekable();
while let Some((block_hash, justification)) = finalized_blocks.next() {
let block_header = self.blockchain.expect_header(block_hash)?;
meta_updates.push(self.finalize_block_with_transaction(
&mut transaction,
Expand All @@ -1463,6 +1468,7 @@ impl<Block: BlockT> Backend<Block> {
Some(last_finalized_hash),
justification,
&mut current_transaction_justifications,
finalized_blocks.peek().is_none(),
)?);
last_finalized_hash = block_hash;
last_finalized_num = *block_header.number();
Expand Down Expand Up @@ -1642,6 +1648,7 @@ impl<Block: BlockT> Backend<Block> {
hash,
operation.commit_state,
&mut current_transaction_justifications,
true,
)?;
} else {
// canonicalize blocks which are old enough, regardless of finality.
Expand Down Expand Up @@ -1766,16 +1773,18 @@ impl<Block: BlockT> Backend<Block> {
Ok(())
}

// write stuff to a transaction after a new block is finalized.
// this canonicalizes finalized blocks. Fails if called with a block which
// was not a child of the last finalized block.
// Write stuff to a transaction after a new block is finalized. This canonicalizes finalized
// blocks. Fails if called with a block which was not a child of the last finalized block.
/// `remove_displaced` can be set to `false` if this is not the last of many subsequent calls
/// for performance reasons.
fn note_finalized(
&self,
transaction: &mut Transaction<DbHash>,
f_header: &Block::Header,
f_hash: Block::Hash,
with_state: bool,
current_transaction_justifications: &mut HashMap<Block::Hash, Justification>,
remove_displaced: bool,
) -> ClientResult<()> {
let f_num = *f_header.number();

Expand All @@ -1800,13 +1809,19 @@ impl<Block: BlockT> Backend<Block> {
apply_state_commit(transaction, commit);
}

let new_displaced = self.blockchain.displaced_leaves_after_finalizing(f_hash, f_num)?;
if remove_displaced {
let new_displaced = self.blockchain.displaced_leaves_after_finalizing(f_hash, f_num)?;

self.blockchain.leaves.write().remove_displaced_leaves(FinalizationOutcome::new(
new_displaced.displaced_leaves.iter().copied(),
));
self.blockchain.leaves.write().remove_displaced_leaves(FinalizationOutcome::new(
new_displaced.displaced_leaves.iter().copied(),
));

if !matches!(self.blocks_pruning, BlocksPruning::KeepAll) {
self.prune_displaced_branches(transaction, &new_displaced)?;
}
}

self.prune_blocks(transaction, f_num, &new_displaced, current_transaction_justifications)?;
self.prune_blocks(transaction, f_num, current_transaction_justifications)?;

Ok(())
}
Expand All @@ -1815,39 +1830,29 @@ impl<Block: BlockT> Backend<Block> {
&self,
transaction: &mut Transaction<DbHash>,
finalized_number: NumberFor<Block>,
displaced: &DisplacedLeavesAfterFinalization<Block>,
current_transaction_justifications: &mut HashMap<Block::Hash, Justification>,
) -> ClientResult<()> {
match self.blocks_pruning {
BlocksPruning::KeepAll => {},
BlocksPruning::Some(blocks_pruning) => {
// Always keep the last finalized block
let keep = std::cmp::max(blocks_pruning, 1);
if finalized_number >= keep.into() {
let number = finalized_number.saturating_sub(keep.into());

// Before we prune a block, check if it is pinned
if let Some(hash) = self.blockchain.hash(number)? {
self.blockchain.insert_persisted_body_if_pinned(hash)?;

// If the block was finalized in this transaction, it will not be in the db
// yet.
if let Some(justification) =
current_transaction_justifications.remove(&hash)
{
self.blockchain.insert_justifications_if_pinned(hash, justification);
} else {
self.blockchain.insert_persisted_justifications_if_pinned(hash)?;
}
};
if let BlocksPruning::Some(blocks_pruning) = self.blocks_pruning {
// Always keep the last finalized block
let keep = std::cmp::max(blocks_pruning, 1);
if finalized_number >= keep.into() {
let number = finalized_number.saturating_sub(keep.into());

// Before we prune a block, check if it is pinned
if let Some(hash) = self.blockchain.hash(number)? {
self.blockchain.insert_persisted_body_if_pinned(hash)?;

// If the block was finalized in this transaction, it will not be in the db
// yet.
if let Some(justification) = current_transaction_justifications.remove(&hash) {
self.blockchain.insert_justifications_if_pinned(hash, justification);
} else {
self.blockchain.insert_persisted_justifications_if_pinned(hash)?;
}
};

self.prune_block(transaction, BlockId::<Block>::number(number))?;
}
self.prune_displaced_branches(transaction, displaced)?;
},
BlocksPruning::KeepFinalized => {
self.prune_displaced_branches(transaction, displaced)?;
},
self.prune_block(transaction, BlockId::<Block>::number(number))?;
}
}
Ok(())
}
Expand Down Expand Up @@ -2108,6 +2113,7 @@ impl<Block: BlockT> sc_client_api::backend::Backend<Block> for Backend<Block> {
None,
justification,
&mut current_transaction_justifications,
true,
)?;

self.storage.db.commit(transaction)?;
Expand Down

0 comments on commit c452058

Please sign in to comment.