2017-10-22 07:10:23 +00:00
|
|
|
// Functions for waiting for processes completed.
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "builtin_wait.h"
|
|
|
|
|
|
|
|
#include <sys/wait.h>
|
|
|
|
|
2018-03-24 17:21:15 +00:00
|
|
|
#include <algorithm>
|
2017-10-22 07:10:23 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "builtin.h"
|
|
|
|
#include "common.h"
|
2019-05-05 05:12:31 +00:00
|
|
|
#include "parser.h"
|
2017-10-22 07:10:23 +00:00
|
|
|
#include "proc.h"
|
2019-05-26 02:05:24 +00:00
|
|
|
#include "signal.h"
|
2017-10-22 07:10:23 +00:00
|
|
|
#include "wgetopt.h"
|
|
|
|
#include "wutil.h"
|
|
|
|
|
2021-05-16 02:08:11 +00:00
|
|
|
/// \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();
|
2018-03-24 17:21:15 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +00:00
|
|
|
/// \return true if a wait handle matches a pid and/or a process name.
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-16 02:08:11 +00:00
|
|
|
return matched;
|
2017-10-22 07:10:23 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +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
|
|
|
}
|
|
|
|
}
|
2021-05-16 02:08:11 +00:00
|
|
|
return result;
|
2017-10-22 07:10:23 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 02:08:11 +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;
|
|
|
|
|
2020-06-20 16:23:36 +00:00
|
|
|
sigchecker_t sigint(topic_t::sighupint);
|
2021-05-16 02:08:11 +00:00
|
|
|
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;
|
|
|
|
}
|
2019-05-26 02:05:24 +00:00
|
|
|
if (sigint.check()) {
|
2018-10-01 16:59:27 +00:00
|
|
|
return 128 + SIGINT;
|
2017-10-22 07:10:23 +00:00
|
|
|
}
|
2019-04-30 03:58:58 +00:00
|
|
|
proc_wait_any(parser);
|
2017-10-22 07:10:23 +00:00
|
|
|
}
|
2021-05-16 02:08:11 +00:00
|
|
|
DIE("Unreachable");
|
2017-10-22 07:10:23 +00:00
|
|
|
}
|
|
|
|
|
2018-03-24 17:21:15 +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;
|
|
|
|
}
|
|
|
|
|
2021-02-14 02:41:09 +00:00
|
|
|
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";
|
2019-11-19 02:34:50 +00:00
|
|
|
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'},
|
2019-11-19 02:34:50 +00:00
|
|
|
{nullptr, 0, nullptr, 0}};
|
2017-10-22 07:10:23 +00:00
|
|
|
|
|
|
|
int opt;
|
|
|
|
wgetopter_t w;
|
2019-11-19 02:34:50 +00:00
|
|
|
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) {
|
2021-05-16 02:08:11 +00:00
|
|
|
// 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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-16 02:08:11 +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
|
|
|
}
|