fish-shell/src/builtin_wait.cpp

190 lines
6.6 KiB
C++
Raw Normal View History

2017-10-22 07:10:23 +00:00
// Functions for waiting for processes completed.
#include "builtin_wait.h"
#include <sys/wait.h>
#include <algorithm>
2017-10-22 07:10:23 +00:00
#include <vector>
#include "builtin.h"
#include "common.h"
#include "parser.h"
2017-10-22 07:10:23 +00:00
#include "proc.h"
#include "signal.h"
2017-10-22 07:10:23 +00:00
#include "wgetopt.h"
#include "wutil.h"
/// \return true if we can wait on a job.
static bool can_wait_on_job(const std::shared_ptr<job_t> &j) {
return j->is_constructed() && !j->is_foreground() && !j->is_stopped();
}
2021-05-16 05:05:16 +00:00
/// \return true if a wait handle matches a pid or a process name. Exactly one should be passed.
static bool wait_handle_matches(pid_t pid, const wchar_t *proc_name, const wait_handle_ref_t &wh) {
assert((pid > 0 || proc_name) && "Must specify either pid or proc_name");
return (pid > 0 && contains(wh->pids, pid)) ||
(proc_name && contains(wh->proc_base_names, proc_name));
2017-10-22 07:10:23 +00:00
}
/// Walk the list of jobs, looking for a process with \p pid (if nonzero) or \p proc_name (if not
/// null). Append all matching wait handles to \p handles.
/// \return true if we found a matching job (even if not waitable), false if not.
static bool find_wait_handles(pid_t pid, const wchar_t *proc_name, const parser_t &parser,
std::vector<wait_handle_ref_t> *handles) {
assert((pid > 0 || proc_name) && "Must specify either pid or proc_name");
// Has a job already completed?
bool matched = false;
for (const auto &wh : parser.get_recorded_wait_handles()) {
if (wait_handle_matches(pid, proc_name, wh)) {
handles->push_back(wh);
matched = true;
2017-10-22 07:10:23 +00:00
}
}
// Is there a running job match?
for (const auto &j : parser.jobs()) {
if (can_wait_on_job(j) && wait_handle_matches(pid, proc_name, j->get_wait_handle())) {
handles->push_back(j->get_wait_handle());
matched = true;
2017-10-22 07:10:23 +00:00
}
}
if (!matched) {
// Maybe we could have matched, but a job was stopped or otherwise unwaitable.
for (const auto &j : parser.jobs()) {
if (wait_handle_matches(pid, proc_name, j->get_wait_handle())) {
matched = true;
break;
2017-10-22 07:10:23 +00:00
}
}
}
return matched;
2017-10-22 07:10:23 +00:00
}
/// \return all wait handles for all jobs, current and already completed (!).
static std::vector<wait_handle_ref_t> get_all_wait_handles(const parser_t &parser) {
std::vector<wait_handle_ref_t> result;
const auto &whs = parser.get_recorded_wait_handles();
result.insert(result.end(), whs.begin(), whs.end());
for (const auto &j : parser.jobs()) {
if (can_wait_on_job(j)) {
result.push_back(j->get_wait_handle());
2017-10-22 07:10:23 +00:00
}
}
return result;
2017-10-22 07:10:23 +00:00
}
static inline bool is_completed(const wait_handle_ref_t &wh) { return wh->completed; }
/// Wait for the given wait handles to be marked as completed.
/// If \p any_flag is set, wait for the first one; otherwise wait for all.
/// \return a status code.
static int wait_for_completion(parser_t &parser, const std::vector<wait_handle_ref_t> &whs,
bool any_flag) {
if (whs.empty()) return 0;
sigchecker_t sigint(topic_t::sighupint);
for (;;) {
if (any_flag ? std::any_of(whs.begin(), whs.end(), is_completed)
: std::all_of(whs.begin(), whs.end(), is_completed)) {
// Remove completed wait handles (at most 1 if any_flag is set).
for (const auto &wh : whs) {
if (is_completed(wh)) {
parser.wait_handle_remove(wh);
if (any_flag) break;
}
}
return 0;
}
if (sigint.check()) {
2018-10-01 16:59:27 +00:00
return 128 + SIGINT;
2017-10-22 07:10:23 +00:00
}
proc_wait_any(parser);
2017-10-22 07:10:23 +00:00
}
DIE("Unreachable");
2017-10-22 07:10:23 +00:00
}
/// Tests if all characters in the wide string are numeric.
static bool iswnumeric(const wchar_t *n) {
for (; *n; n++) {
if (*n < L'0' || *n > L'9') {
return false;
}
}
return true;
}
maybe_t<int> builtin_wait(parser_t &parser, io_streams_t &streams, const wchar_t **argv) {
2017-10-22 07:10:23 +00:00
const wchar_t *cmd = argv[0];
int argc = builtin_count_args(argv);
bool any_flag = false; // flag for -n option
2020-03-31 19:16:42 +00:00
bool print_help = false;
2017-10-22 07:10:23 +00:00
2020-03-31 19:16:42 +00:00
static const wchar_t *const short_options = L":nh";
static const struct woption long_options[] = {{L"any", no_argument, nullptr, 'n'},
2020-03-31 19:16:42 +00:00
{L"help", no_argument, nullptr, 'h'},
{nullptr, 0, nullptr, 0}};
2017-10-22 07:10:23 +00:00
int opt;
wgetopter_t w;
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) {
2017-10-22 07:10:23 +00:00
switch (opt) {
case 'n':
any_flag = true;
break;
2020-03-31 19:16:42 +00:00
case 'h':
print_help = true;
break;
2017-10-22 07:10:23 +00:00
case ':': {
builtin_missing_argument(parser, streams, cmd, argv[w.woptind - 1]);
return STATUS_INVALID_ARGS;
}
case '?': {
builtin_unknown_option(parser, streams, cmd, argv[w.woptind - 1]);
return STATUS_INVALID_ARGS;
}
default: {
DIE("unexpected retval from wgetopt_long");
}
}
}
2020-03-31 19:16:42 +00:00
if (print_help) {
builtin_print_help(parser, streams, cmd);
return STATUS_CMD_OK;
}
2017-10-22 07:10:23 +00:00
if (w.woptind == argc) {
// No jobs specified.
// Note this may succeed with an empty wait list.
return wait_for_completion(parser, get_all_wait_handles(parser), any_flag);
}
// Get the list of wait handles for our waiting.
std::vector<wait_handle_ref_t> wait_handles;
for (int i = w.woptind; i < argc; i++) {
if (iswnumeric(argv[i])) {
// argument is pid
pid_t pid = fish_wcstoi(argv[i]);
if (errno || pid <= 0) {
streams.err.append_format(_(L"%ls: '%ls' is not a valid process id\n"), cmd,
argv[i]);
continue;
}
if (!find_wait_handles(pid, nullptr, parser, &wait_handles)) {
streams.err.append_format(_(L"%ls: Could not find a job with process id '%d'\n"),
cmd, pid);
}
} else {
// argument is process name
if (!find_wait_handles(0, argv[i], parser, &wait_handles)) {
streams.err.append_format(
_(L"%ls: Could not find child processes with the name '%ls'\n"), cmd, argv[i]);
2017-10-22 07:10:23 +00:00
}
}
}
if (wait_handles.empty()) return STATUS_INVALID_ARGS;
return wait_for_completion(parser, wait_handles, any_flag);
2017-10-22 07:10:23 +00:00
}