Skip to content

Commit 8f9c30f

Browse files
authored
feat: add socket buffer size configuration (#22)
1 parent b9b0c1f commit 8f9c30f

File tree

6 files changed

+126
-11
lines changed

6 files changed

+126
-11
lines changed

Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ rustls = { version = "0.23.23", features = ["ring"] }
2121
tokio = { version = "1.43.0", features = ["full"] }
2222
# included to satisfy napi dependencies
2323
ctor = "0.3.6"
24+
socket2 = "0.5.8"
2425

2526
[build-dependencies]
2627
napi-build = "2.1.4"

rust/config.rs

+45-5
Original file line numberDiff line numberDiff line change
@@ -64,25 +64,55 @@ pub struct Config {
6464
/// Timeout for the initial handshake when establishing a connection.
6565
/// The actual timeout is the minimum of this and the [`Config::max_idle_timeout`].
6666
pub handshake_timeout: u32,
67+
6768
/// Maximum duration of inactivity in ms to accept before timing out the connection.
6869
pub max_idle_timeout: u32,
70+
6971
/// Period of inactivity before sending a keep-alive packet.
7072
/// Must be set lower than the idle_timeout of both
7173
/// peers to be effective.
7274
///
7375
/// See [`quinn::TransportConfig::keep_alive_interval`] for more
7476
/// info.
7577
pub keep_alive_interval: u32,
78+
7679
/// Maximum number of incoming bidirectional streams that may be open
7780
/// concurrently by the remote peer.
7881
pub max_concurrent_stream_limit: u32,
7982

80-
/// Max unacknowledged data in bytes that may be sent on a single stream.
83+
/// Maximum number of bytes the peer may transmit without acknowledgement on any one stream
84+
/// before becoming blocked.
85+
///
86+
/// This should be set to at least the expected connection latency multiplied by the maximum
87+
/// desired throughput. Setting this smaller than `max_connection_data` helps ensure that a single
88+
/// stream doesn't monopolize receive buffers, which may otherwise occur if the application
89+
/// chooses not to read from a large stream for a time while still requiring data on other
90+
/// streams.
8191
pub max_stream_data: u32,
8292

83-
/// Max unacknowledged data in bytes that may be sent in total on all streams
84-
/// of a connection.
93+
/// Maximum number of bytes the peer may transmit across all streams of a connection before
94+
/// becoming blocked.
95+
///
96+
/// This should be set to at least the expected connection latency multiplied by the maximum
97+
/// desired throughput. Larger values can be useful to allow maximum throughput within a
98+
/// stream while another is blocked.
8599
pub max_connection_data: u32,
100+
101+
/// OS socket receive buffer size.
102+
///
103+
/// If this is set higher than the OS maximum, it will be clamped to the maximum allowed size.
104+
pub receive_buffer_size: u32,
105+
106+
/// OS socket send buffer size.
107+
///
108+
/// If this is set higher than the OS maximum, it will be clamped to the maximum allowed size.
109+
pub send_buffer_size: u32,
110+
}
111+
112+
#[derive(Clone, Copy)]
113+
pub struct SocketConfig {
114+
pub receive_buffer_size: u32,
115+
pub send_buffer_size: u32,
86116
}
87117

88118
/// Configuration used by the QUIC library
@@ -92,6 +122,7 @@ pub struct QuinnConfig {
92122
pub(crate) client_config: quinn::ClientConfig,
93123
pub(crate) server_config: quinn::ServerConfig,
94124
pub(crate) endpoint_config: quinn::EndpointConfig,
125+
pub(crate) socket_config: SocketConfig,
95126
}
96127

97128
#[napi]
@@ -113,20 +144,25 @@ impl TryFrom<Config> for QuinnConfig {
113144
max_connection_data,
114145
max_stream_data,
115146
handshake_timeout: _,
147+
receive_buffer_size,
148+
send_buffer_size,
116149
} = config;
117150

118151
let keypair = libp2p_identity::Keypair::from_protobuf_encoding(&private_key_proto)
119152
.map_err(|e| ConfigError::InvalidPrivateKey(e))?;
120153

121154
let mut transport = quinn::TransportConfig::default();
155+
156+
// Disable features we don't use/want
122157
// Disable uni-directional streams.
123158
transport.max_concurrent_uni_streams(0u32.into());
124-
transport.max_concurrent_bidi_streams(max_concurrent_stream_limit.into());
125159
// Disable datagrams.
126160
transport.datagram_receive_buffer_size(None);
161+
transport.allow_spin(false);
162+
163+
transport.max_concurrent_bidi_streams(max_concurrent_stream_limit.into());
127164
transport.keep_alive_interval(Some(Duration::from_millis(keep_alive_interval.into())));
128165
transport.max_idle_timeout(Some(quinn::VarInt::from_u32(max_idle_timeout).into()));
129-
transport.allow_spin(false);
130166
transport.stream_receive_window(max_stream_data.into());
131167
transport.receive_window(max_connection_data.into());
132168
transport.mtu_discovery_config(Default::default());
@@ -170,6 +206,10 @@ impl TryFrom<Config> for QuinnConfig {
170206
client_config,
171207
server_config,
172208
endpoint_config,
209+
socket_config: SocketConfig {
210+
receive_buffer_size,
211+
send_buffer_size,
212+
},
173213
})
174214
}
175215
}

