Skip to content

Commit d154185

Browse files
committed
I/O safety.
Introduce `OwnedFd` and `BorrowedFd`, and the `AsFd` trait, and implementations of `AsFd`, `From<OwnedFd>` and `From<T> for OwnedFd` for relevant types, along with Windows counterparts for handles and sockets. Tracking issue: - <rust-lang#87074> RFC: - <https://github.com/rust-lang/rfcs/blob/master/text/3128-io-safety.md>
1 parent 2451f42 commit d154185

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

43 files changed

+2360
-556
lines changed

library/std/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -285,6 +285,7 @@
285285
#![feature(int_log)]
286286
#![feature(into_future)]
287287
#![feature(intra_doc_pointers)]
288+
#![feature(io_safety)]
288289
#![feature(iter_zip)]
289290
#![feature(lang_items)]
290291
#![feature(linkage)]

library/std/src/net/udp/tests.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ use crate::io::ErrorKind;
22
use crate::net::test::{next_test_ip4, next_test_ip6};
33
use crate::net::*;
44
use crate::sync::mpsc::channel;
5-
use crate::sys_common::AsInner;
65
use crate::thread;
76
use crate::time::{Duration, Instant};
87

@@ -173,7 +172,7 @@ fn debug() {
173172
let socket_addr = next_test_ip4();
174173

175174
let udpsock = t!(UdpSocket::bind(&socket_addr));
176-
let udpsock_inner = udpsock.0.socket().as_inner();
175+
let udpsock_inner = udpsock.0.socket().as_raw();
177176
let compare = format!("UdpSocket {{ addr: {:?}, {}: {:?} }}", socket_addr, name, udpsock_inner);
178177
assert_eq!(format!("{:?}", udpsock), compare);
179178
}

library/std/src/os/unix/io/fd.rs

