#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[derive(Deserialize, Serialize)]
#[repr(transparent)]
pub struct ReceiveBufferSizeInBytes(NonZeroU32);
impl TryFrom<NonZeroU32> for ReceiveBufferSizeInBytes
{
type Error = ParseNumberError;
#[inline(always)]
fn try_from(value: NonZeroU32) -> Result<Self, Self::Error>
{
if unlikely!(value < Self::UsualInclusiveMinimumForTransmissionControlProtocol.0)
{
Err(ParseNumberError::TooSmall)
}
else if unlikely!(value > Self::InclusiveMaximumMaximum.0)
{
Err(ParseNumberError::TooLarge)
}
else
{
Ok(Self(value))
}
}
}
impl ParseNumber for ReceiveBufferSizeInBytes
{
#[inline(always)]
fn parse_number(bytes: &[u8], radix: Radix, parse_byte: impl Fn(Radix, u8) -> Result<u8, ParseNumberError>) -> Result<Self, ParseNumberError>
{
let raw_value = NonZeroU32::parse_number(bytes, radix, parse_byte)?;
Self::try_from(raw_value)
}
}
impl ReceiveBufferSizeInBytes
{
pub const UsualGlobalMaximum: Self = Self(new_non_zero_u32(212_992));
#[inline(always)]
pub fn global_maximum(proc_path: &ProcPath) -> Self
{
Self(Self::sys_net_core_rmem_max_file_path(proc_path).read_value().unwrap())
}
#[inline(always)]
pub fn set_global_maximum(self, proc_path: &ProcPath) -> io::Result<()>
{
assert_effective_user_id_is_root("write to /proc/sys/net/core/rmem_max");
let file_path = Self::sys_net_core_rmem_max_file_path(proc_path);
if file_path.exists()
{
file_path.write_value(UnpaddedDecimalInteger(self.0))
}
else
{
Ok(())
}
}
pub const UsualGlobalDefault: Self = Self(new_non_zero_u32(212_992));
#[inline(always)]
pub fn global_default(proc_path: &ProcPath) -> Self
{
Self(Self::sys_net_core_rmem_default_file_path(proc_path).read_value().unwrap())
}
#[inline(always)]
pub fn set_global_default(self, proc_path: &ProcPath) -> io::Result<()>
{
assert_effective_user_id_is_root("write to /proc/sys/net/core/rmem_default");
let file_path = Self::sys_net_core_rmem_default_file_path(proc_path);
if file_path.exists()
{
file_path.write_value(UnpaddedDecimalInteger(self.0))
}
else
{
Ok(())
}
}
pub const UsualInclusiveMinimumForTransmissionControlProtocol: Self = Self(new_non_zero_u32(4_096));
pub const UsualGlobalDefaultForTransmissionControlProtocol: Self = Self(new_non_zero_u32(131_072));
pub const UsualInclusiveMaximumForTransmissionControlProtocol: Self = Self(new_non_zero_u32(6_291_456));
#[inline(always)]
pub fn global_tcp_minimum_default_and_maximum(proc_path: &ProcPath) -> io::Result<(Self, Self, Self)>
{
let line = Self::sys_net_ipv4_tcp_rmem_file_path(proc_path).read_raw_without_line_feed()?;
let mut fields = line.split_bytes_n(3, b'\t');
#[inline(always)]
fn parse_field<'a>(fields: &mut impl Iterator<Item=&'a [u8]>) -> io::Result<ReceiveBufferSizeInBytes>
{
ReceiveBufferSizeInBytes::from_bytes(fields.next().unwrap()).map_err(io_error_other)
}
let minimum = parse_field(&mut fields)?;
let default = parse_field(&mut fields)?;
let maximum = parse_field(&mut fields)?;
Ok((minimum, default, maximum))
}
#[inline(always)]
pub fn set_global_tcp_minimum_default_and_maximum(proc_path: &ProcPath, (minimum, default, maximum): (Self, Self, Self)) -> io::Result<()>
{
assert_effective_user_id_is_root("write to /proc/sys/net/ipv4/tcp_rmem");
let file_path = Self::sys_net_ipv4_tcp_rmem_file_path(proc_path);
if file_path.exists()
{
file_path.write_value(format!("{}\t{}\t{}", minimum.0, default.0, maximum.0))
}
else
{
Ok(())
}
}
pub const InclusiveMaximumMaximum: Self = Self(new_non_zero_u32(i32::MAX as u32));
#[inline(always)]
const fn adjust_for_tcp_set_sock_opt(self) -> u32
{
let receive_buffer_size_in_bytes = self.0.get();
receive_buffer_size_in_bytes / 2
}
#[inline(always)]
fn sys_net_core_rmem_max_file_path(proc_path: &ProcPath) -> PathBuf
{
proc_path.sys_net_core_file_path("rmem_max")
}
#[inline(always)]
fn sys_net_core_rmem_default_file_path(proc_path: &ProcPath) -> PathBuf
{
proc_path.sys_net_core_file_path("rmem_default")
}
#[inline(always)]
fn sys_net_ipv4_tcp_rmem_file_path(proc_path: &ProcPath) -> PathBuf
{
proc_path.sys_net_ipv4_file_path("tcp_rmem")
}
}