mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-03 16:48:45 +00:00
Use nix & Results
This commit is contained in:
parent
971d774e67
commit
b9ba9e57e8
5 changed files with 49 additions and 43 deletions
|
@ -26,11 +26,11 @@ pub fn eval(parser: &Parser, streams: &mut IoStreams, args: &mut [&wstr]) -> Opt
|
||||||
let mut stdout_fill = None;
|
let mut stdout_fill = None;
|
||||||
if streams.out_is_piped {
|
if streams.out_is_piped {
|
||||||
match IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDOUT_FILENO) {
|
match IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDOUT_FILENO) {
|
||||||
None => {
|
Err(_) => {
|
||||||
// We were unable to create a pipe, probably fd exhaustion.
|
// We were unable to create a pipe, probably fd exhaustion.
|
||||||
return STATUS_CMD_ERROR;
|
return STATUS_CMD_ERROR;
|
||||||
}
|
}
|
||||||
Some(buffer) => {
|
Ok(buffer) => {
|
||||||
stdout_fill = Some(buffer.clone());
|
stdout_fill = Some(buffer.clone());
|
||||||
ios.push(buffer);
|
ios.push(buffer);
|
||||||
}
|
}
|
||||||
|
@ -41,11 +41,11 @@ pub fn eval(parser: &Parser, streams: &mut IoStreams, args: &mut [&wstr]) -> Opt
|
||||||
let mut stderr_fill = None;
|
let mut stderr_fill = None;
|
||||||
if streams.err_is_piped {
|
if streams.err_is_piped {
|
||||||
match IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDERR_FILENO) {
|
match IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDERR_FILENO) {
|
||||||
None => {
|
Err(_) => {
|
||||||
// We were unable to create a pipe, probably fd exhaustion.
|
// We were unable to create a pipe, probably fd exhaustion.
|
||||||
return STATUS_CMD_ERROR;
|
return STATUS_CMD_ERROR;
|
||||||
}
|
}
|
||||||
Some(buffer) => {
|
Ok(buffer) => {
|
||||||
stderr_fill = Some(buffer.clone());
|
stderr_fill = Some(buffer.clone());
|
||||||
ios.push(buffer);
|
ios.push(buffer);
|
||||||
}
|
}
|
||||||
|
|
|
@ -136,7 +136,7 @@ pub fn exec_job(parser: &Parser, job: &Job, block_io: IoChain) -> bool {
|
||||||
let mut proc_pipes = PartialPipes::default();
|
let mut proc_pipes = PartialPipes::default();
|
||||||
std::mem::swap(&mut proc_pipes.read, &mut pipe_next_read);
|
std::mem::swap(&mut proc_pipes.read, &mut pipe_next_read);
|
||||||
if !p.is_last_in_job {
|
if !p.is_last_in_job {
|
||||||
let Some(pipes) = make_autoclose_pipes() else {
|
let Ok(pipes) = make_autoclose_pipes() else {
|
||||||
FLOGF!(warning, "%ls", wgettext!(PIPE_ERROR));
|
FLOGF!(warning, "%ls", wgettext!(PIPE_ERROR));
|
||||||
perror("pipe");
|
perror("pipe");
|
||||||
aborted_pipeline = true;
|
aborted_pipeline = true;
|
||||||
|
@ -1048,12 +1048,12 @@ fn exec_block_or_func_process(
|
||||||
if piped_output_needs_buffering {
|
if piped_output_needs_buffering {
|
||||||
// Be careful to handle failure, e.g. too many open fds.
|
// Be careful to handle failure, e.g. too many open fds.
|
||||||
match IoBufferfill::create() {
|
match IoBufferfill::create() {
|
||||||
Some(tmp) => {
|
Ok(tmp) => {
|
||||||
// Teach the job about its bufferfill, and add it to our io chain.
|
// Teach the job about its bufferfill, and add it to our io chain.
|
||||||
io_chain.push(tmp.clone());
|
io_chain.push(tmp.clone());
|
||||||
block_output_bufferfill = Some(tmp);
|
block_output_bufferfill = Some(tmp);
|
||||||
}
|
}
|
||||||
None => return Err(()),
|
Err(_) => return Err(()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1477,8 +1477,7 @@ fn exec_subshell_internal(
|
||||||
|
|
||||||
// IO buffer creation may fail (e.g. if we have too many open files to make a pipe), so this may
|
// IO buffer creation may fail (e.g. if we have too many open files to make a pipe), so this may
|
||||||
// be null.
|
// be null.
|
||||||
let Some(bufferfill) =
|
let Ok(bufferfill) = IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDOUT_FILENO)
|
||||||
IoBufferfill::create_opts(parser.libdata().pods.read_limit, STDOUT_FILENO)
|
|
||||||
else {
|
else {
|
||||||
*break_expand = true;
|
*break_expand = true;
|
||||||
return STATUS_CMD_ERROR.unwrap();
|
return STATUS_CMD_ERROR.unwrap();
|
||||||
|
|
|
@ -62,7 +62,7 @@ impl FdEventSignaller {
|
||||||
#[cfg(not(HAVE_EVENTFD))]
|
#[cfg(not(HAVE_EVENTFD))]
|
||||||
{
|
{
|
||||||
// Implementation using pipes.
|
// Implementation using pipes.
|
||||||
let Some(pipes) = make_autoclose_pipes() else {
|
let Ok(pipes) = make_autoclose_pipes() else {
|
||||||
perror("pipe");
|
perror("pipe");
|
||||||
exit_without_destructors(1);
|
exit_without_destructors(1);
|
||||||
};
|
};
|
||||||
|
|
64
src/fds.rs
64
src/fds.rs
|
@ -6,9 +6,8 @@ use crate::tests::prelude::*;
|
||||||
use crate::wchar::prelude::*;
|
use crate::wchar::prelude::*;
|
||||||
use crate::wutil::perror;
|
use crate::wutil::perror;
|
||||||
use errno::{errno, set_errno};
|
use errno::{errno, set_errno};
|
||||||
use libc::{
|
use libc::{c_int, EINTR, FD_CLOEXEC, F_GETFD, F_GETFL, F_SETFD, F_SETFL, O_NONBLOCK};
|
||||||
c_int, EINTR, FD_CLOEXEC, F_DUPFD_CLOEXEC, F_GETFD, F_GETFL, F_SETFD, F_SETFL, O_NONBLOCK,
|
use nix::fcntl::FcntlArg;
|
||||||
};
|
|
||||||
use nix::{fcntl::OFlag, unistd};
|
use nix::{fcntl::OFlag, unistd};
|
||||||
use std::ffi::CStr;
|
use std::ffi::CStr;
|
||||||
use std::io::{self, Read, Write};
|
use std::io::{self, Read, Write};
|
||||||
|
@ -130,35 +129,39 @@ pub struct AutoClosePipes {
|
||||||
|
|
||||||
/// Construct a pair of connected pipes, set to close-on-exec.
|
/// Construct a pair of connected pipes, set to close-on-exec.
|
||||||
/// \return None on fd exhaustion.
|
/// \return None on fd exhaustion.
|
||||||
pub fn make_autoclose_pipes() -> Option<AutoClosePipes> {
|
pub fn make_autoclose_pipes() -> nix::Result<AutoClosePipes> {
|
||||||
let mut pipes: [c_int; 2] = [-1, -1];
|
|
||||||
|
|
||||||
#[allow(unused_mut, unused_assignments)]
|
#[allow(unused_mut, unused_assignments)]
|
||||||
let mut already_cloexec = false;
|
let mut already_cloexec = false;
|
||||||
#[cfg(HAVE_PIPE2)]
|
#[cfg(HAVE_PIPE2)]
|
||||||
{
|
let pipes = match nix::unistd::pipe2(OFlag::O_CLOEXEC) {
|
||||||
if unsafe { libc::pipe2(&mut pipes[0], libc::O_CLOEXEC) } < 0 {
|
Ok(pipes) => {
|
||||||
|
already_cloexec = true;
|
||||||
|
pipes
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
FLOG!(warning, PIPE_ERROR);
|
FLOG!(warning, PIPE_ERROR);
|
||||||
perror("pipe2");
|
perror("pipe2");
|
||||||
return None;
|
return Err(err);
|
||||||
}
|
}
|
||||||
already_cloexec = true;
|
};
|
||||||
}
|
|
||||||
#[cfg(not(HAVE_PIPE2))]
|
#[cfg(not(HAVE_PIPE2))]
|
||||||
if unsafe { libc::pipe(&mut pipes[0]) } < 0 {
|
let pipes = match nix::unistd::pipe() {
|
||||||
FLOG!(warning, PIPE_ERROR);
|
Ok(pipes) => pipes,
|
||||||
perror("pipe2");
|
Err(err) => {
|
||||||
return None;
|
FLOG!(warning, PIPE_ERROR);
|
||||||
}
|
perror("pipe2");
|
||||||
|
return Err(err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let readp = unsafe { OwnedFd::from_raw_fd(pipes[0]) };
|
let readp = unsafe { OwnedFd::from_raw_fd(pipes.0) };
|
||||||
let writep = unsafe { OwnedFd::from_raw_fd(pipes[1]) };
|
let writep = unsafe { OwnedFd::from_raw_fd(pipes.1) };
|
||||||
|
|
||||||
// Ensure our fds are out of the user range.
|
// Ensure our fds are out of the user range.
|
||||||
let readp = heightenize_fd(readp, already_cloexec)?;
|
let readp = heightenize_fd(readp, already_cloexec)?;
|
||||||
let writep = heightenize_fd(writep, already_cloexec)?;
|
let writep = heightenize_fd(writep, already_cloexec)?;
|
||||||
|
|
||||||
Some(AutoClosePipes {
|
Ok(AutoClosePipes {
|
||||||
read: readp,
|
read: readp,
|
||||||
write: writep,
|
write: writep,
|
||||||
})
|
})
|
||||||
|
@ -170,23 +173,26 @@ pub fn make_autoclose_pipes() -> Option<AutoClosePipes> {
|
||||||
/// setting it again.
|
/// setting it again.
|
||||||
/// \return the fd, which always has CLOEXEC set; or an invalid fd on failure, in
|
/// \return the fd, which always has CLOEXEC set; or an invalid fd on failure, in
|
||||||
/// which case an error will have been printed, and the input fd closed.
|
/// which case an error will have been printed, and the input fd closed.
|
||||||
fn heightenize_fd(fd: OwnedFd, input_has_cloexec: bool) -> Option<OwnedFd> {
|
fn heightenize_fd(fd: OwnedFd, input_has_cloexec: bool) -> nix::Result<OwnedFd> {
|
||||||
let raw_fd = fd.as_raw_fd();
|
let raw_fd = fd.as_raw_fd();
|
||||||
|
|
||||||
if raw_fd >= FIRST_HIGH_FD {
|
if raw_fd >= FIRST_HIGH_FD {
|
||||||
if !input_has_cloexec {
|
if !input_has_cloexec {
|
||||||
set_cloexec(raw_fd, true);
|
set_cloexec(raw_fd, true);
|
||||||
}
|
}
|
||||||
return Some(fd);
|
return Ok(fd);
|
||||||
}
|
|
||||||
// Here we are asking the kernel to give us a cloexec fd.
|
|
||||||
let newfd = unsafe { libc::fcntl(raw_fd, F_DUPFD_CLOEXEC, FIRST_HIGH_FD) };
|
|
||||||
if newfd < 0 {
|
|
||||||
perror("fcntl");
|
|
||||||
return None;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Some(unsafe { OwnedFd::from_raw_fd(newfd) })
|
// Here we are asking the kernel to give us a cloexec fd.
|
||||||
|
let newfd = match nix::fcntl::fcntl(raw_fd, FcntlArg::F_DUPFD_CLOEXEC(FIRST_HIGH_FD)) {
|
||||||
|
Ok(newfd) => newfd,
|
||||||
|
Err(err) => {
|
||||||
|
perror("fcntl");
|
||||||
|
return Err(err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(unsafe { OwnedFd::from_raw_fd(newfd) })
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets CLO_EXEC on a given fd according to the value of \p should_set.
|
/// Sets CLO_EXEC on a given fd according to the value of \p should_set.
|
||||||
|
@ -292,7 +298,7 @@ fn test_pipes() {
|
||||||
// Note pipe creation may fail due to fd exhaustion; don't fail in that case.
|
// Note pipe creation may fail due to fd exhaustion; don't fail in that case.
|
||||||
let mut pipes = vec![];
|
let mut pipes = vec![];
|
||||||
for _i in 0..10 {
|
for _i in 0..10 {
|
||||||
if let Some(pipe) = make_autoclose_pipes() {
|
if let Ok(pipe) = make_autoclose_pipes() {
|
||||||
pipes.push(pipe);
|
pipes.push(pipe);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,7 @@ use libc::{EAGAIN, EINTR, ENOENT, ENOTDIR, EPIPE, EWOULDBLOCK, STDOUT_FILENO};
|
||||||
use nix::fcntl::OFlag;
|
use nix::fcntl::OFlag;
|
||||||
use nix::sys::stat::Mode;
|
use nix::sys::stat::Mode;
|
||||||
use std::cell::{RefCell, UnsafeCell};
|
use std::cell::{RefCell, UnsafeCell};
|
||||||
|
use std::io;
|
||||||
use std::os::fd::{AsRawFd, IntoRawFd, OwnedFd, RawFd};
|
use std::os::fd::{AsRawFd, IntoRawFd, OwnedFd, RawFd};
|
||||||
use std::sync::atomic::{AtomicU64, Ordering};
|
use std::sync::atomic::{AtomicU64, Ordering};
|
||||||
use std::sync::{Arc, Condvar, Mutex, MutexGuard};
|
use std::sync::{Arc, Condvar, Mutex, MutexGuard};
|
||||||
|
@ -345,14 +346,14 @@ pub struct IoBufferfill {
|
||||||
impl IoBufferfill {
|
impl IoBufferfill {
|
||||||
/// Create an io_bufferfill_t which, when written from, fills a buffer with the contents.
|
/// Create an io_bufferfill_t which, when written from, fills a buffer with the contents.
|
||||||
/// \returns nullptr on failure, e.g. too many open fds.
|
/// \returns nullptr on failure, e.g. too many open fds.
|
||||||
pub fn create() -> Option<Arc<IoBufferfill>> {
|
pub fn create() -> io::Result<Arc<IoBufferfill>> {
|
||||||
Self::create_opts(0, STDOUT_FILENO)
|
Self::create_opts(0, STDOUT_FILENO)
|
||||||
}
|
}
|
||||||
/// Create an io_bufferfill_t which, when written from, fills a buffer with the contents.
|
/// Create an io_bufferfill_t which, when written from, fills a buffer with the contents.
|
||||||
/// \returns nullptr on failure, e.g. too many open fds.
|
/// \returns nullptr on failure, e.g. too many open fds.
|
||||||
///
|
///
|
||||||
/// \param target the fd which this will be dup2'd to - typically stdout.
|
/// \param target the fd which this will be dup2'd to - typically stdout.
|
||||||
pub fn create_opts(buffer_limit: usize, target: RawFd) -> Option<Arc<IoBufferfill>> {
|
pub fn create_opts(buffer_limit: usize, target: RawFd) -> io::Result<Arc<IoBufferfill>> {
|
||||||
assert!(target >= 0, "Invalid target fd");
|
assert!(target >= 0, "Invalid target fd");
|
||||||
|
|
||||||
// Construct our pipes.
|
// Construct our pipes.
|
||||||
|
@ -365,13 +366,13 @@ impl IoBufferfill {
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
FLOG!(warning, PIPE_ERROR);
|
FLOG!(warning, PIPE_ERROR);
|
||||||
perror_io("fcntl", &e);
|
perror_io("fcntl", &e);
|
||||||
return None;
|
return Err(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Our fillthread gets the read end of the pipe; out_pipe gets the write end.
|
// Our fillthread gets the read end of the pipe; out_pipe gets the write end.
|
||||||
let buffer = Arc::new(IoBuffer::new(buffer_limit));
|
let buffer = Arc::new(IoBuffer::new(buffer_limit));
|
||||||
begin_filling(&buffer, pipes.read);
|
begin_filling(&buffer, pipes.read);
|
||||||
Some(Arc::new(IoBufferfill {
|
Ok(Arc::new(IoBufferfill {
|
||||||
target,
|
target,
|
||||||
write_fd: pipes.write,
|
write_fd: pipes.write,
|
||||||
buffer,
|
buffer,
|
||||||
|
|
Loading…
Reference in a new issue