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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
use std::cmp::Ordering;
use std::io;
use std::mem;
use std::os::unix::prelude::*;
use std::path::Path;
use libc::{self, c_int, c_ulong};
use cvt;
#[cfg(any(target_os = "linux", target_os = "android"))]
use libc::{SOCK_CLOEXEC, SOCK_NONBLOCK};
#[cfg(not(any(target_os = "linux", target_os = "android")))]
const SOCK_CLOEXEC: c_int = 0;
#[cfg(not(any(target_os = "linux", target_os = "android")))]
const SOCK_NONBLOCK: c_int = 0;
pub struct Socket {
fd: c_int,
}
impl Socket {
pub fn new(ty: c_int) -> io::Result<Socket> {
unsafe {
if cfg!(target_os = "linux") || cfg!(target_os = "android") {
let flags = ty | SOCK_CLOEXEC | SOCK_NONBLOCK;
match cvt(libc::socket(libc::AF_UNIX, flags, 0)) {
Ok(fd) => return Ok(Socket { fd: fd }),
Err(ref e) if e.raw_os_error() == Some(libc::EINVAL) => {}
Err(e) => return Err(e),
}
}
let fd = Socket { fd: try!(cvt(libc::socket(libc::AF_UNIX, ty, 0))) };
try!(cvt(libc::ioctl(fd.fd, libc::FIOCLEX)));
let mut nonblocking = 1 as c_ulong;
try!(cvt(libc::ioctl(fd.fd, libc::FIONBIO, &mut nonblocking)));
Ok(fd)
}
}
pub fn pair(ty: c_int) -> io::Result<(Socket, Socket)> {
unsafe {
let mut fds = [0, 0];
if cfg!(target_os = "linux") || cfg!(target_os = "android") {
let flags = ty | SOCK_CLOEXEC | SOCK_NONBLOCK;
match cvt(libc::socketpair(libc::AF_UNIX, flags, 0, fds.as_mut_ptr())) {
Ok(_) => {
return Ok((Socket { fd: fds[0] }, Socket { fd: fds[1] }))
}
Err(ref e) if e.raw_os_error() == Some(libc::EINVAL) => {},
Err(e) => return Err(e),
}
}
try!(cvt(libc::socketpair(libc::AF_UNIX, ty, 0, fds.as_mut_ptr())));
let a = Socket { fd: fds[0] };
let b = Socket { fd: fds[1] };
try!(cvt(libc::ioctl(a.fd, libc::FIOCLEX)));
try!(cvt(libc::ioctl(b.fd, libc::FIOCLEX)));
let mut nonblocking = 1 as c_ulong;
try!(cvt(libc::ioctl(a.fd, libc::FIONBIO, &mut nonblocking)));
try!(cvt(libc::ioctl(b.fd, libc::FIONBIO, &mut nonblocking)));
Ok((a, b))
}
}
pub fn fd(&self) -> c_int {
self.fd
}
pub fn into_fd(self) -> c_int {
let ret = self.fd;
mem::forget(self);
ret
}
}
impl Drop for Socket {
fn drop(&mut self) {
unsafe {
let _ = libc::close(self.fd);
}
}
}
pub unsafe fn sockaddr_un(path: &Path)
-> io::Result<(libc::sockaddr_un, libc::socklen_t)> {
let mut addr: libc::sockaddr_un = mem::zeroed();
addr.sun_family = libc::AF_UNIX as libc::sa_family_t;
let bytes = path.as_os_str().as_bytes();
match (bytes.get(0), bytes.len().cmp(&addr.sun_path.len())) {
(Some(&0), Ordering::Greater) => {
return Err(io::Error::new(io::ErrorKind::InvalidInput,
"path must be no longer than SUN_LEN"));
}
(_, Ordering::Greater) | (_, Ordering::Equal) => {
return Err(io::Error::new(io::ErrorKind::InvalidInput,
"path must be shorter than SUN_LEN"));
}
_ => {}
}
for (dst, src) in addr.sun_path.iter_mut().zip(bytes.iter()) {
*dst = *src as libc::c_char;
}
let mut len = sun_path_offset() + bytes.len();
match bytes.get(0) {
Some(&0) | None => {}
Some(_) => len += 1,
}
Ok((addr, len as libc::socklen_t))
}
fn sun_path_offset() -> usize {
unsafe {
let addr: libc::sockaddr_un = mem::uninitialized();
let base = &addr as *const _ as usize;
let path = &addr.sun_path as *const _ as usize;
path - base
}
}