2013-10-22 17:13:18 -05:00
|
|
|
// 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 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, 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
|
|
|
|
|
2013-11-05 17:48:27 -06:00
|
|
|
use std::libc::{size_t, c_int, c_uint, c_void, c_char, uintptr_t, c_double};
|
2013-10-22 17:13:18 -05:00
|
|
|
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;
|
|
|
|
|
2013-11-09 13:02:16 -06:00
|
|
|
pub static EACCES: c_int = -4092;
|
|
|
|
pub static ECONNREFUSED: c_int = -4078;
|
|
|
|
pub static ECONNRESET: c_int = -4077;
|
|
|
|
pub static ENOTCONN: c_int = -4053;
|
|
|
|
pub static EPIPE: c_int = -4047;
|
|
|
|
pub static ECONNABORTED: c_int = -4079;
|
|
|
|
pub static ECANCELED: c_int = -4081;
|
|
|
|
pub static EBADF: c_int = -4083;
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
#[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;
|
2013-11-05 02:27:41 -06:00
|
|
|
pub static ECANCELED : c_int = -libc::ECANCELED;
|
2013-11-07 17:24:30 -06:00
|
|
|
pub static EBADF : c_int = -libc::EBADF;
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
2013-11-01 11:36:21 -05:00
|
|
|
#[cfg(unix)]
|
|
|
|
pub type uv_buf_len_t = libc::size_t;
|
|
|
|
#[cfg(windows)]
|
2013-11-09 13:02:16 -06:00
|
|
|
pub type uv_buf_len_t = libc::c_ulong;
|
2013-11-01 11:36:21 -05:00
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
// see libuv/include/uv-unix.h
|
|
|
|
#[cfg(unix)]
|
|
|
|
pub struct uv_buf_t {
|
|
|
|
base: *u8,
|
2013-11-09 13:02:16 -06:00
|
|
|
len: uv_buf_len_t,
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// see libuv/include/uv-win.h
|
|
|
|
#[cfg(windows)]
|
|
|
|
pub struct uv_buf_t {
|
2013-11-09 13:02:16 -06:00
|
|
|
len: uv_buf_len_t,
|
2013-10-22 17:13:18 -05:00
|
|
|
base: *u8,
|
|
|
|
}
|
|
|
|
|
2013-11-03 12:39:39 -06:00
|
|
|
#[repr(C)]
|
|
|
|
pub enum uv_run_mode {
|
|
|
|
RUN_DEFAULT = 0,
|
|
|
|
RUN_ONCE,
|
|
|
|
RUN_NOWAIT,
|
|
|
|
}
|
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
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;
|
2013-11-05 02:27:41 -06:00
|
|
|
pub type uv_req_t = c_void;
|
2013-10-22 17:13:18 -05:00
|
|
|
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,
|
2013-11-07 17:26:47 -06:00
|
|
|
suggested_size: size_t,
|
|
|
|
buf: *mut uv_buf_t);
|
2013-10-22 17:13:18 -05:00
|
|
|
pub type uv_read_cb = extern "C" fn(stream: *uv_stream_t,
|
|
|
|
nread: ssize_t,
|
2013-11-07 17:26:47 -06:00
|
|
|
buf: *uv_buf_t);
|
2013-10-22 17:13:18 -05:00
|
|
|
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,
|
2013-11-07 17:26:47 -06:00
|
|
|
buf: *uv_buf_t,
|
2013-10-22 17:13:18 -05:00
|
|
|
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,
|
2013-11-07 17:26:47 -06:00
|
|
|
exit_status: i64,
|
2013-10-22 17:13:18 -05:00
|
|
|
term_signal: c_int);
|
|
|
|
pub type uv_signal_cb = extern "C" fn(handle: *uv_signal_t,
|
|
|
|
signum: c_int);
|
2013-10-30 01:31:07 -05:00
|
|
|
pub type uv_fs_cb = extern "C" fn(req: *uv_fs_t);
|
2013-10-22 17:13:18 -05:00
|
|
|
|
|
|
|
pub type sockaddr = 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;
|
|
|
|
|
2013-11-03 12:39:39 -06:00
|
|
|
#[repr(C)]
|
2013-10-22 17:13:18 -05:00
|
|
|
#[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
|
|
|
|
}
|
|
|
|
|
2013-11-03 12:39:39 -06:00
|
|
|
#[repr(C)]
|
2013-10-22 17:13:18 -05:00
|
|
|
#[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
|
2013-11-03 12:39:39 -06:00
|
|
|
#[repr(C)]
|
2013-10-22 17:13:18 -05:00
|
|
|
#[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
|
|
|
|
}
|
|
|
|
|
2013-11-03 12:39:39 -06:00
|
|
|
#[repr(C)]
|
2013-10-22 17:13:18 -05:00
|
|
|
#[deriving(Eq)]
|
|
|
|
pub enum uv_membership {
|
|
|
|
UV_LEAVE_GROUP,
|
|
|
|
UV_JOIN_GROUP
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn malloc_handle(handle: uv_handle_type) -> *c_void {
|
|
|
|
assert!(handle != UV_UNKNOWN_HANDLE && handle != UV_HANDLE_TYPE_MAX);
|
2013-11-03 12:39:39 -06:00
|
|
|
let size = uv_handle_size(handle);
|
2013-10-22 17:13:18 -05:00
|
|
|
let p = malloc(size);
|
|
|
|
assert!(p.is_not_null());
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn free_handle(v: *c_void) {
|
|
|
|
free(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn malloc_req(req: uv_req_type) -> *c_void {
|
|
|
|
assert!(req != UV_UNKNOWN_REQ && req != UV_REQ_TYPE_MAX);
|
2013-11-03 12:39:39 -06:00
|
|
|
let size = uv_req_size(req);
|
2013-10-22 17:13:18 -05:00
|
|
|
let p = malloc(size);
|
|
|
|
assert!(p.is_not_null());
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn free_req(v: *c_void) {
|
|
|
|
free(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn handle_sanity_check() {
|
|
|
|
unsafe {
|
|
|
|
assert_eq!(UV_HANDLE_TYPE_MAX as uint, rust_uv_handle_type_max());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn request_sanity_check() {
|
|
|
|
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 {
|
|
|
|
return rust_uv_loop_new();
|
|
|
|
}
|
|
|
|
|
2013-11-03 12:39:39 -06:00
|
|
|
pub unsafe fn uv_write(req: *uv_write_t,
|
|
|
|
stream: *uv_stream_t,
|
2013-10-22 17:13:18 -05:00
|
|
|
buf_in: &[uv_buf_t],
|
|
|
|
cb: uv_write_cb) -> c_int {
|
2013-11-06 17:16:04 -06:00
|
|
|
extern {
|
|
|
|
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;
|
|
|
|
}
|
2013-10-22 17:13:18 -05:00
|
|
|
|
|
|
|
let buf_ptr = vec::raw::to_ptr(buf_in);
|
|
|
|
let buf_cnt = buf_in.len() as i32;
|
2013-11-03 12:39:39 -06:00
|
|
|
return uv_write(req, stream, buf_ptr, buf_cnt, cb);
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
|
2013-11-06 17:16:04 -06:00
|
|
|
pub unsafe fn uv_udp_send(req: *uv_udp_send_t,
|
|
|
|
handle: *uv_udp_t,
|
|
|
|
buf_in: &[uv_buf_t],
|
|
|
|
addr: *sockaddr,
|
|
|
|
cb: uv_udp_send_cb) -> c_int {
|
|
|
|
extern {
|
|
|
|
fn uv_udp_send(req: *uv_write_t, stream: *uv_stream_t,
|
|
|
|
buf_in: *uv_buf_t, buf_cnt: c_int, addr: *sockaddr,
|
|
|
|
cb: uv_udp_send_cb) -> c_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
let buf_ptr = vec::raw::to_ptr(buf_in);
|
|
|
|
let buf_cnt = buf_in.len() as i32;
|
|
|
|
return uv_udp_send(req, handle, buf_ptr, buf_cnt, addr, cb);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn get_udp_handle_from_send_req(send_req: *uv_udp_send_t) -> *uv_udp_t {
|
|
|
|
return rust_uv_get_udp_handle_from_send_req(send_req);
|
|
|
|
}
|
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
pub unsafe fn process_pid(p: *uv_process_t) -> c_int {
|
2013-11-06 17:16:04 -06:00
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
return rust_uv_process_pid(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn set_stdio_container_flags(c: *uv_stdio_container_t,
|
|
|
|
flags: libc::c_int) {
|
2013-11-06 17:16:04 -06:00
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
rust_set_stdio_container_flags(c, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn set_stdio_container_fd(c: *uv_stdio_container_t,
|
|
|
|
fd: libc::c_int) {
|
2013-11-06 17:16:04 -06:00
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
rust_set_stdio_container_fd(c, fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn set_stdio_container_stream(c: *uv_stdio_container_t,
|
|
|
|
stream: *uv_stream_t) {
|
|
|
|
rust_set_stdio_container_stream(c, stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
// data access helpers
|
|
|
|
pub unsafe fn get_result_from_fs_req(req: *uv_fs_t) -> c_int {
|
|
|
|
rust_uv_get_result_from_fs_req(req)
|
|
|
|
}
|
|
|
|
pub unsafe fn get_ptr_from_fs_req(req: *uv_fs_t) -> *libc::c_void {
|
|
|
|
rust_uv_get_ptr_from_fs_req(req)
|
|
|
|
}
|
2013-10-30 01:31:07 -05:00
|
|
|
pub unsafe fn get_path_from_fs_req(req: *uv_fs_t) -> *c_char {
|
|
|
|
rust_uv_get_path_from_fs_req(req)
|
|
|
|
}
|
2013-10-22 17:13:18 -05:00
|
|
|
pub unsafe fn get_loop_from_fs_req(req: *uv_fs_t) -> *uv_loop_t {
|
|
|
|
rust_uv_get_loop_from_fs_req(req)
|
|
|
|
}
|
|
|
|
pub unsafe fn get_loop_from_getaddrinfo_req(req: *uv_getaddrinfo_t) -> *uv_loop_t {
|
|
|
|
rust_uv_get_loop_from_getaddrinfo_req(req)
|
|
|
|
}
|
|
|
|
pub unsafe fn get_loop_for_uv_handle<T>(handle: *T) -> *c_void {
|
|
|
|
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 {
|
|
|
|
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 {
|
|
|
|
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 {
|
|
|
|
rust_uv_get_data_for_uv_loop(loop_ptr)
|
|
|
|
}
|
|
|
|
pub unsafe fn set_data_for_uv_loop(loop_ptr: *c_void, data: *c_void) {
|
|
|
|
rust_uv_set_data_for_uv_loop(loop_ptr, data);
|
|
|
|
}
|
|
|
|
pub unsafe fn get_data_for_uv_handle<T>(handle: *T) -> *c_void {
|
|
|
|
return rust_uv_get_data_for_uv_handle(handle as *c_void);
|
|
|
|
}
|
|
|
|
pub unsafe fn set_data_for_uv_handle<T, U>(handle: *T, data: *U) {
|
|
|
|
rust_uv_set_data_for_uv_handle(handle as *c_void, data as *c_void);
|
|
|
|
}
|
|
|
|
pub unsafe fn get_data_for_req<T>(req: *T) -> *c_void {
|
|
|
|
return rust_uv_get_data_for_req(req as *c_void);
|
|
|
|
}
|
|
|
|
pub unsafe fn set_data_for_req<T, U>(req: *T, data: *U) {
|
|
|
|
rust_uv_set_data_for_req(req as *c_void, data as *c_void);
|
|
|
|
}
|
2013-11-03 12:39:39 -06:00
|
|
|
pub unsafe fn populate_stat(req_in: *uv_fs_t, stat_out: *uv_stat_t) {
|
|
|
|
rust_uv_populate_uv_stat(req_in, stat_out)
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
2013-11-05 21:16:48 -06:00
|
|
|
pub unsafe fn guess_handle(handle: c_int) -> c_int {
|
|
|
|
rust_uv_guess_handle(handle)
|
|
|
|
}
|
2013-10-22 17:13:18 -05:00
|
|
|
|
|
|
|
|
2013-10-31 01:57:27 -05:00
|
|
|
// uv_support is the result of compiling rust_uv.cpp
|
2013-11-09 13:02:16 -06:00
|
|
|
//
|
|
|
|
// Note that this is in a cfg'd block so it doesn't get linked during testing.
|
|
|
|
// There's a bit of a conundrum when testing in that we're actually assuming
|
|
|
|
// that the tests are running in a uv loop, but they were created from the
|
|
|
|
// statically linked uv to the original rustuv crate. When we create the test
|
|
|
|
// executable, on some platforms if we re-link against uv, it actually creates
|
|
|
|
// second copies of everything. We obviously don't want this, so instead of
|
|
|
|
// dying horribly during testing, we allow all of the test rustuv's references
|
|
|
|
// to get resolved to the original rustuv crate.
|
2013-10-31 01:57:27 -05:00
|
|
|
#[link_args = "-luv_support -luv"]
|
2013-11-09 13:02:16 -06:00
|
|
|
#[cfg(not(test))]
|
|
|
|
extern {}
|
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
extern {
|
2013-11-03 12:39:39 -06:00
|
|
|
fn rust_uv_loop_new() -> *c_void;
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2013-11-07 17:26:47 -06:00
|
|
|
// dealing with sockaddr things
|
|
|
|
pub fn rust_sockaddr_size() -> c_int;
|
|
|
|
pub fn rust_malloc_ip4_addr(s: *c_char, port: c_int) -> *sockaddr;
|
|
|
|
pub fn rust_malloc_ip6_addr(s: *c_char, port: c_int) -> *sockaddr;
|
|
|
|
pub fn rust_ip4_port(src: *sockaddr) -> c_uint;
|
|
|
|
pub fn rust_ip6_port(src: *sockaddr) -> c_uint;
|
|
|
|
pub fn rust_is_ipv4_sockaddr(addr: *sockaddr) -> c_int;
|
|
|
|
pub fn rust_is_ipv6_sockaddr(addr: *sockaddr) -> c_int;
|
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
fn rust_uv_handle_type_max() -> uintptr_t;
|
|
|
|
fn rust_uv_req_type_max() -> uintptr_t;
|
|
|
|
fn rust_uv_get_udp_handle_from_send_req(req: *uv_udp_send_t) -> *uv_udp_t;
|
2013-11-07 17:26:47 -06:00
|
|
|
|
2013-10-22 17:13:18 -05:00
|
|
|
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;
|
2013-10-30 01:31:07 -05:00
|
|
|
fn rust_uv_get_path_from_fs_req(req: *uv_fs_t) -> *c_char;
|
2013-10-22 17:13:18 -05:00
|
|
|
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;
|
2013-11-03 12:39:39 -06:00
|
|
|
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;
|
2013-10-22 17:13:18 -05:00
|
|
|
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);
|
2013-11-04 16:03:32 -06:00
|
|
|
fn rust_uv_process_pid(p: *uv_process_t) -> c_int;
|
2013-11-05 21:16:48 -06:00
|
|
|
fn rust_uv_guess_handle(fd: c_int) -> c_int;
|
2013-11-07 17:26:47 -06:00
|
|
|
|
2013-11-06 17:16:04 -06:00
|
|
|
// generic uv functions
|
|
|
|
pub fn uv_loop_delete(l: *uv_loop_t);
|
|
|
|
pub fn uv_handle_size(ty: uv_handle_type) -> size_t;
|
|
|
|
pub fn uv_req_size(ty: uv_req_type) -> size_t;
|
|
|
|
pub fn uv_run(l: *uv_loop_t, mode: uv_run_mode) -> c_int;
|
|
|
|
pub fn uv_close(h: *uv_handle_t, cb: uv_close_cb);
|
|
|
|
pub fn uv_walk(l: *uv_loop_t, cb: uv_walk_cb, arg: *c_void);
|
|
|
|
pub fn uv_buf_init(base: *c_char, len: c_uint) -> uv_buf_t;
|
|
|
|
pub fn uv_strerror(err: c_int) -> *c_char;
|
|
|
|
pub fn uv_err_name(err: c_int) -> *c_char;
|
|
|
|
pub fn uv_listen(s: *uv_stream_t, backlog: c_int,
|
|
|
|
cb: uv_connection_cb) -> c_int;
|
|
|
|
pub fn uv_accept(server: *uv_stream_t, client: *uv_stream_t) -> c_int;
|
|
|
|
pub fn uv_read_start(stream: *uv_stream_t,
|
|
|
|
on_alloc: uv_alloc_cb,
|
|
|
|
on_read: uv_read_cb) -> c_int;
|
|
|
|
pub fn uv_read_stop(stream: *uv_stream_t) -> c_int;
|
|
|
|
|
|
|
|
// idle bindings
|
|
|
|
pub fn uv_idle_init(l: *uv_loop_t, i: *uv_idle_t) -> c_int;
|
|
|
|
pub fn uv_idle_start(i: *uv_idle_t, cb: uv_idle_cb) -> c_int;
|
|
|
|
pub fn uv_idle_stop(i: *uv_idle_t) -> c_int;
|
|
|
|
|
|
|
|
// async bindings
|
|
|
|
pub fn uv_async_init(l: *uv_loop_t, a: *uv_async_t,
|
|
|
|
cb: uv_async_cb) -> c_int;
|
|
|
|
pub fn uv_async_send(a: *uv_async_t);
|
|
|
|
|
|
|
|
// tcp bindings
|
|
|
|
pub fn uv_tcp_init(l: *uv_loop_t, h: *uv_tcp_t) -> c_int;
|
|
|
|
pub fn uv_tcp_connect(c: *uv_connect_t, h: *uv_tcp_t,
|
|
|
|
addr: *sockaddr, cb: uv_connect_cb) -> c_int;
|
|
|
|
pub fn uv_tcp_bind(t: *uv_tcp_t, addr: *sockaddr) -> c_int;
|
|
|
|
pub fn uv_ip4_name(src: *sockaddr, dst: *c_char,
|
|
|
|
size: size_t) -> c_int;
|
|
|
|
pub fn uv_ip6_name(src: *sockaddr, dst: *c_char,
|
|
|
|
size: size_t) -> c_int;
|
|
|
|
pub fn uv_tcp_nodelay(h: *uv_tcp_t, enable: c_int) -> c_int;
|
|
|
|
pub fn uv_tcp_keepalive(h: *uv_tcp_t, enable: c_int,
|
|
|
|
delay: c_uint) -> c_int;
|
|
|
|
pub fn uv_tcp_simultaneous_accepts(h: *uv_tcp_t, enable: c_int) -> c_int;
|
|
|
|
pub fn uv_tcp_getsockname(h: *uv_tcp_t, name: *sockaddr,
|
|
|
|
len: *mut c_int) -> c_int;
|
|
|
|
pub fn uv_tcp_getpeername(h: *uv_tcp_t, name: *sockaddr,
|
|
|
|
len: *mut c_int) -> c_int;
|
|
|
|
pub fn uv_ip4_addr(ip: *c_char, port: c_int, addr: *sockaddr) -> c_int;
|
|
|
|
pub fn uv_ip6_addr(ip: *c_char, port: c_int, addr: *sockaddr) -> c_int;
|
|
|
|
|
|
|
|
// udp bindings
|
|
|
|
pub fn uv_udp_init(l: *uv_loop_t, h: *uv_udp_t) -> c_int;
|
|
|
|
pub fn uv_udp_bind(h: *uv_udp_t, addr: *sockaddr, flags: c_uint) -> c_int;
|
|
|
|
pub fn uv_udp_recv_start(server: *uv_udp_t,
|
|
|
|
on_alloc: uv_alloc_cb,
|
|
|
|
on_recv: uv_udp_recv_cb) -> c_int;
|
|
|
|
pub fn uv_udp_set_membership(handle: *uv_udp_t, multicast_addr: *c_char,
|
|
|
|
interface_addr: *c_char,
|
|
|
|
membership: uv_membership) -> c_int;
|
|
|
|
pub fn uv_udp_recv_stop(server: *uv_udp_t) -> c_int;
|
|
|
|
pub fn uv_udp_set_multicast_loop(handle: *uv_udp_t, on: c_int) -> c_int;
|
|
|
|
pub fn uv_udp_set_multicast_ttl(handle: *uv_udp_t, ttl: c_int) -> c_int;
|
|
|
|
pub fn uv_udp_set_ttl(handle: *uv_udp_t, ttl: c_int) -> c_int;
|
|
|
|
pub fn uv_udp_set_broadcast(handle: *uv_udp_t, on: c_int) -> c_int;
|
|
|
|
pub fn uv_udp_getsockname(h: *uv_udp_t, name: *sockaddr,
|
|
|
|
len: *mut c_int) -> c_int;
|
|
|
|
|
|
|
|
// timer bindings
|
|
|
|
pub fn uv_timer_init(l: *uv_loop_t, t: *uv_timer_t) -> c_int;
|
|
|
|
pub fn uv_timer_start(t: *uv_timer_t, cb: uv_timer_cb,
|
|
|
|
timeout: libc::uint64_t,
|
|
|
|
repeat: libc::uint64_t) -> c_int;
|
|
|
|
pub fn uv_timer_stop(handle: *uv_timer_t) -> c_int;
|
|
|
|
|
|
|
|
// fs operations
|
|
|
|
pub 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;
|
|
|
|
pub fn uv_fs_unlink(loop_ptr: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_write(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, buf: *c_void,
|
|
|
|
len: size_t, offset: i64, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_read(l: *uv_loop_t, req: *uv_fs_t, fd: c_int, buf: *c_void,
|
|
|
|
len: size_t, offset: i64, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_close(l: *uv_loop_t, req: *uv_fs_t, fd: c_int,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_stat(l: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_fstat(l: *uv_loop_t, req: *uv_fs_t, fd: c_int,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_mkdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
mode: c_int, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_rmdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_readdir(l: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
flags: c_int, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_req_cleanup(req: *uv_fs_t);
|
|
|
|
pub fn uv_fs_fsync(handle: *uv_loop_t, req: *uv_fs_t, file: c_int,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_fdatasync(handle: *uv_loop_t, req: *uv_fs_t, file: c_int,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_ftruncate(handle: *uv_loop_t, req: *uv_fs_t, file: c_int,
|
|
|
|
offset: i64, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_readlink(handle: *uv_loop_t, req: *uv_fs_t, file: *c_char,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_symlink(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char,
|
|
|
|
dst: *c_char, flags: c_int, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_rename(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char,
|
|
|
|
dst: *c_char, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_utime(handle: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
atime: c_double, mtime: c_double,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_link(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char,
|
|
|
|
dst: *c_char, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_chown(handle: *uv_loop_t, req: *uv_fs_t, src: *c_char,
|
|
|
|
uid: uv_uid_t, gid: uv_gid_t, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_chmod(handle: *uv_loop_t, req: *uv_fs_t, path: *c_char,
|
|
|
|
mode: c_int, cb: uv_fs_cb) -> c_int;
|
|
|
|
pub fn uv_fs_lstat(handle: *uv_loop_t, req: *uv_fs_t, file: *c_char,
|
|
|
|
cb: uv_fs_cb) -> c_int;
|
|
|
|
|
|
|
|
// getaddrinfo
|
|
|
|
pub 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;
|
|
|
|
pub fn uv_freeaddrinfo(ai: *addrinfo);
|
|
|
|
|
|
|
|
// process spawning
|
|
|
|
pub fn uv_spawn(loop_ptr: *uv_loop_t, outptr: *uv_process_t,
|
|
|
|
options: *uv_process_options_t) -> c_int;
|
|
|
|
pub fn uv_process_kill(p: *uv_process_t, signum: c_int) -> c_int;
|
|
|
|
|
|
|
|
// pipes
|
|
|
|
pub fn uv_pipe_init(l: *uv_loop_t, p: *uv_pipe_t, ipc: c_int) -> c_int;
|
|
|
|
pub fn uv_pipe_open(pipe: *uv_pipe_t, file: c_int) -> c_int;
|
|
|
|
pub fn uv_pipe_bind(pipe: *uv_pipe_t, name: *c_char) -> c_int;
|
|
|
|
pub fn uv_pipe_connect(req: *uv_connect_t, handle: *uv_pipe_t,
|
|
|
|
name: *c_char, cb: uv_connect_cb);
|
|
|
|
|
|
|
|
// tty
|
|
|
|
pub fn uv_tty_init(l: *uv_loop_t, tty: *uv_tty_t, fd: c_int,
|
|
|
|
readable: c_int) -> c_int;
|
|
|
|
pub fn uv_tty_set_mode(tty: *uv_tty_t, mode: c_int) -> c_int;
|
|
|
|
pub fn uv_tty_get_winsize(tty: *uv_tty_t, width: *c_int,
|
|
|
|
height: *c_int) -> c_int;
|
|
|
|
|
|
|
|
// signals
|
|
|
|
pub fn uv_signal_init(loop_: *uv_loop_t, handle: *uv_signal_t) -> c_int;
|
|
|
|
pub fn uv_signal_start(h: *uv_signal_t, cb: uv_signal_cb,
|
|
|
|
signum: c_int) -> c_int;
|
|
|
|
pub fn uv_signal_stop(handle: *uv_signal_t) -> c_int;
|
2013-11-07 17:26:47 -06:00
|
|
|
}
|
2013-11-03 12:39:39 -06:00
|
|
|
|
2013-10-31 01:57:27 -05:00
|
|
|
// 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 {}
|