2017-06-14 20:45:57 +00:00
|
|
|
// Implementation of the fg builtin.
|
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "builtin_fg.h"
|
|
|
|
|
2019-11-19 01:11:16 +00:00
|
|
|
#include <cerrno>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2019-03-12 21:06:01 +00:00
|
|
|
#include <cwchar>
|
2017-06-14 20:45:57 +00:00
|
|
|
|
|
|
|
#include "builtin.h"
|
|
|
|
#include "common.h"
|
|
|
|
#include "env.h"
|
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
|
|
|
#include "io.h"
|
2020-07-19 23:41:58 +00:00
|
|
|
#include "job_group.h"
|
2018-09-14 07:36:26 +00:00
|
|
|
#include "parser.h"
|
2017-06-14 20:45:57 +00:00
|
|
|
#include "proc.h"
|
|
|
|
#include "reader.h"
|
2017-06-17 03:42:33 +00:00
|
|
|
#include "tokenizer.h"
|
2017-06-17 04:00:24 +00:00
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2017-06-14 20:45:57 +00:00
|
|
|
|
|
|
|
/// Builtin for putting a job in the foreground.
|
2020-07-18 17:25:43 +00:00
|
|
|
maybe_t<int> builtin_fg(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
2017-06-14 20:45:57 +00:00
|
|
|
const wchar_t *cmd = argv[0];
|
|
|
|
int argc = builtin_count_args(argv);
|
2017-06-17 03:42:33 +00:00
|
|
|
help_only_cmd_opts_t opts;
|
2017-06-14 20:45:57 +00:00
|
|
|
|
|
|
|
int optind;
|
2017-06-17 03:42:33 +00:00
|
|
|
int retval = parse_help_only_cmd_opts(opts, &optind, argc, argv, parser, streams);
|
2017-06-14 20:45:57 +00:00
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
|
|
|
|
if (opts.print_help) {
|
2019-10-20 09:38:17 +00:00
|
|
|
builtin_print_help(parser, streams, cmd);
|
2017-06-14 20:45:57 +00:00
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
2018-12-31 03:25:16 +00:00
|
|
|
job_t *job = nullptr;
|
2017-06-14 20:45:57 +00:00
|
|
|
if (optind == argc) {
|
2019-01-02 18:10:54 +00:00
|
|
|
// Select last constructed job (i.e. first job in the job queue) that can be brought
|
|
|
|
// to the foreground.
|
2018-12-31 03:25:16 +00:00
|
|
|
|
2019-05-05 05:12:31 +00:00
|
|
|
for (const auto &j : parser.jobs()) {
|
2018-10-02 17:30:23 +00:00
|
|
|
if (j->is_constructed() && (!j->is_completed()) &&
|
2019-06-23 19:39:29 +00:00
|
|
|
((j->is_stopped() || (!j->is_foreground())) && j->wants_job_control())) {
|
2018-12-31 03:25:16 +00:00
|
|
|
job = j.get();
|
2017-06-14 20:45:57 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-12-31 03:25:16 +00:00
|
|
|
if (!job) {
|
2017-06-14 20:45:57 +00:00
|
|
|
streams.err.append_format(_(L"%ls: There are no suitable jobs\n"), cmd);
|
|
|
|
}
|
|
|
|
} else if (optind + 1 < argc) {
|
|
|
|
// Specifying more than one job to put to the foreground is a syntax error, we still
|
2019-01-02 18:10:54 +00:00
|
|
|
// try to locate the job $argv[1], since we need to determine which error message to
|
|
|
|
// emit (ambigous job specification vs malformed job id).
|
2018-12-31 06:43:26 +00:00
|
|
|
bool found_job = false;
|
2019-01-02 18:10:54 +00:00
|
|
|
int pid = fish_wcstoi(argv[optind]);
|
|
|
|
if (errno == 0 && pid > 0) {
|
2020-02-08 20:39:03 +00:00
|
|
|
found_job = (parser.job_get_from_pid(pid) != nullptr);
|
2017-06-14 20:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (found_job) {
|
|
|
|
streams.err.append_format(_(L"%ls: Ambiguous job\n"), cmd);
|
|
|
|
} else {
|
|
|
|
streams.err.append_format(_(L"%ls: '%ls' is not a job\n"), cmd, argv[optind]);
|
|
|
|
}
|
|
|
|
|
2019-01-02 18:10:54 +00:00
|
|
|
job = nullptr;
|
2019-03-26 18:13:01 +00:00
|
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
2017-06-14 20:45:57 +00:00
|
|
|
} else {
|
|
|
|
int pid = abs(fish_wcstoi(argv[optind]));
|
|
|
|
if (errno) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_NOT_NUMBER, cmd, argv[optind]);
|
2019-03-26 18:13:01 +00:00
|
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
2017-06-14 20:45:57 +00:00
|
|
|
} else {
|
2020-02-08 20:39:03 +00:00
|
|
|
job = parser.job_get_from_pid(pid);
|
2018-12-31 03:25:16 +00:00
|
|
|
if (!job || !job->is_constructed() || job->is_completed()) {
|
2017-06-14 20:45:57 +00:00
|
|
|
streams.err.append_format(_(L"%ls: No suitable job: %d\n"), cmd, pid);
|
2018-12-31 03:25:16 +00:00
|
|
|
job = nullptr;
|
2019-06-23 19:39:29 +00:00
|
|
|
} else if (!job->wants_job_control()) {
|
2020-11-22 13:39:48 +00:00
|
|
|
streams.err.append_format(_(L"%ls: Can't put job %d, '%ls' to foreground because "
|
|
|
|
L"it is not under job control\n"),
|
2018-12-31 03:25:16 +00:00
|
|
|
cmd, pid, job->command_wcstr());
|
2019-01-02 18:10:54 +00:00
|
|
|
job = nullptr;
|
2017-06-14 20:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-31 03:25:16 +00:00
|
|
|
if (!job) {
|
2017-06-14 20:45:57 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (streams.err_is_redirected) {
|
Introduce the internal jobs for functions
This PR is aimed at improving how job ids are assigned. In particular,
previous to this commit, a job id would be consumed by functions (and
thus aliases). Since it's usual to use functions as command wrappers
this results in awkward job id assignments.
For example if the user is like me and just made the jump from vim -> neovim
then the user might create the following alias:
```
alias vim=nvim
```
Previous to this commit if the user ran `vim` after setting up this
alias, backgrounded (^Z) and ran `jobs` then the output might be:
```
Job Group State Command
2 60267 stopped nvim $argv
```
If the user subsequently opened another vim (nvim) session, backgrounded
and ran jobs then they might see what follows:
```
Job Group State Command
4 70542 stopped nvim $argv
2 60267 stopped nvim $argv
```
These job ids feel unnatural, especially when transitioning away from
e.g. bash where job ids are sequentially incremented (and aliases/functions
don't consume a job id).
See #6053 for more details.
As @ridiculousfish pointed out in
https://github.com/fish-shell/fish-shell/issues/6053#issuecomment-559899400,
we want to elide a job's job id if it corresponds to a single function in the
foreground. This translates to the following prerequisites:
- A job must correspond to a single process (i.e. the job continuation
must be empty)
- A job must be in the foreground (i.e. `&` wasn't appended)
- The job's single process must resolve to a function invocation
If all of these conditions are true then we should mark a job as
"internal" and somehow remove it from consideration when any
infrastructure tries to interact with jobs / job ids.
I saw two paths to implement these requirements:
- At the time of job creation calculate whether or not a job is
"internal" and use a separate list of job ids to track their ids.
Additionally introduce a new flag denoting that a job is internal so
that e.g. `jobs` doesn't list internal jobs
- I started implementing this route but quickly realized I was
computing the same information that would be computed later on (e.g.
"is this job a single process" and "is this jobs statement a
function"). Specifically I was computing data that populate_job_process
would end up computing later anyway. Additionally this added some
weird complexities to the job system (after the change there were two
job id lists AND an additional flag that had to be taken into
consideration)
- Once a function is about to be executed we release the current jobs
job id if the prerequisites are satisfied (which at this point have
been fully computed).
- I opted for this solution since it seems cleaner. In this
implementation "releasing a job id" is done by both calling
`release_job_id` and by marking the internal job_id member variable to
-1. The former operation allows subsequent child jobs to reuse that
same job id (so e.g. the situation described in Motivation doesn't
occur), and the latter ensures that no other job / job id
infrastructure will interact with these jobs because valid jobs have
positive job ids. The second operation causes job_id to become
non-const which leads to the list of code changes outside of `exec.c`
(i.e. a codemod from `job_t::job_id` -> `job_t::job_id()` and moving the
old member variable to a non-const private `job_t::job_id_`)
Note: Its very possible I missed something and setting the job id to -1
will break some other infrastructure, please let me know if so!
I tried to run `make/ninja lint`, but a bunch of non-relevant issues
appeared (e.g. `fatal error: 'config.h' file not found`). I did
successfully clang-format (`git clang-format -f`) and run tests, though.
This PR closes #6053.
2019-12-29 15:46:07 +00:00
|
|
|
streams.err.append_format(FG_MSG, job->job_id(), job->command_wcstr());
|
2017-06-14 20:45:57 +00:00
|
|
|
} else {
|
|
|
|
// If we aren't redirecting, send output to real stderr, since stuff in sb_err won't get
|
|
|
|
// printed until the command finishes.
|
Introduce the internal jobs for functions
This PR is aimed at improving how job ids are assigned. In particular,
previous to this commit, a job id would be consumed by functions (and
thus aliases). Since it's usual to use functions as command wrappers
this results in awkward job id assignments.
For example if the user is like me and just made the jump from vim -> neovim
then the user might create the following alias:
```
alias vim=nvim
```
Previous to this commit if the user ran `vim` after setting up this
alias, backgrounded (^Z) and ran `jobs` then the output might be:
```
Job Group State Command
2 60267 stopped nvim $argv
```
If the user subsequently opened another vim (nvim) session, backgrounded
and ran jobs then they might see what follows:
```
Job Group State Command
4 70542 stopped nvim $argv
2 60267 stopped nvim $argv
```
These job ids feel unnatural, especially when transitioning away from
e.g. bash where job ids are sequentially incremented (and aliases/functions
don't consume a job id).
See #6053 for more details.
As @ridiculousfish pointed out in
https://github.com/fish-shell/fish-shell/issues/6053#issuecomment-559899400,
we want to elide a job's job id if it corresponds to a single function in the
foreground. This translates to the following prerequisites:
- A job must correspond to a single process (i.e. the job continuation
must be empty)
- A job must be in the foreground (i.e. `&` wasn't appended)
- The job's single process must resolve to a function invocation
If all of these conditions are true then we should mark a job as
"internal" and somehow remove it from consideration when any
infrastructure tries to interact with jobs / job ids.
I saw two paths to implement these requirements:
- At the time of job creation calculate whether or not a job is
"internal" and use a separate list of job ids to track their ids.
Additionally introduce a new flag denoting that a job is internal so
that e.g. `jobs` doesn't list internal jobs
- I started implementing this route but quickly realized I was
computing the same information that would be computed later on (e.g.
"is this job a single process" and "is this jobs statement a
function"). Specifically I was computing data that populate_job_process
would end up computing later anyway. Additionally this added some
weird complexities to the job system (after the change there were two
job id lists AND an additional flag that had to be taken into
consideration)
- Once a function is about to be executed we release the current jobs
job id if the prerequisites are satisfied (which at this point have
been fully computed).
- I opted for this solution since it seems cleaner. In this
implementation "releasing a job id" is done by both calling
`release_job_id` and by marking the internal job_id member variable to
-1. The former operation allows subsequent child jobs to reuse that
same job id (so e.g. the situation described in Motivation doesn't
occur), and the latter ensures that no other job / job id
infrastructure will interact with these jobs because valid jobs have
positive job ids. The second operation causes job_id to become
non-const which leads to the list of code changes outside of `exec.c`
(i.e. a codemod from `job_t::job_id` -> `job_t::job_id()` and moving the
old member variable to a non-const private `job_t::job_id_`)
Note: Its very possible I missed something and setting the job id to -1
will break some other infrastructure, please let me know if so!
I tried to run `make/ninja lint`, but a bunch of non-relevant issues
appeared (e.g. `fatal error: 'config.h' file not found`). I did
successfully clang-format (`git clang-format -f`) and run tests, though.
This PR closes #6053.
2019-12-29 15:46:07 +00:00
|
|
|
std::fwprintf(stderr, FG_MSG, job->job_id(), job->command_wcstr());
|
2017-06-14 20:45:57 +00:00
|
|
|
}
|
|
|
|
|
2020-03-08 03:44:58 +00:00
|
|
|
wcstring ft = tok_command(job->command());
|
2019-05-05 10:09:25 +00:00
|
|
|
// For compatibility with fish 2.0's $_, now replaced with `status current-command`
|
2020-03-08 03:44:58 +00:00
|
|
|
if (!ft.empty()) parser.set_var_and_fire(L"_", ENV_EXPORT, std::move(ft));
|
2019-05-05 03:20:52 +00:00
|
|
|
reader_write_title(job->command(), parser);
|
2017-06-14 20:45:57 +00:00
|
|
|
|
2019-05-05 05:12:31 +00:00
|
|
|
parser.job_promote(job);
|
2020-07-12 00:01:52 +00:00
|
|
|
job->group->set_is_foreground(true);
|
2017-06-14 20:45:57 +00:00
|
|
|
|
2020-07-27 00:55:00 +00:00
|
|
|
job->continue_job(parser);
|
2017-06-14 20:45:57 +00:00
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|