// Functions for executing the eval builtin. #include "config.h" // IWYU pragma: keep #include #include #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 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 stdout_fill{}; if (streams.out_is_piped) { stdout_fill = io_bufferfill_t::create(fd_set_t{}, 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 stderr_fill{}; if (streams.err_is_piped) { stderr_fill = io_bufferfill_t::create(fd_set_t{}, 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) { std::shared_ptr output = io_bufferfill_t::finish(std::move(stdout_fill)); streams.out.append_narrow_buffer(output->take_buffer()); } if (stderr_fill) { std::shared_ptr errput = io_bufferfill_t::finish(std::move(stderr_fill)); streams.err.append_narrow_buffer(errput->take_buffer()); } return status; }