mirror of
https://github.com/fish-shell/fish-shell
synced 2024-12-31 23:28:45 +00:00
Factor internal process short-circuiting together
When executing a buffered block or builtin, the usual approach is to execute, collect output in a string, and then output that string to stdout or whatever the redirections say. Similarly for stderr. If we get no output, then we can elide the outputting which means skipping the background thread. In this case we just mark the process as finished immediately. We do this in multiple locations which is confusing. Factor them all together into a new function run_internal_process_or_short_circuit.
This commit is contained in:
parent
fe4f7fea5c
commit
47b87dbeb7
1 changed files with 23 additions and 22 deletions
45
src/exec.cpp
45
src/exec.cpp
|
@ -237,7 +237,7 @@ static void on_process_created(const std::shared_ptr<job_t> &j, pid_t child_pid)
|
||||||
/// stdout and \p errdata to stderr, respecting the io chain \p ios. For example if target_fd is 1
|
/// stdout and \p errdata to stderr, respecting the io chain \p ios. For example if target_fd is 1
|
||||||
/// (stdout), and there is a dup2 3->1, then we need to write to fd 3. Then exit the internal
|
/// (stdout), and there is a dup2 3->1, then we need to write to fd 3. Then exit the internal
|
||||||
/// process.
|
/// process.
|
||||||
static void run_internal_process(process_t *p, std::string outdata, std::string errdata,
|
static void run_internal_process(process_t *p, std::string &&outdata, std::string &&errdata,
|
||||||
const io_chain_t &ios) {
|
const io_chain_t &ios) {
|
||||||
p->check_generations_before_launch();
|
p->check_generations_before_launch();
|
||||||
|
|
||||||
|
@ -328,6 +328,23 @@ static void run_internal_process(process_t *p, std::string outdata, std::string
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// If \p outdata or \p errdata are both empty, then mark the process as completed immediately.
|
||||||
|
/// Otherwise, run an internal process.
|
||||||
|
static void run_internal_process_or_short_circuit(parser_t &parser, const std::shared_ptr<job_t> &j,
|
||||||
|
process_t *p, std::string &&outdata,
|
||||||
|
std::string &&errdata, const io_chain_t &ios) {
|
||||||
|
if (outdata.empty() && errdata.empty()) {
|
||||||
|
p->completed = true;
|
||||||
|
if (p->is_last_in_job) {
|
||||||
|
FLOGF(exec_job_status, L"Set status of job %d (%ls) to %d using short circuit",
|
||||||
|
j->job_id(), j->preview().c_str(), p->status);
|
||||||
|
parser.set_last_statuses(j->get_statuses());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
run_internal_process(p, std::move(outdata), std::move(errdata), ios);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Call fork() as part of executing a process \p p in a job \j. Execute \p child_action in the
|
/// Call fork() as part of executing a process \p p in a job \j. Execute \p child_action in the
|
||||||
/// context of the child. Returns true if fork succeeded, false if fork failed.
|
/// context of the child. Returns true if fork succeeded, false if fork failed.
|
||||||
static bool fork_child_for_process(const std::shared_ptr<job_t> &job, process_t *p,
|
static bool fork_child_for_process(const std::shared_ptr<job_t> &job, process_t *p,
|
||||||
|
@ -505,19 +522,9 @@ static bool handle_builtin_output(parser_t &parser, const std::shared_ptr<job_t>
|
||||||
if (!outbuff.empty()) fflush(stdout);
|
if (!outbuff.empty()) fflush(stdout);
|
||||||
if (!errbuff.empty()) fflush(stderr);
|
if (!errbuff.empty()) fflush(stderr);
|
||||||
|
|
||||||
if (outbuff.empty() && errbuff.empty()) {
|
// Construct and run our background process.
|
||||||
// We do not need to construct a background process.
|
run_internal_process_or_short_circuit(parser, j, p, std::move(outbuff), std::move(errbuff),
|
||||||
// TODO: factor this job-status-setting stuff into a single place.
|
*io_chain);
|
||||||
p->completed = true;
|
|
||||||
if (p->is_last_in_job) {
|
|
||||||
FLOGF(exec_job_status, L"Set status of job %d (%ls) to %d using short circuit",
|
|
||||||
j->job_id(), j->preview().c_str(), p->status);
|
|
||||||
parser.set_last_statuses(j->get_statuses());
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Construct and run our background process.
|
|
||||||
run_internal_process(p, std::move(outbuff), std::move(errbuff), *io_chain);
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -791,14 +798,8 @@ static bool exec_block_or_func_process(parser_t &parser, const std::shared_ptr<j
|
||||||
buffer_contents = block_output_buffer->buffer().newline_serialized();
|
buffer_contents = block_output_buffer->buffer().newline_serialized();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!buffer_contents.empty()) {
|
run_internal_process_or_short_circuit(parser, j, p, std::move(buffer_contents),
|
||||||
run_internal_process(p, std::move(buffer_contents), {} /*errdata*/, io_chain);
|
{} /* errdata */, io_chain);
|
||||||
} else {
|
|
||||||
if (p->is_last_in_job) {
|
|
||||||
parser.set_last_statuses(j->get_statuses());
|
|
||||||
}
|
|
||||||
p->completed = true;
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue