Codebase list fish / HEAD src / builtin_eval.cpp
HEAD

Tree @HEAD (Download .tar.gz)

builtin_eval.cpp @HEADraw · history · blame

// Functions for executing the eval builtin.
#include "config.h"  // IWYU pragma: keep

#include <cerrno>
#include <cstddef>

#include "builtin.h"
#include "common.h"
#include "exec.h"
#include "fallback.h"  // IWYU pragma: keep
#include "io.h"
#include "parser.h"
#include "proc.h"
#include "wgetopt.h"
#include "wutil.h"  // IWYU pragma: keep

/// Implementation of eval builtin.
maybe_t<int> builtin_eval(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
    int argc = builtin_count_args(argv);
    if (argc <= 1) {
        return STATUS_CMD_OK;
    }

    wcstring new_cmd;
    for (int i = 1; i < argc; ++i) {
        if (i > 1) new_cmd += L' ';
        new_cmd += argv[i];
    }

    // Copy the full io chain; we may append bufferfills.
    io_chain_t ios = *streams.io_chain;

    // If stdout is piped, then its output must go to the streams, not to the io_chain in our
    // streams, because the pipe may be intended to be consumed by a process which
    // is not yet launched (#6806). If stdout is NOT redirected, it must see the tty (#6955). So
    // create a bufferfill for stdout if and only if stdout is piped.
    // Note do not do this if stdout is merely redirected (say, to a file); we don't want to
    // buffer in that case.
    shared_ptr<io_bufferfill_t> stdout_fill{};
    if (streams.out_is_piped) {
        stdout_fill = io_bufferfill_t::create(parser.libdata().read_limit, STDOUT_FILENO);
        if (!stdout_fill) {
            // We were unable to create a pipe, probably fd exhaustion.
            return STATUS_CMD_ERROR;
        }
        ios.push_back(stdout_fill);
    }

    // Of course the same applies to stderr.
    shared_ptr<io_bufferfill_t> stderr_fill{};
    if (streams.err_is_piped) {
        stderr_fill = io_bufferfill_t::create(parser.libdata().read_limit, STDERR_FILENO);
        if (!stderr_fill) {
            return STATUS_CMD_ERROR;
        }
        ios.push_back(stderr_fill);
    }

    int status = STATUS_CMD_OK;
    auto res = parser.eval(new_cmd, ios, streams.job_group);
    if (res.was_empty) {
        // Issue #5692, in particular, to catch `eval ""`, `eval "begin; end;"`, etc.
        // where we have an argument but nothing is executed.
        status = STATUS_CMD_OK;
    } else {
        status = res.status.status_value();
    }

    // Finish the bufferfills - exhaust and close our pipes.
    // Copy the output from the bufferfill back to the streams.
    // Note it is important that we hold no other references to the bufferfills here - they need to
    // deallocate to close.
    ios.clear();
    if (stdout_fill) {
        separated_buffer_t output = io_bufferfill_t::finish(std::move(stdout_fill));
        streams.out.append_narrow_buffer(std::move(output));
    }
    if (stderr_fill) {
        separated_buffer_t errput = io_bufferfill_t::finish(std::move(stderr_fill));
        streams.err.append_narrow_buffer(std::move(errput));
    }
    return status;
}