// Copyright 2015 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use prelude::v1::*; use ffi::CStr; use io; use libc::{self, c_int, size_t, sockaddr, socklen_t}; use net::{SocketAddr, Shutdown}; use str; use sys::fd::FileDesc; use sys_common::{AsInner, FromInner, IntoInner}; use sys_common::net::{getsockopt, setsockopt}; use time::Duration; pub use sys::{cvt, cvt_r}; pub extern crate libc as netc; pub type wrlen_t = size_t; // See below for the usage of SOCK_CLOEXEC, but this constant is only defined on // Linux currently (e.g. support doesn't exist on other platforms). In order to // get name resolution to work and things to compile we just define a dummy // SOCK_CLOEXEC here for other platforms. Note that the dummy constant isn't // actually ever used (the blocks below are wrapped in `if cfg!` as well. #[cfg(target_os = "linux")] use libc::SOCK_CLOEXEC; #[cfg(not(target_os = "linux"))] const SOCK_CLOEXEC: c_int = 0; #[cfg(any(target_os = "openbsd", taret_os = "freebsd"))] use libc::SO_KEEPALIVE as TCP_KEEPALIVE; #[cfg(any(target_os = "macos", taret_os = "ios"))] use libc::TCP_KEEPALIVE; #[cfg(not(any(target_os = "openbsd", target_os = "freebsd", target_os = "macos", target_os = "ios")))] use libc::TCP_KEEPIDLE as TCP_KEEPALIVE; pub struct Socket(FileDesc); pub fn init() {} pub fn cvt_gai(err: c_int) -> io::Result<()> { if err == 0 { return Ok(()) } let detail = unsafe { str::from_utf8(CStr::from_ptr(libc::gai_strerror(err)).to_bytes()).unwrap() .to_owned() }; Err(io::Error::new(io::ErrorKind::Other, &format!("failed to lookup address information: {}", detail)[..])) } impl Socket { pub fn new(addr: &SocketAddr, ty: c_int) -> io::Result { let fam = match *addr { SocketAddr::V4(..) => libc::AF_INET, SocketAddr::V6(..) => libc::AF_INET6, }; unsafe { // On linux we first attempt to pass the SOCK_CLOEXEC flag to // atomically create the socket and set it as CLOEXEC. Support for // this option, however, was added in 2.6.27, and we still support // 2.6.18 as a kernel, so if the returned error is EINVAL we // fallthrough to the fallback. if cfg!(target_os = "linux") { match cvt(libc::socket(fam, ty | SOCK_CLOEXEC, 0)) { Ok(fd) => return Ok(Socket(FileDesc::new(fd))), Err(ref e) if e.raw_os_error() == Some(libc::EINVAL) => {} Err(e) => return Err(e), } } let fd = try!(cvt(libc::socket(fam, ty, 0))); let fd = FileDesc::new(fd); fd.set_cloexec(); Ok(Socket(fd)) } } pub fn accept(&self, storage: *mut sockaddr, len: *mut socklen_t) -> io::Result { // Unfortunately the only known way right now to accept a socket and // atomically set the CLOEXEC flag is to use the `accept4` syscall on // Linux. This was added in 2.6.28, however, and because we support // 2.6.18 we must detect this support dynamically. if cfg!(target_os = "linux") { weak! { fn accept4(c_int, *mut sockaddr, *mut socklen_t, c_int) -> c_int } if let Some(accept) = accept4.get() { let res = cvt_r(|| unsafe { accept(self.0.raw(), storage, len, SOCK_CLOEXEC) }); match res { Ok(fd) => return Ok(Socket(FileDesc::new(fd))), Err(ref e) if e.raw_os_error() == Some(libc::ENOSYS) => {} Err(e) => return Err(e), } } } let fd = try!(cvt_r(|| unsafe { libc::accept(self.0.raw(), storage, len) })); let fd = FileDesc::new(fd); fd.set_cloexec(); Ok(Socket(fd)) } pub fn duplicate(&self) -> io::Result { self.0.duplicate().map(Socket) } pub fn read(&self, buf: &mut [u8]) -> io::Result { self.0.read(buf) } pub fn set_timeout(&self, dur: Option, kind: libc::c_int) -> io::Result<()> { let timeout = match dur { Some(dur) => { if dur.as_secs() == 0 && dur.subsec_nanos() == 0 { return Err(io::Error::new(io::ErrorKind::InvalidInput, "cannot set a 0 duration timeout")); } let secs = if dur.as_secs() > libc::time_t::max_value() as u64 { libc::time_t::max_value() } else { dur.as_secs() as libc::time_t }; let mut timeout = libc::timeval { tv_sec: secs, tv_usec: (dur.subsec_nanos() / 1000) as libc::suseconds_t, }; if timeout.tv_sec == 0 && timeout.tv_usec == 0 { timeout.tv_usec = 1; } timeout } None => { libc::timeval { tv_sec: 0, tv_usec: 0, } } }; setsockopt(self, libc::SOL_SOCKET, kind, timeout) } pub fn timeout(&self, kind: libc::c_int) -> io::Result> { let raw: libc::timeval = try!(getsockopt(self, libc::SOL_SOCKET, kind)); if raw.tv_sec == 0 && raw.tv_usec == 0 { Ok(None) } else { let sec = raw.tv_sec as u64; let nsec = (raw.tv_usec as u32) * 1000; Ok(Some(Duration::new(sec, nsec))) } } pub fn shutdown(&self, how: Shutdown) -> io::Result<()> { let how = match how { Shutdown::Write => libc::SHUT_WR, Shutdown::Read => libc::SHUT_RD, Shutdown::Both => libc::SHUT_RDWR, }; try!(cvt(unsafe { libc::shutdown(self.0.raw(), how) })); Ok(()) } pub fn set_keepalive(&self, keepalive: Option) -> io::Result<()> { try!(setsockopt(self, libc::SOL_SOCKET, libc::SO_KEEPALIVE, keepalive.is_some() as libc::c_int)); if let Some(dur) = keepalive { let mut raw = dur.as_secs(); if dur.subsec_nanos() > 0 { raw = raw.saturating_add(1); } let raw = if raw > libc::c_int::max_value() as u64 { libc::c_int::max_value() } else { raw as libc::c_int }; try!(setsockopt(self, libc::IPPROTO_TCP, TCP_KEEPALIVE, raw)); } Ok(()) } pub fn keepalive(&self) -> io::Result> { let raw: c_int = try!(getsockopt(self, libc::SOL_SOCKET, libc::SO_KEEPALIVE)); if raw == 0 { return Ok(None); } let raw: c_int = try!(getsockopt(self, libc::IPPROTO_TCP, TCP_KEEPALIVE)); Ok(Some(Duration::from_secs(raw as u64))) } pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> { let mut nonblocking = nonblocking as libc::c_ulong; cvt(unsafe { libc::ioctl(*self.as_inner(), libc::FIONBIO, &mut nonblocking) }).map(|_| ()) } } impl AsInner for Socket { fn as_inner(&self) -> &c_int { self.0.as_inner() } } impl FromInner for Socket { fn from_inner(fd: c_int) -> Socket { Socket(FileDesc::new(fd)) } } impl IntoInner for Socket { fn into_inner(self) -> c_int { self.0.into_raw() } }