std/os/fd/raw.rs
1//! Raw Unix-like file descriptors.
2
3#![stable(feature = "rust1", since = "1.0.0")]
4
5#[cfg(target_os = "hermit")]
6use hermit_abi as libc;
7
8#[cfg(target_os = "hermit")]
9use crate::os::hermit::io::OwnedFd;
10#[cfg(not(target_os = "hermit"))]
11use crate::os::raw;
12#[cfg(all(doc, not(target_arch = "wasm32")))]
13use crate::os::unix::io::AsFd;
14#[cfg(unix)]
15use crate::os::unix::io::OwnedFd;
16#[cfg(target_os = "wasi")]
17use crate::os::wasi::io::OwnedFd;
18use crate::sys_common::{AsInner, IntoInner};
19use crate::{fs, io};
20
21/// Raw file descriptors.
22#[stable(feature = "rust1", since = "1.0.0")]
23#[cfg(not(target_os = "hermit"))]
24pub type RawFd = raw::c_int;
25#[stable(feature = "rust1", since = "1.0.0")]
26#[cfg(target_os = "hermit")]
27pub type RawFd = i32;
28
29/// A trait to extract the raw file descriptor from an underlying object.
30///
31/// This is only available on unix and WASI platforms and must be imported in
32/// order to call the method. Windows platforms have a corresponding
33/// `AsRawHandle` and `AsRawSocket` set of traits.
34#[stable(feature = "rust1", since = "1.0.0")]
35pub trait AsRawFd {
36 /// Extracts the raw file descriptor.
37 ///
38 /// This function is typically used to **borrow** an owned file descriptor.
39 /// When used in this way, this method does **not** pass ownership of the
40 /// raw file descriptor to the caller, and the file descriptor is only
41 /// guaranteed to be valid while the original object has not yet been
42 /// destroyed.
43 ///
44 /// However, borrowing is not strictly required. See [`AsFd::as_fd`]
45 /// for an API which strictly borrows a file descriptor.
46 ///
47 /// # Example
48 ///
49 /// ```no_run
50 /// use std::fs::File;
51 /// # use std::io;
52 /// #[cfg(any(unix, target_os = "wasi"))]
53 /// use std::os::fd::{AsRawFd, RawFd};
54 ///
55 /// let mut f = File::open("foo.txt")?;
56 /// // Note that `raw_fd` is only valid as long as `f` exists.
57 /// #[cfg(any(unix, target_os = "wasi"))]
58 /// let raw_fd: RawFd = f.as_raw_fd();
59 /// # Ok::<(), io::Error>(())
60 /// ```
61 #[stable(feature = "rust1", since = "1.0.0")]
62 fn as_raw_fd(&self) -> RawFd;
63}
64
65/// A trait to express the ability to construct an object from a raw file
66/// descriptor.
67#[stable(feature = "from_raw_os", since = "1.1.0")]
68pub trait FromRawFd {
69 /// Constructs a new instance of `Self` from the given raw file
70 /// descriptor.
71 ///
72 /// This function is typically used to **consume ownership** of the
73 /// specified file descriptor. When used in this way, the returned object
74 /// will take responsibility for closing it when the object goes out of
75 /// scope.
76 ///
77 /// However, consuming ownership is not strictly required. Use a
78 /// [`From<OwnedFd>::from`] implementation for an API which strictly
79 /// consumes ownership.
80 ///
81 /// # Safety
82 ///
83 /// The `fd` passed in must be an [owned file descriptor][io-safety];
84 /// in particular, it must be open.
85 ///
86 /// [io-safety]: io#io-safety
87 ///
88 /// # Example
89 ///
90 /// ```no_run
91 /// use std::fs::File;
92 /// # use std::io;
93 /// #[cfg(any(unix, target_os = "wasi"))]
94 /// use std::os::fd::{FromRawFd, IntoRawFd, RawFd};
95 ///
96 /// let f = File::open("foo.txt")?;
97 /// # #[cfg(any(unix, target_os = "wasi"))]
98 /// let raw_fd: RawFd = f.into_raw_fd();
99 /// // SAFETY: no other functions should call `from_raw_fd`, so there
100 /// // is only one owner for the file descriptor.
101 /// # #[cfg(any(unix, target_os = "wasi"))]
102 /// let f = unsafe { File::from_raw_fd(raw_fd) };
103 /// # Ok::<(), io::Error>(())
104 /// ```
105 #[stable(feature = "from_raw_os", since = "1.1.0")]
106 unsafe fn from_raw_fd(fd: RawFd) -> Self;
107}
108
109/// A trait to express the ability to consume an object and acquire ownership of
110/// its raw file descriptor.
111#[stable(feature = "into_raw_os", since = "1.4.0")]
112pub trait IntoRawFd {
113 /// Consumes this object, returning the raw underlying file descriptor.
114 ///
115 /// This function is typically used to **transfer ownership** of the underlying
116 /// file descriptor to the caller. When used in this way, callers are then the unique
117 /// owners of the file descriptor and must close it once it's no longer needed.
118 ///
119 /// However, transferring ownership is not strictly required. Use a
120 /// [`Into<OwnedFd>::into`] implementation for an API which strictly
121 /// transfers ownership.
122 ///
123 /// # Example
124 ///
125 /// ```no_run
126 /// use std::fs::File;
127 /// # use std::io;
128 /// #[cfg(any(unix, target_os = "wasi"))]
129 /// use std::os::fd::{IntoRawFd, RawFd};
130 ///
131 /// let f = File::open("foo.txt")?;
132 /// #[cfg(any(unix, target_os = "wasi"))]
133 /// let raw_fd: RawFd = f.into_raw_fd();
134 /// # Ok::<(), io::Error>(())
135 /// ```
136 #[must_use = "losing the raw file descriptor may leak resources"]
137 #[stable(feature = "into_raw_os", since = "1.4.0")]
138 fn into_raw_fd(self) -> RawFd;
139}
140
141#[stable(feature = "raw_fd_reflexive_traits", since = "1.48.0")]
142impl AsRawFd for RawFd {
143 #[inline]
144 fn as_raw_fd(&self) -> RawFd {
145 *self
146 }
147}
148#[stable(feature = "raw_fd_reflexive_traits", since = "1.48.0")]
149impl IntoRawFd for RawFd {
150 #[inline]
151 fn into_raw_fd(self) -> RawFd {
152 self
153 }
154}
155#[stable(feature = "raw_fd_reflexive_traits", since = "1.48.0")]
156impl FromRawFd for RawFd {
157 #[inline]
158 unsafe fn from_raw_fd(fd: RawFd) -> RawFd {
159 fd
160 }
161}
162
163#[stable(feature = "rust1", since = "1.0.0")]
164impl AsRawFd for fs::File {
165 #[inline]
166 fn as_raw_fd(&self) -> RawFd {
167 self.as_inner().as_raw_fd()
168 }
169}
170#[stable(feature = "from_raw_os", since = "1.1.0")]
171impl FromRawFd for fs::File {
172 #[inline]
173 unsafe fn from_raw_fd(fd: RawFd) -> fs::File {
174 unsafe { fs::File::from(OwnedFd::from_raw_fd(fd)) }
175 }
176}
177#[stable(feature = "into_raw_os", since = "1.4.0")]
178impl IntoRawFd for fs::File {
179 #[inline]
180 fn into_raw_fd(self) -> RawFd {
181 self.into_inner().into_inner().into_raw_fd()
182 }
183}
184
185#[stable(feature = "asraw_stdio", since = "1.21.0")]
186impl AsRawFd for io::Stdin {
187 #[inline]
188 fn as_raw_fd(&self) -> RawFd {
189 libc::STDIN_FILENO
190 }
191}
192
193#[stable(feature = "asraw_stdio", since = "1.21.0")]
194impl AsRawFd for io::Stdout {
195 #[inline]
196 fn as_raw_fd(&self) -> RawFd {
197 libc::STDOUT_FILENO
198 }
199}
200
201#[stable(feature = "asraw_stdio", since = "1.21.0")]
202impl AsRawFd for io::Stderr {
203 #[inline]
204 fn as_raw_fd(&self) -> RawFd {
205 libc::STDERR_FILENO
206 }
207}
208
209#[stable(feature = "asraw_stdio_locks", since = "1.35.0")]
210impl<'a> AsRawFd for io::StdinLock<'a> {
211 #[inline]
212 fn as_raw_fd(&self) -> RawFd {
213 libc::STDIN_FILENO
214 }
215}
216
217#[stable(feature = "asraw_stdio_locks", since = "1.35.0")]
218impl<'a> AsRawFd for io::StdoutLock<'a> {
219 #[inline]
220 fn as_raw_fd(&self) -> RawFd {
221 libc::STDOUT_FILENO
222 }
223}
224
225#[stable(feature = "asraw_stdio_locks", since = "1.35.0")]
226impl<'a> AsRawFd for io::StderrLock<'a> {
227 #[inline]
228 fn as_raw_fd(&self) -> RawFd {
229 libc::STDERR_FILENO
230 }
231}
232
233/// This impl allows implementing traits that require `AsRawFd` on Arc.
234/// ```
235/// # #[cfg(any(unix, target_os = "wasi"))] mod group_cfg {
236/// # #[cfg(target_os = "wasi")]
237/// # use std::os::wasi::io::AsRawFd;
238/// # #[cfg(unix)]
239/// # use std::os::unix::io::AsRawFd;
240/// use std::net::UdpSocket;
241/// use std::sync::Arc;
242/// trait MyTrait: AsRawFd {
243/// }
244/// impl MyTrait for Arc<UdpSocket> {}
245/// impl MyTrait for Box<UdpSocket> {}
246/// # }
247/// ```
248#[stable(feature = "asrawfd_ptrs", since = "1.63.0")]
249impl<T: AsRawFd> AsRawFd for crate::sync::Arc<T> {
250 #[inline]
251 fn as_raw_fd(&self) -> RawFd {
252 (**self).as_raw_fd()
253 }
254}
255
256#[stable(feature = "asfd_rc", since = "1.69.0")]
257impl<T: AsRawFd> AsRawFd for crate::rc::Rc<T> {
258 #[inline]
259 fn as_raw_fd(&self) -> RawFd {
260 (**self).as_raw_fd()
261 }
262}
263
264#[unstable(feature = "unique_rc_arc", issue = "112566")]
265impl<T: AsRawFd + ?Sized> AsRawFd for crate::rc::UniqueRc<T> {
266 #[inline]
267 fn as_raw_fd(&self) -> RawFd {
268 (**self).as_raw_fd()
269 }
270}
271
272#[stable(feature = "asrawfd_ptrs", since = "1.63.0")]
273impl<T: AsRawFd> AsRawFd for Box<T> {
274 #[inline]
275 fn as_raw_fd(&self) -> RawFd {
276 (**self).as_raw_fd()
277 }
278}