2018-06-13 21:52:23 -07:00
|
|
|
//! The `blob_fetch_stage` pulls blobs from UDP sockets and sends it to a channel.
|
|
|
|
|
2019-08-20 17:16:06 -07:00
|
|
|
use crate::recycler::Recycler;
|
|
|
|
use crate::result;
|
|
|
|
use crate::result::Error;
|
2018-12-07 19:16:27 -08:00
|
|
|
use crate::service::Service;
|
2019-08-20 17:16:06 -07:00
|
|
|
use crate::streamer::{self, BlobSender, PacketReceiver, PacketSender};
|
2018-06-13 21:52:23 -07:00
|
|
|
use std::net::UdpSocket;
|
2019-03-04 20:50:02 -08:00
|
|
|
use std::sync::atomic::AtomicBool;
|
2019-08-20 17:16:06 -07:00
|
|
|
use std::sync::mpsc::{channel, RecvTimeoutError};
|
2018-06-13 21:52:23 -07:00
|
|
|
use std::sync::Arc;
|
2019-08-20 17:16:06 -07:00
|
|
|
use std::thread::{self, Builder, JoinHandle};
|
2018-06-13 21:52:23 -07:00
|
|
|
|
|
|
|
pub struct BlobFetchStage {
|
2018-07-03 21:14:08 -07:00
|
|
|
thread_hdls: Vec<JoinHandle<()>>,
|
2018-06-13 21:52:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl BlobFetchStage {
|
2019-03-04 20:50:02 -08:00
|
|
|
pub fn new(socket: Arc<UdpSocket>, sender: &BlobSender, exit: &Arc<AtomicBool>) -> Self {
|
2019-01-31 13:43:22 -08:00
|
|
|
Self::new_multi_socket(vec![socket], sender, exit)
|
2018-06-13 21:52:23 -07:00
|
|
|
}
|
|
|
|
pub fn new_multi_socket(
|
2018-08-28 16:32:40 -07:00
|
|
|
sockets: Vec<Arc<UdpSocket>>,
|
2019-01-31 13:43:22 -08:00
|
|
|
sender: &BlobSender,
|
2019-03-04 20:50:02 -08:00
|
|
|
exit: &Arc<AtomicBool>,
|
2019-01-31 13:43:22 -08:00
|
|
|
) -> Self {
|
2018-06-13 21:52:23 -07:00
|
|
|
let thread_hdls: Vec<_> = sockets
|
|
|
|
.into_iter()
|
2019-03-04 20:50:02 -08:00
|
|
|
.map(|socket| streamer::blob_receiver(socket, &exit, sender.clone()))
|
2018-09-18 08:02:57 -07:00
|
|
|
.collect();
|
2018-06-13 21:52:23 -07:00
|
|
|
|
2019-03-04 20:50:02 -08:00
|
|
|
Self { thread_hdls }
|
2018-06-13 21:52:23 -07:00
|
|
|
}
|
2019-08-20 17:16:06 -07:00
|
|
|
|
|
|
|
fn handle_forwarded_packets(
|
|
|
|
recvr: &PacketReceiver,
|
|
|
|
sendr: &PacketSender,
|
|
|
|
) -> result::Result<()> {
|
|
|
|
let msgs = recvr.recv()?;
|
|
|
|
let mut batch = vec![msgs];
|
|
|
|
while let Ok(more) = recvr.try_recv() {
|
|
|
|
batch.push(more);
|
|
|
|
}
|
|
|
|
|
|
|
|
batch
|
|
|
|
.iter_mut()
|
|
|
|
.for_each(|b| b.packets.iter_mut().for_each(|p| p.meta.forward = true));
|
|
|
|
|
|
|
|
for packets in batch {
|
|
|
|
if sendr.send(packets).is_err() {
|
|
|
|
return Err(Error::SendError);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_multi_socket_packet(
|
|
|
|
sockets: Vec<Arc<UdpSocket>>,
|
|
|
|
forward_sockets: Vec<Arc<UdpSocket>>,
|
|
|
|
sender: &PacketSender,
|
|
|
|
exit: &Arc<AtomicBool>,
|
|
|
|
) -> Self {
|
|
|
|
let recycler = Recycler::default();
|
|
|
|
let tvu_threads = sockets.into_iter().map(|socket| {
|
|
|
|
streamer::receiver(
|
|
|
|
socket,
|
|
|
|
&exit,
|
|
|
|
sender.clone(),
|
|
|
|
recycler.clone(),
|
|
|
|
"blob_fetch_stage",
|
|
|
|
)
|
|
|
|
});
|
|
|
|
|
|
|
|
let (forward_sender, forward_receiver) = channel();
|
|
|
|
let tvu_forwards_threads = forward_sockets.into_iter().map(|socket| {
|
|
|
|
streamer::receiver(
|
|
|
|
socket,
|
|
|
|
&exit,
|
|
|
|
forward_sender.clone(),
|
|
|
|
recycler.clone(),
|
|
|
|
"blob_fetch_stage",
|
|
|
|
)
|
|
|
|
});
|
|
|
|
|
|
|
|
let sender = sender.clone();
|
|
|
|
let fwd_thread_hdl = Builder::new()
|
|
|
|
.name("solana-tvu-fetch-stage-fwd-rcvr".to_string())
|
|
|
|
.spawn(move || loop {
|
|
|
|
if let Err(e) = Self::handle_forwarded_packets(&forward_receiver, &sender) {
|
|
|
|
match e {
|
|
|
|
Error::RecvTimeoutError(RecvTimeoutError::Disconnected) => break,
|
|
|
|
Error::RecvTimeoutError(RecvTimeoutError::Timeout) => (),
|
|
|
|
Error::RecvError(_) => break,
|
|
|
|
Error::SendError => break,
|
|
|
|
_ => error!("{:?}", e),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let mut thread_hdls: Vec<_> = tvu_threads.chain(tvu_forwards_threads).collect();
|
|
|
|
thread_hdls.push(fwd_thread_hdl);
|
|
|
|
|
|
|
|
Self { thread_hdls }
|
|
|
|
}
|
2018-06-13 21:52:23 -07:00
|
|
|
}
|
2018-07-03 21:14:08 -07:00
|
|
|
|
|
|
|
impl Service for BlobFetchStage {
|
2018-09-13 14:00:17 -07:00
|
|
|
type JoinReturnType = ();
|
2018-07-03 21:14:08 -07:00
|
|
|
|
|
|
|
fn join(self) -> thread::Result<()> {
|
2018-09-13 14:00:17 -07:00
|
|
|
for thread_hdl in self.thread_hdls {
|
2018-07-03 21:14:08 -07:00
|
|
|
thread_hdl.join()?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|