2019-01-28 21:26:22 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
|
|
|
#include "redirection.h"
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
2019-12-13 00:44:24 +00:00
|
|
|
#include "io.h"
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "wutil.h"
|
|
|
|
|
2019-01-28 21:26:22 +00:00
|
|
|
dup2_list_t::~dup2_list_t() = default;
|
|
|
|
|
2019-12-13 00:44:24 +00:00
|
|
|
maybe_t<int> redirection_spec_t::get_target_as_fd() const {
|
|
|
|
errno = 0;
|
|
|
|
int result = fish_wcstoi(target.c_str());
|
|
|
|
if (errno || result < 0) return none();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int redirection_spec_t::oflags() const {
|
|
|
|
switch (mode) {
|
|
|
|
case redirection_mode_t::append:
|
|
|
|
return O_CREAT | O_APPEND | O_WRONLY;
|
|
|
|
case redirection_mode_t::overwrite:
|
|
|
|
return O_CREAT | O_WRONLY | O_TRUNC;
|
|
|
|
case redirection_mode_t::noclob:
|
|
|
|
return O_CREAT | O_EXCL | O_WRONLY;
|
|
|
|
case redirection_mode_t::input:
|
|
|
|
return O_RDONLY;
|
|
|
|
case redirection_mode_t::fd:
|
2019-12-14 02:42:08 +00:00
|
|
|
default:
|
2019-12-13 00:44:24 +00:00
|
|
|
DIE("Not a file redirection");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-29 22:49:05 +00:00
|
|
|
dup2_list_t dup2_list_t::resolve_chain(const io_chain_t &io_chain) {
|
2019-01-28 21:26:22 +00:00
|
|
|
ASSERT_IS_NOT_FORKED_CHILD();
|
|
|
|
dup2_list_t result;
|
|
|
|
for (const auto &io_ref : io_chain) {
|
|
|
|
switch (io_ref->io_mode) {
|
|
|
|
case io_mode_t::file: {
|
2019-12-13 01:27:48 +00:00
|
|
|
const io_file_t *io = static_cast<const io_file_t *>(io_ref.get());
|
|
|
|
result.add_dup2(io->file_fd(), io->fd);
|
2019-01-28 21:26:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case io_mode_t::close: {
|
|
|
|
const io_close_t *io = static_cast<const io_close_t *>(io_ref.get());
|
|
|
|
result.add_close(io->fd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case io_mode_t::fd: {
|
|
|
|
const io_fd_t *io = static_cast<const io_fd_t *>(io_ref.get());
|
2019-12-29 23:14:08 +00:00
|
|
|
result.add_dup2(io->source_fd, io->fd);
|
2019-01-28 21:26:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case io_mode_t::pipe: {
|
|
|
|
const io_pipe_t *io = static_cast<const io_pipe_t *>(io_ref.get());
|
2019-02-01 00:05:42 +00:00
|
|
|
result.add_dup2(io->pipe_fd(), io->fd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case io_mode_t::bufferfill: {
|
|
|
|
const io_bufferfill_t *io = static_cast<const io_bufferfill_t *>(io_ref.get());
|
|
|
|
result.add_dup2(io->write_fd(), io->fd);
|
2019-01-28 21:26:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-29 22:49:05 +00:00
|
|
|
return result;
|
2019-01-28 21:26:22 +00:00
|
|
|
}
|
2019-02-02 20:52:51 +00:00
|
|
|
|
|
|
|
int dup2_list_t::fd_for_target_fd(int target) const {
|
|
|
|
// Paranoia.
|
|
|
|
if (target < 0) {
|
|
|
|
return target;
|
|
|
|
}
|
|
|
|
// Note we can simply walk our action list backwards, looking for src -> target dups.
|
|
|
|
int cursor = target;
|
|
|
|
for (auto iter = actions_.rbegin(); iter != actions_.rend(); ++iter) {
|
|
|
|
if (iter->target == cursor) {
|
|
|
|
// cursor is replaced by iter->src
|
|
|
|
cursor = iter->src;
|
|
|
|
} else if (iter->src == cursor && iter->target < 0) {
|
|
|
|
// cursor is closed.
|
|
|
|
cursor = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return cursor;
|
|
|
|
}
|