2016-04-20 02:49:15 +00:00
|
|
|
// Functions for executing the jobs builtin.
|
2016-04-21 06:00:54 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <sys/time.h>
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2019-11-19 01:11:16 +00:00
|
|
|
#include <cerrno>
|
|
|
|
#include <cstddef>
|
|
|
|
|
2006-05-26 11:38:11 +00:00
|
|
|
#include "builtin.h"
|
|
|
|
#include "common.h"
|
2016-04-20 02:49:15 +00:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2016-04-21 06:00:54 +00:00
|
|
|
#include "io.h"
|
2019-05-05 05:12:31 +00:00
|
|
|
#include "parser.h"
|
2016-04-20 02:49:15 +00:00
|
|
|
#include "proc.h"
|
|
|
|
#include "wgetopt.h"
|
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2006-07-19 22:55:49 +00:00
|
|
|
|
2016-04-21 06:00:54 +00:00
|
|
|
class parser_t;
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
/// Print modes for the jobs builtin.
|
|
|
|
enum {
|
|
|
|
JOBS_DEFAULT, // print lots of general info
|
|
|
|
JOBS_PRINT_PID, // print pid of each process in job
|
|
|
|
JOBS_PRINT_COMMAND, // print command name of each process in job
|
|
|
|
JOBS_PRINT_GROUP, // print group id of job
|
2019-05-05 10:09:25 +00:00
|
|
|
JOBS_PRINT_NOTHING, // print nothing (exit status only)
|
2016-04-20 02:49:15 +00:00
|
|
|
};
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
/// Calculates the cpu usage (in percent) of the specified job.
|
|
|
|
static int cpu_use(const job_t *j) {
|
|
|
|
double u = 0;
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2017-01-23 17:28:34 +00:00
|
|
|
for (const process_ptr_t &p : j->processes) {
|
2012-11-19 00:30:30 +00:00
|
|
|
struct timeval t;
|
2020-05-30 20:59:44 +00:00
|
|
|
unsigned long jiffies;
|
2019-11-19 02:34:50 +00:00
|
|
|
gettimeofday(&t, nullptr);
|
2017-01-23 17:28:34 +00:00
|
|
|
jiffies = proc_get_jiffies(p.get());
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
double t1 = 1000000.0 * p->last_time.tv_sec + p->last_time.tv_usec;
|
|
|
|
double t2 = 1000000.0 * t.tv_sec + t.tv_usec;
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2020-05-30 20:59:44 +00:00
|
|
|
// Check for a race condition that can cause negative CPU usage to be reported (#7066)
|
2020-05-30 21:05:29 +00:00
|
|
|
unsigned long cached_last_jiffies = p->last_jiffies;
|
|
|
|
if (t2 < t1 || jiffies < cached_last_jiffies) {
|
2020-05-30 20:59:44 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-03-12 21:06:01 +00:00
|
|
|
// std::fwprintf( stderr, L"t1 %f t2 %f p1 %d p2 %d\n", t1, t2, jiffies, p->last_jiffies );
|
2020-05-30 21:05:29 +00:00
|
|
|
u += (static_cast<double>(jiffies - cached_last_jiffies)) / (t2 - t1);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
return u * 1000000;
|
2006-05-26 11:38:11 +00:00
|
|
|
}
|
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
/// Print information about the specified job.
|
|
|
|
static void builtin_jobs_print(const job_t *j, int mode, int header, io_streams_t &streams) {
|
2020-05-29 21:51:48 +00:00
|
|
|
int pgid = INVALID_PID;
|
|
|
|
if (auto job_pgid = j->get_pgid()) {
|
|
|
|
pgid = *job_pgid;
|
|
|
|
}
|
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
switch (mode) {
|
2018-03-04 21:20:14 +00:00
|
|
|
case JOBS_PRINT_NOTHING: {
|
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case JOBS_DEFAULT: {
|
|
|
|
if (header) {
|
|
|
|
// Print table header before first job.
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(_(L"Job\tGroup\t"));
|
2019-05-12 21:59:30 +00:00
|
|
|
if (have_proc_stat()) {
|
2019-04-29 13:29:21 +00:00
|
|
|
streams.out.append(_(L"CPU\t"));
|
|
|
|
}
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(_(L"State\tCommand\n"));
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2020-05-29 21:51:48 +00:00
|
|
|
streams.out.append_format(L"%d\t%d\t", j->job_id(), pgid);
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2019-05-12 21:59:30 +00:00
|
|
|
if (have_proc_stat()) {
|
2019-04-29 13:29:21 +00:00
|
|
|
streams.out.append_format(L"%d%%\t", cpu_use(j));
|
|
|
|
}
|
|
|
|
|
2018-10-02 17:30:23 +00:00
|
|
|
streams.out.append(j->is_stopped() ? _(L"stopped") : _(L"running"));
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(L"\t");
|
|
|
|
streams.out.append(j->command_wcstr());
|
|
|
|
streams.out.append(L"\n");
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case JOBS_PRINT_GROUP: {
|
|
|
|
if (header) {
|
|
|
|
// Print table header before first job.
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(_(L"Group\n"));
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
2020-05-29 21:51:48 +00:00
|
|
|
streams.out.append_format(L"%d\n", pgid);
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case JOBS_PRINT_PID: {
|
|
|
|
if (header) {
|
|
|
|
// Print table header before first job.
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(_(L"Process\n"));
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-01-23 17:28:34 +00:00
|
|
|
for (const process_ptr_t &p : j->processes) {
|
2016-04-20 02:49:15 +00:00
|
|
|
streams.out.append_format(L"%d\n", p->pid);
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case JOBS_PRINT_COMMAND: {
|
|
|
|
if (header) {
|
|
|
|
// Print table header before first job.
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(_(L"Command\n"));
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
|
|
|
|
2017-01-23 17:28:34 +00:00
|
|
|
for (const process_ptr_t &p : j->processes) {
|
2016-04-20 02:49:15 +00:00
|
|
|
streams.out.append_format(L"%ls\n", p->argv0());
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2016-10-30 00:25:48 +00:00
|
|
|
default: {
|
|
|
|
DIE("unexpected mode");
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2006-05-26 11:38:11 +00:00
|
|
|
}
|
|
|
|
|
2017-11-10 19:41:02 +00:00
|
|
|
/// The jobs builtin. Used for printing running jobs. Defined in builtin_jobs.c.
|
2020-07-18 17:25:43 +00:00
|
|
|
maybe_t<int> builtin_jobs(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
2017-06-14 19:26:05 +00:00
|
|
|
wchar_t *cmd = argv[0];
|
2017-06-10 19:30:09 +00:00
|
|
|
int argc = builtin_count_args(argv);
|
2018-12-31 06:43:26 +00:00
|
|
|
bool found = false;
|
2016-04-20 02:49:15 +00:00
|
|
|
int mode = JOBS_DEFAULT;
|
2012-11-19 00:30:30 +00:00
|
|
|
int print_last = 0;
|
|
|
|
|
2018-09-29 03:45:56 +00:00
|
|
|
static const wchar_t *const short_options = L":cghlpq";
|
2019-11-19 02:34:50 +00:00
|
|
|
static const struct woption long_options[] = {{L"command", no_argument, nullptr, 'c'},
|
|
|
|
{L"group", no_argument, nullptr, 'g'},
|
|
|
|
{L"help", no_argument, nullptr, 'h'},
|
|
|
|
{L"last", no_argument, nullptr, 'l'},
|
|
|
|
{L"pid", no_argument, nullptr, 'p'},
|
|
|
|
{L"quiet", no_argument, nullptr, 'q'},
|
|
|
|
{nullptr, 0, nullptr, 0}};
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-06-10 19:30:09 +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) {
|
2016-04-20 02:49:15 +00:00
|
|
|
switch (opt) {
|
|
|
|
case 'p': {
|
|
|
|
mode = JOBS_PRINT_PID;
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
2016-04-20 02:49:15 +00:00
|
|
|
}
|
2018-03-04 21:20:14 +00:00
|
|
|
case 'q': {
|
|
|
|
mode = JOBS_PRINT_NOTHING;
|
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case 'c': {
|
|
|
|
mode = JOBS_PRINT_COMMAND;
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
2016-04-20 02:49:15 +00:00
|
|
|
}
|
|
|
|
case 'g': {
|
|
|
|
mode = JOBS_PRINT_GROUP;
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
2016-04-20 02:49:15 +00:00
|
|
|
}
|
|
|
|
case 'l': {
|
2012-11-19 08:31:03 +00:00
|
|
|
print_last = 1;
|
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case 'h': {
|
2019-10-20 09:38:17 +00:00
|
|
|
builtin_print_help(parser, streams, cmd);
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_CMD_OK;
|
2016-04-20 02:49:15 +00:00
|
|
|
}
|
2017-06-30 04:49:57 +00:00
|
|
|
case ':': {
|
2017-07-01 21:03:47 +00:00
|
|
|
builtin_missing_argument(parser, streams, cmd, argv[w.woptind - 1]);
|
2017-06-30 04:49:57 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case '?': {
|
2017-06-14 19:26:05 +00:00
|
|
|
builtin_unknown_option(parser, streams, cmd, argv[w.woptind - 1]);
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
2016-04-20 02:49:15 +00:00
|
|
|
}
|
2016-10-30 00:25:48 +00:00
|
|
|
default: {
|
2017-06-10 19:30:09 +00:00
|
|
|
DIE("unexpected retval from wgetopt_long");
|
2016-10-30 00:25:48 +00:00
|
|
|
}
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2006-05-26 11:38:11 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
if (print_last) {
|
|
|
|
// Ignore unconstructed jobs, i.e. ourself.
|
2019-05-05 05:12:31 +00:00
|
|
|
for (const auto &j : parser.jobs()) {
|
2019-04-10 04:29:58 +00:00
|
|
|
if (j->is_visible()) {
|
2018-12-31 03:25:16 +00:00
|
|
|
builtin_jobs_print(j.get(), mode, !streams.out_is_redirected, streams);
|
2019-09-11 05:17:23 +00:00
|
|
|
return STATUS_CMD_OK;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-11 05:17:23 +00:00
|
|
|
return STATUS_CMD_ERROR;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
} else {
|
|
|
|
if (w.woptind < argc) {
|
2016-10-02 00:21:40 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = w.woptind; i < argc; i++) {
|
2018-04-14 21:47:05 +00:00
|
|
|
const job_t *j = nullptr;
|
|
|
|
|
|
|
|
if (argv[i][0] == L'%') {
|
2020-02-08 20:47:13 +00:00
|
|
|
int job_id = fish_wcstoi(argv[i] + 1);
|
|
|
|
if (errno || job_id < -1) {
|
2019-05-05 10:09:25 +00:00
|
|
|
streams.err.append_format(_(L"%ls: '%ls' is not a valid job id"), cmd,
|
|
|
|
argv[i]);
|
2018-04-14 21:47:05 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
2020-02-08 20:47:13 +00:00
|
|
|
j = parser.job_get(job_id);
|
2019-05-05 10:09:25 +00:00
|
|
|
} else {
|
2018-04-14 21:47:05 +00:00
|
|
|
int pid = fish_wcstoi(argv[i]);
|
|
|
|
if (errno || pid < 0) {
|
2019-05-05 10:09:25 +00:00
|
|
|
streams.err.append_format(_(L"%ls: '%ls' is not a valid process id\n"), cmd,
|
|
|
|
argv[i]);
|
2018-04-14 21:47:05 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
2020-02-08 20:39:03 +00:00
|
|
|
j = parser.job_get_from_pid(pid);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
|
2018-10-02 17:30:23 +00:00
|
|
|
if (j && !j->is_completed() && j->is_constructed()) {
|
2015-11-25 13:37:48 +00:00
|
|
|
builtin_jobs_print(j, mode, false, streams);
|
2018-12-31 06:43:26 +00:00
|
|
|
found = true;
|
2016-04-20 02:49:15 +00:00
|
|
|
} else {
|
2020-03-26 06:00:31 +00:00
|
|
|
if (mode != JOBS_PRINT_NOTHING) {
|
|
|
|
streams.err.append_format(_(L"%ls: No suitable job: %ls\n"), cmd, argv[i]);
|
|
|
|
}
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_CMD_ERROR;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
} else {
|
2019-05-05 05:12:31 +00:00
|
|
|
for (const auto &j : parser.jobs()) {
|
2016-04-20 02:49:15 +00:00
|
|
|
// Ignore unconstructed jobs, i.e. ourself.
|
2019-04-10 04:29:58 +00:00
|
|
|
if (j->is_visible()) {
|
2019-05-05 10:09:25 +00:00
|
|
|
builtin_jobs_print(j.get(), mode, !found && !streams.out_is_redirected,
|
|
|
|
streams);
|
2018-12-31 06:43:26 +00:00
|
|
|
found = true;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
if (!found) {
|
|
|
|
// Do not babble if not interactive.
|
2018-03-04 21:20:14 +00:00
|
|
|
if (!streams.out_is_redirected && mode != JOBS_PRINT_NOTHING) {
|
2016-04-20 02:49:15 +00:00
|
|
|
streams.out.append_format(_(L"%ls: There are no jobs\n"), argv[0]);
|
2015-11-25 13:37:48 +00:00
|
|
|
}
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_CMD_ERROR;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_CMD_OK;
|
2006-05-26 11:38:11 +00:00
|
|
|
}
|