aboutsummaryrefslogtreecommitdiff
path: root/src/sys/unix/tcp.rs
diff options
context:
space:
mode:
Diffstat (limited to 'src/sys/unix/tcp.rs')
-rw-r--r--src/sys/unix/tcp.rs432
1 files changed, 29 insertions, 403 deletions
diff --git a/src/sys/unix/tcp.rs b/src/sys/unix/tcp.rs
index 59642c6..c4d7e94 100644
--- a/src/sys/unix/tcp.rs
+++ b/src/sys/unix/tcp.rs
@@ -1,428 +1,57 @@
use std::convert::TryInto;
use std::io;
-use std::mem;
use std::mem::{size_of, MaybeUninit};
use std::net::{self, SocketAddr};
use std::os::unix::io::{AsRawFd, FromRawFd};
-use std::time::Duration;
use crate::sys::unix::net::{new_socket, socket_addr, to_socket_addr};
-use crate::net::TcpKeepalive;
-#[cfg(any(target_os = "openbsd", target_os = "netbsd", target_os = "haiku"))]
-use libc::SO_KEEPALIVE as KEEPALIVE_TIME;
-#[cfg(any(target_os = "macos", target_os = "ios"))]
-use libc::TCP_KEEPALIVE as KEEPALIVE_TIME;
-#[cfg(not(any(
- target_os = "macos",
- target_os = "ios",
- target_os = "openbsd",
- target_os = "netbsd",
- target_os = "haiku"
-)))]
-use libc::TCP_KEEPIDLE as KEEPALIVE_TIME;
-pub type TcpSocket = libc::c_int;
-
-pub(crate) fn new_v4_socket() -> io::Result<TcpSocket> {
- new_socket(libc::AF_INET, libc::SOCK_STREAM)
-}
-
-pub(crate) fn new_v6_socket() -> io::Result<TcpSocket> {
- new_socket(libc::AF_INET6, libc::SOCK_STREAM)
+pub(crate) fn new_for_addr(address: SocketAddr) -> io::Result<libc::c_int> {
+ let domain = match address {
+ SocketAddr::V4(_) => libc::AF_INET,
+ SocketAddr::V6(_) => libc::AF_INET6,
+ };
+ new_socket(domain, libc::SOCK_STREAM)
}
-pub(crate) fn bind(socket: TcpSocket, addr: SocketAddr) -> io::Result<()> {
+pub(crate) fn bind(socket: &net::TcpListener, addr: SocketAddr) -> io::Result<()> {
let (raw_addr, raw_addr_length) = socket_addr(&addr);
- syscall!(bind(socket, raw_addr.as_ptr(), raw_addr_length))?;
+ syscall!(bind(socket.as_raw_fd(), raw_addr.as_ptr(), raw_addr_length))?;
Ok(())
}
-pub(crate) fn connect(socket: TcpSocket, addr: SocketAddr) -> io::Result<net::TcpStream> {
+pub(crate) fn connect(socket: &net::TcpStream, addr: SocketAddr) -> io::Result<()> {
let (raw_addr, raw_addr_length) = socket_addr(&addr);
- match syscall!(connect(socket, raw_addr.as_ptr(), raw_addr_length)) {
- Err(err) if err.raw_os_error() != Some(libc::EINPROGRESS) => {
- Err(err)
- }
- _ => {
- Ok(unsafe { net::TcpStream::from_raw_fd(socket) })
- }
+ match syscall!(connect(
+ socket.as_raw_fd(),
+ raw_addr.as_ptr(),
+ raw_addr_length
+ )) {
+ Err(err) if err.raw_os_error() != Some(libc::EINPROGRESS) => Err(err),
+ _ => Ok(()),
}
}
-pub(crate) fn listen(socket: TcpSocket, backlog: u32) -> io::Result<net::TcpListener> {
+pub(crate) fn listen(socket: &net::TcpListener, backlog: u32) -> io::Result<()> {
let backlog = backlog.try_into().unwrap_or(i32::max_value());
- syscall!(listen(socket, backlog))?;
- Ok(unsafe { net::TcpListener::from_raw_fd(socket) })
-}
-
-pub(crate) fn close(socket: TcpSocket) {
- let _ = unsafe { net::TcpStream::from_raw_fd(socket) };
+ syscall!(listen(socket.as_raw_fd(), backlog))?;
+ Ok(())
}
-pub(crate) fn set_reuseaddr(socket: TcpSocket, reuseaddr: bool) -> io::Result<()> {
+pub(crate) fn set_reuseaddr(socket: &net::TcpListener, reuseaddr: bool) -> io::Result<()> {
let val: libc::c_int = if reuseaddr { 1 } else { 0 };
syscall!(setsockopt(
- socket,
+ socket.as_raw_fd(),
libc::SOL_SOCKET,
libc::SO_REUSEADDR,
&val as *const libc::c_int as *const libc::c_void,
size_of::<libc::c_int>() as libc::socklen_t,
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_reuseaddr(socket: TcpSocket) -> io::Result<bool> {
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
-
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_REUSEADDR,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(optval != 0)
-}
-
-#[cfg(all(unix, not(any(target_os = "solaris", target_os = "illumos"))))]
-pub(crate) fn set_reuseport(socket: TcpSocket, reuseport: bool) -> io::Result<()> {
- let val: libc::c_int = if reuseport { 1 } else { 0 };
-
- syscall!(setsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_REUSEPORT,
- &val as *const libc::c_int as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t,
- ))
- .map(|_| ())
-}
-
-#[cfg(all(unix, not(any(target_os = "solaris", target_os = "illumos"))))]
-pub(crate) fn get_reuseport(socket: TcpSocket) -> io::Result<bool> {
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
-
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_REUSEPORT,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(optval != 0)
-}
-
-pub(crate) fn get_localaddr(socket: TcpSocket) -> io::Result<SocketAddr> {
- let mut addr: libc::sockaddr_storage = unsafe { std::mem::zeroed() };
- let mut length = size_of::<libc::sockaddr_storage>() as libc::socklen_t;
-
- syscall!(getsockname(
- socket,
- &mut addr as *mut _ as *mut _,
- &mut length
))?;
-
- unsafe { to_socket_addr(&addr) }
-}
-
-pub(crate) fn set_linger(socket: TcpSocket, dur: Option<Duration>) -> io::Result<()> {
- let val: libc::linger = libc::linger {
- l_onoff: if dur.is_some() { 1 } else { 0 },
- l_linger: dur
- .map(|dur| dur.as_secs() as libc::c_int)
- .unwrap_or_default(),
- };
- syscall!(setsockopt(
- socket,
- libc::SOL_SOCKET,
- #[cfg(target_vendor = "apple")]
- libc::SO_LINGER_SEC,
- #[cfg(not(target_vendor = "apple"))]
- libc::SO_LINGER,
- &val as *const libc::linger as *const libc::c_void,
- size_of::<libc::linger>() as libc::socklen_t,
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_linger(socket: TcpSocket) -> io::Result<Option<Duration>> {
- let mut val: libc::linger = unsafe { std::mem::zeroed() };
- let mut len = mem::size_of::<libc::linger>() as libc::socklen_t;
-
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- #[cfg(target_vendor = "apple")]
- libc::SO_LINGER_SEC,
- #[cfg(not(target_vendor = "apple"))]
- libc::SO_LINGER,
- &mut val as *mut _ as *mut _,
- &mut len,
- ))?;
-
- if val.l_onoff == 0 {
- Ok(None)
- } else {
- Ok(Some(Duration::from_secs(val.l_linger as u64)))
- }
-}
-
-pub(crate) fn set_recv_buffer_size(socket: TcpSocket, size: u32) -> io::Result<()> {
- let size = size.try_into().ok().unwrap_or_else(i32::max_value);
- syscall!(setsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_RCVBUF,
- &size as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_recv_buffer_size(socket: TcpSocket) -> io::Result<u32> {
- let mut optval: libc::c_int = 0;
- let mut optlen = size_of::<libc::c_int>() as libc::socklen_t;
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_RCVBUF,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(optval as u32)
-}
-
-pub(crate) fn set_send_buffer_size(socket: TcpSocket, size: u32) -> io::Result<()> {
- let size = size.try_into().ok().unwrap_or_else(i32::max_value);
- syscall!(setsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_SNDBUF,
- &size as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_send_buffer_size(socket: TcpSocket) -> io::Result<u32> {
- let mut optval: libc::c_int = 0;
- let mut optlen = size_of::<libc::c_int>() as libc::socklen_t;
-
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_SNDBUF,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(optval as u32)
-}
-
-pub(crate) fn set_keepalive(socket: TcpSocket, keepalive: bool) -> io::Result<()> {
- let val: libc::c_int = if keepalive { 1 } else { 0 };
- syscall!(setsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_KEEPALIVE,
- &val as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_keepalive(socket: TcpSocket) -> io::Result<bool> {
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
-
- syscall!(getsockopt(
- socket,
- libc::SOL_SOCKET,
- libc::SO_KEEPALIVE,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(optval != 0)
-}
-
-pub(crate) fn set_keepalive_params(socket: TcpSocket, keepalive: TcpKeepalive) -> io::Result<()> {
- if let Some(dur) = keepalive.time {
- set_keepalive_time(socket, dur)?;
- }
-
- #[cfg(any(
- target_os = "linux",
- target_os = "macos",
- target_os = "ios",
- target_os = "freebsd",
- target_os = "netbsd",
- ))]
- {
- if let Some(dur) = keepalive.interval {
- set_keepalive_interval(socket, dur)?;
- }
-
- if let Some(retries) = keepalive.retries {
- set_keepalive_retries(socket, retries)?;
- }
- }
-
-
Ok(())
}
-fn set_keepalive_time(socket: TcpSocket, time: Duration) -> io::Result<()> {
- let time_secs = time
- .as_secs()
- .try_into()
- .ok()
- .unwrap_or_else(i32::max_value);
- syscall!(setsockopt(
- socket,
- libc::IPPROTO_TCP,
- KEEPALIVE_TIME,
- &(time_secs as libc::c_int) as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-pub(crate) fn get_keepalive_time(socket: TcpSocket) -> io::Result<Option<Duration>> {
- if !get_keepalive(socket)? {
- return Ok(None);
- }
-
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
- syscall!(getsockopt(
- socket,
- libc::IPPROTO_TCP,
- KEEPALIVE_TIME,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(Some(Duration::from_secs(optval as u64)))
-}
-
-/// Linux, FreeBSD, and NetBSD support setting the keepalive interval via
-/// `TCP_KEEPINTVL`.
-/// See:
-/// - https://man7.org/linux/man-pages/man7/tcp.7.html
-/// - https://www.freebsd.org/cgi/man.cgi?query=tcp#end
-/// - http://man.netbsd.org/tcp.4#DESCRIPTION
-///
-/// OpenBSD does not:
-/// https://man.openbsd.org/tcp
-#[cfg(any(
- target_os = "linux",
- target_os = "macos",
- target_os = "ios",
- target_os = "freebsd",
- target_os = "netbsd",
-))]
-fn set_keepalive_interval(socket: TcpSocket, interval: Duration) -> io::Result<()> {
- let interval_secs = interval
- .as_secs()
- .try_into()
- .ok()
- .unwrap_or_else(i32::max_value);
- syscall!(setsockopt(
- socket,
- libc::IPPROTO_TCP,
- libc::TCP_KEEPINTVL,
- &(interval_secs as libc::c_int) as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-#[cfg(any(
- target_os = "linux",
- target_os = "macos",
- target_os = "ios",
- target_os = "freebsd",
- target_os = "netbsd",
-))]
-pub(crate) fn get_keepalive_interval(socket: TcpSocket) -> io::Result<Option<Duration>> {
- if !get_keepalive(socket)? {
- return Ok(None);
- }
-
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
- syscall!(getsockopt(
- socket,
- libc::IPPROTO_TCP,
- libc::TCP_KEEPINTVL,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(Some(Duration::from_secs(optval as u64)))
-}
-
-/// Linux, macOS/iOS, FreeBSD, and NetBSD support setting the number of TCP
-/// keepalive retries via `TCP_KEEPCNT`.
-/// See:
-/// - https://man7.org/linux/man-pages/man7/tcp.7.html
-/// - https://www.freebsd.org/cgi/man.cgi?query=tcp#end
-/// - http://man.netbsd.org/tcp.4#DESCRIPTION
-///
-/// OpenBSD does not:
-/// https://man.openbsd.org/tcp
-#[cfg(any(
- target_os = "linux",
- target_os = "macos",
- target_os = "ios",
- target_os = "freebsd",
- target_os = "netbsd",
-))]
-fn set_keepalive_retries(socket: TcpSocket, retries: u32) -> io::Result<()> {
- let retries = retries.try_into().ok().unwrap_or_else(i32::max_value);
- syscall!(setsockopt(
- socket,
- libc::IPPROTO_TCP,
- libc::TCP_KEEPCNT,
- &(retries as libc::c_int) as *const _ as *const libc::c_void,
- size_of::<libc::c_int>() as libc::socklen_t
- ))
- .map(|_| ())
-}
-
-#[cfg(any(
- target_os = "linux",
- target_os = "macos",
- target_os = "ios",
- target_os = "freebsd",
- target_os = "netbsd",
-))]
-pub(crate) fn get_keepalive_retries(socket: TcpSocket) -> io::Result<Option<u32>> {
- if !get_keepalive(socket)? {
- return Ok(None);
- }
-
- let mut optval: libc::c_int = 0;
- let mut optlen = mem::size_of::<libc::c_int>() as libc::socklen_t;
- syscall!(getsockopt(
- socket,
- libc::IPPROTO_TCP,
- libc::TCP_KEEPCNT,
- &mut optval as *mut _ as *mut _,
- &mut optlen,
- ))?;
-
- Ok(Some(optval as u32))
-}
-
-pub fn accept(listener: &net::TcpListener) -> io::Result<(net::TcpStream, SocketAddr)> {
+pub(crate) fn accept(listener: &net::TcpListener) -> io::Result<(net::TcpStream, SocketAddr)> {
let mut addr: MaybeUninit<libc::sockaddr_storage> = MaybeUninit::uninit();
let mut length = size_of::<libc::sockaddr_storage>() as libc::socklen_t;
@@ -456,13 +85,10 @@ pub fn accept(listener: &net::TcpListener) -> io::Result<(net::TcpStream, Socket
// OSes inherit the non-blocking flag from the listener, so we just have to
// set `CLOEXEC`.
#[cfg(any(
- all(
- target_arch = "x86",
- target_os = "android"
- ),
- target_os = "ios",
- target_os = "macos",
- target_os = "solaris"
+ all(target_arch = "x86", target_os = "android"),
+ target_os = "ios",
+ target_os = "macos",
+ target_os = "redox"
))]
let stream = {
syscall!(accept(
@@ -473,11 +99,11 @@ pub fn accept(listener: &net::TcpListener) -> io::Result<(net::TcpStream, Socket
.map(|socket| unsafe { net::TcpStream::from_raw_fd(socket) })
.and_then(|s| {
syscall!(fcntl(s.as_raw_fd(), libc::F_SETFD, libc::FD_CLOEXEC))?;
-
+
// See https://github.com/tokio-rs/mio/issues/1450
- #[cfg(all(target_arch = "x86",target_os = "android"))]
+ #[cfg(all(target_arch = "x86", target_os = "android"))]
syscall!(fcntl(s.as_raw_fd(), libc::F_SETFL, libc::O_NONBLOCK))?;
-
+
Ok(s)
})
}?;