mirror of
https://github.com/fish-shell/fish-shell.git
synced 2025-01-20 20:59:47 +08:00
4899086b3c
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.
217 lines
4.6 KiB
C++
217 lines
4.6 KiB
C++
#ifndef FISH_IO_H
|
|
#define FISH_IO_H
|
|
|
|
#include <vector>
|
|
#if __cplusplus > 199711L
|
|
// C++11
|
|
#include <memory>
|
|
using std::shared_ptr;
|
|
#else
|
|
// C++03
|
|
#include <tr1/memory>
|
|
using std::tr1::shared_ptr;
|
|
#endif
|
|
|
|
/**
|
|
Describes what type of IO operation an io_data_t represents
|
|
*/
|
|
enum io_mode_t
|
|
{
|
|
IO_FILE, IO_PIPE, IO_FD, IO_BUFFER, IO_CLOSE
|
|
};
|
|
|
|
/** Represents an FD redirection */
|
|
class io_data_t
|
|
{
|
|
private:
|
|
/* No assignment or copying allowed */
|
|
io_data_t(const io_data_t &rhs);
|
|
void operator=(const io_data_t &rhs);
|
|
|
|
protected:
|
|
io_data_t(io_mode_t m, int f) :
|
|
io_mode(m),
|
|
fd(f)
|
|
{
|
|
}
|
|
|
|
public:
|
|
/** Type of redirect */
|
|
const io_mode_t io_mode;
|
|
/** FD to redirect */
|
|
const int fd;
|
|
|
|
virtual void print() const = 0;
|
|
virtual ~io_data_t() = 0;
|
|
};
|
|
|
|
class io_close_t : public io_data_t
|
|
{
|
|
public:
|
|
io_close_t(int f) :
|
|
io_data_t(IO_CLOSE, f)
|
|
{
|
|
}
|
|
|
|
virtual void print() const;
|
|
};
|
|
|
|
class io_fd_t : public io_data_t
|
|
{
|
|
public:
|
|
/** fd to redirect specified fd to */
|
|
const int old_fd;
|
|
/** Whether to close old_fd */
|
|
const bool close_old;
|
|
|
|
virtual void print() const;
|
|
|
|
io_fd_t(int f, int old, bool close = false) :
|
|
io_data_t(IO_FD, f),
|
|
old_fd(old),
|
|
close_old(close)
|
|
{
|
|
}
|
|
};
|
|
|
|
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 */
|
|
const int flags;
|
|
|
|
virtual void print() const;
|
|
|
|
io_file_t(int f, const wcstring &fname, int fl = 0) :
|
|
io_data_t(IO_FILE, f),
|
|
filename_cstr(wcs2str(fname)),
|
|
flags(fl)
|
|
{
|
|
}
|
|
|
|
virtual ~io_file_t()
|
|
{
|
|
free((void *)filename_cstr);
|
|
}
|
|
};
|
|
|
|
class io_pipe_t : public io_data_t
|
|
{
|
|
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;
|
|
}
|
|
|
|
public:
|
|
int pipe_fd[2];
|
|
const bool is_input;
|
|
|
|
virtual void print() const;
|
|
|
|
io_pipe_t(int f, bool i):
|
|
io_data_t(IO_PIPE, f),
|
|
is_input(i)
|
|
{
|
|
pipe_fd[0] = pipe_fd[1] = -1;
|
|
}
|
|
};
|
|
|
|
class io_buffer_t : public io_pipe_t
|
|
{
|
|
private:
|
|
/** buffer to save output in */
|
|
std::vector<char> out_buffer;
|
|
|
|
io_buffer_t(int f, bool i):
|
|
io_pipe_t(IO_BUFFER, f, i),
|
|
out_buffer()
|
|
{
|
|
}
|
|
|
|
public:
|
|
virtual void print() const;
|
|
|
|
virtual ~io_buffer_t();
|
|
|
|
/** Function to append to the buffer */
|
|
void out_buffer_append(const char *ptr, size_t count)
|
|
{
|
|
out_buffer.insert(out_buffer.end(), ptr, ptr + count);
|
|
}
|
|
|
|
/** Function to get a pointer to the buffer */
|
|
char *out_buffer_ptr(void)
|
|
{
|
|
return out_buffer.empty() ? NULL : &out_buffer.at(0);
|
|
}
|
|
|
|
const char *out_buffer_ptr(void) const
|
|
{
|
|
return out_buffer.empty() ? NULL : &out_buffer.at(0);
|
|
}
|
|
|
|
/** Function to get the size of the buffer */
|
|
size_t out_buffer_size(void) const
|
|
{
|
|
return out_buffer.size();
|
|
}
|
|
|
|
/**
|
|
Close output pipe, and read from input pipe until eof.
|
|
*/
|
|
void read();
|
|
|
|
/**
|
|
Create a IO_BUFFER type io redirection, complete with a pipe and a
|
|
vector<char> for output. The default file descriptor used is 1 for
|
|
output buffering and 0 for input buffering.
|
|
|
|
\param is_input set this parameter to zero if the buffer should be
|
|
used to buffer the output of a command, or non-zero to buffer the
|
|
input to a command.
|
|
|
|
\param fd when -1, determined from is_input.
|
|
*/
|
|
static io_buffer_t *create(bool is_input, int fd = -1);
|
|
};
|
|
|
|
class io_chain_t : public std::vector<shared_ptr<io_data_t> >
|
|
{
|
|
public:
|
|
io_chain_t();
|
|
io_chain_t(const shared_ptr<io_data_t> &);
|
|
|
|
void remove(const shared_ptr<const io_data_t> &element);
|
|
void push_back(const shared_ptr<io_data_t> &element);
|
|
void push_front(const shared_ptr<io_data_t> &element);
|
|
void append(const io_chain_t &chain);
|
|
|
|
shared_ptr<const io_data_t> get_io_for_fd(int fd) const;
|
|
shared_ptr<io_data_t> get_io_for_fd(int fd);
|
|
};
|
|
|
|
/**
|
|
Remove the specified io redirection from the chain
|
|
*/
|
|
void io_remove(io_chain_t &list, const shared_ptr<const io_data_t> &element);
|
|
|
|
/** Destroys an io_chain */
|
|
void io_chain_destroy(io_chain_t &chain);
|
|
|
|
/**
|
|
Return the last io redirection in the chain for the specified file descriptor.
|
|
*/
|
|
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);
|
|
|
|
|
|
/** Print debug information about the specified IO redirection chain to stderr. */
|
|
void io_print(const io_chain_t &chain);
|
|
|
|
#endif
|