Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Fix BTCRelay Chain Reorganization to Prioritize Work Over Length #1198

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
5 changes: 4 additions & 1 deletion crates/btc-relay/src/benchmarking.rs
Original file line number Diff line number Diff line change
Expand Up @@ -140,8 +140,11 @@ pub mod benchmarks {
}
}

BtcRelay::<T>::insert_block_hash(0, DIFFICULTY_ADJUSTMENT_INTERVAL, init_block_hash);

// new fork up to block before swapping the main chain
for _ in 1..(BestBlockHeight::<T>::get() + SECURE_BITCOIN_CONFIRMATIONS) {
for chain_id in 1..(BestBlockHeight::<T>::get() + SECURE_BITCOIN_CONFIRMATIONS) {
BtcRelay::<T>::insert_block_hash(chain_id, DIFFICULTY_ADJUSTMENT_INTERVAL, init_block_hash);
let block = add_new_block_to_relay::<T>(caller.clone(), init_block_hash, f as usize);
init_block_hash = block.header.hash;
}
Expand Down
168 changes: 165 additions & 3 deletions crates/btc-relay/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ use frame_support::{
transactional,
};
use frame_system::{ensure_signed, pallet_prelude::BlockNumberFor};
use sp_arithmetic::traits::Saturating;
use sp_core::{H256, U256};
use sp_runtime::traits::{CheckedAdd, CheckedDiv, CheckedSub, One};
use sp_std::{
Expand Down Expand Up @@ -320,6 +321,11 @@ pub mod pallet {
InvalidCoinbasePosition,
}

/// Store ChainWork
/// mapping block hash -> chain work
#[pallet::storage]
pub(super) type ChainWork<T: Config> = StorageMap<_, Blake2_128Concat, H256Le, u32, OptionQuery>;

/// Store Bitcoin block headers
#[pallet::storage]
pub(super) type BlockHeaders<T: Config> =
Expand Down Expand Up @@ -398,6 +404,8 @@ pub mod pallet {
StableParachainConfirmations::<T>::put(self.parachain_confirmations);
DisableDifficultyCheck::<T>::put(self.disable_difficulty_check);
DisableInclusionCheck::<T>::put(self.disable_inclusion_check);
// ToDo: insert genesis block hash and block work as zero
ChainWork::<T>::insert(H256Le::zero(), 0);
}
}
}
Expand Down Expand Up @@ -767,6 +775,14 @@ impl<T: Config> Pallet<T> {
ChainCounter::<T>::get()
}

fn get_block_chainwork(block_hash: H256Le) -> Option<u32> {
ChainWork::<T>::get(block_hash)
}

fn set_block_chainwork(block_hash: H256Le, chain_work: u32) {
ChainWork::<T>::insert(block_hash, chain_work)
}

/// Get a block hash from a blockchain
///
/// # Arguments
Expand All @@ -780,8 +796,35 @@ impl<T: Config> Pallet<T> {
Ok(ChainsHashes::<T>::get(chain_id, block_height))
}

fn get_block_hash_from_forks(chain_id: u32, block_height: u32) -> Result<H256Le, DispatchError> {
// Get the blockchain associated with the given chain_id.
let chain = Self::get_block_chain_from_id(chain_id)?;

// If the requested block height is before the chain's start height,
// recursively fetch the block hash from the previous chain.
if block_height < chain.start_height {
// Get the hash of the block at the start height of the current chain.
let start_height_hash = Self::get_block_hash(chain_id, chain.start_height)?;

// Get the previous block's header from its hash.
let prev_hash = Self::get_block_header_from_hash(start_height_hash)?
.block_header
.hash_prev_block;

// Get the chain_id of the previous chain.
let prev_chain_id = Self::get_block_header_from_hash(prev_hash)?.chain_id;

// Recursively call the function for the previous chain and requested block height.
Self::get_block_hash_from_forks(prev_chain_id, block_height)
} else {
// If the requested block height is within the current chain's range,
// directly fetch the block hash from the current chain.
Self::get_block_hash(chain_id, block_height)
}
}

