Thread pool for par_iter in EntrySlice::verify (#4732)
* Use thread pool for entry verify par iter * some performance metrics * check batch size and use CPU for smaller batches
This commit is contained in:
@ -8,20 +8,32 @@ use crate::result::Result;
|
|||||||
use bincode::{deserialize, serialized_size};
|
use bincode::{deserialize, serialized_size};
|
||||||
use chrono::prelude::Utc;
|
use chrono::prelude::Utc;
|
||||||
use rayon::prelude::*;
|
use rayon::prelude::*;
|
||||||
|
use rayon::ThreadPool;
|
||||||
use solana_budget_api::budget_instruction;
|
use solana_budget_api::budget_instruction;
|
||||||
|
use solana_metrics::inc_new_counter_warn;
|
||||||
use solana_sdk::hash::{Hash, Hasher};
|
use solana_sdk::hash::{Hash, Hasher};
|
||||||
use solana_sdk::signature::{Keypair, KeypairUtil};
|
use solana_sdk::signature::{Keypair, KeypairUtil};
|
||||||
use solana_sdk::transaction::Transaction;
|
use solana_sdk::transaction::Transaction;
|
||||||
use std::borrow::Borrow;
|
use std::borrow::Borrow;
|
||||||
|
use std::cell::RefCell;
|
||||||
use std::sync::mpsc::{Receiver, Sender};
|
use std::sync::mpsc::{Receiver, Sender};
|
||||||
use std::sync::{Arc, RwLock};
|
use std::sync::{Arc, RwLock};
|
||||||
|
|
||||||
#[cfg(feature = "cuda")]
|
#[cfg(feature = "cuda")]
|
||||||
use crate::sigverify::poh_verify_many;
|
use crate::sigverify::poh_verify_many;
|
||||||
|
use solana_sdk::timing;
|
||||||
#[cfg(feature = "cuda")]
|
#[cfg(feature = "cuda")]
|
||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
#[cfg(feature = "cuda")]
|
#[cfg(feature = "cuda")]
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
use std::time::Instant;
|
||||||
|
|
||||||
|
pub const NUM_THREADS: u32 = 10;
|
||||||
|
|
||||||
|
thread_local!(static PAR_THREAD_POOL: RefCell<ThreadPool> = RefCell::new(rayon::ThreadPoolBuilder::new()
|
||||||
|
.num_threads(sys_info::cpu_num().unwrap_or(NUM_THREADS) as usize)
|
||||||
|
.build()
|
||||||
|
.unwrap()));
|
||||||
|
|
||||||
pub type EntrySender = Sender<Vec<Entry>>;
|
pub type EntrySender = Sender<Vec<Entry>>;
|
||||||
pub type EntryReceiver = Receiver<Vec<Entry>>;
|
pub type EntryReceiver = Receiver<Vec<Entry>>;
|
||||||
@ -211,6 +223,7 @@ where
|
|||||||
// an EntrySlice is a slice of Entries
|
// an EntrySlice is a slice of Entries
|
||||||
pub trait EntrySlice {
|
pub trait EntrySlice {
|
||||||
/// Verifies the hashes and counts of a slice of transactions are all consistent.
|
/// Verifies the hashes and counts of a slice of transactions are all consistent.
|
||||||
|
fn verify_cpu(&self, start_hash: &Hash) -> bool;
|
||||||
fn verify(&self, start_hash: &Hash) -> bool;
|
fn verify(&self, start_hash: &Hash) -> bool;
|
||||||
fn to_shared_blobs(&self) -> Vec<SharedBlob>;
|
fn to_shared_blobs(&self) -> Vec<SharedBlob>;
|
||||||
fn to_blobs(&self) -> Vec<Blob>;
|
fn to_blobs(&self) -> Vec<Blob>;
|
||||||
@ -219,30 +232,53 @@ pub trait EntrySlice {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl EntrySlice for [Entry] {
|
impl EntrySlice for [Entry] {
|
||||||
#[cfg(not(feature = "cuda"))]
|
fn verify_cpu(&self, start_hash: &Hash) -> bool {
|
||||||
fn verify(&self, start_hash: &Hash) -> bool {
|
let now = Instant::now();
|
||||||
let genesis = [Entry {
|
let genesis = [Entry {
|
||||||
num_hashes: 0,
|
num_hashes: 0,
|
||||||
hash: *start_hash,
|
hash: *start_hash,
|
||||||
transactions: vec![],
|
transactions: vec![],
|
||||||
}];
|
}];
|
||||||
let entry_pairs = genesis.par_iter().chain(self).zip(self);
|
let entry_pairs = genesis.par_iter().chain(self).zip(self);
|
||||||
entry_pairs.all(|(x0, x1)| {
|
let res = PAR_THREAD_POOL.with(|thread_pool| {
|
||||||
let r = x1.verify(&x0.hash);
|
thread_pool.borrow().install(|| {
|
||||||
if !r {
|
entry_pairs.all(|(x0, x1)| {
|
||||||
warn!(
|
let r = x1.verify(&x0.hash);
|
||||||
"entry invalid!: x0: {:?}, x1: {:?} num txs: {}",
|
if !r {
|
||||||
x0.hash,
|
warn!(
|
||||||
x1.hash,
|
"entry invalid!: x0: {:?}, x1: {:?} num txs: {}",
|
||||||
x1.transactions.len()
|
x0.hash,
|
||||||
);
|
x1.hash,
|
||||||
}
|
x1.transactions.len()
|
||||||
r
|
);
|
||||||
})
|
}
|
||||||
|
r
|
||||||
|
})
|
||||||
|
})
|
||||||
|
});
|
||||||
|
inc_new_counter_warn!(
|
||||||
|
"entry_verify-duration",
|
||||||
|
timing::duration_as_ms(&now.elapsed()) as usize
|
||||||
|
);
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(feature = "cuda"))]
|
||||||
|
fn verify(&self, start_hash: &Hash) -> bool {
|
||||||
|
self.verify_cpu(start_hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "cuda")]
|
#[cfg(feature = "cuda")]
|
||||||
fn verify(&self, start_hash: &Hash) -> bool {
|
fn verify(&self, start_hash: &Hash) -> bool {
|
||||||
|
inc_new_counter_warn!("entry_verify-num_entries", self.len() as usize);
|
||||||
|
|
||||||
|
// Use CPU verify if the batch length is < 1K
|
||||||
|
if self.len() < 1024 {
|
||||||
|
return self.verify_cpu(start_hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
let start = Instant::now();
|
||||||
|
|
||||||
let genesis = [Entry {
|
let genesis = [Entry {
|
||||||
num_hashes: 0,
|
num_hashes: 0,
|
||||||
hash: *start_hash,
|
hash: *start_hash,
|
||||||
@ -250,7 +286,7 @@ impl EntrySlice for [Entry] {
|
|||||||
}];
|
}];
|
||||||
|
|
||||||
let hashes: Vec<Hash> = genesis
|
let hashes: Vec<Hash> = genesis
|
||||||
.par_iter()
|
.iter()
|
||||||
.chain(self)
|
.chain(self)
|
||||||
.map(|entry| entry.hash)
|
.map(|entry| entry.hash)
|
||||||
.take(self.len())
|
.take(self.len())
|
||||||
@ -265,6 +301,7 @@ impl EntrySlice for [Entry] {
|
|||||||
let hashes = Arc::new(Mutex::new(hashes));
|
let hashes = Arc::new(Mutex::new(hashes));
|
||||||
let hashes_clone = hashes.clone();
|
let hashes_clone = hashes.clone();
|
||||||
|
|
||||||
|
let gpu_wait = Instant::now();
|
||||||
let gpu_verify_thread = thread::spawn(move || {
|
let gpu_verify_thread = thread::spawn(move || {
|
||||||
let mut hashes = hashes_clone.lock().unwrap();
|
let mut hashes = hashes_clone.lock().unwrap();
|
||||||
let res;
|
let res;
|
||||||
@ -281,36 +318,51 @@ impl EntrySlice for [Entry] {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let tx_hashes: Vec<Option<Hash>> = self
|
let tx_hashes: Vec<Option<Hash>> = PAR_THREAD_POOL.with(|thread_pool| {
|
||||||
.into_par_iter()
|
thread_pool.borrow().install(|| {
|
||||||
.map(|entry| {
|
self.into_par_iter()
|
||||||
if entry.transactions.is_empty() {
|
.map(|entry| {
|
||||||
None
|
if entry.transactions.is_empty() {
|
||||||
} else {
|
None
|
||||||
Some(hash_transactions(&entry.transactions))
|
} else {
|
||||||
}
|
Some(hash_transactions(&entry.transactions))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
})
|
})
|
||||||
.collect();
|
});
|
||||||
|
|
||||||
gpu_verify_thread.join().unwrap();
|
gpu_verify_thread.join().unwrap();
|
||||||
|
inc_new_counter_warn!(
|
||||||
|
"entry_verify-gpu_thread",
|
||||||
|
timing::duration_as_ms(&gpu_wait.elapsed()) as usize
|
||||||
|
);
|
||||||
|
|
||||||
let hashes = Arc::try_unwrap(hashes).unwrap().into_inner().unwrap();
|
let hashes = Arc::try_unwrap(hashes).unwrap().into_inner().unwrap();
|
||||||
hashes
|
let res =
|
||||||
.into_par_iter()
|
PAR_THREAD_POOL.with(|thread_pool| {
|
||||||
.zip(tx_hashes)
|
thread_pool.borrow().install(|| {
|
||||||
.zip(self)
|
hashes.into_par_iter().zip(tx_hashes).zip(self).all(
|
||||||
.all(|((hash, tx_hash), answer)| {
|
|((hash, tx_hash), answer)| {
|
||||||
if answer.num_hashes == 0 {
|
if answer.num_hashes == 0 {
|
||||||
hash == answer.hash
|
hash == answer.hash
|
||||||
} else {
|
} else {
|
||||||
let mut poh = Poh::new(hash, None);
|
let mut poh = Poh::new(hash, None);
|
||||||
if let Some(mixin) = tx_hash {
|
if let Some(mixin) = tx_hash {
|
||||||
poh.record(mixin).unwrap().hash == answer.hash
|
poh.record(mixin).unwrap().hash == answer.hash
|
||||||
} else {
|
} else {
|
||||||
poh.tick().unwrap().hash == answer.hash
|
poh.tick().unwrap().hash == answer.hash
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
},
|
||||||
|
)
|
||||||
|
})
|
||||||
|
});
|
||||||
|
inc_new_counter_warn!(
|
||||||
|
"entry_verify-duration",
|
||||||
|
timing::duration_as_ms(&start.elapsed()) as usize
|
||||||
|
);
|
||||||
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_blobs(&self) -> Vec<Blob> {
|
fn to_blobs(&self) -> Vec<Blob> {
|
||||||
|
Reference in New Issue
Block a user