2023-01-14 22:56:24 +00:00
|
|
|
use crate::ffi;
|
|
|
|
use nix::unistd;
|
2023-02-18 01:21:44 +00:00
|
|
|
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
|
2023-01-14 22:56:24 +00:00
|
|
|
|
|
|
|
/// A helper type for managing and automatically closing a file descriptor
|
2023-02-18 01:21:44 +00:00
|
|
|
///
|
|
|
|
/// This was implemented in rust as a port of the existing C++ code but it didn't take its place
|
|
|
|
/// (yet) and there's still the original cpp implementation in `src/fds.h`, so its name is
|
|
|
|
/// disambiguated because some code uses a mix of both for interop purposes.
|
|
|
|
pub struct AutoCloseFd {
|
2023-01-14 22:56:24 +00:00
|
|
|
fd_: RawFd,
|
|
|
|
}
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
#[cxx::bridge]
|
|
|
|
mod autoclose_fd_t {
|
|
|
|
extern "Rust" {
|
|
|
|
#[cxx_name = "autoclose_fd_t2"]
|
|
|
|
type AutoCloseFd;
|
|
|
|
|
|
|
|
#[cxx_name = "valid"]
|
|
|
|
fn is_valid(&self) -> bool;
|
|
|
|
fn close(&mut self);
|
|
|
|
fn fd(&self) -> i32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AutoCloseFd {
|
2023-01-14 22:56:24 +00:00
|
|
|
// Closes the fd if not already closed.
|
|
|
|
pub fn close(&mut self) {
|
|
|
|
if self.fd_ != -1 {
|
|
|
|
_ = unistd::close(self.fd_);
|
|
|
|
self.fd_ = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the fd.
|
|
|
|
pub fn fd(&self) -> RawFd {
|
|
|
|
self.fd_
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the fd, transferring ownership to the caller.
|
|
|
|
pub fn acquire(&mut self) -> RawFd {
|
|
|
|
let temp = self.fd_;
|
|
|
|
self.fd_ = -1;
|
|
|
|
temp
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resets to a new fd, taking ownership.
|
|
|
|
pub fn reset(&mut self, fd: RawFd) {
|
|
|
|
if fd == self.fd_ {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.close();
|
|
|
|
self.fd_ = fd;
|
|
|
|
}
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
// Returns if this has a valid fd.
|
|
|
|
pub fn is_valid(&self) -> bool {
|
2023-01-14 22:56:24 +00:00
|
|
|
self.fd_ >= 0
|
|
|
|
}
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
// Create a new AutoCloseFd instance taking ownership of the passed fd
|
|
|
|
pub fn new(fd: RawFd) -> Self {
|
|
|
|
AutoCloseFd { fd_: fd }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new AutoCloseFd without an open fd
|
|
|
|
pub fn empty() -> Self {
|
|
|
|
AutoCloseFd { fd_: -1 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromRawFd for AutoCloseFd {
|
|
|
|
unsafe fn from_raw_fd(fd: RawFd) -> Self {
|
|
|
|
AutoCloseFd { fd_: fd }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRawFd for AutoCloseFd {
|
|
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
|
|
self.fd()
|
2023-01-14 22:56:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
impl Default for AutoCloseFd {
|
|
|
|
fn default() -> AutoCloseFd {
|
|
|
|
AutoCloseFd { fd_: -1 }
|
2023-01-14 22:56:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
impl Drop for AutoCloseFd {
|
2023-01-14 22:56:24 +00:00
|
|
|
fn drop(&mut self) {
|
|
|
|
self.close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Helper type returned from make_autoclose_pipes.
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct autoclose_pipes_t {
|
|
|
|
/// Read end of the pipe.
|
2023-02-18 01:21:44 +00:00
|
|
|
pub read: AutoCloseFd,
|
2023-01-14 22:56:24 +00:00
|
|
|
|
|
|
|
/// Write end of the pipe.
|
2023-02-18 01:21:44 +00:00
|
|
|
pub write: AutoCloseFd,
|
2023-01-14 22:56:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Construct a pair of connected pipes, set to close-on-exec.
|
|
|
|
/// \return None on fd exhaustion.
|
|
|
|
pub fn make_autoclose_pipes() -> Option<autoclose_pipes_t> {
|
|
|
|
let pipes = ffi::make_pipes_ffi();
|
|
|
|
|
2023-02-18 01:21:44 +00:00
|
|
|
let readp = AutoCloseFd::new(pipes.read);
|
|
|
|
let writep = AutoCloseFd::new(pipes.write);
|
|
|
|
if !readp.is_valid() || !writep.is_valid() {
|
2023-01-14 22:56:24 +00:00
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(autoclose_pipes_t {
|
|
|
|
read: readp,
|
|
|
|
write: writep,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|