mirror of
https://github.com/hinto-janai/cuprate.git
synced 2025-01-20 17:54:31 +00:00
cross block bp(+) batch verification
This commit is contained in:
parent
1f94a90c45
commit
764c4663a0
5 changed files with 50 additions and 6 deletions
|
@ -20,6 +20,8 @@ use crate::{
|
|||
BlockChainContextRequest, BlockChainContextResponse, ExtendedConsensusError,
|
||||
VerifyBlockResponse,
|
||||
};
|
||||
use crate::batch_verifier::MultiThreadedBatchVerifier;
|
||||
use crate::transactions::verify_tx_semantic;
|
||||
|
||||
/// Batch prepares a list of blocks for verification.
|
||||
#[instrument(level = "debug", name = "batch_prep_blocks", skip_all, fields(amt = blocks.len()))]
|
||||
|
@ -168,7 +170,9 @@ where
|
|||
tracing::debug!("Calculating PoW and prepping transaction");
|
||||
|
||||
let blocks = rayon_spawn_async(move || {
|
||||
blocks
|
||||
let batch_verifier = MultiThreadedBatchVerifier::new(rayon::current_num_threads());
|
||||
|
||||
let res = blocks
|
||||
.into_par_iter()
|
||||
.zip(difficulties)
|
||||
.zip(txs)
|
||||
|
@ -183,11 +187,16 @@ where
|
|||
// Check the PoW
|
||||
check_block_pow(&block.pow_hash, difficultly).map_err(ConsensusError::Block)?;
|
||||
|
||||
let hf = block.hf_version;
|
||||
|
||||
// Now setup the txs.
|
||||
let txs = txs
|
||||
.into_par_iter()
|
||||
.map(|tx| {
|
||||
let tx = new_tx_verification_data(tx)?;
|
||||
let mut tx = new_tx_verification_data(tx)?;
|
||||
|
||||
verify_tx_semantic(&mut tx, hf, &batch_verifier)?;
|
||||
|
||||
Ok::<_, ConsensusError>((tx.tx_hash, tx))
|
||||
})
|
||||
.collect::<Result<HashMap<_, _>, _>>()?;
|
||||
|
@ -201,7 +210,13 @@ where
|
|||
|
||||
Ok((block, ordered_txs))
|
||||
})
|
||||
.collect::<Result<Vec<_>, ExtendedConsensusError>>()
|
||||
.collect::<Result<Vec<_>, ExtendedConsensusError>>()?;
|
||||
|
||||
if !batch_verifier.verify() {
|
||||
return Err(ExtendedConsensusError::OneOrMoreBatchVerificationStatementsInvalid);
|
||||
}
|
||||
|
||||
Ok(res)
|
||||
})
|
||||
.await?;
|
||||
|
||||
|
|
|
@ -38,7 +38,7 @@ use crate::{
|
|||
pub mod contextual_data;
|
||||
mod free;
|
||||
|
||||
pub use free::new_tx_verification_data;
|
||||
pub use free::{new_tx_verification_data, verify_tx_semantic};
|
||||
|
||||
/// A struct representing the type of validation that needs to be completed for this transaction.
|
||||
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
|
||||
|
@ -333,6 +333,19 @@ fn transactions_needing_verification(
|
|||
.push((Arc::clone(tx), VerificationNeeded::SemanticAndContextual));
|
||||
continue;
|
||||
}
|
||||
CachedVerificationState::SemanticallyValidAtHF(hf) => {
|
||||
if current_hf != *hf {
|
||||
drop(guard);
|
||||
full_validation_transactions
|
||||
.push((Arc::clone(tx), VerificationNeeded::SemanticAndContextual));
|
||||
continue;
|
||||
}
|
||||
|
||||
drop(guard);
|
||||
full_validation_transactions
|
||||
.push((Arc::clone(tx), VerificationNeeded::Contextual));
|
||||
continue;
|
||||
}
|
||||
CachedVerificationState::ValidAtHashAndHF { block_hash, hf } => {
|
||||
if current_hf != *hf {
|
||||
drop(guard);
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
use std::sync::Mutex as StdMutex;
|
||||
use std::sync::{Arc, Mutex as StdMutex};
|
||||
|
||||
use monero_serai::{
|
||||
ringct::{bulletproofs::Bulletproof, RctType},
|
||||
|
@ -6,7 +6,8 @@ use monero_serai::{
|
|||
};
|
||||
|
||||
use cuprate_consensus_rules::{transactions::TransactionError, ConsensusError};
|
||||
use cuprate_types::{CachedVerificationState, TransactionVerificationData, TxVersion};
|
||||
use cuprate_types::{CachedVerificationState, HardFork, TransactionVerificationData, TxVersion};
|
||||
use crate::batch_verifier::MultiThreadedBatchVerifier;
|
||||
|
||||
/// Creates a new [`TransactionVerificationData`] from a [`Transaction`].
|
||||
///
|
||||
|
@ -36,6 +37,18 @@ pub fn new_tx_verification_data(
|
|||
})
|
||||
}
|
||||
|
||||
pub fn verify_tx_semantic(
|
||||
tx: &mut TransactionVerificationData,
|
||||
hf: HardFork,
|
||||
multi_threaded_batch_verifier: &MultiThreadedBatchVerifier,
|
||||
) -> Result<(), ConsensusError> {
|
||||
cuprate_consensus_rules::transactions::check_transaction_semantic(&tx.tx, tx.tx_blob.len(), tx.tx_weight, &tx.tx_hash, hf, multi_threaded_batch_verifier)?;
|
||||
|
||||
*tx.cached_verification_state.get_mut().unwrap() = CachedVerificationState::SemanticallyValidAtHF(hf);
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Calculates the weight of a [`Transaction`].
|
||||
///
|
||||
/// This is more efficient that [`Transaction::weight`] if you already have the transaction blob.
|
||||
|
|
|
@ -103,6 +103,7 @@ impl From<CachedVerificationState> for RawCachedVerificationState {
|
|||
raw_hf: 0,
|
||||
raw_valid_past_timestamp: [0; 8],
|
||||
},
|
||||
CachedVerificationState::SemanticallyValidAtHF(hf) => todo!(),
|
||||
CachedVerificationState::ValidAtHashAndHF { block_hash, hf } => Self {
|
||||
raw_valid_at_hash: block_hash,
|
||||
raw_hf: hf.as_u8(),
|
||||
|
|
|
@ -37,6 +37,7 @@ impl TxVersion {
|
|||
pub enum CachedVerificationState {
|
||||
/// The transaction has not been validated.
|
||||
NotVerified,
|
||||
SemanticallyValidAtHF(HardFork),
|
||||
/// The transaction is valid* if the block represented by this hash is in the blockchain and the [`HardFork`]
|
||||
/// is the same.
|
||||
///
|
||||
|
@ -68,6 +69,7 @@ impl CachedVerificationState {
|
|||
pub const fn verified_at_block_hash(&self) -> Option<[u8; 32]> {
|
||||
match self {
|
||||
Self::NotVerified => None,
|
||||
Self::SemanticallyValidAtHF(_) => None,
|
||||
Self::ValidAtHashAndHF { block_hash, .. }
|
||||
| Self::ValidAtHashAndHFWithTimeBasedLock { block_hash, .. } => Some(*block_hash),
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue