mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-14 14:03:58 +00:00
Port io
This commit is contained in:
parent
ecb0ab5f34
commit
912f10ceb0
3 changed files with 970 additions and 1 deletions
|
@ -196,7 +196,15 @@ macro_rules! FLOGF {
|
|||
}
|
||||
}
|
||||
|
||||
pub(crate) use {FLOG, FLOGF};
|
||||
macro_rules! should_flog {
|
||||
($category:ident) => {
|
||||
crate::flog::categories::$category
|
||||
.enabled
|
||||
.load(std::sync::atomic::Ordering::Relaxed)
|
||||
};
|
||||
}
|
||||
|
||||
pub(crate) use {should_flog, FLOG, FLOGF};
|
||||
|
||||
/// For each category, if its name matches the wildcard, set its enabled to the given sense.
|
||||
fn apply_one_wildcard(wc_esc: &wstr, sense: bool) {
|
||||
|
|
960
fish-rust/src/io.rs
Normal file
960
fish-rust/src/io.rs
Normal file
|
@ -0,0 +1,960 @@
|
|||
use crate::builtins::shared::{STATUS_CMD_ERROR, STATUS_CMD_OK, STATUS_READ_TOO_MUCH};
|
||||
use crate::common::{str2wcstring, wcs2string, EMPTY_STRING};
|
||||
use crate::fd_monitor::{
|
||||
FdMonitor, FdMonitorItem, FdMonitorItemId, ItemWakeReason, NativeCallback,
|
||||
};
|
||||
use crate::fds::{make_autoclose_pipes, wopen_cloexec, AutoCloseFd, PIPE_ERROR};
|
||||
use crate::ffi;
|
||||
use crate::flog::{should_flog, FLOG, FLOGF};
|
||||
use crate::global_safety::RelaxedAtomicBool;
|
||||
use crate::job_group::JobGroup;
|
||||
use crate::path::path_apply_working_directory;
|
||||
use crate::redirection::{RedirectionMode, RedirectionSpecList};
|
||||
use crate::signal::sigchecker_t;
|
||||
use crate::topic_monitor::topic_t;
|
||||
use crate::wchar::{wstr, WString, L};
|
||||
use crate::wutil::{perror, wdirname, wstat, wwrite_to_fd};
|
||||
use errno::Errno;
|
||||
use libc::{EAGAIN, EEXIST, EINTR, ENOENT, ENOTDIR, EPIPE, EWOULDBLOCK, O_EXCL, STDERR_FILENO};
|
||||
use std::cell::UnsafeCell;
|
||||
use std::sync::{Arc, Condvar, Mutex, MutexGuard, RwLock, RwLockReadGuard};
|
||||
use std::{os::fd::RawFd, rc::Rc};
|
||||
use widestring_suffix::widestrs;
|
||||
|
||||
/// separated_buffer_t represents a buffer of output from commands, prepared to be turned into a
|
||||
/// variable. For example, command substitutions output into one of these. Most commands just
|
||||
/// produce a stream of bytes, and those get stored directly. However other commands produce
|
||||
/// explicitly separated output, in particular `string` like `string collect` and `string split0`.
|
||||
/// The buffer tracks a sequence of elements. Some elements are explicitly separated and should not
|
||||
/// be further split; other elements have inferred separation and may be split by IFS (or not,
|
||||
/// depending on its value).
|
||||
#[derive(Clone, Copy, PartialEq, Eq)]
|
||||
pub enum SeparationType {
|
||||
/// this element should be further separated by IFS
|
||||
inferred,
|
||||
/// this element is explicitly separated and should not be further split
|
||||
explicitly,
|
||||
}
|
||||
|
||||
pub struct BufferElement {
|
||||
contents: Vec<u8>,
|
||||
separation: SeparationType,
|
||||
}
|
||||
|
||||
impl BufferElement {
|
||||
pub fn new(contents: Vec<u8>, separation: SeparationType) -> Self {
|
||||
BufferElement {
|
||||
contents,
|
||||
separation,
|
||||
}
|
||||
}
|
||||
pub fn is_explicitly_separated(&self) -> bool {
|
||||
self.separation == SeparationType::explicitly
|
||||
}
|
||||
}
|
||||
|
||||
/// A separated_buffer_t contains a list of elements, some of which may be separated explicitly and
|
||||
/// others which must be separated further by the user (e.g. via IFS).
|
||||
pub struct SeparatedBuffer {
|
||||
/// Limit on how much data we'll buffer. Zero means no limit.
|
||||
buffer_limit: usize,
|
||||
/// Current size of all contents.
|
||||
contents_size: usize,
|
||||
/// List of buffer elements.
|
||||
elements: Vec<BufferElement>,
|
||||
/// True if we're discarding input because our buffer_limit has been exceeded.
|
||||
discard: bool,
|
||||
}
|
||||
|
||||
impl SeparatedBuffer {
|
||||
pub fn new(limit: usize) -> Self {
|
||||
SeparatedBuffer {
|
||||
buffer_limit: limit,
|
||||
contents_size: 0,
|
||||
elements: vec![],
|
||||
discard: false,
|
||||
}
|
||||
}
|
||||
|
||||
/// \return the buffer limit size, or 0 for no limit.
|
||||
pub fn limit(&self) -> usize {
|
||||
self.buffer_limit
|
||||
}
|
||||
|
||||
/// \return the contents size.
|
||||
pub fn size(&self) -> usize {
|
||||
self.contents_size
|
||||
}
|
||||
|
||||
/// \return whether the output has been discarded.
|
||||
pub fn discarded(&self) -> bool {
|
||||
self.discard
|
||||
}
|
||||
|
||||
/// Serialize the contents to a single string, where explicitly separated elements have a
|
||||
/// newline appended.
|
||||
pub fn newline_serialized(&self) -> Vec<u8> {
|
||||
let mut result = vec![];
|
||||
result.reserve(self.size());
|
||||
for elem in &self.elements {
|
||||
result.extend_from_slice(&elem.contents);
|
||||
if elem.is_explicitly_separated() {
|
||||
result.push(b'\n');
|
||||
}
|
||||
}
|
||||
result
|
||||
}
|
||||
|
||||
/// \return the list of elements.
|
||||
pub fn elements(&self) -> &[BufferElement] {
|
||||
&self.elements
|
||||
}
|
||||
|
||||
/// Append the given data with separation type \p sep.
|
||||
pub fn append(&mut self, data: &[u8], sep: SeparationType) -> bool {
|
||||
if !self.try_add_size(data.len()) {
|
||||
return false;
|
||||
}
|
||||
// Try merging with the last element.
|
||||
if sep == SeparationType::inferred && self.last_inferred() {
|
||||
self.elements
|
||||
.last_mut()
|
||||
.unwrap()
|
||||
.contents
|
||||
.extend_from_slice(data);
|
||||
} else {
|
||||
self.elements.push(BufferElement::new(data.to_vec(), sep));
|
||||
}
|
||||
true
|
||||
}
|
||||
|
||||
/// Remove all elements and unset the discard flag.
|
||||
pub fn clear(&mut self) {
|
||||
self.elements.clear();
|
||||
self.contents_size = 0;
|
||||
self.discard = false;
|
||||
}
|
||||
|
||||
/// \return true if our last element has an inferred separation type.
|
||||
fn last_inferred(&self) -> bool {
|
||||
!self.elements.is_empty() && !self.elements.last().unwrap().is_explicitly_separated()
|
||||
}
|
||||
|
||||
/// If our last element has an inferred separation, return a pointer to it; else nullptr.
|
||||
/// This is useful for appending one inferred separation to another.
|
||||
fn last_if_inferred(&self) -> Option<&BufferElement> {
|
||||
if self.last_inferred() {
|
||||
self.elements.last()
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
/// Mark that we are about to add the given size \p delta to the buffer. \return true if we
|
||||
/// succeed, false if we exceed buffer_limit.
|
||||
fn try_add_size(&mut self, delta: usize) -> bool {
|
||||
if self.discard {
|
||||
return false;
|
||||
}
|
||||
let proposed_size = self.contents_size + delta;
|
||||
if proposed_size < delta || (self.buffer_limit > 0 && proposed_size > self.buffer_limit) {
|
||||
self.clear();
|
||||
self.discard = true;
|
||||
return false;
|
||||
}
|
||||
self.contents_size = proposed_size;
|
||||
true
|
||||
}
|
||||
}
|
||||
|
||||
/// Describes what type of IO operation an io_data_t represents.
|
||||
#[derive(Clone, Copy)]
|
||||
pub enum IoMode {
|
||||
file,
|
||||
pipe,
|
||||
fd,
|
||||
close,
|
||||
bufferfill,
|
||||
}
|
||||
|
||||
/// Represents a FD redirection.
|
||||
pub trait IoData {
|
||||
/// Type of redirect.
|
||||
fn io_mode(&self) -> IoMode;
|
||||
/// FD to redirect.
|
||||
fn fd(&self) -> RawFd;
|
||||
/// Source fd. This is dup2'd to fd, or if it is -1, then fd is closed.
|
||||
/// That is, we call dup2(source_fd, fd).
|
||||
fn source_fd(&self) -> RawFd;
|
||||
fn print(&self);
|
||||
// The address of the object, for comparison.
|
||||
fn as_ptr(&self) -> *const ();
|
||||
}
|
||||
|
||||
pub struct IoClose {
|
||||
fd: RawFd,
|
||||
}
|
||||
impl IoClose {
|
||||
pub fn new(fd: RawFd) -> Self {
|
||||
IoClose { fd }
|
||||
}
|
||||
}
|
||||
impl IoData for IoClose {
|
||||
fn io_mode(&self) -> IoMode {
|
||||
IoMode::close
|
||||
}
|
||||
fn fd(&self) -> RawFd {
|
||||
self.fd
|
||||
}
|
||||
fn source_fd(&self) -> RawFd {
|
||||
-1
|
||||
}
|
||||
fn print(&self) {
|
||||
fwprintf!(STDERR_FILENO, "close %d\n", self.fd)
|
||||
}
|
||||
fn as_ptr(&self) -> *const () {
|
||||
(self as *const Self).cast()
|
||||
}
|
||||
}
|
||||
|
||||
pub struct IoFd {
|
||||
fd: RawFd,
|
||||
source_fd: RawFd,
|
||||
}
|
||||
impl IoFd {
|
||||
/// fd to redirect specified fd to. For example, in 2>&1, source_fd is 1, and io_data_t::fd
|
||||
/// is 2.
|
||||
pub fn new(fd: RawFd, source_fd: RawFd) -> Self {
|
||||
IoFd { fd, source_fd }
|
||||
}
|
||||
}
|
||||
impl IoData for IoFd {
|
||||
fn io_mode(&self) -> IoMode {
|
||||
IoMode::fd
|
||||
}
|
||||
fn fd(&self) -> RawFd {
|
||||
self.fd
|
||||
}
|
||||
fn source_fd(&self) -> RawFd {
|
||||
self.source_fd
|
||||
}
|
||||
fn print(&self) {
|
||||
fwprintf!(STDERR_FILENO, "FD map %d -> %d\n", self.source_fd, self.fd)
|
||||
}
|
||||
fn as_ptr(&self) -> *const () {
|
||||
(self as *const Self).cast()
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents a redirection to or from an opened file.
|
||||
pub struct IoFile {
|
||||
fd: RawFd,
|
||||
// The fd for the file which we are writing to or reading from.
|
||||
file_fd: AutoCloseFd,
|
||||
}
|
||||
impl IoFile {
|
||||
pub fn new(fd: RawFd, file_fd: AutoCloseFd) -> Self {
|
||||
IoFile { fd, file_fd }
|
||||
// Invalid file redirections are replaced with a closed fd, so the following
|
||||
// assertion isn't guaranteed to pass:
|
||||
// assert(file_fd_.valid() && "File is not valid");
|
||||
}
|
||||
}
|
||||
impl IoData for IoFile {
|
||||
fn io_mode(&self) -> IoMode {
|
||||
IoMode::file
|
||||
}
|
||||
fn fd(&self) -> RawFd {
|
||||
self.fd
|
||||
}
|
||||
fn source_fd(&self) -> RawFd {
|
||||
self.file_fd.fd()
|
||||
}
|
||||
fn print(&self) {
|
||||
fwprintf!(STDERR_FILENO, "file %d -> %d\n", self.file_fd.fd(), self.fd)
|
||||
}
|
||||
fn as_ptr(&self) -> *const () {
|
||||
(self as *const Self).cast()
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents (one end) of a pipe.
|
||||
pub struct IoPipe {
|
||||
fd: RawFd,
|
||||
// The pipe's fd. Conceptually this is dup2'd to io_data_t::fd.
|
||||
pipe_fd: AutoCloseFd,
|
||||
/// Whether this is an input pipe. This is used only for informational purposes.
|
||||
is_input: bool,
|
||||
}
|
||||
impl IoPipe {
|
||||
pub fn new(fd: RawFd, is_input: bool, pipe_fd: AutoCloseFd) -> Self {
|
||||
assert!(pipe_fd.is_valid(), "Pipe is not valid");
|
||||
IoPipe {
|
||||
fd,
|
||||
pipe_fd,
|
||||
is_input,
|
||||
}
|
||||
}
|
||||
}
|
||||
impl IoData for IoPipe {
|
||||
fn io_mode(&self) -> IoMode {
|
||||
IoMode::pipe
|
||||
}
|
||||
fn fd(&self) -> RawFd {
|
||||
self.fd
|
||||
}
|
||||
fn source_fd(&self) -> RawFd {
|
||||
self.pipe_fd.fd()
|
||||
}
|
||||
fn print(&self) {
|
||||
fwprintf!(
|
||||
STDERR_FILENO,
|
||||
"pipe {%d} (input: %s) -> %d\n",
|
||||
self.source_fd(),
|
||||
if self.is_input { "yes" } else { "no" },
|
||||
self.fd
|
||||
)
|
||||
}
|
||||
fn as_ptr(&self) -> *const () {
|
||||
(self as *const Self).cast()
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents filling an io_buffer_t. Very similar to io_pipe_t.
|
||||
pub struct IoBufferfill {
|
||||
target: RawFd,
|
||||
|
||||
/// Write end. The other end is connected to an io_buffer_t.
|
||||
write_fd: AutoCloseFd,
|
||||
|
||||
/// The receiving buffer.
|
||||
buffer: Arc<RwLock<IoBuffer>>,
|
||||
}
|
||||
impl IoBufferfill {
|
||||
/// 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.
|
||||
///
|
||||
/// \param target the fd which this will be dup2'd to - typically stdout.
|
||||
pub fn create(buffer_limit: usize, target: RawFd) -> Option<Rc<IoBufferfill>> {
|
||||
assert!(target >= 0, "Invalid target fd");
|
||||
|
||||
// Construct our pipes.
|
||||
let pipes = make_autoclose_pipes()?;
|
||||
// Our buffer will read from the read end of the pipe. This end must be non-blocking. This is
|
||||
// because our fillthread needs to poll to decide if it should shut down, and also accept input
|
||||
// from direct buffer transfers.
|
||||
if ffi::make_fd_nonblocking(autocxx::c_int(pipes.read.fd())).0 != 0 {
|
||||
FLOG!(warning, PIPE_ERROR);
|
||||
perror("fcntl");
|
||||
return None;
|
||||
}
|
||||
// Our fillthread gets the read end of the pipe; out_pipe gets the write end.
|
||||
let mut buffer = Arc::new(RwLock::new(IoBuffer::new(buffer_limit)));
|
||||
begin_filling(&mut buffer, pipes.read);
|
||||
assert!(pipes.write.is_valid(), "fd is not valid");
|
||||
Some(Rc::new(IoBufferfill {
|
||||
target,
|
||||
write_fd: pipes.write,
|
||||
buffer,
|
||||
}))
|
||||
}
|
||||
|
||||
pub fn buffer(&self) -> RwLockReadGuard<'_, IoBuffer> {
|
||||
self.buffer.read().unwrap()
|
||||
}
|
||||
|
||||
/// Reset the receiver (possibly closing the write end of the pipe), and complete the fillthread
|
||||
/// of the buffer. \return the buffer.
|
||||
pub fn finish(filler: IoBufferfill) -> SeparatedBuffer {
|
||||
// The io filler is passed in. This typically holds the only instance of the write side of the
|
||||
// pipe used by the buffer's fillthread (except for that side held by other processes). Get the
|
||||
// buffer out of the bufferfill and clear the shared_ptr; this will typically widow the pipe.
|
||||
// Then allow the buffer to finish.
|
||||
filler
|
||||
.buffer
|
||||
.write()
|
||||
.unwrap()
|
||||
.complete_background_fillthread_and_take_buffer()
|
||||
}
|
||||
}
|
||||
impl IoData for IoBufferfill {
|
||||
fn io_mode(&self) -> IoMode {
|
||||
IoMode::bufferfill
|
||||
}
|
||||
fn fd(&self) -> RawFd {
|
||||
self.target
|
||||
}
|
||||
fn source_fd(&self) -> RawFd {
|
||||
self.write_fd.fd()
|
||||
}
|
||||
fn print(&self) {
|
||||
fwprintf!(
|
||||
STDERR_FILENO,
|
||||
"bufferfill %d -> %d\n",
|
||||
self.write_fd.fd(),
|
||||
self.fd()
|
||||
)
|
||||
}
|
||||
fn as_ptr(&self) -> *const () {
|
||||
(self as *const Self).cast()
|
||||
}
|
||||
}
|
||||
|
||||
/// An io_buffer_t is a buffer which can populate itself by reading from an fd.
|
||||
/// It is not an io_data_t.
|
||||
pub struct IoBuffer {
|
||||
/// Buffer storing what we have read.
|
||||
buffer: Mutex<SeparatedBuffer>,
|
||||
|
||||
/// Atomic flag indicating our fillthread should shut down.
|
||||
shutdown_fillthread: RelaxedAtomicBool,
|
||||
|
||||
/// A promise, allowing synchronization with the background fill operation.
|
||||
/// The operation has a reference to this as well, and fulfills this promise when it exits.
|
||||
fill_waiter: Option<Arc<(Mutex<bool>, Condvar)>>,
|
||||
|
||||
/// The item id of our background fillthread fd monitor item.
|
||||
item_id: FdMonitorItemId,
|
||||
}
|
||||
|
||||
impl IoBuffer {
|
||||
pub fn new(limit: usize) -> Self {
|
||||
IoBuffer {
|
||||
buffer: Mutex::new(SeparatedBuffer::new(limit)),
|
||||
shutdown_fillthread: RelaxedAtomicBool::new(false),
|
||||
fill_waiter: None,
|
||||
item_id: FdMonitorItemId::from(0),
|
||||
}
|
||||
}
|
||||
|
||||
/// Append a string to the buffer.
|
||||
pub fn append(&mut self, data: &[u8], typ: SeparationType) -> bool {
|
||||
self.buffer.lock().unwrap().append(data, typ)
|
||||
}
|
||||
|
||||
/// \return true if output was discarded due to exceeding the read limit.
|
||||
pub fn discarded(&self) -> bool {
|
||||
self.buffer.lock().unwrap().discarded()
|
||||
}
|
||||
|
||||
/// Read some, filling the buffer. The buffer is passed in to enforce that the append lock is
|
||||
/// held. \return positive on success, 0 if closed, -1 on error (in which case errno will be
|
||||
/// set).
|
||||
pub fn read_once(fd: RawFd, buffer: &mut MutexGuard<'_, SeparatedBuffer>) -> isize {
|
||||
assert!(fd >= 0, "Invalid fd");
|
||||
errno::set_errno(Errno(0));
|
||||
let mut bytes = [b'\0'; 4096 * 4];
|
||||
|
||||
// We want to swallow EINTR only; in particular EAGAIN needs to be returned back to the caller.
|
||||
let amt = loop {
|
||||
let amt = unsafe {
|
||||
libc::read(
|
||||
fd,
|
||||
std::ptr::addr_of_mut!(bytes).cast(),
|
||||
std::mem::size_of_val(&bytes),
|
||||
)
|
||||
};
|
||||
if amt < 0 && errno::errno().0 == EINTR {
|
||||
continue;
|
||||
}
|
||||
break amt;
|
||||
};
|
||||
if amt < 0 && ![EAGAIN, EWOULDBLOCK].contains(&errno::errno().0) {
|
||||
perror("read");
|
||||
} else if amt > 0 {
|
||||
buffer.append(
|
||||
&bytes[0..usize::try_from(amt).unwrap()],
|
||||
SeparationType::inferred,
|
||||
);
|
||||
}
|
||||
amt
|
||||
}
|
||||
|
||||
/// End the background fillthread operation, and return the buffer, transferring ownership.
|
||||
pub fn complete_background_fillthread_and_take_buffer(&mut self) -> SeparatedBuffer {
|
||||
// Mark that our fillthread is done, then wake it up.
|
||||
assert!(self.fillthread_running(), "Should have a fillthread");
|
||||
assert!(
|
||||
self.item_id != FdMonitorItemId::from(0),
|
||||
"Should have a valid item ID"
|
||||
);
|
||||
self.shutdown_fillthread.store(true);
|
||||
fd_monitor().poke_item(self.item_id);
|
||||
|
||||
// Wait for the fillthread to fulfill its promise, and then clear the future so we know we no
|
||||
// longer have one.
|
||||
|
||||
let (mutex, condvar) = &**(self.fill_waiter.as_ref().unwrap());
|
||||
{
|
||||
let mut done = mutex.lock().unwrap();
|
||||
while !*done {
|
||||
done = condvar.wait(done).unwrap();
|
||||
}
|
||||
}
|
||||
self.fill_waiter = None;
|
||||
|
||||
// Return our buffer, transferring ownership.
|
||||
let mut locked_buff = self.buffer.lock().unwrap();
|
||||
let mut result = SeparatedBuffer::new(locked_buff.limit());
|
||||
std::mem::swap(&mut result, &mut locked_buff);
|
||||
locked_buff.clear();
|
||||
result
|
||||
}
|
||||
|
||||
/// Helper to return whether the fillthread is running.
|
||||
pub fn fillthread_running(&self) -> bool {
|
||||
return self.fill_waiter.is_some();
|
||||
}
|
||||
}
|
||||
|
||||
/// Begin the fill operation, reading from the given fd in the background.
|
||||
fn begin_filling(iobuffer: &mut Arc<RwLock<IoBuffer>>, fd: AutoCloseFd) {
|
||||
assert!(
|
||||
!iobuffer.read().unwrap().fillthread_running(),
|
||||
"Already have a fillthread"
|
||||
);
|
||||
|
||||
// We want to fill buffer_ by reading from fd. fd is the read end of a pipe; the write end is
|
||||
// owned by another process, or something else writing in fish.
|
||||
// Pass fd to an fd_monitor. It will add fd to its select() loop, and give us a callback when
|
||||
// the fd is readable, or when our item is poked. The usual path is that we will get called
|
||||
// back, read a bit from the fd, and append it to the buffer. Eventually the write end of the
|
||||
// pipe will be closed - probably the other process exited - and fd will be widowed; read() will
|
||||
// then return 0 and we will stop reading.
|
||||
// In exotic circumstances the write end of the pipe will not be closed; this may happen in
|
||||
// e.g.:
|
||||
// cmd ( background & ; echo hi )
|
||||
// Here the background process will inherit the write end of the pipe and hold onto it forever.
|
||||
// In this case, when complete_background_fillthread() is called, the callback will be invoked
|
||||
// with item_wake_reason_t::poke, and we will notice that the shutdown flag is set (this
|
||||
// indicates that the command substitution is done); in this case we will read until we get
|
||||
// EAGAIN and then give up.
|
||||
|
||||
// Construct a promise. We will fulfill it in our fill thread, and wait for it in
|
||||
// complete_background_fillthread(). Note that TSan complains if the promise's dtor races with
|
||||
// the future's call to wait(), so we store the promise, not just its future (#7681).
|
||||
let promise = Arc::new((Mutex::new(false), Condvar::new()));
|
||||
iobuffer.write().unwrap().fill_waiter = Some(promise.clone());
|
||||
|
||||
// Run our function to read until the receiver is closed.
|
||||
// It's OK to capture 'buffer' because 'this' waits for the promise in its dtor.
|
||||
let item_callback: Option<NativeCallback> = {
|
||||
let iobuffer = iobuffer.clone();
|
||||
Some(Box::new(
|
||||
move |fd: &mut AutoCloseFd, reason: ItemWakeReason| {
|
||||
// Only check the shutdown flag if we timed out or were poked.
|
||||
// It's important that if select() indicated we were readable, that we call select() again
|
||||
// allowing it to time out. Note the typical case is that the fd will be closed, in which
|
||||
// case select will return immediately.
|
||||
let mut done = false;
|
||||
if reason == ItemWakeReason::Readable {
|
||||
// select() reported us as readable; read a bit.
|
||||
let iobuf = iobuffer.write().unwrap();
|
||||
let mut buf = iobuf.buffer.lock().unwrap();
|
||||
let ret = IoBuffer::read_once(fd.fd(), &mut buf);
|
||||
done =
|
||||
ret == 0 || (ret < 0 && ![EAGAIN, EWOULDBLOCK].contains(&errno::errno().0));
|
||||
} else if iobuffer.read().unwrap().shutdown_fillthread.load() {
|
||||
// Here our caller asked us to shut down; read while we keep getting data.
|
||||
// This will stop when the fd is closed or if we get EAGAIN.
|
||||
let iobuf = iobuffer.write().unwrap();
|
||||
let mut buf = iobuf.buffer.lock().unwrap();
|
||||
loop {
|
||||
let ret = IoBuffer::read_once(fd.fd(), &mut buf);
|
||||
if ret <= 0 {
|
||||
break;
|
||||
}
|
||||
}
|
||||
done = true;
|
||||
}
|
||||
if done {
|
||||
fd.close();
|
||||
let (mutex, condvar) = &*promise;
|
||||
let mut done = mutex.lock().unwrap();
|
||||
*done = true;
|
||||
condvar.notify_one();
|
||||
}
|
||||
},
|
||||
))
|
||||
};
|
||||
|
||||
iobuffer.write().unwrap().item_id =
|
||||
fd_monitor().add(FdMonitorItem::new(fd, None, item_callback));
|
||||
}
|
||||
|
||||
pub type IoDataRef = Rc<dyn IoData>;
|
||||
|
||||
#[derive(Default)]
|
||||
pub struct IoChain(pub Vec<IoDataRef>);
|
||||
|
||||
impl IoChain {
|
||||
pub fn new() -> Self {
|
||||
Default::default()
|
||||
}
|
||||
pub fn remove(&mut self, element: &IoDataRef) {
|
||||
let element = Rc::as_ptr(element) as *const ();
|
||||
self.0.retain(|e| {
|
||||
let e = Rc::as_ptr(e) as *const ();
|
||||
!std::ptr::eq(e, element)
|
||||
});
|
||||
}
|
||||
pub fn push(&mut self, element: IoDataRef) {
|
||||
self.0.push(element);
|
||||
}
|
||||
pub fn append(&mut self, chain: &IoChain) -> bool {
|
||||
self.0.extend_from_slice(&chain.0);
|
||||
true
|
||||
}
|
||||
|
||||
/// \return the last io redirection in the chain for the specified file descriptor, or nullptr
|
||||
/// if none.
|
||||
pub fn io_for_fd(&self, fd: RawFd) -> Option<IoDataRef> {
|
||||
self.0.iter().rev().find(|data| data.fd() == fd).cloned()
|
||||
}
|
||||
|
||||
/// Attempt to resolve a list of redirection specs to IOs, appending to 'this'.
|
||||
/// \return true on success, false on error, in which case an error will have been printed.
|
||||
#[widestrs]
|
||||
pub fn append_from_specs(&mut self, specs: &RedirectionSpecList, pwd: &wstr) -> bool {
|
||||
let mut have_error = false;
|
||||
for spec in specs {
|
||||
match spec.mode {
|
||||
RedirectionMode::fd => {
|
||||
if spec.is_close() {
|
||||
self.push(Rc::new(IoClose::new(spec.fd)));
|
||||
} else {
|
||||
let target_fd = spec
|
||||
.get_target_as_fd()
|
||||
.expect("fd redirection should have been validated already");
|
||||
self.push(Rc::new(IoFd::new(spec.fd, target_fd)));
|
||||
}
|
||||
}
|
||||
_ => {
|
||||
// We have a path-based redireciton. Resolve it to a file.
|
||||
// Mark it as CLO_EXEC because we don't want it to be open in any child.
|
||||
let path = path_apply_working_directory(&spec.target, pwd);
|
||||
let oflags = spec.oflags();
|
||||
let file = AutoCloseFd::new(wopen_cloexec(&path, oflags, OPEN_MASK));
|
||||
if !file.is_valid() {
|
||||
if (oflags & O_EXCL) != 0 && errno::errno().0 == EEXIST {
|
||||
FLOGF!(warning, NOCLOB_ERROR, spec.target);
|
||||
} else {
|
||||
if should_flog!(warning) {
|
||||
FLOGF!(warning, FILE_ERROR, spec.target);
|
||||
let err = errno::errno().0;
|
||||
// If the error is that the file doesn't exist
|
||||
// or there's a non-directory component,
|
||||
// find the first problematic component for a better message.
|
||||
if [ENOENT, ENOTDIR].contains(&err) {
|
||||
let mut dname = spec.target.clone();
|
||||
while !dname.is_empty() {
|
||||
let next = wdirname(dname.clone());
|
||||
if let Some(md) = wstat(&next) {
|
||||
if !md.is_dir() {
|
||||
FLOGF!(
|
||||
warning,
|
||||
"Path '%ls' is not a directory"L,
|
||||
next
|
||||
);
|
||||
} else {
|
||||
FLOGF!(
|
||||
warning,
|
||||
"Path '%ls' does not exist"L,
|
||||
dname
|
||||
);
|
||||
}
|
||||
break;
|
||||
}
|
||||
dname = next;
|
||||
}
|
||||
} else {
|
||||
perror("open");
|
||||
}
|
||||
}
|
||||
}
|
||||
// If opening a file fails, insert a closed FD instead of the file redirection
|
||||
// and return false. This lets execution potentially recover and at least gives
|
||||
// the shell a chance to gracefully regain control of the shell (see #7038).
|
||||
self.push(Rc::new(IoClose::new(spec.fd)));
|
||||
have_error = true;
|
||||
continue;
|
||||
}
|
||||
self.push(Rc::new(IoFile::new(spec.fd, file)));
|
||||
}
|
||||
}
|
||||
}
|
||||
!have_error
|
||||
}
|
||||
|
||||
/// Output debugging information to stderr.
|
||||
pub fn print(&self) {
|
||||
if self.0.is_empty() {
|
||||
fwprintf!(
|
||||
STDERR_FILENO,
|
||||
"Empty chain %s\n",
|
||||
format!("{:p}", std::ptr::addr_of!(self))
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
fwprintf!(
|
||||
STDERR_FILENO,
|
||||
"Chain %s (%ld items):\n",
|
||||
format!("{:p}", std::ptr::addr_of!(self)),
|
||||
self.0.len()
|
||||
);
|
||||
for (i, io) in self.0.iter().enumerate() {
|
||||
fwprintf!(STDERR_FILENO, "\t%lu: fd:%d, ", i, io.fd());
|
||||
io.print();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Base class representing the output that a builtin can generate.
|
||||
/// This has various subclasses depending on the ultimate output destination.
|
||||
pub trait OutputStream {
|
||||
/// Required override point. The output stream receives a string \p s with \p amt chars.
|
||||
fn append(&mut self, s: &wstr) -> bool;
|
||||
|
||||
/// \return any internally buffered contents.
|
||||
/// This is only implemented for a string_output_stream; others flush data to their underlying
|
||||
/// receiver (fd, or separated buffer) immediately and so will return an empty string here.
|
||||
fn contents(&self) -> &wstr {
|
||||
&EMPTY_STRING
|
||||
}
|
||||
|
||||
/// Flush any unwritten data to the underlying device, and return an error code.
|
||||
/// A 0 code indicates success. The base implementation returns 0.
|
||||
fn flush_and_check_error(&mut self) -> libc::c_int {
|
||||
STATUS_CMD_OK.unwrap()
|
||||
}
|
||||
|
||||
/// An optional override point. This is for explicit separation.
|
||||
/// \param want_newline this is true if the output item should be ended with a newline. This
|
||||
/// is only relevant if we are printing the output to a stream,
|
||||
fn append_with_separation(
|
||||
&mut self,
|
||||
s: &wstr,
|
||||
typ: SeparationType,
|
||||
want_newline: bool,
|
||||
) -> bool {
|
||||
if typ == SeparationType::explicitly && want_newline {
|
||||
// Try calling "append" less - it might write() to an fd
|
||||
let mut buf = s.to_owned();
|
||||
buf.push('\n');
|
||||
self.append(&buf)
|
||||
} else {
|
||||
self.append(s)
|
||||
}
|
||||
}
|
||||
|
||||
fn append_char(&mut self, c: char) -> bool {
|
||||
self.append(wstr::from_char_slice(&[c]))
|
||||
}
|
||||
fn push_back(&mut self, c: char) -> bool {
|
||||
self.append_char(c)
|
||||
}
|
||||
fn push(&mut self, c: char) -> bool {
|
||||
self.append(wstr::from_char_slice(&[c]))
|
||||
}
|
||||
|
||||
// Append data from a narrow buffer, widening it.
|
||||
fn append_narrow_buffer(&mut self, buffer: &SeparatedBuffer) -> bool {
|
||||
for rhs_elem in buffer.elements() {
|
||||
if !self.append_with_separation(
|
||||
&str2wcstring(&rhs_elem.contents),
|
||||
rhs_elem.separation,
|
||||
false,
|
||||
) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
true
|
||||
}
|
||||
}
|
||||
|
||||
/// A null output stream which ignores all writes.
|
||||
pub struct NullOutputStream {}
|
||||
|
||||
impl OutputStream for NullOutputStream {
|
||||
fn append(&mut self, _s: &wstr) -> bool {
|
||||
true
|
||||
}
|
||||
}
|
||||
|
||||
/// An output stream for builtins which outputs to an fd.
|
||||
/// Note the fd may be something like stdout; there is no ownership implied here.
|
||||
pub struct FdOutputStream {
|
||||
/// The file descriptor to write to.
|
||||
fd: RawFd,
|
||||
|
||||
/// Used to check if a SIGINT has been received when EINTR is encountered
|
||||
sigcheck: sigchecker_t,
|
||||
|
||||
/// Whether we have received an error.
|
||||
errored: bool,
|
||||
}
|
||||
impl FdOutputStream {
|
||||
/// Construct from a file descriptor, which must be nonegative.
|
||||
pub fn new(fd: RawFd) -> Self {
|
||||
assert!(fd >= 0, "Invalid fd");
|
||||
FdOutputStream {
|
||||
fd,
|
||||
sigcheck: sigchecker_t::new(topic_t::sighupint),
|
||||
errored: false,
|
||||
}
|
||||
}
|
||||
}
|
||||
impl OutputStream for FdOutputStream {
|
||||
fn append(&mut self, s: &wstr) -> bool {
|
||||
if self.errored {
|
||||
return false;
|
||||
}
|
||||
if wwrite_to_fd(s, self.fd).is_none() {
|
||||
// Some of our builtins emit multiple screens worth of data sent to a pager (the primary
|
||||
// example being the `history` builtin) and receiving SIGINT should be considered normal and
|
||||
// non-exceptional (user request to abort via Ctrl-C), meaning we shouldn't print an error.
|
||||
if errno::errno().0 == EINTR && self.sigcheck.check() {
|
||||
// We have two options here: we can either return false without setting errored_ to
|
||||
// true (*this* write will be silently aborted but the onus is on the caller to check
|
||||
// the return value and skip future calls to `append()`) or we can flag the entire
|
||||
// output stream as errored, causing us to both return false and skip any future writes.
|
||||
// We're currently going with the latter, especially seeing as no callers currently
|
||||
// check the result of `append()` (since it was always a void function before).
|
||||
} else if errno::errno().0 != EPIPE {
|
||||
perror("write");
|
||||
}
|
||||
self.errored = true;
|
||||
}
|
||||
!self.errored
|
||||
}
|
||||
|
||||
fn flush_and_check_error(&mut self) -> libc::c_int {
|
||||
// Return a generic 1 on any write failure.
|
||||
if self.errored {
|
||||
STATUS_CMD_ERROR
|
||||
} else {
|
||||
STATUS_CMD_OK
|
||||
}
|
||||
.unwrap()
|
||||
}
|
||||
}
|
||||
|
||||
/// A simple output stream which buffers into a wcstring.
|
||||
#[derive(Default)]
|
||||
pub struct StringOutputStream {
|
||||
contents: WString,
|
||||
}
|
||||
impl OutputStream for StringOutputStream {
|
||||
fn append(&mut self, s: &wstr) -> bool {
|
||||
self.contents.push_utfstr(s);
|
||||
true
|
||||
}
|
||||
/// \return the wcstring containing the output.
|
||||
fn contents(&self) -> &wstr {
|
||||
&self.contents
|
||||
}
|
||||
}
|
||||
|
||||
/// An output stream for builtins which writes into a separated buffer.
|
||||
pub struct BufferedOutputStream {
|
||||
/// The buffer we are filling.
|
||||
buffer: Arc<RwLock<IoBuffer>>,
|
||||
}
|
||||
impl BufferedOutputStream {
|
||||
pub fn new(buffer: Arc<RwLock<IoBuffer>>) -> Self {
|
||||
Self { buffer }
|
||||
}
|
||||
}
|
||||
impl OutputStream for BufferedOutputStream {
|
||||
fn append(&mut self, s: &wstr) -> bool {
|
||||
self.buffer
|
||||
.write()
|
||||
.unwrap()
|
||||
.append(&wcs2string(s), SeparationType::inferred)
|
||||
}
|
||||
fn append_with_separation(
|
||||
&mut self,
|
||||
s: &wstr,
|
||||
typ: SeparationType,
|
||||
_want_newline: bool,
|
||||
) -> bool {
|
||||
self.buffer.write().unwrap().append(&wcs2string(s), typ)
|
||||
}
|
||||
fn flush_and_check_error(&mut self) -> libc::c_int {
|
||||
if self.buffer.read().unwrap().discarded() {
|
||||
return STATUS_READ_TOO_MUCH.unwrap();
|
||||
}
|
||||
0
|
||||
}
|
||||
}
|
||||
|
||||
pub struct IoStreams<'a> {
|
||||
// Streams for out and err.
|
||||
pub out: &'a dyn OutputStream,
|
||||
pub err: &'a dyn OutputStream,
|
||||
|
||||
// fd representing stdin. This is not closed by the destructor.
|
||||
// Note: if stdin is explicitly closed by `<&-` then this is -1!
|
||||
pub stdin_fd: RawFd,
|
||||
|
||||
// Whether stdin is "directly redirected," meaning it is the recipient of a pipe (foo | cmd) or
|
||||
// direct redirection (cmd < foo.txt). An "indirect redirection" would be e.g.
|
||||
// begin ; cmd ; end < foo.txt
|
||||
// If stdin is closed (cmd <&-) this is false.
|
||||
pub stdin_is_directly_redirected: bool,
|
||||
|
||||
// Indicates whether stdout and stderr are specifically piped.
|
||||
// If this is set, then the is_redirected flags must also be set.
|
||||
pub out_is_piped: bool,
|
||||
pub err_is_piped: bool,
|
||||
|
||||
// Indicates whether stdout and stderr are at all redirected (e.g. to a file or piped).
|
||||
pub out_is_redirected: bool,
|
||||
pub err_is_redirected: bool,
|
||||
|
||||
// Actual IO redirections. This is only used by the source builtin. Unowned.
|
||||
io_chain: *const IoChain,
|
||||
|
||||
// The job group of the job, if any. This enables builtins which run more code like eval() to
|
||||
// share pgid.
|
||||
// FIXME: this is awkwardly placed.
|
||||
job_group: Option<Rc<JobGroup>>,
|
||||
}
|
||||
|
||||
impl<'a> IoStreams<'a> {
|
||||
pub fn new(out: &'a dyn OutputStream, err: &'a dyn OutputStream) -> Self {
|
||||
IoStreams {
|
||||
out,
|
||||
err,
|
||||
stdin_fd: -1,
|
||||
stdin_is_directly_redirected: false,
|
||||
out_is_piped: false,
|
||||
err_is_piped: false,
|
||||
out_is_redirected: false,
|
||||
err_is_redirected: false,
|
||||
io_chain: std::ptr::null(),
|
||||
job_group: None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// File redirection error message.
|
||||
const FILE_ERROR: &wstr = L!("An error occurred while redirecting file '%ls'");
|
||||
const NOCLOB_ERROR: &wstr = L!("The file '%ls' already exists");
|
||||
|
||||
/// Base open mode to pass to calls to open.
|
||||
const OPEN_MASK: libc::c_int = 0o666;
|
||||
|
||||
/// Provide the fd monitor used for background fillthread operations.
|
||||
fn fd_monitor() -> &'static mut FdMonitor {
|
||||
// Deliberately leaked to avoid shutdown dtors.
|
||||
static mut FDM: *const UnsafeCell<FdMonitor> = std::ptr::null();
|
||||
unsafe {
|
||||
if FDM.is_null() {
|
||||
FDM = Box::into_raw(Box::new(UnsafeCell::new(FdMonitor::new())))
|
||||
}
|
||||
}
|
||||
let ptr: *mut FdMonitor = unsafe { (*FDM).get() };
|
||||
unsafe { &mut *ptr }
|
||||
}
|
|
@ -32,6 +32,7 @@ mod ffi_tests;
|
|||
mod flog;
|
||||
mod future_feature_flags;
|
||||
mod global_safety;
|
||||
mod io;
|
||||
mod job_group;
|
||||
mod locale;
|
||||
mod nix;
|
||||
|
|
Loading…
Reference in a new issue