rust/lib.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,8 @@ impl Server {
3030
pub fn new(config: &config::QuinnConfig, ip: String, port: u16) -> Result<Self> {
3131
let ip_addr = ip.parse::<IpAddr>().map_err(to_err)?;
3232
let socket_addr = SocketAddr::new(ip_addr, port);
33-
let socket = std::net::UdpSocket::bind(socket_addr)?;
33+
let socket = socket::create_socket(config.socket_config, socket_addr)?;
34+
3435
let socket = block_on(async move{
3536
socket::UdpSocket::wrap_udp_socket(socket)
3637
})?;
@@ -93,7 +94,7 @@ impl Client {
9394
SocketFamily::Ipv6 => SocketAddr::new(std::net::Ipv6Addr::UNSPECIFIED.into(), 0),
9495
};
9596
let mut endpoint = block_on(async move {
96-
let socket = std::net::UdpSocket::bind(bind_addr)?;
97+
let socket = socket::create_socket(config.socket_config,bind_addr)?;
9798
let endpoint = quinn::Endpoint::new(
9899
config.endpoint_config.clone(),
99100
None,

rust/socket.rs

+46
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,19 @@ use std::{
2323
use quinn::udp;
2424
use quinn::{AsyncUdpSocket, UdpPoller};
2525
use tokio::{io::Interest, sync::RwLock};
26+
use socket2::Socket;
27+
28+
use crate::config;
29+
30+
pub fn create_socket(config: config::SocketConfig, socket_addr: std::net::SocketAddr) -> napi::Result<std::net::UdpSocket> {
31+
let socket = std::net::UdpSocket::bind(socket_addr)?;
32+
33+
let socket = Socket::from(socket);
34+
socket.set_send_buffer_size(config.send_buffer_size as usize).unwrap();
35+
socket.set_recv_buffer_size(config.receive_buffer_size as usize).unwrap();
36+
let socket = std::net::UdpSocket::from(socket);
37+
Ok(socket)
38+
}
2639

2740
#[derive(Debug)]
2841
pub struct UdpSocket {
@@ -244,3 +257,36 @@ impl<MakeFut, Fut> std::fmt::Debug for UdpPollHelper<MakeFut, Fut> {
244257
f.debug_struct("UdpPollHelper").finish_non_exhaustive()
245258
}
246259
}
260+
261+
#[cfg(test)]
262+
mod tests {
263+
use super::*;
264+
265+
/// Check how buffer sizes are set. On my (linux, ubuntu 24.04.2) machine, this is the output:
266+
///
267+
/// send buffer size: 212992
268+
/// receive buffer size: 212992
269+
/// updating buffer sizes: 1000000
270+
/// send buffer size: 425984
271+
/// receive buffer size: 425984
272+
#[test]
273+
fn check_default_buffer_sizes() {
274+
let socket = std::net::UdpSocket::bind("127.0.0.1:0").unwrap();
275+
let socket = Socket::from(socket);
276+
let send_buffer_size = socket.send_buffer_size().unwrap();
277+
let receive_buffer_size = socket.recv_buffer_size().unwrap();
278+
println!("send buffer size: {}", send_buffer_size);
279+
println!("receive buffer size: {}", receive_buffer_size);
280+
281+
let new_size: usize = 1_000_000;
282+
println!("updating buffer sizes: {}", new_size);
283+
284+
socket.set_send_buffer_size(new_size).unwrap();
285+
socket.set_recv_buffer_size(new_size).unwrap();
286+
287+
let send_buffer_size = socket.send_buffer_size().unwrap();
288+
let receive_buffer_size = socket.recv_buffer_size().unwrap();
289+
println!("send buffer size: {}", send_buffer_size);
290+
println!("receive buffer size: {}", receive_buffer_size);
291+
}
292+
}

src/index.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -51,5 +51,7 @@ export const defaultOptions: QuicOptions = {
5151
keepAliveInterval: 5_000,
5252
maxConcurrentStreamLimit: 256,
5353
maxStreamData: 10_000_000,
54-
maxConnectionData: 15_000_000
54+
maxConnectionData: 15_000_000,
55+
receiveBufferSize: 500_000,
56+
sendBufferSize: 500_000,
5557
}

src/napi.d.ts

+28-3
Original file line numberDiff line numberDiff line change
@@ -382,13 +382,38 @@ export interface Config {
382382
* concurrently by the remote peer.
383383
*/
384384
maxConcurrentStreamLimit: number
385-
/** Max unacknowledged data in bytes that may be sent on a single stream. */
385+
/**
386+
* Maximum number of bytes the peer may transmit without acknowledgement on any one stream
387+
* before becoming blocked.
388+
*
389+
* This should be set to at least the expected connection latency multiplied by the maximum
390+
* desired throughput. Setting this smaller than `max_connection_data` helps ensure that a single
391+
* stream doesn't monopolize receive buffers, which may otherwise occur if the application
392+
* chooses not to read from a large stream for a time while still requiring data on other
393+
* streams.
394+
*/
386395
maxStreamData: number
387396
/**
388-
* Max unacknowledged data in bytes that may be sent in total on all streams
389-
* of a connection.
397+
* Maximum number of bytes the peer may transmit across all streams of a connection before
398+
* becoming blocked.
399+
*
400+
* This should be set to at least the expected connection latency multiplied by the maximum
401+
* desired throughput. Larger values can be useful to allow maximum throughput within a
402+
* stream while another is blocked.
390403
*/
391404
maxConnectionData: number
405+
/**
406+
* OS socket receive buffer size.
407+
*
408+
* If this is set higher than the OS maximum, it will be clamped to the maximum allowed size.
409+
*/
410+
receiveBufferSize: number
411+
/**
412+
* OS socket send buffer size.
413+
*
414+
* If this is set higher than the OS maximum, it will be clamped to the maximum allowed size.
415+
*/
416+
sendBufferSize: number
392417
}
393418

394419
export declare const enum SocketFamily {

0 commit comments

Comments
 (0)