2019-01-28 21:26:22 +00:00
|
|
|
#ifndef FISH_REDIRECTION_H
|
|
|
|
#define FISH_REDIRECTION_H
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2019-01-28 21:26:22 +00:00
|
|
|
#include "common.h"
|
2019-02-16 10:30:21 +00:00
|
|
|
#include "maybe.h"
|
2019-01-28 21:26:22 +00:00
|
|
|
|
2019-12-13 00:44:24 +00:00
|
|
|
/// This file supports specifying and applying redirections.
|
|
|
|
|
|
|
|
enum class redirection_mode_t {
|
|
|
|
overwrite, // normal redirection: > file.txt
|
|
|
|
append, // appending redirection: >> file.txt
|
|
|
|
input, // input redirection: < file.txt
|
|
|
|
fd, // fd redirection: 2>&1
|
|
|
|
noclob // noclobber redirection: >? file.txt
|
|
|
|
};
|
|
|
|
|
|
|
|
class io_chain_t;
|
|
|
|
|
|
|
|
/// A struct which represents a redirection specification from the user.
|
|
|
|
/// Here the file descriptors don't represent open files - it's purely textual.
|
|
|
|
struct redirection_spec_t {
|
|
|
|
/// The redirected fd, or -1 on overflow.
|
|
|
|
/// In the common case of a pipe, this is 1 (STDOUT_FILENO).
|
|
|
|
/// For example, in the case of "3>&1" this will be 3.
|
|
|
|
int fd{-1};
|
|
|
|
|
|
|
|
/// The redirection mode.
|
|
|
|
redirection_mode_t mode{redirection_mode_t::overwrite};
|
|
|
|
|
|
|
|
/// The target of the redirection.
|
|
|
|
/// For example in "3>&1", this will be "1".
|
|
|
|
/// In "< file.txt" this will be "file.txt".
|
|
|
|
wcstring target{};
|
|
|
|
|
|
|
|
/// \return if this is a close-type redirection.
|
|
|
|
bool is_close() const { return mode == redirection_mode_t::fd && target == L"-"; }
|
|
|
|
|
|
|
|
/// Attempt to parse target as an fd. Return the fd, or none() if none.
|
|
|
|
maybe_t<int> get_target_as_fd() const;
|
|
|
|
|
|
|
|
/// \return the open flags for this redirection.
|
|
|
|
int oflags() const;
|
|
|
|
|
|
|
|
redirection_spec_t(int fd, redirection_mode_t mode, wcstring target)
|
|
|
|
: fd(fd), mode(mode), target(std::move(target)) {}
|
|
|
|
};
|
|
|
|
using redirection_spec_list_t = std::vector<redirection_spec_t>;
|
2019-01-28 21:26:22 +00:00
|
|
|
|
|
|
|
/// A class representing a sequence of basic redirections.
|
|
|
|
class dup2_list_t {
|
2019-01-28 22:35:56 +00:00
|
|
|
public:
|
2019-01-28 21:26:22 +00:00
|
|
|
/// A type that represents the action dup2(src, target).
|
|
|
|
/// If target is negative, this represents close(src).
|
|
|
|
/// Note none of the fds here are considered 'owned'.
|
|
|
|
struct action_t {
|
|
|
|
int src;
|
|
|
|
int target;
|
|
|
|
};
|
|
|
|
|
2019-01-28 22:35:56 +00:00
|
|
|
private:
|
2019-01-28 21:26:22 +00:00
|
|
|
/// The list of actions.
|
|
|
|
std::vector<action_t> actions_;
|
|
|
|
|
|
|
|
/// The list of fds that we opened, and are responsible for closing.
|
|
|
|
std::vector<autoclose_fd_t> opened_fds_;
|
|
|
|
|
|
|
|
/// Append a dup2 action.
|
|
|
|
void add_dup2(int src, int target) {
|
|
|
|
assert(src >= 0 && target >= 0 && "Invalid fd in add_dup2");
|
|
|
|
if (src != target) {
|
|
|
|
actions_.push_back(action_t{src, target});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Append a close action.
|
|
|
|
void add_close(int fd) {
|
|
|
|
assert(fd >= 0 && "Invalid fd in add_close");
|
|
|
|
actions_.push_back(action_t{fd, -1});
|
|
|
|
}
|
|
|
|
|
|
|
|
dup2_list_t() = default;
|
|
|
|
|
2019-02-16 10:30:21 +00:00
|
|
|
public:
|
2019-01-28 21:26:22 +00:00
|
|
|
~dup2_list_t();
|
|
|
|
|
|
|
|
/// Disable copying because we own our fds.
|
|
|
|
dup2_list_t(const dup2_list_t &) = delete;
|
|
|
|
void operator=(const dup2_list_t &) = delete;
|
|
|
|
|
|
|
|
dup2_list_t(dup2_list_t &&) = default;
|
|
|
|
dup2_list_t &operator=(dup2_list_t &&) = default;
|
|
|
|
|
|
|
|
/// \return the list of dup2 actions.
|
|
|
|
const std::vector<action_t> &get_actions() const { return actions_; }
|
|
|
|
|
|
|
|
/// Produce a dup_fd_list_t from an io_chain. This may not be called before fork().
|
2019-02-10 06:14:42 +00:00
|
|
|
/// The result contains the list of fd actions (dup2 and close), as well as the list
|
|
|
|
/// of fds opened.
|
2019-01-28 21:26:22 +00:00
|
|
|
static maybe_t<dup2_list_t> resolve_chain(const io_chain_t &);
|
2019-02-02 20:52:51 +00:00
|
|
|
|
|
|
|
/// \return the fd ultimately dup'd to a target fd, or -1 if the target is closed.
|
2019-05-05 10:09:25 +00:00
|
|
|
/// For example, if target fd is 1, and we have a dup2 chain 5->3 and 3->1, then we will
|
|
|
|
/// return 5. If the target is not referenced in the chain, returns target.
|
2019-02-02 20:52:51 +00:00
|
|
|
int fd_for_target_fd(int target) const;
|
2019-01-28 21:26:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|