blob: b78d8647978e940334c4d3592dbecf39b8c8b9c8 [file] [log] [blame]
use std::io;
use std::mem::size_of;
use std::net::{self, SocketAddr, SocketAddrV4, SocketAddrV6};
use std::time::Duration;
use std::os::windows::io::FromRawSocket;
use std::os::windows::raw::SOCKET as StdSocket; // winapi uses usize, stdlib uses u32/u64.
use winapi::ctypes::{c_char, c_int, c_ushort};
use winapi::shared::ws2def::{SOCKADDR_STORAGE, AF_INET, SOCKADDR_IN};
use winapi::shared::ws2ipdef::SOCKADDR_IN6_LH;
use winapi::shared::minwindef::{BOOL, TRUE, FALSE};
use winapi::um::winsock2::{
self, closesocket, linger, setsockopt, getsockopt, getsockname, PF_INET, PF_INET6, SOCKET, SOCKET_ERROR,
SOCK_STREAM, SOL_SOCKET, SO_LINGER, SO_REUSEADDR,
};
use crate::sys::windows::net::{init, new_socket, socket_addr};
pub(crate) type TcpSocket = SOCKET;
pub(crate) fn new_v4_socket() -> io::Result<TcpSocket> {
init();
new_socket(PF_INET, SOCK_STREAM)
}
pub(crate) fn new_v6_socket() -> io::Result<TcpSocket> {
init();
new_socket(PF_INET6, SOCK_STREAM)
}
pub(crate) fn bind(socket: TcpSocket, addr: SocketAddr) -> io::Result<()> {
use winsock2::bind;
let (raw_addr, raw_addr_length) = socket_addr(&addr);
syscall!(
bind(socket, raw_addr, raw_addr_length),
PartialEq::eq,
SOCKET_ERROR
)?;
Ok(())
}
pub(crate) fn connect(socket: TcpSocket, addr: SocketAddr) -> io::Result<net::TcpStream> {
use winsock2::connect;
let (raw_addr, raw_addr_length) = socket_addr(&addr);
let res = syscall!(
connect(socket, raw_addr, raw_addr_length),
PartialEq::eq,
SOCKET_ERROR
);
match res {
Err(err) if err.kind() != io::ErrorKind::WouldBlock => {
Err(err)
}
_ => {
Ok(unsafe { net::TcpStream::from_raw_socket(socket as StdSocket) })
}
}
}
pub(crate) fn listen(socket: TcpSocket, backlog: u32) -> io::Result<net::TcpListener> {
use winsock2::listen;
use std::convert::TryInto;
let backlog = backlog.try_into().unwrap_or(i32::max_value());
syscall!(listen(socket, backlog), PartialEq::eq, SOCKET_ERROR)?;
Ok(unsafe { net::TcpListener::from_raw_socket(socket as StdSocket) })
}
pub(crate) fn close(socket: TcpSocket) {
let _ = unsafe { closesocket(socket) };
}
pub(crate) fn set_reuseaddr(socket: TcpSocket, reuseaddr: bool) -> io::Result<()> {
let val: BOOL = if reuseaddr { TRUE } else { FALSE };
match unsafe { setsockopt(
socket,
SOL_SOCKET,
SO_REUSEADDR,
&val as *const _ as *const c_char,
size_of::<BOOL>() as c_int,
) } {
SOCKET_ERROR => Err(io::Error::last_os_error()),
_ => Ok(()),
}
}
pub(crate) fn get_reuseaddr(socket: TcpSocket) -> io::Result<bool> {
let mut optval: c_char = 0;
let mut optlen = size_of::<BOOL>() as c_int;
match unsafe { getsockopt(
socket,
SOL_SOCKET,
SO_REUSEADDR,
&mut optval as *mut _ as *mut _,
&mut optlen,
) } {
SOCKET_ERROR => Err(io::Error::last_os_error()),
_ => Ok(optval != 0),
}
}
pub(crate) fn get_localaddr(socket: TcpSocket) -> io::Result<SocketAddr> {
let mut addr: SOCKADDR_STORAGE = unsafe { std::mem::zeroed() };
let mut length = std::mem::size_of_val(&addr) as c_int;
match unsafe { getsockname(
socket,
&mut addr as *mut _ as *mut _,
&mut length
) } {
SOCKET_ERROR => Err(io::Error::last_os_error()),
_ => {
let storage: *const SOCKADDR_STORAGE = (&addr) as *const _;
if addr.ss_family as c_int == AF_INET {
let sock_addr : SocketAddrV4 = unsafe { *(storage as *const SOCKADDR_IN as *const _) };
Ok(sock_addr.into())
} else {
let sock_addr : SocketAddrV6 = unsafe { *(storage as *const SOCKADDR_IN6_LH as *const _) };
Ok(sock_addr.into())
}
},
}
}
pub(crate) fn set_linger(socket: TcpSocket, dur: Option<Duration>) -> io::Result<()> {
let val: linger = linger {
l_onoff: if dur.is_some() { 1 } else { 0 },
l_linger: dur.map(|dur| dur.as_secs() as c_ushort).unwrap_or_default(),
};
match unsafe { setsockopt(
socket,
SOL_SOCKET,
SO_LINGER,
&val as *const _ as *const c_char,
size_of::<linger>() as c_int,
) } {
SOCKET_ERROR => Err(io::Error::last_os_error()),
_ => Ok(()),
}
}
pub(crate) fn accept(listener: &net::TcpListener) -> io::Result<(net::TcpStream, SocketAddr)> {
// The non-blocking state of `listener` is inherited. See
// https://docs.microsoft.com/en-us/windows/win32/api/winsock2/nf-winsock2-accept#remarks.
listener.accept()
}