1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
extern crate libc;
extern crate mio;
extern crate tokio_core;
use std::{fs, io, os};
use tokio_core::reactor::{Handle, PollEvented};
#[derive(Debug)]
pub struct File(pub fs::File);
impl File {
pub fn new_nb(mut file: fs::File) -> io::Result<Self> {
set_file_nonblocking(&mut file, true)?;
Ok(File(file))
}
pub fn into_io(self, handle: &Handle) -> io::Result<PollEvented<Self>> {
Ok(PollEvented::new(self, handle)?)
}
pub fn into_reader(self, handle: &Handle)
-> io::Result<io::BufReader<PollEvented<Self>>> {
Ok(io::BufReader::new(self.into_io(handle)?))
}
}
impl mio::Evented for File {
fn register(&self, poll: &mio::Poll, token: mio::Token,
interest: mio::Ready, opts: mio::PollOpt)
-> io::Result<()> {
mio::unix::EventedFd(&os::unix::io::AsRawFd::as_raw_fd(&self.0))
.register(poll, token, interest, opts)
}
fn reregister(&self, poll: &mio::Poll, token: mio::Token,
interest: mio::Ready, opts: mio::PollOpt)
-> io::Result<()> {
mio::unix::EventedFd(&os::unix::io::AsRawFd::as_raw_fd(&self.0))
.reregister(poll, token, interest, opts)
}
fn deregister(&self, poll: &mio::Poll) -> io::Result<()> {
mio::unix::EventedFd(&os::unix::io::AsRawFd::as_raw_fd(&self.0))
.deregister(poll)
}
}
impl io::Read for File {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
self.0.read(buf)
}
}
impl io::Write for File {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.0.write(buf)
}
fn flush(&mut self) -> io::Result<()> {
self.0.flush()
}
}
pub fn set_file_nonblocking(file: &mut fs::File, nonblocking: bool)
-> io::Result<()> {
let fd = os::unix::io::AsRawFd::as_raw_fd(file);
set_fd_nonblocking(fd, nonblocking)
}
fn set_fd_nonblocking(fd: os::unix::io::RawFd, nonblocking: bool)
-> io::Result<()> {
unsafe {
let previous = libc::fcntl(fd, libc::F_GETFL);
if previous < 0 {
return Err(io::Error::last_os_error());
}
let new = if nonblocking {
previous | libc::O_NONBLOCK
} else {
previous & !libc::O_NONBLOCK
};
if libc::fcntl(fd, libc::F_SETFL, new) < 0 {
return Err(io::Error::last_os_error());
}
Ok(())
}
}