mirror of
https://github.com/YGGverse/aquatic.git
synced 2026-03-31 09:45:31 +00:00
* WIP: add udp uring support * WIP: fix udp uring address parsing * WIP: udp uring: resubmit recv when needed * WIP: udp uring: add OutMessageStorage, send swarm responses * WIP: udp uring: increase ring entries to 1024 * WIP: udp uring: add constants * WIP: udp uring: use sqpoll, avoid kernel calls * WIP: udp uring: disable sqpoll * WIP: udp uring: use VecDeque for local responses * udp uring: enable setup_coop_taskrun * udp uring: add RecvMsgStorage * udp: improve split of uring and mio implementations * udp uring: clean up * udp uring: initial ipv6 support * udp uring: improve helper structs * udp uring: clean up, use constants for important data * udp: share create_socket fn between implementations * udp uring: improve send buffer free index finding * udp uring: work on SendBuffers.try_add * udp uring: split into modules * udp uring: Rename RecvMsgMultiHelper to RecvHelper * udp uring: improve SendBuffers * udp uring: fix copyright attribution in buf_ring module * udp uring: stop always consuming 100% cpu * udp uring: clean up * udp uring: add handle_recv_cqe * udp uring: move local_responses into SocketWorker * udp uring: move timeout_timespec into SocketWorker * Update TODO * udp: make io-uring optional * Update TODO * udp uring: enqueue timeout before sends * udp uring: move likely empty buffer tracking logic into SendBuffers * udp uring: improve error handling and logging * udp uring: keep one timeout submitted at a time * udp uring: update pending_scrape_valid_until * udp uring: add second timeout for cleaning * Update TODO * udp uring: store resubmittable squeue entries in a Vec * udp uring: add comment, remove a log statement * Update TODO * Update TODO * udp: io_uring: fall back to mio if io_uring support not recent enough * udp: uring: add bytes_received statistics * udp: uring: add bytes_sent statistics * udp: uring: add more statistics * Update TODO * udp: uring: improve SendBuffers code * udp: uring: remove unneeded squeue sync calls * udp: uring: replace buf_ring impl with one from tokio-uring * udp: uring: store ring in TLS so it can be used in Drop impls * udp: uring: store BufRing in SocketWorker * udp: uring: silence buf_ring dead code warnings, improve comment * Update TODO * udp: uring: improve CurrentRing docs, use anonymous struct field * udp: uring: improve ring setup * udp: uring: get ipv6 working * udp: uring: make ring entry count configurable, use more send entries * udp: uring: log number of pending responses (info level) * udp: uring: improve comment on send_buffer_entries calculation * udp: improve config comments * udp: uring: add to responses stats when they are confirmed as sent * Update TODO * udp: uring: enable IoUring setup_submit_all * Update README
128 lines
3.4 KiB
Rust
128 lines
3.4 KiB
Rust
mod mio;
|
|
mod storage;
|
|
#[cfg(feature = "io-uring")]
|
|
mod uring;
|
|
mod validator;
|
|
|
|
use anyhow::Context;
|
|
use aquatic_common::{
|
|
privileges::PrivilegeDropper, CanonicalSocketAddr, PanicSentinel, ServerStartInstant,
|
|
};
|
|
use crossbeam_channel::Receiver;
|
|
use socket2::{Domain, Protocol, Socket, Type};
|
|
|
|
use crate::{
|
|
common::{ConnectedRequestSender, ConnectedResponse, State},
|
|
config::Config,
|
|
};
|
|
|
|
pub use self::validator::ConnectionValidator;
|
|
|
|
/// Bytes of data transmitted when sending an IPv4 UDP packet, in addition to payload size
|
|
///
|
|
/// Consists of:
|
|
/// - 8 bit ethernet frame
|
|
/// - 14 + 4 bit MAC header and checksum
|
|
/// - 20 bit IPv4 header
|
|
/// - 8 bit udp header
|
|
const EXTRA_PACKET_SIZE_IPV4: usize = 8 + 18 + 20 + 8;
|
|
|
|
/// Bytes of data transmitted when sending an IPv4 UDP packet, in addition to payload size
|
|
///
|
|
/// Consists of:
|
|
/// - 8 bit ethernet frame
|
|
/// - 14 + 4 bit MAC header and checksum
|
|
/// - 40 bit IPv6 header
|
|
/// - 8 bit udp header
|
|
const EXTRA_PACKET_SIZE_IPV6: usize = 8 + 18 + 40 + 8;
|
|
|
|
pub fn run_socket_worker(
|
|
sentinel: PanicSentinel,
|
|
shared_state: State,
|
|
config: Config,
|
|
validator: ConnectionValidator,
|
|
server_start_instant: ServerStartInstant,
|
|
request_sender: ConnectedRequestSender,
|
|
response_receiver: Receiver<(ConnectedResponse, CanonicalSocketAddr)>,
|
|
priv_dropper: PrivilegeDropper,
|
|
) {
|
|
#[cfg(feature = "io-uring")]
|
|
match self::uring::supported_on_current_kernel() {
|
|
Ok(()) => {
|
|
self::uring::SocketWorker::run(
|
|
sentinel,
|
|
shared_state,
|
|
config,
|
|
validator,
|
|
server_start_instant,
|
|
request_sender,
|
|
response_receiver,
|
|
priv_dropper,
|
|
);
|
|
|
|
return;
|
|
}
|
|
Err(err) => {
|
|
::log::warn!(
|
|
"Falling back to mio because of lacking kernel io_uring support: {:#}",
|
|
err
|
|
);
|
|
}
|
|
}
|
|
|
|
self::mio::SocketWorker::run(
|
|
sentinel,
|
|
shared_state,
|
|
config,
|
|
validator,
|
|
server_start_instant,
|
|
request_sender,
|
|
response_receiver,
|
|
priv_dropper,
|
|
);
|
|
}
|
|
|
|
fn create_socket(
|
|
config: &Config,
|
|
priv_dropper: PrivilegeDropper,
|
|
) -> anyhow::Result<::std::net::UdpSocket> {
|
|
let socket = if config.network.address.is_ipv4() {
|
|
Socket::new(Domain::IPV4, Type::DGRAM, Some(Protocol::UDP))?
|
|
} else {
|
|
Socket::new(Domain::IPV6, Type::DGRAM, Some(Protocol::UDP))?
|
|
};
|
|
|
|
if config.network.only_ipv6 {
|
|
socket
|
|
.set_only_v6(true)
|
|
.with_context(|| "socket: set only ipv6")?;
|
|
}
|
|
|
|
socket
|
|
.set_reuse_port(true)
|
|
.with_context(|| "socket: set reuse port")?;
|
|
|
|
socket
|
|
.set_nonblocking(true)
|
|
.with_context(|| "socket: set nonblocking")?;
|
|
|
|
let recv_buffer_size = config.network.socket_recv_buffer_size;
|
|
|
|
if recv_buffer_size != 0 {
|
|
if let Err(err) = socket.set_recv_buffer_size(recv_buffer_size) {
|
|
::log::error!(
|
|
"socket: failed setting recv buffer to {}: {:?}",
|
|
recv_buffer_size,
|
|
err
|
|
);
|
|
}
|
|
}
|
|
|
|
socket
|
|
.bind(&config.network.address.into())
|
|
.with_context(|| format!("socket: bind to {}", config.network.address))?;
|
|
|
|
priv_dropper.after_socket_creation()?;
|
|
|
|
Ok(socket.into())
|
|
}
|