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
use std::convert::From;
use std::io::{self, Read, Write};
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
pub struct TunIo(RawFd);
impl From<RawFd> for TunIo {
fn from(fd: RawFd) -> Self {
Self(fd)
}
}
impl FromRawFd for TunIo {
unsafe fn from_raw_fd(fd: RawFd) -> Self {
Self(fd)
}
}
impl AsRawFd for TunIo {
fn as_raw_fd(&self) -> RawFd {
self.0
}
}
impl Read for TunIo {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
self.recv(buf)
}
}
impl Write for TunIo {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.send(buf)
}
fn flush(&mut self) -> io::Result<()> {
let ret = unsafe { libc::fsync(self.0) };
if ret < 0 {
return Err(io::Error::last_os_error());
}
Ok(())
}
}
impl TunIo {
pub fn recv(&self, buf: &mut [u8]) -> io::Result<usize> {
let n = unsafe { libc::read(self.0, buf.as_ptr() as *mut _, buf.len() as _) };
if n < 0 {
return Err(io::Error::last_os_error());
}
Ok(n as _)
}
pub fn send(&self, buf: &[u8]) -> io::Result<usize> {
let n = unsafe { libc::write(self.0, buf.as_ptr() as *const _, buf.len() as _) };
if n < 0 {
return Err(io::Error::last_os_error());
}
Ok(n as _)
}
}
impl Drop for TunIo {
fn drop(&mut self) {
unsafe { libc::close(self.0) };
}
}