+329
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,329 @@
1+
//! Owned and borrowed file descriptors.
2+
3+
#![unstable(feature = "io_safety", issue = "87074")]
4+
5+
use super::raw::{AsRawFd, FromRawFd, IntoRawFd, RawFd};
6+
use crate::fmt;
7+
use crate::fs;
8+
use crate::marker::PhantomData;
9+
use crate::mem::forget;
10+
use crate::os::raw;
11+
use crate::sys_common::{AsInner, FromInner, IntoInner};
12+
13+
/// A borrowed file descriptor.
14+
///
15+
/// This has a lifetime parameter to tie it to the lifetime of something that
16+
/// owns the file descriptor.
17+
///
18+
/// This uses `repr(transparent)` and has the representation of a host file
19+
/// descriptor, so it can be used in FFI in places where a file descriptor is
20+
/// passed as an argument, it is not captured or consumed, and it never has the
21+
/// value `-1`.
22+
#[derive(Copy, Clone)]
23+
#[repr(transparent)]
24+
#[rustc_layout_scalar_valid_range_start(0)]
25+
// libstd/os/raw/mod.rs assures me that every libstd-supported platform has a
26+
// 32-bit c_int. Below is -2, in two's complement, but that only works out
27+
// because c_int is 32 bits.
28+
#[rustc_layout_scalar_valid_range_end(0xFF_FF_FF_FE)]
29+
#[unstable(feature = "io_safety", issue = "87074")]
30+
pub struct BorrowedFd<'fd> {
31+
raw: RawFd,
32+
_phantom: PhantomData<&'fd OwnedFd>,
33+
}
34+
35+
/// An owned file descriptor.
36+
///
37+
/// This closes the file descriptor on drop.
38+
///
39+
/// This uses `repr(transparent)` and has the representation of a host file
40+
/// descriptor, so it can be used in FFI in places where a file descriptor is
41+
/// passed as a consumed argument or returned as an owned value, and it never
42+
/// has the value `-1`.
43+
#[repr(transparent)]
44+
#[rustc_layout_scalar_valid_range_start(0)]
45+
// libstd/os/raw/mod.rs assures me that every libstd-supported platform has a
46+
// 32-bit c_int. Below is -2, in two's complement, but that only works out
47+
// because c_int is 32 bits.
48+
#[rustc_layout_scalar_valid_range_end(0xFF_FF_FF_FE)]
49+
#[unstable(feature = "io_safety", issue = "87074")]
50+
pub struct OwnedFd {
51+
raw: RawFd,
52+
}
53+
54+
impl BorrowedFd<'_> {
55+
/// Return a `BorrowedFd` holding the given raw file descriptor.
56+
///
57+
/// # Safety
58+
///
59+
/// The resource pointed to by `raw` must remain open for the duration of
60+
/// the returned `BorrowedFd`, and it must not have the value `-1`.
61+
#[inline]
62+
#[unstable(feature = "io_safety", issue = "87074")]
63+
pub unsafe fn borrow_raw_fd(raw: RawFd) -> Self {
64+
assert_ne!(raw, -1_i32 as RawFd);
65+
Self { raw, _phantom: PhantomData }
66+
}
67+
}
68+
69+
#[unstable(feature = "io_safety", issue = "87074")]
70+
impl AsRawFd for BorrowedFd<'_> {
71+
#[inline]
72+
fn as_raw_fd(&self) -> RawFd {
73+
self.raw
74+
}
75+
}
76+
77+
#[unstable(feature = "io_safety", issue = "87074")]
78+
impl AsRawFd for OwnedFd {
79+
#[inline]
80+
fn as_raw_fd(&self) -> RawFd {
81+
self.raw
82+
}
83+
}
84+
85+
#[unstable(feature = "io_safety", issue = "87074")]
86+
impl IntoRawFd for OwnedFd {
87+
#[inline]
88+
fn into_raw_fd(self) -> RawFd {
89+
let raw = self.raw;
90+
forget(self);
91+
raw
92+
}
93+
}
94+
95+
#[unstable(feature = "io_safety", issue = "87074")]
96+
impl FromRawFd for OwnedFd {
97+
/// Constructs a new instance of `Self` from the given raw file descriptor.
98+
///
99+
/// # Safety
100+
///
101+
/// The resource pointed to by `raw` must be open and suitable for assuming
102+
/// ownership.
103+
#[inline]
104+
unsafe fn from_raw_fd(raw: RawFd) -> Self {
105+
assert_ne!(raw, -1i32);
106+
// SAFETY: we just asserted that the value is in the valid range and isn't `-1` (the only value bigger than `0xFF_FF_FF_FE` unsigned)
107+
Self { raw }
108+
}
109+
}
110+
111+
#[unstable(feature = "io_safety", issue = "87074")]
112+
impl Drop for OwnedFd {
113+
#[inline]
114+
fn drop(&mut self) {
115+
unsafe {
116+
// Note that errors are ignored when closing a file descriptor. The
117+
// reason for this is that if an error occurs we don't actually know if
118+
// the file descriptor was closed or not, and if we retried (for
119+
// something like EINTR), we might close another valid file descriptor
120+
// opened after we closed ours.
121+
let _ = libc::close(self.raw as raw::c_int);
122+
}
123+
}
124+
}
125+
126+
#[unstable(feature = "io_safety", issue = "87074")]
127+
impl fmt::Debug for BorrowedFd<'_> {
128+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
129+
f.debug_struct("BorrowedFd").field("fd", &self.raw).finish()
130+
}
131+
}
132+
133+
#[unstable(feature = "io_safety", issue = "87074")]
134+
impl fmt::Debug for OwnedFd {
135+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
136+
f.debug_struct("OwnedFd").field("fd", &self.raw).finish()
137+
}
138+
}
139+
140+
/// A trait to borrow the file descriptor from an underlying object.
141+
///
142+
/// This is only available on unix platforms and must be imported in order to
143+
/// call the method. Windows platforms have a corresponding `AsHandle` and
144+
/// `AsSocket` set of traits.
145+
#[unstable(feature = "io_safety", issue = "87074")]
146+
pub trait AsFd {
147+
/// Borrows the file descriptor.
148+
///
149+
/// # Example
150+
///
151+
/// ```rust,no_run
152+
/// #![feature(io_safety)]
153+
/// use std::fs::File;
154+
/// # use std::io;
155+
/// use std::os::unix::io::{AsFd, BorrowedFd};
156+
///
157+
/// let mut f = File::open("foo.txt")?;
158+
/// let borrowed_fd: BorrowedFd<'_> = f.as_fd();
159+
/// # Ok::<(), io::Error>(())
160+
/// ```
161+
#[unstable(feature = "io_safety", issue = "87074")]
162+
fn as_fd(&self) -> BorrowedFd<'_>;
163+
}
164+
165+
#[unstable(feature = "io_safety", issue = "87074")]
166+
impl AsFd for BorrowedFd<'_> {
167+
#[inline]
168+
fn as_fd(&self) -> BorrowedFd<'_> {
169+
*self
170+
}
171+
}
172+
173+
#[unstable(feature = "io_safety", issue = "87074")]
174+
impl AsFd for OwnedFd {
175+
#[inline]
176+
fn as_fd(&self) -> BorrowedFd<'_> {
177+
unsafe { BorrowedFd::borrow_raw_fd(self.as_raw_fd()) }
178+
}
179+
}
180+
181+
#[unstable(feature = "io_safety", issue = "87074")]
182+
impl AsFd for fs::File {
183+
#[inline]
184+
fn as_fd(&self) -> BorrowedFd<'_> {
185+
self.as_inner().as_fd()
186+
}
187+
}
188+
189+
#[unstable(feature = "io_safety", issue = "87074")]
190+
impl From<fs::File> for OwnedFd {
191+
#[inline]
192+
fn from(file: fs::File) -> OwnedFd {
193+
file.into_inner().into_inner().into_inner()
194+
}
195+
}
196+
197+
#[unstable(feature = "io_safety", issue = "87074")]
198+
impl From<OwnedFd> for fs::File {
199+
#[inline]
200+
fn from(owned_fd: OwnedFd) -> Self {
201+
Self::from_inner(FromInner::from_inner(FromInner::from_inner(owned_fd)))
202+
}
203+
}
204+
205+
#[unstable(feature = "io_safety", issue = "87074")]
206+
impl AsFd for crate::net::TcpStream {
207+
#[inline]
208+
fn as_fd(&self) -> BorrowedFd<'_> {
209+
self.as_inner().socket().as_fd()
210+
}
211+
}
212+
213+
#[unstable(feature = "io_safety", issue = "87074")]
214+
impl From<crate::net::TcpStream> for OwnedFd {
215+
#[inline]
216+
fn from(tcp_stream: crate::net::TcpStream) -> OwnedFd {
217+
tcp_stream.into_inner().into_socket().into_inner().into_inner().into()
218+
}
219+
}
220+
221+
#[unstable(feature = "io_safety", issue = "87074")]
222+
impl From<OwnedFd> for crate::net::TcpStream {
223+
#[inline]
224+
fn from(owned_fd: OwnedFd) -> Self {
225+
Self::from_inner(FromInner::from_inner(FromInner::from_inner(FromInner::from_inner(
226+
owned_fd,
227+
))))
228+
}
229+
}
230+
231+
#[unstable(feature = "io_safety", issue = "87074")]
232+
impl AsFd for crate::net::TcpListener {
233+
#[inline]
234+
fn as_fd(&self) -> BorrowedFd<'_> {
235+
self.as_inner().socket().as_fd()
236+
}
237+
}
238+
239+
#[unstable(feature = "io_safety", issue = "87074")]
240+
impl From<crate::net::TcpListener> for OwnedFd {
241+
#[inline]
242+
fn from(tcp_listener: crate::net::TcpListener) -> OwnedFd {
243+
tcp_listener.into_inner().into_socket().into_inner().into_inner().into()
244+
}
245+
}
246+
247+
#[unstable(feature = "io_safety", issue = "87074")]
248+
impl From<OwnedFd> for crate::net::TcpListener {
249+
#[inline]
250+
fn from(owned_fd: OwnedFd) -> Self {
251+
Self::from_inner(FromInner::from_inner(FromInner::from_inner(FromInner::from_inner(
252+
owned_fd,
253+
))))
254+
}
255+
}
256+
257+
#[unstable(feature = "io_safety", issue = "87074")]
258+
impl AsFd for crate::net::UdpSocket {
259+
#[inline]
260+
fn as_fd(&self) -> BorrowedFd<'_> {
261+
self.as_inner().socket().as_fd()
262+
}
263+
}
264+
265+
#[unstable(feature = "io_safety", issue = "87074")]
266+
impl From<crate::net::UdpSocket> for OwnedFd {
267+
#[inline]
268+
fn from(udp_socket: crate::net::UdpSocket) -> OwnedFd {
269+
udp_socket.into_inner().into_socket().into_inner().into_inner().into()
270+
}
271+
}
272+
273+
#[unstable(feature = "io_safety", issue = "87074")]
274+
impl From<OwnedFd> for crate::net::UdpSocket {
275+
#[inline]
276+
fn from(owned_fd: OwnedFd) -> Self {
277+
Self::from_inner(FromInner::from_inner(FromInner::from_inner(FromInner::from_inner(
278+
owned_fd,
279+
))))
280+
}
281+
}
282+
283+
#[unstable(feature = "io_safety", issue = "87074")]
284+
impl AsFd for crate::process::ChildStdin {
285+
#[inline]
286+
fn as_fd(&self) -> BorrowedFd<'_> {
287+
self.as_inner().as_fd()
288+
}
289+
}
290+
291+
#[unstable(feature = "io_safety", issue = "87074")]
292+
impl From<crate::process::ChildStdin> for OwnedFd {
293+
#[inline]
294+
fn from(child_stdin: crate::process::ChildStdin) -> OwnedFd {
295+
child_stdin.into_inner().into_inner().into_inner()
296+
}
297+
}
298+
299+
#[unstable(feature = "io_safety", issue = "87074")]
300+
impl AsFd for crate::process::ChildStdout {
301+
#[inline]
302+
fn as_fd(&self) -> BorrowedFd<'_> {
303+
self.as_inner().as_fd()
304+
}
305+
}
306+
307+
#[unstable(feature = "io_safety", issue = "87074")]
308+
impl From<crate::process::ChildStdout> for OwnedFd {
309+
#[inline]
310+
fn from(child_stdout: crate::process::ChildStdout) -> OwnedFd {
311+
child_stdout.into_inner().into_inner().into_inner()
312+
}
313+
}
314+
315+
#[unstable(feature = "io_safety", issue = "87074")]
316+
impl AsFd for crate::process::ChildStderr {
317+
#[inline]
318+
fn as_fd(&self) -> BorrowedFd<'_> {
319+
self.as_inner().as_fd()
320+
}
321+
}
322+
323+
#[unstable(feature = "io_safety", issue = "87074")]
324+
impl From<crate::process::ChildStderr> for OwnedFd {
325+
#[inline]
326+
fn from(child_stderr: crate::process::ChildStderr) -> OwnedFd {
327+
child_stderr.into_inner().into_inner().into_inner()
328+
}
329+
}

0 commit comments

Comments
 (0)