2024-05-31 00:52:12 +00:00
|
|
|
use std::{cell::RefCell, ops::DerefMut};
|
2023-11-18 14:00:33 +00:00
|
|
|
|
|
|
|
use multiexp::BatchVerifier as InternalBatchVerifier;
|
|
|
|
use rayon::prelude::*;
|
|
|
|
use thread_local::ThreadLocal;
|
|
|
|
|
|
|
|
use crate::ConsensusError;
|
|
|
|
|
2024-05-31 00:52:12 +00:00
|
|
|
/// A multithreaded batch verifier.
|
2023-11-18 14:00:33 +00:00
|
|
|
pub struct MultiThreadedBatchVerifier {
|
2024-05-31 00:52:12 +00:00
|
|
|
internal: ThreadLocal<RefCell<InternalBatchVerifier<(), dalek_ff_group::EdwardsPoint>>>,
|
2023-11-18 14:00:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl MultiThreadedBatchVerifier {
|
|
|
|
/// Create a new multithreaded batch verifier,
|
|
|
|
pub fn new(numb_threads: usize) -> MultiThreadedBatchVerifier {
|
|
|
|
MultiThreadedBatchVerifier {
|
|
|
|
internal: ThreadLocal::with_capacity(numb_threads),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-27 23:50:18 +00:00
|
|
|
pub fn queue_statement<R>(
|
2023-11-18 14:00:33 +00:00
|
|
|
&self,
|
|
|
|
stmt: impl FnOnce(
|
2023-12-27 23:50:18 +00:00
|
|
|
&mut InternalBatchVerifier<(), dalek_ff_group::EdwardsPoint>,
|
|
|
|
) -> Result<R, ConsensusError>,
|
|
|
|
) -> Result<R, ConsensusError> {
|
2023-11-18 14:00:33 +00:00
|
|
|
let verifier_cell = self
|
|
|
|
.internal
|
2024-05-31 00:52:12 +00:00
|
|
|
.get_or(|| RefCell::new(InternalBatchVerifier::new(8)));
|
|
|
|
// TODO: this is not ok as a rayon par_iter could be called in stmt.
|
|
|
|
stmt(verifier_cell.borrow_mut().deref_mut())
|
2023-11-18 14:00:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn verify(self) -> bool {
|
|
|
|
self.internal
|
|
|
|
.into_iter()
|
2024-05-31 00:52:12 +00:00
|
|
|
.map(RefCell::into_inner)
|
2023-11-18 14:00:33 +00:00
|
|
|
.par_bridge()
|
2024-02-15 16:03:04 +00:00
|
|
|
.find_any(|batch_verifier| !batch_verifier.verify_vartime())
|
2023-11-18 14:00:33 +00:00
|
|
|
.is_none()
|
|
|
|
}
|
|
|
|
}
|