2005-10-08 11:20:51 +00:00
|
|
|
#ifndef FISH_IO_H
|
|
|
|
#define FISH_IO_H
|
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
#include <stdarg.h>
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
2017-02-14 04:37:27 +00:00
|
|
|
|
2019-02-01 09:58:06 +00:00
|
|
|
#include <vector>
|
2019-02-03 01:53:40 +00:00
|
|
|
// Note that we have to include something to get any _LIBCPP_VERSION defined so we can detect libc++
|
|
|
|
// So it's key that vector go above. If we didn't need vector for other reasons, we might include
|
|
|
|
// ciso646, which does nothing
|
|
|
|
#if defined(_LIBCPP_VERSION) || __cplusplus > 199711L
|
|
|
|
// C++11 or libc++ (which is a C++11-only library, but the memory header works OK in C++03)
|
|
|
|
#include <memory>
|
|
|
|
using std::shared_ptr;
|
|
|
|
#else
|
|
|
|
// C++03 or libstdc++
|
|
|
|
#include <tr1/memory>
|
|
|
|
using std::tr1::shared_ptr;
|
|
|
|
#endif
|
2012-03-04 10:35:30 +00:00
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include "common.h"
|
2017-07-27 03:17:04 +00:00
|
|
|
#include "env.h"
|
2015-07-25 15:14:25 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
/// separated_buffer_t is composed of a sequence of elements, some of which may be explicitly
|
|
|
|
/// separated (e.g. through string spit0) and some of which the separation is inferred. This enum
|
|
|
|
/// tracks the type.
|
|
|
|
enum class separation_type_t {
|
|
|
|
/// This element's separation should be inferred, e.g. through IFS.
|
|
|
|
inferred,
|
|
|
|
/// This element was explicitly separated and should not be separated further.
|
|
|
|
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).
|
|
|
|
template <typename StringType>
|
|
|
|
class separated_buffer_t {
|
2018-05-30 04:11:50 +00:00
|
|
|
public:
|
2018-05-30 04:11:34 +00:00
|
|
|
struct element_t {
|
|
|
|
StringType contents;
|
|
|
|
separation_type_t separation;
|
|
|
|
|
|
|
|
element_t(StringType contents, separation_type_t sep)
|
|
|
|
: contents(std::move(contents)), separation(sep) {}
|
|
|
|
|
|
|
|
bool is_explicitly_separated() const { return separation == separation_type_t::explicitly; }
|
|
|
|
};
|
|
|
|
|
2018-05-30 04:11:50 +00:00
|
|
|
private:
|
2018-05-30 04:11:34 +00:00
|
|
|
/// Limit on how much data we'll buffer. Zero means no limit.
|
|
|
|
size_t buffer_limit_;
|
|
|
|
|
|
|
|
/// Current size of all contents.
|
|
|
|
size_t contents_size_{0};
|
|
|
|
|
|
|
|
/// List of buffer elements.
|
|
|
|
std::vector<element_t> elements_;
|
|
|
|
|
|
|
|
/// True if we're discarding input because our buffer_limit has been exceeded.
|
|
|
|
bool discard = false;
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
bool try_add_size(size_t delta) {
|
|
|
|
if (discard) return false;
|
|
|
|
contents_size_ += delta;
|
|
|
|
if (contents_size_ < delta) {
|
|
|
|
// Overflow!
|
|
|
|
set_discard();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (buffer_limit_ > 0 && contents_size_ > buffer_limit_) {
|
|
|
|
set_discard();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// separated_buffer_t may not be copied.
|
|
|
|
separated_buffer_t(const separated_buffer_t &) = delete;
|
|
|
|
void operator=(const separated_buffer_t &) = delete;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// Construct a separated_buffer_t with the given buffer limit \p limit, or 0 for no limit.
|
|
|
|
separated_buffer_t(size_t limit) : buffer_limit_(limit) {}
|
|
|
|
|
|
|
|
/// \return the buffer limit size, or 0 for no limit.
|
|
|
|
size_t limit() const { return buffer_limit_; }
|
|
|
|
|
|
|
|
/// \return the contents size.
|
|
|
|
size_t size() const { return contents_size_; }
|
|
|
|
|
|
|
|
/// \return whether the output has been discarded.
|
|
|
|
bool discarded() const { return discard; }
|
|
|
|
|
|
|
|
/// Mark the contents as discarded.
|
|
|
|
void set_discard() {
|
|
|
|
elements_.clear();
|
|
|
|
contents_size_ = 0;
|
|
|
|
discard = true;
|
|
|
|
}
|
|
|
|
|
2018-10-24 14:59:24 +00:00
|
|
|
void reset_discard() {
|
|
|
|
discard = false;
|
|
|
|
}
|
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
/// Serialize the contents to a single string, where explicitly separated elements have a
|
|
|
|
/// newline appended.
|
|
|
|
StringType newline_serialized() const {
|
|
|
|
StringType result;
|
|
|
|
result.reserve(size());
|
|
|
|
for (const auto &elem : elements_) {
|
|
|
|
result.append(elem.contents);
|
|
|
|
if (elem.is_explicitly_separated()) {
|
|
|
|
result.push_back('\n');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \return the list of elements.
|
|
|
|
const std::vector<element_t> &elements() const { return elements_; }
|
|
|
|
|
|
|
|
/// Append an element with range [begin, end) and the given separation type \p sep.
|
|
|
|
template <typename Iterator>
|
|
|
|
void append(Iterator begin, Iterator end, separation_type_t sep = separation_type_t::inferred) {
|
|
|
|
if (!try_add_size(std::distance(begin, end))) return;
|
|
|
|
// Try merging with the last element.
|
|
|
|
if (sep == separation_type_t::inferred && !elements_.empty() && !elements_.back().is_explicitly_separated()) {
|
|
|
|
elements_.back().contents.append(begin, end);
|
|
|
|
} else {
|
|
|
|
elements_.emplace_back(StringType(begin, end), sep);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Append a string \p str with the given separation type \p sep.
|
|
|
|
void append(const StringType &str, separation_type_t sep = separation_type_t::inferred) {
|
|
|
|
append(str.begin(), str.end(), sep);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Given that this is a narrow stream, convert a wide stream \p rhs to narrow and then append
|
|
|
|
// it.
|
|
|
|
template <typename RHSStringType>
|
|
|
|
void append_wide_buffer(const separated_buffer_t<RHSStringType> &rhs) {
|
|
|
|
for (const auto &rhs_elem : rhs.elements()) {
|
|
|
|
append(wcs2string(rhs_elem.contents), rhs_elem.separation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
/// Describes what type of IO operation an io_data_t represents.
|
2019-02-03 01:53:40 +00:00
|
|
|
enum class io_mode_t { file, pipe, fd, buffer, close };
|
2005-10-08 11:20:51 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
/// Represents an FD redirection.
|
|
|
|
class io_data_t {
|
|
|
|
private:
|
|
|
|
// No assignment or copying allowed.
|
2013-01-07 15:04:55 +00:00
|
|
|
io_data_t(const io_data_t &rhs);
|
2012-08-05 20:24:33 +00:00
|
|
|
void operator=(const io_data_t &rhs);
|
2013-01-04 10:03:41 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
protected:
|
|
|
|
io_data_t(io_mode_t m, int f) : io_mode(m), fd(f) {}
|
2013-01-15 09:39:20 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
public:
|
|
|
|
/// Type of redirect.
|
2013-01-17 06:46:02 +00:00
|
|
|
const io_mode_t io_mode;
|
2016-05-02 03:32:40 +00:00
|
|
|
/// FD to redirect.
|
2013-02-04 12:07:16 +00:00
|
|
|
const int fd;
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2013-01-15 09:39:20 +00:00
|
|
|
virtual void print() const = 0;
|
|
|
|
virtual ~io_data_t() = 0;
|
2012-02-10 02:43:36 +00:00
|
|
|
};
|
2013-01-09 08:02:04 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
class io_close_t : public io_data_t {
|
|
|
|
public:
|
2019-01-31 20:12:46 +00:00
|
|
|
explicit io_close_t(int f) : io_data_t(io_mode_t::close, f) {}
|
2013-01-09 08:02:04 +00:00
|
|
|
|
2018-02-19 02:50:35 +00:00
|
|
|
void print() const override;
|
2013-01-09 08:02:04 +00:00
|
|
|
};
|
2013-01-15 07:37:33 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
class io_fd_t : public io_data_t {
|
|
|
|
public:
|
|
|
|
/// fd to redirect specified fd to. For example, in 2>&1, old_fd is 1, and io_data_t::fd is 2.
|
2013-02-04 12:14:48 +00:00
|
|
|
const int old_fd;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
|
|
|
/// Whether this redirection was supplied by a script. For example, 'cmd <&3' would have
|
|
|
|
/// user_supplied set to true. But a redirection that comes about through transmogrification
|
|
|
|
/// would not.
|
2015-01-08 18:44:05 +00:00
|
|
|
const bool user_supplied;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-02-19 02:50:35 +00:00
|
|
|
void print() const override;
|
2013-01-15 07:37:33 +00:00
|
|
|
|
2019-01-31 20:12:46 +00:00
|
|
|
io_fd_t(int f, int old, bool us)
|
|
|
|
: io_data_t(io_mode_t::fd, f), old_fd(old), user_supplied(us) {}
|
2013-01-15 07:37:33 +00:00
|
|
|
};
|
2013-01-15 08:18:03 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
class io_file_t : public io_data_t {
|
|
|
|
public:
|
|
|
|
/// Filename, malloc'd. This needs to be used after fork, so don't use wcstring here.
|
|
|
|
const char *const filename_cstr;
|
|
|
|
/// file creation flags to send to open.
|
2013-02-04 12:15:32 +00:00
|
|
|
const int flags;
|
2013-01-21 02:34:18 +00:00
|
|
|
|
2018-02-19 02:50:35 +00:00
|
|
|
void print() const override;
|
2013-01-15 08:18:03 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
io_file_t(int f, const wcstring &fname, int fl = 0)
|
2019-01-31 20:12:46 +00:00
|
|
|
: io_data_t(io_mode_t::file, f), filename_cstr(wcs2str(fname)), flags(fl) {}
|
2013-01-15 08:18:03 +00:00
|
|
|
|
2018-02-19 02:50:35 +00:00
|
|
|
~io_file_t() override { free((void *)filename_cstr); }
|
2013-01-15 08:18:03 +00:00
|
|
|
};
|
2005-10-08 11:20:51 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
class io_pipe_t : public io_data_t {
|
2019-02-03 01:53:40 +00:00
|
|
|
protected:
|
|
|
|
io_pipe_t(io_mode_t m, int f, bool i) : io_data_t(m, f), is_input(i) {
|
|
|
|
pipe_fd[0] = pipe_fd[1] = -1;
|
|
|
|
}
|
2013-01-15 09:31:36 +00:00
|
|
|
|
2019-02-01 00:05:42 +00:00
|
|
|
public:
|
2019-02-03 01:53:40 +00:00
|
|
|
int pipe_fd[2];
|
|
|
|
const bool is_input;
|
2013-01-15 09:31:36 +00:00
|
|
|
|
2019-02-01 00:05:42 +00:00
|
|
|
void print() const override;
|
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
io_pipe_t(int f, bool i) : io_data_t(io_mode_t::pipe, f), is_input(i) {
|
|
|
|
pipe_fd[0] = pipe_fd[1] = -1;
|
|
|
|
}
|
2019-02-01 00:05:42 +00:00
|
|
|
};
|
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
class io_chain_t;
|
2018-05-28 06:56:20 +00:00
|
|
|
class output_stream_t;
|
2019-02-03 01:53:40 +00:00
|
|
|
class io_buffer_t : public io_pipe_t {
|
2016-05-02 03:32:40 +00:00
|
|
|
private:
|
2018-05-28 08:27:26 +00:00
|
|
|
separated_buffer_t<std::string> buffer_;
|
2013-01-15 08:44:31 +00:00
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
explicit io_buffer_t(int f, size_t limit)
|
|
|
|
: io_pipe_t(io_mode_t::buffer, f, false /* not input */), buffer_(limit) {
|
2018-10-24 14:59:24 +00:00
|
|
|
// Explicitly reset the discard flag because we share this buffer.
|
|
|
|
buffer_.reset_discard();
|
|
|
|
}
|
2013-01-15 08:44:31 +00:00
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
public:
|
|
|
|
void print() const override;
|
|
|
|
|
|
|
|
~io_buffer_t() override;
|
2013-01-15 09:02:46 +00:00
|
|
|
|
2018-05-28 08:27:26 +00:00
|
|
|
/// Access the underlying buffer.
|
2019-02-03 01:53:40 +00:00
|
|
|
const separated_buffer_t<std::string> &buffer() const { return buffer_; }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-28 08:27:26 +00:00
|
|
|
/// Function to append to the buffer.
|
2019-02-03 01:53:40 +00:00
|
|
|
void append(const char *ptr, size_t count) { buffer_.append(ptr, ptr + count); }
|
|
|
|
|
|
|
|
/// Ensures that the pipes do not conflict with any fd redirections in the chain.
|
|
|
|
bool avoid_conflicts_with_io_chain(const io_chain_t &ios);
|
2013-01-15 09:07:30 +00:00
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
/// Close output pipe, and read from input pipe until eof.
|
|
|
|
void read();
|
2013-01-15 09:10:40 +00:00
|
|
|
|
2018-05-28 06:56:20 +00:00
|
|
|
/// Appends data from a given output_stream_t.
|
|
|
|
/// Marks the receiver as discarded if the stream was discarded.
|
|
|
|
void append_from_stream(const output_stream_t &stream);
|
2019-02-03 01:53:40 +00:00
|
|
|
|
|
|
|
/// Create a io_mode_t::buffer type io redirection, complete with a pipe and a vector<char> for
|
|
|
|
/// output. The default file descriptor used is STDOUT_FILENO for buffering.
|
|
|
|
///
|
|
|
|
/// \param fd the fd that will be mapped in the child process, typically STDOUT_FILENO
|
|
|
|
/// \param conflicts A set of IO redirections. The function ensures that any pipe it makes does
|
|
|
|
/// not conflict with an fd redirection in this list.
|
|
|
|
static shared_ptr<io_buffer_t> create(int fd, const io_chain_t &conflicts,
|
|
|
|
size_t buffer_limit = 0);
|
2013-01-15 08:44:31 +00:00
|
|
|
};
|
|
|
|
|
2019-02-01 02:49:52 +00:00
|
|
|
class io_chain_t : public std::vector<shared_ptr<io_data_t>> {
|
2016-05-02 03:32:40 +00:00
|
|
|
public:
|
2019-02-01 02:49:52 +00:00
|
|
|
using std::vector<shared_ptr<io_data_t>>::vector;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2013-01-07 15:04:55 +00:00
|
|
|
void remove(const shared_ptr<const io_data_t> &element);
|
2019-02-01 02:49:52 +00:00
|
|
|
void push_back(shared_ptr<io_data_t> element);
|
|
|
|
void push_front(shared_ptr<io_data_t> element);
|
Big fat refactoring of how redirections work. In fish 1.x and 2.0.0, the redirections for a process were flattened into a big list associated with the job, so there was no way to tell which redirections applied to each process. Each process therefore got all the redirections associated with the job. See https://github.com/fish-shell/fish-shell/issues/877 for how this could manifest.
With this change, jobs only track their block-level redirections. Process level redirections are correctly associated with the process, and at exec time we stitch them together (block, pipe, and process redirects).
This fixes the weird issues where redirects bleed across pipelines (like #877), and also allows us to play with the order in which redirections are applied, since the final list is constructed right before it's needed. This lets us put pipes after block level redirections but before process level redirections, so that a 2>&1-type redirection gets picked up after the pipe, i.e. it should fix https://github.com/fish-shell/fish-shell/issues/110
This is a significant change. The tests all pass. Cross your fingers.
2013-08-19 23:16:41 +00:00
|
|
|
void append(const io_chain_t &chain);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2013-01-07 15:04:55 +00:00
|
|
|
shared_ptr<const io_data_t> get_io_for_fd(int fd) const;
|
|
|
|
shared_ptr<io_data_t> get_io_for_fd(int fd);
|
2012-08-15 07:57:56 +00:00
|
|
|
};
|
2005-10-08 11:20:51 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
/// Return the last io redirection in the chain for the specified file descriptor.
|
2013-01-07 15:04:55 +00:00
|
|
|
shared_ptr<const io_data_t> io_chain_get(const io_chain_t &src, int fd);
|
|
|
|
shared_ptr<io_data_t> io_chain_get(io_chain_t &src, int fd);
|
2005-10-08 11:20:51 +00:00
|
|
|
|
2019-02-03 01:53:40 +00:00
|
|
|
/// Given a pair of fds, if an fd is used by the given io chain, duplicate that fd repeatedly until
|
|
|
|
/// we find one that does not conflict, or we run out of fds. Returns the new fds by reference,
|
|
|
|
/// closing the old ones. If we get an error, returns false (in which case both fds are closed and
|
|
|
|
/// set to -1).
|
|
|
|
bool pipe_avoid_conflicts_with_io_chain(int fds[2], const io_chain_t &ios);
|
2019-01-28 21:26:22 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
/// Class representing the output that a builtin can generate.
|
|
|
|
class output_stream_t {
|
|
|
|
private:
|
2018-05-30 04:11:34 +00:00
|
|
|
/// Storage for our data.
|
|
|
|
separated_buffer_t<wcstring> buffer_;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
// No copying.
|
|
|
|
output_stream_t(const output_stream_t &s) = delete;
|
|
|
|
void operator=(const output_stream_t &s) = delete;
|
2017-07-27 03:17:04 +00:00
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
public:
|
2018-05-30 04:11:34 +00:00
|
|
|
output_stream_t(size_t buffer_limit) : buffer_(buffer_limit) {}
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
void append(const wcstring &s) { buffer_.append(s.begin(), s.end()); }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:50 +00:00
|
|
|
separated_buffer_t<wcstring> &buffer() { return buffer_; }
|
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
const separated_buffer_t<wcstring> &buffer() const { return buffer_; }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
void append(const wchar_t *s) { append(s, wcslen(s)); }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
void append(wchar_t s) { append(&s, 1); }
|
2017-07-27 03:17:04 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
void append(const wchar_t *s, size_t amt) { buffer_.append(s, s + amt); }
|
|
|
|
|
|
|
|
void push_back(wchar_t c) { append(c); }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
|
|
|
void append_format(const wchar_t *format, ...) {
|
2015-09-21 18:24:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, format);
|
2018-05-30 04:11:34 +00:00
|
|
|
append_formatv(format, va);
|
2015-09-21 18:24:49 +00:00
|
|
|
va_end(va);
|
|
|
|
}
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
void append_formatv(const wchar_t *format, va_list va) { append(vformat_string(format, va)); }
|
2017-07-27 03:17:04 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
bool empty() const { return buffer_.size() == 0; }
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2018-05-30 04:11:34 +00:00
|
|
|
wcstring contents() const { return buffer_.newline_serialized(); }
|
2015-09-21 18:24:49 +00:00
|
|
|
};
|
|
|
|
|
2016-05-02 03:32:40 +00:00
|
|
|
struct io_streams_t {
|
2015-09-21 18:24:49 +00:00
|
|
|
output_stream_t out;
|
|
|
|
output_stream_t err;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2015-09-21 18:24:49 +00:00
|
|
|
// fd representing stdin. This is not closed by the destructor.
|
|
|
|
int stdin_fd;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
|
|
|
// 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
|
2015-09-25 21:17:53 +00:00
|
|
|
bool stdin_is_directly_redirected;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
|
|
|
// Indicates whether stdout and stderr are redirected (e.g. to a file or piped).
|
2015-09-21 18:24:49 +00:00
|
|
|
bool out_is_redirected;
|
|
|
|
bool err_is_redirected;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2015-09-21 18:24:49 +00:00
|
|
|
// Actual IO redirections. This is only used by the source builtin. Unowned.
|
|
|
|
const io_chain_t *io_chain;
|
2016-05-02 03:32:40 +00:00
|
|
|
|
2017-07-27 03:17:04 +00:00
|
|
|
io_streams_t(size_t read_limit)
|
|
|
|
: out(read_limit),
|
|
|
|
err(read_limit),
|
|
|
|
stdin_fd(-1),
|
2016-05-02 03:32:40 +00:00
|
|
|
stdin_is_directly_redirected(false),
|
|
|
|
out_is_redirected(false),
|
|
|
|
err_is_redirected(false),
|
|
|
|
io_chain(NULL) {}
|
2015-09-21 18:24:49 +00:00
|
|
|
};
|
|
|
|
|
2016-04-04 02:02:46 +00:00
|
|
|
#if 0
|
|
|
|
// Print debug information about the specified IO redirection chain to stderr.
|
2012-11-19 00:30:30 +00:00
|
|
|
void io_print(const io_chain_t &chain);
|
2016-04-04 02:02:46 +00:00
|
|
|
#endif
|
2006-05-14 22:29:05 +00:00
|
|
|
|
2005-10-08 11:20:51 +00:00
|
|
|
#endif
|