pub enum Outcome {
LimitRaised {
from: u64,
to: u64,
},
Unsupported,
}
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("Failed to call sysctl to get max supported value configured in sysctl: {0}")]
#[cfg(any(target_os = "macos", target_os = "ios"))]
FailedToCallSysctl(std::io::Error),
#[error("Failed to get current limit: {0}")]
FailedToGetLimit(std::io::Error),
#[error("Failed to set new limit ({from}->{to}): {error}")]
FailedToSetLimit {
from: u64,
to: u64,
error: std::io::Error,
},
}
#[cfg(any(target_os = "macos", target_os = "ios"))]
#[allow(clippy::useless_conversion, non_camel_case_types)]
pub fn raise_fd_limit() -> Result<Outcome, Error> {
use std::cmp;
use std::io;
use std::mem::size_of_val;
use std::ptr::null_mut;
unsafe {
static CTL_KERN: libc::c_int = 1;
static KERN_MAXFILESPERPROC: libc::c_int = 29;
let mut mib: [libc::c_int; 2] = [CTL_KERN, KERN_MAXFILESPERPROC];
let mut maxfiles: libc::c_int = 0;
let mut size: libc::size_t = size_of_val(&maxfiles) as libc::size_t;
if libc::sysctl(&mut mib[0], 2, &mut maxfiles as *mut _ as *mut _, &mut size, null_mut(), 0)
!= 0
{
return Err(Error::FailedToCallSysctl(io::Error::last_os_error()));
}
let mut rlim = libc::rlimit { rlim_cur: 0, rlim_max: 0 };
if libc::getrlimit(libc::RLIMIT_NOFILE, &mut rlim) != 0 {
return Err(Error::FailedToGetLimit(io::Error::last_os_error()));
}
let old_value = rlim.rlim_cur;
rlim.rlim_cur = cmp::min(maxfiles as libc::rlim_t, rlim.rlim_max);
if libc::setrlimit(libc::RLIMIT_NOFILE, &rlim) != 0 {
return Err(Error::FailedToSetLimit {
from: old_value.into(),
to: rlim.rlim_cur.into(),
error: io::Error::last_os_error(),
});
}
Ok(Outcome::LimitRaised { from: old_value.into(), to: rlim.rlim_cur.into() })
}
}
#[cfg(target_os = "linux")]
#[allow(clippy::useless_conversion, non_camel_case_types)]
pub fn raise_fd_limit() -> Result<Outcome, Error> {
use std::io;
unsafe {
let mut rlim = libc::rlimit { rlim_cur: 0, rlim_max: 0 };
if libc::getrlimit(libc::RLIMIT_NOFILE, &mut rlim) != 0 {
return Err(Error::FailedToGetLimit(io::Error::last_os_error()));
}
let old_value = rlim.rlim_cur;
rlim.rlim_cur = rlim.rlim_max;
if libc::setrlimit(libc::RLIMIT_NOFILE, &rlim) != 0 {
return Err(Error::FailedToSetLimit {
from: old_value.into(),
to: rlim.rlim_cur.into(),
error: io::Error::last_os_error(),
});
}
Ok(Outcome::LimitRaised { from: old_value.into(), to: rlim.rlim_cur.into() })
}
}
#[cfg(not(any(target_os = "macos", target_os = "ios", target_os = "linux")))]
pub fn raise_fd_limit() -> Result<Outcome, Error> {
Ok(Outcome::Unsupported)
}