/// Get a block header from its hash
fn get_block_header_from_hash(block_hash: H256Le) -> Result<RichBlockHeader<BlockNumberFor<T>>, DispatchError> {
pub fn get_block_header_from_hash(block_hash: H256Le) -> Result<RichBlockHeader<BlockNumberFor<T>>, DispatchError> {
BlockHeaders::<T>::try_get(block_hash).or(Err(Error::<T>::BlockNotFound.into()))
}

Expand Down Expand Up @@ -877,7 +920,7 @@ impl<T: Config> Pallet<T> {
}
}

fn insert_block_hash(chain_id: u32, block_height: u32, block_hash: H256Le) {
pub fn insert_block_hash(chain_id: u32, block_height: u32, block_hash: H256Le) {
ChainsHashes::<T>::insert(chain_id, block_height, block_hash);
}

Expand Down Expand Up @@ -1181,7 +1224,17 @@ impl<T: Config> Pallet<T> {
// and the current height is more than the
// STABLE_TRANSACTION_CONFIRMATIONS ahead
// we are swapping the main chain
if prev_height.saturating_add(Self::get_stable_transaction_confirmations()) <= current_height {
let current_chain_id = Self::get_chain_id_from_position(current_position)?;
let current_fork_work = Self::get_chainwork(Self::get_block_chain_from_id(current_chain_id)?)?;

let prev_chain_id = Self::get_chain_id_from_position(prev_position)?;
let prev_fork_work = Self::get_chainwork(Self::get_block_chain_from_id(prev_chain_id)?)?;
// update: ensure_no_ongoing_fork
if prev_height.saturating_add(Self::get_stable_transaction_confirmations()) <= current_height
&&
// We should only reorg when the fork has more or equal work than the current main chain
current_fork_work >= prev_fork_work
{
// Swap the mainchain. As an optimization, this function returns the
// new best block hash and its height
let (new_chain_tip_hash, new_chain_tip_height) = Self::swap_main_blockchain(&fork)?;
Expand Down Expand Up @@ -1218,6 +1271,115 @@ impl<T: Config> Pallet<T> {
Ok(())
}

/// The target value represents the mining difficulty required to mine a block with the specified hash.
///
/// # Parameters
///
/// * `block_hash`: A 256-bit block hash.
///
/// # Returns
///
/// * `Result<u32, DispatchError>`: A `Result` containing the retrieved target value as a `u32` if successful, or a
/// `DispatchError` if an error occurs during the retrieval.
fn get_target(block_hash: H256Le) -> Result<u32, DispatchError> {
let block_header = Self::get_block_header_from_hash(block_hash)?;
// Fixme: Should this be kept as a U256 type and other u32 type can be converted to U256?
// Currently taking low_u32/as_u32 loses precision
Ok(block_header.block_header.target.low_u32())
}

/// Calculates the chainwork of a block header at a given height based on the provided block hash.
/// Reference: https://github.com/spesmilo/electrum/blob/cee22abcb5544c5a6fa7f8a8108ccda9c32c2e29/electrum/blockchain.py#L582-L588
///
/// # Arguments
///
/// * `block_hash`: A 256-bit block hash.
///
/// # Returns
///
/// * `Result<u32, DispatchError>`: A `Result` containing the calculated chainwork as a `u32` if successful, or a
/// `DispatchError` if an error occurs during target retrieval.
fn chainwork_of_header_at_height(block_hash: H256Le) -> Result<u32, DispatchError> {
let target = Self::get_target(block_hash)?;
let work = (2_u32.saturating_pow(256).saturating_sub(target).saturating_less_one())
.saturating_div(target.saturating_plus_one())
.saturating_plus_one();
Ok(work)
}

/// The chainwork is a measure of the total computational work done by the chain up to a certain point.
/// Reference: https://github.com/spesmilo/electrum/blob/cee22abcb5544c5a6fa7f8a8108ccda9c32c2e29/electrum/blockchain.py#L589-L614
///
/// # Parameters
///
/// * `chain`: new blockchain element
///
/// # Returns
///
/// * `Result<u32, DispatchError>`: A `Result` containing the calculated cumulative chainwork as a `u32` if
/// successful, or a `DispatchError` if an error occurs during the calculation.
fn get_chainwork(chain: BlockChain) -> Result<u32, DispatchError> {
Copy link
Member

@sander2 sander2 Sep 22, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure it's a good idea to use the electrum function unchanged. In fact, I don't think it'll work, because

  • (1) it's super expensive - it iterates all the way from block zero. We might exceed the block time because of all the iteration
  • (2) We didn't initialize the relay at height 0. The chainwork_of_header_at_height function would fail.

This is a pretty high-risk change, and I would ask you to test it thoroughly. If I'm right and this doesn't work, it could've been caught with a chopsticks test.

Copy link
Member Author

@nakul1010 nakul1010 Sep 22, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, reviewing the code back again even if the loop over the target it would still take btc_best_block/2016 ie 808,849 /2016 = 402 iterations. Ideally, the code should have been added from parachain genesis block to avoid iterations. I will still do the chopsticks test and put the results in another comment.

// Calculate the height of the last retarget point.
let last_retarget = chain
.max_height
.saturating_div(DIFFICULTY_ADJUSTMENT_INTERVAL)
.saturating_mul(DIFFICULTY_ADJUSTMENT_INTERVAL)
.saturating_less_one();

let mut cached_height = last_retarget;

// Iterate backward to find the last block with known chainwork.
while Self::get_block_chainwork(Self::get_block_hash_from_forks(chain.chain_id, cached_height)?).is_none() {
if cached_height == 0 {
break;
}
cached_height = cached_height.saturating_sub(DIFFICULTY_ADJUSTMENT_INTERVAL)
}

// Get the chainwork of the last known block.
let cache_chain_work =
Self::get_block_chainwork(Self::get_block_hash_from_forks(chain.chain_id, cached_height)?);

let mut running_total = if let Some(chain_work) = cache_chain_work {
chain_work
} else {
// Genesis chain work should be 0.
0_u32
};

// Loop to calculate chainwork for blocks between 'cached_height' and 'last_retarget' (0 to 2015).
while cached_height < last_retarget {
cached_height = cached_height.saturating_add(DIFFICULTY_ADJUSTMENT_INTERVAL);

let cached_block_hash = Self::get_block_hash_from_forks(chain.chain_id, cached_height)?;

// Calculate the chainwork for a single header at the current height.
let work_in_single_header = Self::chainwork_of_header_at_height(cached_block_hash)?;

// Calculate the chainwork for a chunk of blocks (DIFFICULTY_ADJUSTMENT_INTERVAL) at the current height.
let work_in_chunk = DIFFICULTY_ADJUSTMENT_INTERVAL.saturating_mul(work_in_single_header);

// Update the running total chainwork.
running_total = running_total.saturating_add(work_in_chunk);

// Store the calculated chainwork for the block.
Self::set_block_chainwork(cached_block_hash, running_total);
}

// Move to the next height.
cached_height = cached_height.saturating_add(DIFFICULTY_ADJUSTMENT_INTERVAL);

// Calculate the chainwork for the last partial chunk of blocks.
let work_in_single_header =
Self::chainwork_of_header_at_height(Self::get_block_hash_from_forks(chain.chain_id, cached_height)?)?;

let work_in_last_partial_chunk = ((chain.max_height % DIFFICULTY_ADJUSTMENT_INTERVAL).saturating_plus_one())
.saturating_mul(work_in_single_header);

// Return the total chainwork, which is the running total plus the chain work of the last partial chunk.
Ok(running_total.saturating_add(work_in_last_partial_chunk))
}

/// Insert a new fork into the Chains mapping sorted by its max height
///
/// # Arguments
Expand Down
47 changes: 47 additions & 0 deletions crates/btc-relay/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -370,6 +370,13 @@ mod store_block_header_tests {

let mut blocks = vec![a2];

BTCRelay::get_target.mock_safe(|_| MockResult::Return(Ok(0_u32)));
BTCRelay::get_block_hash_from_forks.mock_safe(move |chain_id, height| {
assert!(matches!(chain_id, 0 | 1));
assert!(matches!(height, 0 | DIFFICULTY_ADJUSTMENT_INTERVAL));
MockResult::Return(Ok(H256Le::zero()))
});

// create a new fork, and make it overtake the main chain
let mut prev = genesis.hash;
for i in 0..10 {
Expand Down Expand Up @@ -422,6 +429,13 @@ mod store_block_header_tests {
let mut genesis = sample_block_header();
genesis.update_hash().unwrap();

BTCRelay::get_target.mock_safe(|_| MockResult::Return(Ok(0_u32)));
BTCRelay::get_block_hash_from_forks.mock_safe(move |chain_id, height| {
assert!(matches!(chain_id, 0 | 1 | 2));
assert!(matches!(height, 0 | DIFFICULTY_ADJUSTMENT_INTERVAL));
MockResult::Return(Ok(H256Le::zero()))
});

// create the following tree shape:
// f1 --> temp_fork_1
// \-> f2 --> temp_fork_2
Expand Down Expand Up @@ -554,6 +568,13 @@ mod store_block_header_tests {
// "030000005fbd386a5032a0aa1c428416d2d0a9e62b3f31021bb695000000000000000000c46349cf6bddc4451f5df490ad53a83a58018e519579755800845fd4b0e39e79f46197558e41161884eabd86",
].into_iter().map(parse_from_hex).collect();

BTCRelay::get_target.mock_safe(|_| MockResult::Return(Ok(0_u32)));
BTCRelay::get_block_hash_from_forks.mock_safe(move |chain_id, height| {
assert!(matches!(chain_id, 0 | 1));
assert!(matches!(height, 0 | DIFFICULTY_ADJUSTMENT_INTERVAL));
MockResult::Return(Ok(H256Le::zero()))
});

for (idx, block) in reorg_blocks.iter().take(11).enumerate() {
store_header_and_check_invariants(block);

Expand Down Expand Up @@ -688,6 +709,20 @@ fn check_and_do_reorg_new_fork_is_main_chain() {

BTCRelay::swap_main_blockchain.mock_safe(move |_| MockResult::Return(Ok((best_block_hash, fork_block_height))));

BTCRelay::swap_main_blockchain.mock_safe(move |_| MockResult::Return(Ok((best_block_hash, fork_block_height))));

BTCRelay::get_chainwork.mock_safe(move |chain| {
if chain.chain_id == fork_chain_id {
assert_eq!(chain.start_height, main_start_height);
assert_eq!(chain.max_height, fork_block_height);
MockResult::Return(Ok(fork_chain_id))
} else {
assert_eq!(chain.start_height, main_start_height);
assert_eq!(chain.max_height, main_block_height);
MockResult::Return(Ok(main_chain_id))
}
});

assert_ok!(BTCRelay::reorganize_chains(&fork));
// assert that the new main chain is set
let reorg_event = TestEvent::BTCRelay(Event::ChainReorg {
Expand Down Expand Up @@ -730,6 +765,18 @@ fn check_and_do_reorg_new_fork_below_stable_transaction_confirmations() {

BTCRelay::swap_main_blockchain.mock_safe(move |_| MockResult::Return(Ok((best_block_hash, fork_block_height))));

BTCRelay::get_chainwork.mock_safe(move |chain| {
if chain.chain_id == fork_chain_id {
assert_eq!(chain.start_height, main_start_height);
assert_eq!(chain.max_height, fork_block_height);
MockResult::Return(Ok(fork_chain_id))
} else {
assert_eq!(chain.start_height, main_start_height);
assert_eq!(chain.max_height, main_block_height);
MockResult::Return(Ok(main_chain_id))
}
});

assert_ok!(BTCRelay::reorganize_chains(&fork));
// assert that the fork has not overtaken the main chain
let ahead_event = TestEvent::BTCRelay(Event::ForkAheadOfMainChain {
Expand Down
3 changes: 3 additions & 0 deletions parachain/runtime/runtime-tests/src/parachain/btc_relay.rs
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,9 @@ fn integration_test_submit_fork_headers() {
assert_store_main_chain_header_event(index as u32, header.hash, account_of(ALICE));
}

BTCRelayPallet::insert_block_hash(0, DIFFICULTY_ADJUSTMENT_INTERVAL, genesis_header.hash);
BTCRelayPallet::insert_block_hash(FORK_ID, DIFFICULTY_ADJUSTMENT_INTERVAL, test_data[1].hash);

// submit future main chain without genesis
for (index, header) in test_data.iter().enumerate().skip(1 + NUM_FORK_HEADERS as usize) {
SecurityPallet::set_active_block_number(index as u32);
Expand Down