2013-10-22 17:13:18 -05:00
|
|
|
// 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::libc;
|
2013-11-04 18:42:05 -06:00
|
|
|
use std::rt::io::IoError;
|
|
|
|
use std::rt::local::Local;
|
|
|
|
use std::rt::rtio::RtioTTY;
|
|
|
|
use std::rt::sched::{Scheduler, SchedHandle};
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
use stream::StreamWatcher;
|
|
|
|
use super::{Loop, UvError, UvHandle, uv_error_to_io_error};
|
|
|
|
use uvio::HomingIO;
|
2013-10-22 17:13:18 -05:00
|
|
|
use uvll;
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
pub struct TtyWatcher{
|
|
|
|
tty: *uvll::uv_tty_t,
|
|
|
|
stream: StreamWatcher,
|
|
|
|
home: SchedHandle,
|
|
|
|
fd: libc::c_int,
|
|
|
|
}
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
impl TtyWatcher {
|
|
|
|
pub fn new(loop_: &Loop, fd: libc::c_int, readable: bool)
|
|
|
|
-> Result<TtyWatcher, UvError>
|
2013-10-22 17:13:18 -05:00
|
|
|
{
|
2013-11-04 18:42:05 -06:00
|
|
|
let handle = UvHandle::alloc(None::<TtyWatcher>, uvll::UV_TTY);
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
match unsafe {
|
2013-11-05 13:29:45 -06:00
|
|
|
uvll::uv_tty_init(loop_.handle, handle, fd as libc::c_int,
|
2013-11-03 12:39:39 -06:00
|
|
|
readable as libc::c_int)
|
2013-11-04 18:42:05 -06:00
|
|
|
} {
|
2013-10-22 17:13:18 -05:00
|
|
|
0 => {
|
2013-11-04 18:42:05 -06:00
|
|
|
Ok(TtyWatcher {
|
|
|
|
tty: handle,
|
|
|
|
stream: StreamWatcher::new(handle),
|
|
|
|
home: get_handle_to_current_scheduler!(),
|
|
|
|
fd: fd,
|
|
|
|
})
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
n => {
|
2013-11-04 18:42:05 -06:00
|
|
|
unsafe { uvll::free_handle(handle) }
|
2013-10-22 17:13:18 -05:00
|
|
|
Err(UvError(n))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-11-04 18:42:05 -06:00
|
|
|
}
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
impl RtioTTY for TtyWatcher {
|
|
|
|
fn read(&mut self, buf: &mut [u8]) -> Result<uint, IoError> {
|
|
|
|
let _m = self.fire_missiles();
|
|
|
|
self.stream.read(buf).map_err(uv_error_to_io_error)
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
fn write(&mut self, buf: &[u8]) -> Result<(), IoError> {
|
|
|
|
let _m = self.fire_missiles();
|
|
|
|
self.stream.write(buf).map_err(uv_error_to_io_error)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_raw(&mut self, raw: bool) -> Result<(), IoError> {
|
2013-10-22 17:13:18 -05:00
|
|
|
let raw = raw as libc::c_int;
|
2013-11-04 18:42:05 -06:00
|
|
|
let _m = self.fire_missiles();
|
|
|
|
match unsafe { uvll::uv_tty_set_mode(self.tty, raw) } {
|
2013-10-22 17:13:18 -05:00
|
|
|
0 => Ok(()),
|
2013-11-04 18:42:05 -06:00
|
|
|
n => Err(uv_error_to_io_error(UvError(n)))
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
#[allow(unused_mut)]
|
|
|
|
fn get_winsize(&mut self) -> Result<(int, int), IoError> {
|
2013-10-22 17:13:18 -05:00
|
|
|
let mut width: libc::c_int = 0;
|
|
|
|
let mut height: libc::c_int = 0;
|
|
|
|
let widthptr: *libc::c_int = &width;
|
|
|
|
let heightptr: *libc::c_int = &width;
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
let _m = self.fire_missiles();
|
|
|
|
match unsafe { uvll::uv_tty_get_winsize(self.tty,
|
2013-11-03 12:39:39 -06:00
|
|
|
widthptr, heightptr) } {
|
2013-10-22 17:13:18 -05:00
|
|
|
0 => Ok((width as int, height as int)),
|
2013-11-04 18:42:05 -06:00
|
|
|
n => Err(uv_error_to_io_error(UvError(n)))
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
fn isatty(&self) -> bool {
|
|
|
|
unsafe { uvll::uv_guess_handle(self.fd) == uvll::UV_TTY }
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-04 18:42:05 -06:00
|
|
|
impl UvHandle<uvll::uv_tty_t> for TtyWatcher {
|
|
|
|
fn uv_handle(&self) -> *uvll::uv_tty_t { self.tty }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HomingIO for TtyWatcher {
|
|
|
|
fn home<'a>(&'a mut self) -> &'a mut SchedHandle { &mut self.home }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for TtyWatcher {
|
|
|
|
// TTY handles are used for the logger in a task, so this destructor is run
|
|
|
|
// when a task is destroyed. When a task is being destroyed, a local
|
|
|
|
// scheduler isn't available, so we can't do the normal "take the scheduler
|
|
|
|
// and resume once close is done". Instead close operations on a TTY are
|
|
|
|
// asynchronous.
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let _m = self.fire_missiles();
|
|
|
|
self.stream.close(false);
|
|
|
|
}
|
|
|
|
}
|