58a51120a7
This update brings a few months of changes, but primarily a fix for the following situation. When creating a handle to stdin, libuv used to set the stdin handle to nonblocking mode. This would end up affect this stdin handle across all processes that shared it, which mean that stdin become nonblocking for everyone using the same stdin. On linux, this also affected *stdout* because stdin/stdout roughly point at the same thing. This problem became apparent when running the test suite manually on a local computer. The stdtest suite (running with libgreen) would set stdout to nonblocking mode (as described above), and then the next test suite would always fail for a printing failure (because stdout was returning EAGAIN). This has been fixed upstream, joyent/libuv@342e8c, and this update pulls in this fix. This also brings us in line with a recently upstreamed libuv patch. Closes #13336 Closes #13355
324 lines
11 KiB
Rust
324 lines
11 KiB
Rust
// Copyright 2013 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.
|
|
|
|
use std::c_str::CString;
|
|
use std::cast;
|
|
use std::io::IoError;
|
|
use std::io::net::ip::SocketAddr;
|
|
use std::io::process::ProcessConfig;
|
|
use std::io::signal::Signum;
|
|
use std::io::{FileMode, FileAccess, Open, Append, Truncate, Read, Write,
|
|
ReadWrite, FileStat};
|
|
use std::io;
|
|
use libc::c_int;
|
|
use libc::{O_CREAT, O_APPEND, O_TRUNC, O_RDWR, O_RDONLY, O_WRONLY, S_IRUSR,
|
|
S_IWUSR};
|
|
use libc;
|
|
use std::path::Path;
|
|
use std::rt::rtio;
|
|
use std::rt::rtio::{IoFactory, EventLoop};
|
|
use ai = std::io::net::addrinfo;
|
|
|
|
#[cfg(test)] use std::unstable::run_in_bare_thread;
|
|
|
|
use super::{uv_error_to_io_error, Loop};
|
|
|
|
use addrinfo::GetAddrInfoRequest;
|
|
use async::AsyncWatcher;
|
|
use file::{FsRequest, FileWatcher};
|
|
use queue::QueuePool;
|
|
use homing::HomeHandle;
|
|
use idle::IdleWatcher;
|
|
use net::{TcpWatcher, TcpListener, UdpWatcher};
|
|
use pipe::{PipeWatcher, PipeListener};
|
|
use process::Process;
|
|
use signal::SignalWatcher;
|
|
use timer::TimerWatcher;
|
|
use tty::TtyWatcher;
|
|
use uvll;
|
|
|
|
// Obviously an Event Loop is always home.
|
|
pub struct UvEventLoop {
|
|
uvio: UvIoFactory
|
|
}
|
|
|
|
impl UvEventLoop {
|
|
pub fn new() -> UvEventLoop {
|
|
let mut loop_ = Loop::new();
|
|
let handle_pool = QueuePool::new(&mut loop_);
|
|
UvEventLoop {
|
|
uvio: UvIoFactory {
|
|
loop_: loop_,
|
|
handle_pool: Some(handle_pool),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Drop for UvEventLoop {
|
|
fn drop(&mut self) {
|
|
// Must first destroy the pool of handles before we destroy the loop
|
|
// because otherwise the contained async handle will be destroyed after
|
|
// the loop is free'd (use-after-free). We also must free the uv handle
|
|
// after the loop has been closed because during the closing of the loop
|
|
// the handle is required to be used apparently.
|
|
//
|
|
// Lastly, after we've closed the pool of handles we pump the event loop
|
|
// one last time to run any closing callbacks to make sure the loop
|
|
// shuts down cleanly.
|
|
let handle = self.uvio.handle_pool.get_ref().handle();
|
|
drop(self.uvio.handle_pool.take());
|
|
self.run();
|
|
|
|
self.uvio.loop_.close();
|
|
unsafe { uvll::free_handle(handle) }
|
|
}
|
|
}
|
|
|
|
impl EventLoop for UvEventLoop {
|
|
fn run(&mut self) {
|
|
self.uvio.loop_.run();
|
|
}
|
|
|
|
fn callback(&mut self, f: proc()) {
|
|
IdleWatcher::onetime(&mut self.uvio.loop_, f);
|
|
}
|
|
|
|
fn pausable_idle_callback(&mut self, cb: ~rtio::Callback:Send)
|
|
-> ~rtio::PausableIdleCallback:Send
|
|
{
|
|
IdleWatcher::new(&mut self.uvio.loop_,
|
|
cb) as ~rtio::PausableIdleCallback:Send
|
|
}
|
|
|
|
fn remote_callback(&mut self, f: ~rtio::Callback:Send)
|
|
-> ~rtio::RemoteCallback:Send
|
|
{
|
|
~AsyncWatcher::new(&mut self.uvio.loop_, f) as ~rtio::RemoteCallback:Send
|
|
}
|
|
|
|
fn io<'a>(&'a mut self) -> Option<&'a mut rtio::IoFactory> {
|
|
let factory = &mut self.uvio as &mut rtio::IoFactory;
|
|
Some(factory)
|
|
}
|
|
|
|
fn has_active_io(&self) -> bool {
|
|
self.uvio.loop_.get_blockers() > 0
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_callback_run_once() {
|
|
run_in_bare_thread(proc() {
|
|
let mut event_loop = UvEventLoop::new();
|
|
let mut count = 0;
|
|
let count_ptr: *mut int = &mut count;
|
|
event_loop.callback(proc() {
|
|
unsafe { *count_ptr += 1 }
|
|
});
|
|
event_loop.run();
|
|
assert_eq!(count, 1);
|
|
});
|
|
}
|
|
|
|
pub struct UvIoFactory {
|
|
pub loop_: Loop,
|
|
handle_pool: Option<~QueuePool>,
|
|
}
|
|
|
|
impl UvIoFactory {
|
|
pub fn uv_loop<'a>(&mut self) -> *uvll::uv_loop_t { self.loop_.handle }
|
|
|
|
pub fn make_handle(&mut self) -> HomeHandle {
|
|
// It's understood by the homing code that the "local id" is just the
|
|
// pointer of the local I/O factory cast to a uint.
|
|
let id: uint = unsafe { cast::transmute_copy(&self) };
|
|
HomeHandle::new(id, &mut **self.handle_pool.get_mut_ref())
|
|
}
|
|
}
|
|
|
|
impl IoFactory for UvIoFactory {
|
|
// Connect to an address and return a new stream
|
|
// NB: This blocks the task waiting on the connection.
|
|
// It would probably be better to return a future
|
|
fn tcp_connect(&mut self, addr: SocketAddr, timeout: Option<u64>)
|
|
-> Result<~rtio::RtioTcpStream:Send, IoError>
|
|
{
|
|
match TcpWatcher::connect(self, addr, timeout) {
|
|
Ok(t) => Ok(~t as ~rtio::RtioTcpStream:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn tcp_bind(&mut self, addr: SocketAddr) -> Result<~rtio::RtioTcpListener:Send, IoError> {
|
|
match TcpListener::bind(self, addr) {
|
|
Ok(t) => Ok(t as ~rtio::RtioTcpListener:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn udp_bind(&mut self, addr: SocketAddr) -> Result<~rtio::RtioUdpSocket:Send, IoError> {
|
|
match UdpWatcher::bind(self, addr) {
|
|
Ok(u) => Ok(~u as ~rtio::RtioUdpSocket:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn timer_init(&mut self) -> Result<~rtio::RtioTimer:Send, IoError> {
|
|
Ok(TimerWatcher::new(self) as ~rtio::RtioTimer:Send)
|
|
}
|
|
|
|
fn get_host_addresses(&mut self, host: Option<&str>, servname: Option<&str>,
|
|
hint: Option<ai::Hint>) -> Result<~[ai::Info], IoError> {
|
|
let r = GetAddrInfoRequest::run(&self.loop_, host, servname, hint);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
|
|
fn fs_from_raw_fd(&mut self, fd: c_int,
|
|
close: rtio::CloseBehavior) -> ~rtio::RtioFileStream:Send {
|
|
~FileWatcher::new(self, fd, close) as ~rtio::RtioFileStream:Send
|
|
}
|
|
|
|
fn fs_open(&mut self, path: &CString, fm: FileMode, fa: FileAccess)
|
|
-> Result<~rtio::RtioFileStream:Send, IoError> {
|
|
let flags = match fm {
|
|
io::Open => 0,
|
|
io::Append => libc::O_APPEND,
|
|
io::Truncate => libc::O_TRUNC,
|
|
};
|
|
// Opening with a write permission must silently create the file.
|
|
let (flags, mode) = match fa {
|
|
io::Read => (flags | libc::O_RDONLY, 0),
|
|
io::Write => (flags | libc::O_WRONLY | libc::O_CREAT,
|
|
libc::S_IRUSR | libc::S_IWUSR),
|
|
io::ReadWrite => (flags | libc::O_RDWR | libc::O_CREAT,
|
|
libc::S_IRUSR | libc::S_IWUSR),
|
|
};
|
|
|
|
match FsRequest::open(self, path, flags as int, mode as int) {
|
|
Ok(fs) => Ok(~fs as ~rtio::RtioFileStream:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e))
|
|
}
|
|
}
|
|
|
|
fn fs_unlink(&mut self, path: &CString) -> Result<(), IoError> {
|
|
let r = FsRequest::unlink(&self.loop_, path);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_lstat(&mut self, path: &CString) -> Result<FileStat, IoError> {
|
|
let r = FsRequest::lstat(&self.loop_, path);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_stat(&mut self, path: &CString) -> Result<FileStat, IoError> {
|
|
let r = FsRequest::stat(&self.loop_, path);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_mkdir(&mut self, path: &CString,
|
|
perm: io::FilePermission) -> Result<(), IoError> {
|
|
let r = FsRequest::mkdir(&self.loop_, path, perm as c_int);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_rmdir(&mut self, path: &CString) -> Result<(), IoError> {
|
|
let r = FsRequest::rmdir(&self.loop_, path);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_rename(&mut self, path: &CString, to: &CString) -> Result<(), IoError> {
|
|
let r = FsRequest::rename(&self.loop_, path, to);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_chmod(&mut self, path: &CString,
|
|
perm: io::FilePermission) -> Result<(), IoError> {
|
|
let r = FsRequest::chmod(&self.loop_, path, perm as c_int);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_readdir(&mut self, path: &CString, flags: c_int)
|
|
-> Result<Vec<Path>, IoError>
|
|
{
|
|
let r = FsRequest::readdir(&self.loop_, path, flags);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_link(&mut self, src: &CString, dst: &CString) -> Result<(), IoError> {
|
|
let r = FsRequest::link(&self.loop_, src, dst);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_symlink(&mut self, src: &CString, dst: &CString) -> Result<(), IoError> {
|
|
let r = FsRequest::symlink(&self.loop_, src, dst);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_chown(&mut self, path: &CString, uid: int, gid: int) -> Result<(), IoError> {
|
|
let r = FsRequest::chown(&self.loop_, path, uid, gid);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_readlink(&mut self, path: &CString) -> Result<Path, IoError> {
|
|
let r = FsRequest::readlink(&self.loop_, path);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
fn fs_utime(&mut self, path: &CString, atime: u64, mtime: u64)
|
|
-> Result<(), IoError>
|
|
{
|
|
let r = FsRequest::utime(&self.loop_, path, atime, mtime);
|
|
r.map_err(uv_error_to_io_error)
|
|
}
|
|
|
|
fn spawn(&mut self, config: ProcessConfig)
|
|
-> Result<(~rtio::RtioProcess:Send, ~[Option<~rtio::RtioPipe:Send>]), IoError>
|
|
{
|
|
match Process::spawn(self, config) {
|
|
Ok((p, io)) => {
|
|
Ok((p as ~rtio::RtioProcess:Send,
|
|
io.move_iter().map(|i| i.map(|p| ~p as ~rtio::RtioPipe:Send)).collect()))
|
|
}
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn kill(&mut self, pid: libc::pid_t, signum: int) -> Result<(), IoError> {
|
|
Process::kill(pid, signum).map_err(uv_error_to_io_error)
|
|
}
|
|
|
|
fn unix_bind(&mut self, path: &CString) -> Result<~rtio::RtioUnixListener:Send, IoError>
|
|
{
|
|
match PipeListener::bind(self, path) {
|
|
Ok(p) => Ok(p as ~rtio::RtioUnixListener:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn unix_connect(&mut self, path: &CString) -> Result<~rtio::RtioPipe:Send, IoError> {
|
|
match PipeWatcher::connect(self, path) {
|
|
Ok(p) => Ok(~p as ~rtio::RtioPipe:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
|
|
fn tty_open(&mut self, fd: c_int, readable: bool)
|
|
-> Result<~rtio::RtioTTY:Send, IoError> {
|
|
match TtyWatcher::new(self, fd, readable) {
|
|
Ok(tty) => Ok(~tty as ~rtio::RtioTTY:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e))
|
|
}
|
|
}
|
|
|
|
fn pipe_open(&mut self, fd: c_int) -> Result<~rtio::RtioPipe:Send, IoError> {
|
|
match PipeWatcher::open(self, fd) {
|
|
Ok(s) => Ok(~s as ~rtio::RtioPipe:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e))
|
|
}
|
|
}
|
|
|
|
fn signal(&mut self, signum: Signum, channel: Sender<Signum>)
|
|
-> Result<~rtio::RtioSignal:Send, IoError> {
|
|
match SignalWatcher::new(self, signum, channel) {
|
|
Ok(s) => Ok(s as ~rtio::RtioSignal:Send),
|
|
Err(e) => Err(uv_error_to_io_error(e)),
|
|
}
|
|
}
|
|
}
|