// Copyright 2012 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. /*! * Low-level bindings to the libuv library. * * This module contains a set of direct, 'bare-metal' wrappers around * the libuv C-API. * * We're not bothering yet to redefine uv's structs as Rust structs * because they are quite large and change often between versions. * The maintenance burden is just too high. Instead we use the uv's * `uv_handle_size` and `uv_req_size` to find the correct size of the * structs and allocate them on the heap. This can be revisited later. * * There are also a collection of helper functions to ease interacting * with the low-level API. * * As new functionality, existent in uv.h, is added to the rust stdlib, * the mappings should be added in this module. */ #[allow(non_camel_case_types)]; // C types use std::libc::{size_t, c_int, c_uint, c_void, c_char, uintptr_t}; use std::libc::ssize_t; use std::libc::{malloc, free}; use std::libc; use std::vec; pub use self::errors::*; pub static OK: c_int = 0; pub static EOF: c_int = -4095; pub static UNKNOWN: c_int = -4094; // uv-errno.h redefines error codes for windows, but not for unix... #[cfg(windows)] pub mod errors { use std::libc::c_int; pub static EACCES: c_int = -4093; pub static ECONNREFUSED: c_int = -4079; pub static ECONNRESET: c_int = -4078; pub static ENOTCONN: c_int = -4054; pub static EPIPE: c_int = -4048; pub static ECONNABORTED: c_int = -4080; } #[cfg(not(windows))] pub mod errors { use std::libc; use std::libc::c_int; pub static EACCES: c_int = -libc::EACCES; pub static ECONNREFUSED: c_int = -libc::ECONNREFUSED; pub static ECONNRESET: c_int = -libc::ECONNRESET; pub static ENOTCONN: c_int = -libc::ENOTCONN; pub static EPIPE: c_int = -libc::EPIPE; pub static ECONNABORTED: c_int = -libc::ECONNABORTED; } pub static PROCESS_SETUID: c_int = 1 << 0; pub static PROCESS_SETGID: c_int = 1 << 1; pub static PROCESS_WINDOWS_VERBATIM_ARGUMENTS: c_int = 1 << 2; pub static PROCESS_DETACHED: c_int = 1 << 3; pub static PROCESS_WINDOWS_HIDE: c_int = 1 << 4; pub static STDIO_IGNORE: c_int = 0x00; pub static STDIO_CREATE_PIPE: c_int = 0x01; pub static STDIO_INHERIT_FD: c_int = 0x02; pub static STDIO_INHERIT_STREAM: c_int = 0x04; pub static STDIO_READABLE_PIPE: c_int = 0x10; pub static STDIO_WRITABLE_PIPE: c_int = 0x20; // see libuv/include/uv-unix.h #[cfg(unix)] pub struct uv_buf_t { base: *u8, len: libc::size_t, } // see libuv/include/uv-win.h #[cfg(windows)] pub struct uv_buf_t { len: u32, base: *u8, } #[repr(C)] pub enum uv_run_mode { RUN_DEFAULT = 0, RUN_ONCE, RUN_NOWAIT, } pub struct uv_process_options_t { exit_cb: uv_exit_cb, file: *libc::c_char, args: **libc::c_char, env: **libc::c_char, cwd: *libc::c_char, flags: libc::c_uint, stdio_count: libc::c_int, stdio: *uv_stdio_container_t, uid: uv_uid_t, gid: uv_gid_t, } // These fields are private because they must be interfaced with through the // functions below. pub struct uv_stdio_container_t { priv flags: libc::c_int, priv stream: *uv_stream_t, } pub type uv_handle_t = c_void; pub type uv_loop_t = c_void; pub type uv_idle_t = c_void; pub type uv_tcp_t = c_void; pub type uv_udp_t = c_void; pub type uv_connect_t = c_void; pub type uv_connection_t = c_void; pub type uv_write_t = c_void; pub type uv_async_t = c_void; pub type uv_timer_t = c_void; pub type uv_stream_t = c_void; pub type uv_fs_t = c_void; pub type uv_udp_send_t = c_void; pub type uv_getaddrinfo_t = c_void; pub type uv_process_t = c_void; pub type uv_pipe_t = c_void; pub type uv_tty_t = c_void; pub type uv_signal_t = c_void; pub struct uv_timespec_t { tv_sec: libc::c_long, tv_nsec: libc::c_long } pub struct uv_stat_t { st_dev: libc::uint64_t, st_mode: libc::uint64_t, st_nlink: libc::uint64_t, st_uid: libc::uint64_t, st_gid: libc::uint64_t, st_rdev: libc::uint64_t, st_ino: libc::uint64_t, st_size: libc::uint64_t, st_blksize: libc::uint64_t, st_blocks: libc::uint64_t, st_flags: libc::uint64_t, st_gen: libc::uint64_t, st_atim: uv_timespec_t, st_mtim: uv_timespec_t, st_ctim: uv_timespec_t, st_birthtim: uv_timespec_t } impl uv_stat_t { pub fn new() -> uv_stat_t { uv_stat_t { st_dev: 0, st_mode: 0, st_nlink: 0, st_uid: 0, st_gid: 0, st_rdev: 0, st_ino: 0, st_size: 0, st_blksize: 0, st_blocks: 0, st_flags: 0, st_gen: 0, st_atim: uv_timespec_t { tv_sec: 0, tv_nsec: 0 }, st_mtim: uv_timespec_t { tv_sec: 0, tv_nsec: 0 }, st_ctim: uv_timespec_t { tv_sec: 0, tv_nsec: 0 }, st_birthtim: uv_timespec_t { tv_sec: 0, tv_nsec: 0 } } } pub fn is_file(&self) -> bool { ((self.st_mode) & libc::S_IFMT as libc::uint64_t) == libc::S_IFREG as libc::uint64_t } pub fn is_dir(&self) -> bool { ((self.st_mode) & libc::S_IFMT as libc::uint64_t) == libc::S_IFDIR as libc::uint64_t } } pub type uv_idle_cb = extern "C" fn(handle: *uv_idle_t, status: c_int); pub type uv_alloc_cb = extern "C" fn(stream: *uv_stream_t, suggested_size: size_t) -> uv_buf_t; pub type uv_read_cb = extern "C" fn(stream: *uv_stream_t, nread: ssize_t, buf: uv_buf_t); pub type uv_udp_send_cb = extern "C" fn(req: *uv_udp_send_t, status: c_int); pub type uv_udp_recv_cb = extern "C" fn(handle: *uv_udp_t, nread: ssize_t, buf: uv_buf_t, addr: *sockaddr, flags: c_uint); pub type uv_close_cb = extern "C" fn(handle: *uv_handle_t); pub type uv_walk_cb = extern "C" fn(handle: *uv_handle_t, arg: *c_void); pub type uv_async_cb = extern "C" fn(handle: *uv_async_t, status: c_int); pub type uv_connect_cb = extern "C" fn(handle: *uv_connect_t, status: c_int); pub type uv_connection_cb = extern "C" fn(handle: *uv_connection_t, status: c_int); pub type uv_timer_cb = extern "C" fn(handle: *uv_timer_t, status: c_int); pub type uv_write_cb = extern "C" fn(handle: *uv_write_t, status: c_int); pub type uv_getaddrinfo_cb = extern "C" fn(req: *uv_getaddrinfo_t, status: c_int, res: *addrinfo); pub type uv_exit_cb = extern "C" fn(handle: *uv_process_t, exit_status: c_int, term_signal: c_int); pub type uv_signal_cb = extern "C" fn(handle: *uv_signal_t, signum: c_int); pub type uv_fs_cb = extern "C" fn(req: *uv_fs_t); pub type sockaddr = c_void; pub type sockaddr_in = c_void; pub type sockaddr_in6 = c_void; pub type sockaddr_storage = c_void; #[cfg(unix)] pub type socklen_t = c_int; // XXX: This is a standard C type. Could probably be defined in libc #[cfg(target_os = "android")] #[cfg(target_os = "linux")] pub struct addrinfo { ai_flags: c_int, ai_family: c_int, ai_socktype: c_int, ai_protocol: c_int, ai_addrlen: socklen_t, ai_addr: *sockaddr, ai_canonname: *char, ai_next: *addrinfo } #[cfg(target_os = "macos")] #[cfg(target_os = "freebsd")] pub struct addrinfo { ai_flags: c_int, ai_family: c_int, ai_socktype: c_int, ai_protocol: c_int, ai_addrlen: socklen_t, ai_canonname: *char, ai_addr: *sockaddr, ai_next: *addrinfo } #[cfg(windows)] pub struct addrinfo { ai_flags: c_int, ai_family: c_int, ai_socktype: c_int, ai_protocol: c_int, ai_addrlen: size_t, ai_canonname: *char, ai_addr: *sockaddr, ai_next: *addrinfo } #[cfg(unix)] pub type uv_uid_t = libc::types::os::arch::posix88::uid_t; #[cfg(unix)] pub type uv_gid_t = libc::types::os::arch::posix88::gid_t; #[cfg(windows)] pub type uv_uid_t = libc::c_uchar; #[cfg(windows)] pub type uv_gid_t = libc::c_uchar; #[repr(C)] #[deriving(Eq)] pub enum uv_handle_type { UV_UNKNOWN_HANDLE, UV_ASYNC, UV_CHECK, UV_FS_EVENT, UV_FS_POLL, UV_HANDLE, UV_IDLE, UV_NAMED_PIPE, UV_POLL, UV_PREPARE, UV_PROCESS, UV_STREAM, UV_TCP, UV_TIMER, UV_TTY, UV_UDP, UV_SIGNAL, UV_FILE, UV_HANDLE_TYPE_MAX } #[repr(C)] #[cfg(unix)] #[deriving(Eq)] pub enum uv_req_type { UV_UNKNOWN_REQ, UV_REQ, UV_CONNECT, UV_WRITE, UV_SHUTDOWN, UV_UDP_SEND, UV_FS, UV_WORK, UV_GETADDRINFO, UV_REQ_TYPE_MAX } // uv_req_type may have additional fields defined by UV_REQ_TYPE_PRIVATE. // See UV_REQ_TYPE_PRIVATE at libuv/include/uv-win.h #[repr(C)] #[cfg(windows)] #[deriving(Eq)] pub enum uv_req_type { UV_UNKNOWN_REQ, UV_REQ, UV_CONNECT, UV_WRITE, UV_SHUTDOWN, UV_UDP_SEND, UV_FS, UV_WORK, UV_GETADDRINFO, UV_ACCEPT, UV_FS_EVENT_REQ, UV_POLL_REQ, UV_PROCESS_EXIT, UV_READ, UV_UDP_RECV, UV_WAKEUP, UV_SIGNAL_REQ, UV_REQ_TYPE_MAX } #[repr(C)] #[deriving(Eq)] pub enum uv_membership { UV_LEAVE_GROUP, UV_JOIN_GROUP } pub unsafe fn malloc_handle(handle: uv_handle_type) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; assert!(handle != UV_UNKNOWN_HANDLE && handle != UV_HANDLE_TYPE_MAX); let size = uv_handle_size(handle); let p = malloc(size); assert!(p.is_not_null()); return p; } pub unsafe fn free_handle(v: *c_void) { #[fixed_stack_segment]; #[inline(never)]; free(v) } pub unsafe fn malloc_req(req: uv_req_type) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; assert!(req != UV_UNKNOWN_REQ && req != UV_REQ_TYPE_MAX); let size = uv_req_size(req); let p = malloc(size); assert!(p.is_not_null()); return p; } pub unsafe fn free_req(v: *c_void) { #[fixed_stack_segment]; #[inline(never)]; free(v) } #[test] fn handle_sanity_check() { #[fixed_stack_segment]; #[inline(never)]; unsafe { assert_eq!(UV_HANDLE_TYPE_MAX as uint, rust_uv_handle_type_max()); } } #[test] fn request_sanity_check() { #[fixed_stack_segment]; #[inline(never)]; unsafe { assert_eq!(UV_REQ_TYPE_MAX as uint, rust_uv_req_type_max()); } } // XXX Event loops ignore SIGPIPE by default. pub unsafe fn loop_new() -> *c_void { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_loop_new(); } pub unsafe fn udp_bind(server: *uv_udp_t, addr: *sockaddr_in, flags: c_uint) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_udp_bind(server, addr, flags); } pub unsafe fn udp_bind6(server: *uv_udp_t, addr: *sockaddr_in6, flags: c_uint) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_udp_bind6(server, addr, flags); } pub unsafe fn udp_send(req: *uv_udp_send_t, handle: *T, buf_in: &[uv_buf_t], addr: *sockaddr_in, cb: uv_udp_send_cb) -> c_int { #[fixed_stack_segment]; #[inline(never)]; let buf_ptr = vec::raw::to_ptr(buf_in); let buf_cnt = buf_in.len() as i32; return rust_uv_udp_send(req, handle as *c_void, buf_ptr, buf_cnt, addr, cb); } pub unsafe fn udp_send6(req: *uv_udp_send_t, handle: *T, buf_in: &[uv_buf_t], addr: *sockaddr_in6, cb: uv_udp_send_cb) -> c_int { #[fixed_stack_segment]; #[inline(never)]; let buf_ptr = vec::raw::to_ptr(buf_in); let buf_cnt = buf_in.len() as i32; return rust_uv_udp_send6(req, handle as *c_void, buf_ptr, buf_cnt, addr, cb); } pub unsafe fn get_udp_handle_from_send_req(send_req: *uv_udp_send_t) -> *uv_udp_t { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_udp_handle_from_send_req(send_req); } pub unsafe fn udp_getsockname(handle: *uv_udp_t, name: *sockaddr_storage) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_udp_getsockname(handle, name); } pub unsafe fn tcp_connect(connect_ptr: *uv_connect_t, tcp_handle_ptr: *uv_tcp_t, addr_ptr: *sockaddr_in, after_connect_cb: uv_connect_cb) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_connect(connect_ptr, tcp_handle_ptr, after_connect_cb, addr_ptr); } pub unsafe fn tcp_connect6(connect_ptr: *uv_connect_t, tcp_handle_ptr: *uv_tcp_t, addr_ptr: *sockaddr_in6, after_connect_cb: uv_connect_cb) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_connect6(connect_ptr, tcp_handle_ptr, after_connect_cb, addr_ptr); } pub unsafe fn tcp_bind(tcp_server_ptr: *uv_tcp_t, addr_ptr: *sockaddr_in) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_bind(tcp_server_ptr, addr_ptr); } pub unsafe fn tcp_bind6(tcp_server_ptr: *uv_tcp_t, addr_ptr: *sockaddr_in6) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_bind6(tcp_server_ptr, addr_ptr); } pub unsafe fn tcp_getpeername(tcp_handle_ptr: *uv_tcp_t, name: *sockaddr_storage) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_getpeername(tcp_handle_ptr, name); } pub unsafe fn tcp_getsockname(handle: *uv_tcp_t, name: *sockaddr_storage) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_tcp_getsockname(handle, name); } pub unsafe fn uv_write(req: *uv_write_t, stream: *uv_stream_t, buf_in: &[uv_buf_t], cb: uv_write_cb) -> c_int { externfn!(fn uv_write(req: *uv_write_t, stream: *uv_stream_t, buf_in: *uv_buf_t, buf_cnt: c_int, cb: uv_write_cb) -> c_int) let buf_ptr = vec::raw::to_ptr(buf_in); let buf_cnt = buf_in.len() as i32; return uv_write(req, stream, buf_ptr, buf_cnt, cb); } pub unsafe fn is_ip4_addr(addr: *sockaddr) -> bool { #[fixed_stack_segment]; #[inline(never)]; match rust_uv_is_ipv4_sockaddr(addr) { 0 => false, _ => true } } pub unsafe fn is_ip6_addr(addr: *sockaddr) -> bool { #[fixed_stack_segment]; #[inline(never)]; match rust_uv_is_ipv6_sockaddr(addr) { 0 => false, _ => true } } pub unsafe fn malloc_ip4_addr(ip: &str, port: int) -> *sockaddr_in { #[fixed_stack_segment]; #[inline(never)]; do ip.with_c_str |ip_buf| { rust_uv_ip4_addrp(ip_buf as *u8, port as libc::c_int) } } pub unsafe fn malloc_ip6_addr(ip: &str, port: int) -> *sockaddr_in6 { #[fixed_stack_segment]; #[inline(never)]; do ip.with_c_str |ip_buf| { rust_uv_ip6_addrp(ip_buf as *u8, port as libc::c_int) } } pub unsafe fn malloc_sockaddr_storage() -> *sockaddr_storage { #[fixed_stack_segment]; #[inline(never)]; rust_uv_malloc_sockaddr_storage() } pub unsafe fn free_sockaddr_storage(ss: *sockaddr_storage) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_free_sockaddr_storage(ss); } pub unsafe fn free_ip4_addr(addr: *sockaddr_in) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_free_ip4_addr(addr); } pub unsafe fn free_ip6_addr(addr: *sockaddr_in6) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_free_ip6_addr(addr); } pub unsafe fn ip4_port(addr: *sockaddr_in) -> c_uint { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_ip4_port(addr); } pub unsafe fn ip6_port(addr: *sockaddr_in6) -> c_uint { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_ip6_port(addr); } pub unsafe fn process_pid(p: *uv_process_t) -> c_int { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_process_pid(p); } pub unsafe fn set_stdio_container_flags(c: *uv_stdio_container_t, flags: libc::c_int) { #[fixed_stack_segment]; #[inline(never)]; rust_set_stdio_container_flags(c, flags); } pub unsafe fn set_stdio_container_fd(c: *uv_stdio_container_t, fd: libc::c_int) { #[fixed_stack_segment]; #[inline(never)]; rust_set_stdio_container_fd(c, fd); } pub unsafe fn set_stdio_container_stream(c: *uv_stdio_container_t, stream: *uv_stream_t) { #[fixed_stack_segment]; #[inline(never)]; rust_set_stdio_container_stream(c, stream); } // data access helpers pub unsafe fn get_result_from_fs_req(req: *uv_fs_t) -> c_int { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_result_from_fs_req(req) } pub unsafe fn get_ptr_from_fs_req(req: *uv_fs_t) -> *libc::c_void { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_ptr_from_fs_req(req) } pub unsafe fn get_path_from_fs_req(req: *uv_fs_t) -> *c_char { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_path_from_fs_req(req) } pub unsafe fn get_loop_from_fs_req(req: *uv_fs_t) -> *uv_loop_t { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_loop_from_fs_req(req) } pub unsafe fn get_loop_from_getaddrinfo_req(req: *uv_getaddrinfo_t) -> *uv_loop_t { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_loop_from_getaddrinfo_req(req) } pub unsafe fn get_loop_for_uv_handle(handle: *T) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_loop_for_uv_handle(handle as *c_void); } pub unsafe fn get_stream_handle_from_connect_req(connect: *uv_connect_t) -> *uv_stream_t { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_stream_handle_from_connect_req(connect); } pub unsafe fn get_stream_handle_from_write_req(write_req: *uv_write_t) -> *uv_stream_t { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_stream_handle_from_write_req(write_req); } pub unsafe fn get_data_for_uv_loop(loop_ptr: *c_void) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; rust_uv_get_data_for_uv_loop(loop_ptr) } pub unsafe fn set_data_for_uv_loop(loop_ptr: *c_void, data: *c_void) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_set_data_for_uv_loop(loop_ptr, data); } pub unsafe fn get_data_for_uv_handle(handle: *T) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_data_for_uv_handle(handle as *c_void); } pub unsafe fn set_data_for_uv_handle(handle: *T, data: *U) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_set_data_for_uv_handle(handle as *c_void, data as *c_void); } pub unsafe fn get_data_for_req(req: *T) -> *c_void { #[fixed_stack_segment]; #[inline(never)]; return rust_uv_get_data_for_req(req as *c_void); } pub unsafe fn set_data_for_req(req: *T, data: *U) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_set_data_for_req(req as *c_void, data as *c_void); } pub unsafe fn populate_stat(req_in: *uv_fs_t, stat_out: *uv_stat_t) { #[fixed_stack_segment]; #[inline(never)]; rust_uv_populate_uv_stat(req_in, stat_out) } // uv_support is the result of compiling rust_uv.cpp #[link_args = "-luv_support -luv"] extern { fn rust_uv_loop_new() -> *c_void; fn rust_uv_handle_type_max() -> uintptr_t; fn rust_uv_req_type_max() -> uintptr_t; fn rust_uv_ip4_addrp(ip: *u8, port: c_int) -> *sockaddr_in; fn rust_uv_ip6_addrp(ip: *u8, port: c_int) -> *sockaddr_in6; fn rust_uv_free_ip4_addr(addr: *sockaddr_in); fn rust_uv_free_ip6_addr(addr: *sockaddr_in6); fn rust_uv_ip4_port(src: *sockaddr_in) -> c_uint; fn rust_uv_ip6_port(src: *sockaddr_in6) -> c_uint; fn rust_uv_tcp_connect(req: *uv_connect_t, handle: *uv_tcp_t, cb: uv_connect_cb, addr: *sockaddr_in) -> c_int; fn rust_uv_tcp_bind(tcp_server: *uv_tcp_t, addr: *sockaddr_in) -> c_int; fn rust_uv_tcp_connect6(req: *uv_connect_t, handle: *uv_tcp_t, cb: uv_connect_cb, addr: *sockaddr_in6) -> c_int; fn rust_uv_tcp_bind6(tcp_server: *uv_tcp_t, addr: *sockaddr_in6) -> c_int; fn rust_uv_tcp_getpeername(tcp_handle_ptr: *uv_tcp_t, name: *sockaddr_storage) -> c_int; fn rust_uv_tcp_getsockname(handle: *uv_tcp_t, name: *sockaddr_storage) -> c_int; fn rust_uv_udp_bind(server: *uv_udp_t, addr: *sockaddr_in, flags: c_uint) -> c_int; fn rust_uv_udp_bind6(server: *uv_udp_t, addr: *sockaddr_in6, flags: c_uint) -> c_int; fn rust_uv_udp_send(req: *uv_udp_send_t, handle: *uv_udp_t, buf_in: *uv_buf_t, buf_cnt: c_int, addr: *sockaddr_in, cb: uv_udp_send_cb) -> c_int; fn rust_uv_udp_send6(req: *uv_udp_send_t, handle: *uv_udp_t, buf_in: *uv_buf_t, buf_cnt: c_int, addr: *sockaddr_in6, cb: uv_udp_send_cb) -> c_int; fn rust_uv_get_udp_handle_from_send_req(req: *uv_udp_send_t) -> *uv_udp_t; fn rust_uv_udp_getsockname(handle: *uv_udp_t, name: *sockaddr_storage) -> c_int; fn rust_uv_is_ipv4_sockaddr(addr: *sockaddr) -> c_int; fn rust_uv_is_ipv6_sockaddr(addr: *sockaddr) -> c_int; fn rust_uv_malloc_sockaddr_storage() -> *sockaddr_storage; fn rust_uv_free_sockaddr_storage(ss: *sockaddr_storage); fn rust_uv_populate_uv_stat(req_in: *uv_fs_t, stat_out: *uv_stat_t); fn rust_uv_get_result_from_fs_req(req: *uv_fs_t) -> c_int; fn rust_uv_get_ptr_from_fs_req(req: *uv_fs_t) -> *libc::c_void; fn rust_uv_get_path_from_fs_req(req: *uv_fs_t) -> *c_char; fn rust_uv_get_loop_from_fs_req(req: *uv_fs_t) -> *uv_loop_t; fn rust_uv_get_loop_from_getaddrinfo_req(req: *uv_fs_t) -> *uv_loop_t; fn rust_uv_get_stream_handle_from_connect_req(req: *uv_connect_t) -> *uv_stream_t; fn rust_uv_get_stream_handle_from_write_req(req: *uv_write_t) -> *uv_stream_t; fn rust_uv_get_loop_for_uv_handle(handle: *c_void) -> *c_void; fn rust_uv_get_data_for_uv_loop(loop_ptr: *c_void) -> *c_void; fn rust_uv_set_data_for_uv_loop(loop_ptr: *c_void, data: *c_void); fn rust_uv_get_data_for_uv_handle(handle: *c_void) -> *c_void; fn rust_uv_set_data_for_uv_handle(handle: *c_void, data: *c_void); fn rust_uv_get_data_for_req(req: *c_void) -> *c_void; fn rust_uv_set_data_for_req(req: *c_void, data: *c_void); fn rust_set_stdio_container_flags(c: *uv_stdio_container_t, flags: c_int); fn rust_set_stdio_container_fd(c: *uv_stdio_container_t, fd: c_int); fn rust_set_stdio_container_stream(c: *uv_stdio_container_t, stream: *uv_stream_t); } // generic uv functions externfn!(fn uv_loop_delete(l: *uv_loop_t)) externfn!(fn uv_handle_size(ty: uv_handle_type) -> size_t) externfn!(fn uv_req_size(ty: uv_req_type) -> size_t) externfn!(fn uv_run(l: *uv_loop_t, mode: uv_run_mode) -> c_int) externfn!(fn uv_close(h: *uv_handle_t, cb: uv_close_cb)) externfn!(fn uv_walk(l: *uv_loop_t, cb: uv_walk_cb, arg: *c_void)) externfn!(fn uv_buf_init(base: *c_char, len: c_uint) -> uv_buf_t) externfn!(fn uv_strerror(err: c_int) -> *c_char) externfn!(fn uv_err_name(err: c_int) -> *c_char) externfn!(fn uv_listen(s: *uv_stream_t, backlog: c_int, cb: uv_connection_cb) -> c_int) externfn!(fn uv_accept(server: *uv_stream_t, client: *uv_stream_t) -> c_int) externfn!(fn uv_read_start(stream: *uv_stream_t, on_alloc: uv_alloc_cb, on_read: uv_read_cb) -> c_int) externfn!(fn uv_read_stop(stream: *uv_stream_t) -> c_int) // idle bindings externfn!(fn uv_idle_init(l: *uv_loop_t, i: *uv_idle_t) -> c_int) externfn!(fn uv_idle_start(i: *uv_idle_t, cb: uv_idle_cb) -> c_int) externfn!(fn uv_idle_stop(i: *uv_idle_t) -> c_int) // async bindings externfn!(fn uv_async_init(l: *uv_loop_t, a: *uv_async_t, cb: uv_async_cb) -> c_int) externfn!(fn uv_async_send(a: *uv_async_t)) // tcp bindings externfn!(fn uv_tcp_init(l: *uv_loop_t, h: *uv_tcp_t) -> c_int) externfn!(fn uv_ip4_name(src: *sockaddr_in, dst: *c_char, size: size_t) -> c_int) externfn!(fn uv_ip6_name(src: *sockaddr_in6, dst: *c_char, size: size_t) -> c_int) externfn!(fn uv_tcp_nodelay(h: *uv_tcp_t, enable: c_int) -> c_int) externfn!(fn uv_tcp_keepalive(h: *uv_tcp_t, enable: c_int, delay: c_uint) -> c_int) externfn!(fn uv_tcp_simultaneous_accepts(h: *uv_tcp_t, enable: c_int) -> c_int) // udp bindings externfn!(fn uv_udp_init(l: *uv_loop_t, h: *uv_udp_t) -> c_int) externfn!(fn uv_udp_recv_start(server: *uv_udp_t, on_alloc: uv_alloc_cb, on_recv: uv_udp_recv_cb) -> c_int) externfn!(fn uv_udp_set_membership(handle: *uv_udp_t, multicast_addr: *c_char, interface_addr: *c_char, membership: uv_membership) -> c_int) externfn!(fn uv_udp_recv_stop(server: *uv_udp_t) -> c_int) externfn!(fn uv_udp_set_multicast_loop(handle: *uv_udp_t, on: c_int) -> c_int) externfn!(fn uv_udp_set_multicast_ttl(handle: *uv_udp_t, ttl: c_int) -> c_int) externfn!(fn uv_udp_set_ttl(handle: *uv_udp_t, ttl: c_int) -> c_int) externfn!(fn uv_udp_set_broadcast(handle: *uv_udp_t, on: c_int) -> c_int) // timer bindings externfn!(fn uv_timer_init(l: *uv_loop_t, t: *uv_timer_t) -> c_int) externfn!(fn uv_timer_start(t: *uv_timer_t, cb: uv_timer_cb, timeout: libc::uint64_t, repeat: libc::uint64_t) -> c_int) externfn!(fn uv_timer_stop(handle: *uv_timer_t) -> c_int) // fs operations externfn!(fn uv_fs_open(loop_ptr: *uv_loop_t, req: *uv_fs_t, path: *c_char, flags: c_int, mode: c_int, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_unlink(loop_ptr: *uv_loop_t, req: *uv_fs_t, path: *c_char, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_write(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, buf: *c_void, len: c_uint, offset: i64, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_read(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, buf: *c_void, len: c_uint, offset: i64, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_close(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_stat(l: *uv_loop_t, req: *uv_fs_t, path: *c_char, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_fstat(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_mkdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char, mode: c_int, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_rmdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_readdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char, flags: c_int, cb: uv_fs_cb) -> c_int) externfn!(fn uv_fs_req_cleanup(req: *uv_fs_t)) externfn!(fn uv_fs_fsync(handle: *uv_loop_t, req: *uv_fs_t, file: c_int, cb: *u8) -> c_int) externfn!(fn uv_fs_fdatasync(handle: *uv_loop_t, req: *uv_fs_t, file: c_int, cb: *u8) -> c_int) externfn!(fn uv_fs_ftruncate(handle: *uv_loop_t, req: *uv_fs_t, file: c_int, offset: i64, cb: *u8) -> c_int) externfn!(fn uv_fs_readlink(handle: *uv_loop_t, req: *uv_fs_t, file: *c_char, cb: *u8) -> c_int) externfn!(fn uv_fs_symlink(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char, dst: *c_char, flags: c_int, cb: *u8) -> c_int) externfn!(fn uv_fs_link(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char, dst: *c_char, cb: *u8) -> c_int) externfn!(fn uv_fs_chown(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char, uid: uv_uid_t, gid: uv_gid_t, cb: *u8) -> c_int) externfn!(fn uv_fs_lstat(handle: *uv_loop_t, req: *uv_fs_t, file: *c_char, cb: *u8) -> c_int) // getaddrinfo externfn!(fn uv_getaddrinfo(loop_: *uv_loop_t, req: *uv_getaddrinfo_t, getaddrinfo_cb: uv_getaddrinfo_cb, node: *c_char, service: *c_char, hints: *addrinfo) -> c_int) externfn!(fn uv_freeaddrinfo(ai: *addrinfo)) // process spawning externfn!(fn uv_spawn(loop_ptr: *uv_loop_t, outptr: *uv_process_t, options: uv_process_options_t) -> c_int) externfn!(fn uv_process_kill(p: *uv_process_t, signum: c_int) -> c_int) // pipes externfn!(fn uv_pipe_init(l: *uv_loop_t, p: *uv_pipe_t, ipc: c_int) -> c_int) externfn!(fn uv_pipe_open(pipe: *uv_pipe_t, file: c_int) -> c_int) externfn!(fn uv_pipe_bind(pipe: *uv_pipe_t, name: *c_char) -> c_int) externfn!(fn uv_pipe_connect(req: *uv_connect_t, handle: *uv_pipe_t, name: *c_char, cb: uv_connect_cb)) // tty externfn!(fn uv_tty_init(l: *uv_loop_t, tty: *uv_tty_t, fd: c_int, readable: c_int) -> c_int) externfn!(fn uv_tty_set_mode(tty: *uv_tty_t, mode: c_int) -> c_int) externfn!(fn uv_tty_get_winsize(tty: *uv_tty_t, width: *c_int, height: *c_int) -> c_int) externfn!(fn uv_guess_handle(fd: c_int) -> uv_handle_type) // signals externfn!(fn uv_signal_init(loop_: *uv_loop_t, handle: *uv_signal_t) -> c_int) externfn!(fn uv_signal_start(h: *uv_signal_t, cb: uv_signal_cb, signum: c_int) -> c_int) externfn!(fn uv_signal_stop(handle: *uv_signal_t) -> c_int) // libuv requires various system libraries to successfully link on some // platforms #[cfg(target_os = "linux")] #[link_args = "-lpthread"] extern {} #[cfg(target_os = "win32")] #[link_args = "-lWs2_32 -lpsapi -liphlpapi"] extern {}