mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-27 20:25:12 +00:00
Use type safety for pid values
The previous approach of "treat this field as an `Option<NonZeroU32>` and remember to check `p.has_pid()` before accessing it" was a mix of C++ and rust conventions and led to some bugs or incorrect behaviors. * `jobs -p` would previously print both the (correct) external pid and the (incorrect) internal value of `0` if a backgrounded command contained a fish function (e.g. `function foo; end; cat | foo &; jobs`) * Updating/calculating job cpu time and usage was incorrectly including all of fish's cpu usage/time for each function/builtin member of the job pipeline. Closes #10832
This commit is contained in:
parent
080e40aac0
commit
3307672998
4 changed files with 126 additions and 63 deletions
|
@ -19,7 +19,6 @@ use crate::{
|
||||||
};
|
};
|
||||||
use libc::c_int;
|
use libc::c_int;
|
||||||
use std::num::NonZeroU32;
|
use std::num::NonZeroU32;
|
||||||
use std::sync::atomic::Ordering;
|
|
||||||
|
|
||||||
/// Print modes for the jobs builtin.
|
/// Print modes for the jobs builtin.
|
||||||
|
|
||||||
|
@ -36,9 +35,9 @@ enum JobsPrintMode {
|
||||||
/// This may exceed 1 if there are multiple CPUs!
|
/// This may exceed 1 if there are multiple CPUs!
|
||||||
fn cpu_use(j: &Job) -> f64 {
|
fn cpu_use(j: &Job) -> f64 {
|
||||||
let mut u = 0.0;
|
let mut u = 0.0;
|
||||||
for p in j.processes() {
|
for p in j.external_procs() {
|
||||||
let now = timef();
|
let now = timef();
|
||||||
let jiffies = proc_get_jiffies(p.pid.load(Ordering::Relaxed));
|
let jiffies = proc_get_jiffies(p.pid.load().unwrap().as_pid_t());
|
||||||
let last_jiffies = p.last_times.get().jiffies;
|
let last_jiffies = p.last_times.get().jiffies;
|
||||||
let since = now - last_jiffies as f64;
|
let since = now - last_jiffies as f64;
|
||||||
if since > 0.0 && jiffies > last_jiffies {
|
if since > 0.0 && jiffies > last_jiffies {
|
||||||
|
@ -104,8 +103,8 @@ fn builtin_jobs_print(j: &Job, mode: JobsPrintMode, header: bool, streams: &mut
|
||||||
out += wgettext!("Process\n");
|
out += wgettext!("Process\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
for p in j.processes() {
|
for p in j.external_procs() {
|
||||||
out += &sprintf!("%d\n", p.pid.load(Ordering::Relaxed))[..];
|
out += &sprintf!("%d\n", p.pid.load().unwrap().get())[..];
|
||||||
}
|
}
|
||||||
streams.out.append(out);
|
streams.out.append(out);
|
||||||
}
|
}
|
||||||
|
|
21
src/exec.rs
21
src/exec.rs
|
@ -36,8 +36,8 @@ use crate::null_terminated_array::{
|
||||||
use crate::parser::{Block, BlockId, BlockType, EvalRes, Parser};
|
use crate::parser::{Block, BlockId, BlockType, EvalRes, Parser};
|
||||||
use crate::proc::{
|
use crate::proc::{
|
||||||
hup_jobs, is_interactive_session, jobs_requiring_warning_on_exit, no_exec,
|
hup_jobs, is_interactive_session, jobs_requiring_warning_on_exit, no_exec,
|
||||||
print_exit_warning_for_jobs, InternalProc, Job, JobGroupRef, ProcStatus, Process, ProcessType,
|
print_exit_warning_for_jobs, InternalProc, Job, JobGroupRef, Pid, ProcStatus, Process,
|
||||||
TtyTransfer,
|
ProcessType, TtyTransfer,
|
||||||
};
|
};
|
||||||
use crate::reader::{reader_run_count, restore_term_mode};
|
use crate::reader::{reader_run_count, restore_term_mode};
|
||||||
use crate::redirection::{dup2_list_resolve_chain, Dup2List};
|
use crate::redirection::{dup2_list_resolve_chain, Dup2List};
|
||||||
|
@ -715,13 +715,14 @@ fn fork_child_for_process(
|
||||||
job.group().set_pgid(pid);
|
job.group().set_pgid(pid);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
let pid = p.pid().unwrap();
|
||||||
if let Some(pgid) = job.group().get_pgid() {
|
if let Some(pgid) = job.group().get_pgid() {
|
||||||
let err = execute_setpgid(p.pid(), pgid, is_parent);
|
let err = execute_setpgid(pid, pgid, is_parent);
|
||||||
if err != 0 {
|
if err != 0 {
|
||||||
report_setpgid_error(
|
report_setpgid_error(
|
||||||
err,
|
err,
|
||||||
is_parent,
|
is_parent,
|
||||||
p.pid(),
|
pid,
|
||||||
pgid,
|
pgid,
|
||||||
job.job_id().as_num(),
|
job.job_id().as_num(),
|
||||||
&narrow_cmd,
|
&narrow_cmd,
|
||||||
|
@ -869,7 +870,7 @@ fn exec_external_command(
|
||||||
return Err(());
|
return Err(());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
assert!(pid > 0, "Should have either a valid pid, or an error");
|
let pid = Pid::new(pid).expect("Should have either a valid pid, or an error");
|
||||||
|
|
||||||
// This usleep can be used to test for various race conditions
|
// This usleep can be used to test for various race conditions
|
||||||
// (https://github.com/fish-shell/fish-shell/issues/360).
|
// (https://github.com/fish-shell/fish-shell/issues/360).
|
||||||
|
@ -879,7 +880,7 @@ fn exec_external_command(
|
||||||
exec_fork,
|
exec_fork,
|
||||||
"Fork #%d, pid %d: spawn external command '%s' from '%ls'",
|
"Fork #%d, pid %d: spawn external command '%s' from '%ls'",
|
||||||
count,
|
count,
|
||||||
pid,
|
pid.get(),
|
||||||
p.actual_cmd,
|
p.actual_cmd,
|
||||||
file.as_ref()
|
file.as_ref()
|
||||||
.map(|s| s.as_utfstr())
|
.map(|s| s.as_utfstr())
|
||||||
|
@ -887,14 +888,14 @@ fn exec_external_command(
|
||||||
);
|
);
|
||||||
|
|
||||||
// these are all things do_fork() takes care of normally (for forked processes):
|
// these are all things do_fork() takes care of normally (for forked processes):
|
||||||
p.pid.store(pid, Ordering::Relaxed);
|
p.pid.store(pid);
|
||||||
if p.leads_pgrp {
|
if p.leads_pgrp {
|
||||||
j.group().set_pgid(pid);
|
j.group().set_pgid(pid.as_pid_t());
|
||||||
// posix_spawn should in principle set the pgid before returning.
|
// posix_spawn should in principle set the pgid before returning.
|
||||||
// In glibc, posix_spawn uses fork() and the pgid group is set on the child side;
|
// In glibc, posix_spawn uses fork() and the pgid group is set on the child side;
|
||||||
// therefore the parent may not have seen it be set yet.
|
// therefore the parent may not have seen it be set yet.
|
||||||
// Ensure it gets set. See #4715, also https://github.com/Microsoft/WSL/issues/2997.
|
// Ensure it gets set. See #4715, also https://github.com/Microsoft/WSL/issues/2997.
|
||||||
execute_setpgid(pid, pid, true /* is parent */);
|
execute_setpgid(pid.as_pid_t(), pid.as_pid_t(), true /* is parent */);
|
||||||
}
|
}
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
@ -1326,7 +1327,7 @@ fn exec_process_in_job(
|
||||||
exec_external_command(parser, j, p, &process_net_io_chain)?;
|
exec_external_command(parser, j, p, &process_net_io_chain)?;
|
||||||
// It's possible (though unlikely) that this is a background process which recycled a
|
// It's possible (though unlikely) that this is a background process which recycled a
|
||||||
// pid from another, previous background process. Forget any such old process.
|
// pid from another, previous background process. Forget any such old process.
|
||||||
parser.mut_wait_handles().remove_by_pid(p.pid());
|
parser.mut_wait_handles().remove_by_pid(p.pid().unwrap());
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
ProcessType::exec => {
|
ProcessType::exec => {
|
||||||
|
|
|
@ -949,8 +949,8 @@ impl Parser {
|
||||||
/// Returns the job and job index with the given pid.
|
/// Returns the job and job index with the given pid.
|
||||||
pub fn job_get_with_index_from_pid(&self, pid: libc::pid_t) -> Option<(usize, JobRef)> {
|
pub fn job_get_with_index_from_pid(&self, pid: libc::pid_t) -> Option<(usize, JobRef)> {
|
||||||
for (i, job) in self.jobs().iter().enumerate() {
|
for (i, job) in self.jobs().iter().enumerate() {
|
||||||
for p in job.processes().iter() {
|
for p in job.external_procs() {
|
||||||
if p.pid.load(Ordering::Relaxed) == pid {
|
if p.pid.load().unwrap().get() == pid {
|
||||||
return Some((i, job.clone()));
|
return Some((i, job.clone()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
155
src/proc.rs
155
src/proc.rs
|
@ -37,11 +37,12 @@ use portable_atomic::AtomicU64;
|
||||||
use std::cell::{Cell, Ref, RefCell, RefMut};
|
use std::cell::{Cell, Ref, RefCell, RefMut};
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::io::{Read, Write};
|
use std::io::{Read, Write};
|
||||||
|
use std::num::NonZeroU32;
|
||||||
use std::os::fd::RawFd;
|
use std::os::fd::RawFd;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
#[cfg(target_has_atomic = "64")]
|
#[cfg(target_has_atomic = "64")]
|
||||||
use std::sync::atomic::AtomicU64;
|
use std::sync::atomic::AtomicU64;
|
||||||
use std::sync::atomic::{AtomicBool, AtomicI32, AtomicU8, Ordering};
|
use std::sync::atomic::{AtomicBool, AtomicU32, AtomicU8, Ordering};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
/// Types of processes.
|
/// Types of processes.
|
||||||
|
@ -507,6 +508,56 @@ impl Drop for TtyTransfer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[repr(transparent)]
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
||||||
|
pub struct Pid(NonZeroU32);
|
||||||
|
|
||||||
|
impl Pid {
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn new<P: Into<i32>>(pid: P) -> Option<Pid> {
|
||||||
|
let pid = pid.into();
|
||||||
|
assert!(pid >= 0, "Invalid negative PID value!");
|
||||||
|
match NonZeroU32::new(pid as u32) {
|
||||||
|
Some(x) => Some(Pid(x)),
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn get(&self) -> i32 {
|
||||||
|
self.0.get() as i32
|
||||||
|
}
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn as_pid_t(&self) -> libc::pid_t {
|
||||||
|
#[allow(clippy::useless_conversion)]
|
||||||
|
self.get().into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<Pid> for u32 {
|
||||||
|
#[inline(always)]
|
||||||
|
fn into(self) -> Pid {
|
||||||
|
Pid::new(self as i32).unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default, Debug)]
|
||||||
|
pub struct AtomicPid(AtomicU32);
|
||||||
|
|
||||||
|
impl AtomicPid {
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn load(&self) -> Option<Pid> {
|
||||||
|
Pid::new(self.0.load(Ordering::Relaxed) as i32)
|
||||||
|
}
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn store(&self, pid: Pid) {
|
||||||
|
self.0.store(pid.get() as u32, Ordering::Relaxed);
|
||||||
|
}
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn swap(&self, pid: Pid) -> Option<Pid> {
|
||||||
|
Pid::new(self.0.swap(pid.get() as u32, Ordering::Relaxed) as i32)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// A structure representing a single fish process. Contains variables for tracking process state
|
/// A structure representing a single fish process. Contains variables for tracking process state
|
||||||
/// and the process argument list. Actually, a fish process can be either a regular external
|
/// and the process argument list. Actually, a fish process can be either a regular external
|
||||||
/// process, an internal builtin which may or may not spawn a fake IO process during execution, a
|
/// process, an internal builtin which may or may not spawn a fake IO process during execution, a
|
||||||
|
@ -551,9 +602,8 @@ pub struct Process {
|
||||||
/// Generation counts for reaping.
|
/// Generation counts for reaping.
|
||||||
pub gens: GenerationsList,
|
pub gens: GenerationsList,
|
||||||
|
|
||||||
/// Process ID, represented as an AtomicI32. This is actually an Option<AtomicNonZeroI32> with a
|
/// Process ID or `None` where not available.
|
||||||
/// value of zero representing `None`.
|
pub pid: AtomicPid,
|
||||||
pub pid: AtomicI32,
|
|
||||||
|
|
||||||
/// If we are an "internal process," that process.
|
/// If we are an "internal process," that process.
|
||||||
pub internal_proc: RefCell<Option<Arc<InternalProc>>>,
|
pub internal_proc: RefCell<Option<Arc<InternalProc>>>,
|
||||||
|
@ -614,26 +664,22 @@ impl Process {
|
||||||
Default::default()
|
Default::default()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Retrieves the associated [`libc::pid_t`], 0 if unset.
|
/// Retrieves the associated [`libc::pid_t`], `None` if unset.
|
||||||
///
|
#[inline(always)]
|
||||||
/// See [`Process::has_pid()]` to safely check if the process has a pid.
|
pub fn pid(&self) -> Option<libc::pid_t> {
|
||||||
pub fn pid(&self) -> libc::pid_t {
|
self.pid.load().map(|pid| pid.as_pid_t())
|
||||||
let value = self.pid.load(Ordering::Relaxed);
|
|
||||||
#[allow(clippy::useless_conversion)]
|
|
||||||
value.into()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
pub fn has_pid(&self) -> bool {
|
pub fn has_pid(&self) -> bool {
|
||||||
let value = self.pid.load(Ordering::Relaxed);
|
self.pid.load().is_some()
|
||||||
value != 0
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the process' pid. Panics if a pid has already been set.
|
/// Sets the process' pid. Panics if a pid has already been set.
|
||||||
pub fn set_pid(&self, pid: libc::pid_t) {
|
pub fn set_pid(&self, pid: libc::pid_t) {
|
||||||
assert!(pid != 0, "Invalid pid of 0 passed to Process::set_pid()");
|
let pid = Pid::new(pid).expect("Invalid pid passed to Process::set_pid()");
|
||||||
assert!(pid >= 0);
|
let old = self.pid.swap(pid);
|
||||||
let old = self.pid.swap(pid, Ordering::Relaxed);
|
assert!(old.is_none(), "Process::set_pid() called more than once!");
|
||||||
assert!(old == 0, "Process::set_pid() called more than once!");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets argv.
|
/// Sets argv.
|
||||||
|
@ -707,13 +753,13 @@ impl Process {
|
||||||
/// As a process does not know its job id, we pass it in.
|
/// As a process does not know its job id, we pass it in.
|
||||||
/// Note this will return null if the process is not waitable (has no pid).
|
/// Note this will return null if the process is not waitable (has no pid).
|
||||||
pub fn make_wait_handle(&self, jid: InternalJobId) -> Option<WaitHandleRef> {
|
pub fn make_wait_handle(&self, jid: InternalJobId) -> Option<WaitHandleRef> {
|
||||||
if self.typ != ProcessType::external || self.pid.load(Ordering::Relaxed) <= 0 {
|
if self.typ != ProcessType::external || self.pid().is_none() {
|
||||||
// Not waitable.
|
// Not waitable.
|
||||||
None
|
None
|
||||||
} else {
|
} else {
|
||||||
if self.wait_handle.borrow().is_none() {
|
if self.wait_handle.borrow().is_none() {
|
||||||
self.wait_handle.replace(Some(WaitHandle::new(
|
self.wait_handle.replace(Some(WaitHandle::new(
|
||||||
self.pid(),
|
self.pid().unwrap(),
|
||||||
jid,
|
jid,
|
||||||
wbasename(&self.actual_cmd.clone()).to_owned(),
|
wbasename(&self.actual_cmd.clone()).to_owned(),
|
||||||
)));
|
)));
|
||||||
|
@ -820,6 +866,14 @@ impl Job {
|
||||||
&mut self.processes
|
&mut self.processes
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A read-only view of external processes running in the job's process list.
|
||||||
|
///
|
||||||
|
/// Equivalent to `processes().iter().filter(|p| p.pid.is_some())`.
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn external_procs(&self) -> impl Iterator<Item = &ProcessPtr> {
|
||||||
|
self.processes.iter().filter(|p| p.pid.load().is_some())
|
||||||
|
}
|
||||||
|
|
||||||
/// Return whether it is OK to reap a given process. Sometimes we want to defer reaping a
|
/// Return whether it is OK to reap a given process. Sometimes we want to defer reaping a
|
||||||
/// process if it is the group leader and the job is not yet constructed, because then we might
|
/// process if it is the group leader and the job is not yet constructed, because then we might
|
||||||
/// also reap the process group and then we cannot add new processes to the group.
|
/// also reap the process group and then we cannot add new processes to the group.
|
||||||
|
@ -828,7 +882,7 @@ impl Job {
|
||||||
// Can't reap twice.
|
// Can't reap twice.
|
||||||
p.is_completed() ||
|
p.is_completed() ||
|
||||||
// Can't reap the group leader in an under-construction job.
|
// Can't reap the group leader in an under-construction job.
|
||||||
(p.has_pid() && !self.is_constructed() && self.get_pgid() == Some(p.pid()))
|
(!self.is_constructed() && self.get_pgid() == p.pid())
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -859,11 +913,7 @@ impl Job {
|
||||||
/// This will never be fish's own pid.
|
/// This will never be fish's own pid.
|
||||||
// TODO: Return a type-safe result.
|
// TODO: Return a type-safe result.
|
||||||
pub fn get_last_pid(&self) -> Option<libc::pid_t> {
|
pub fn get_last_pid(&self) -> Option<libc::pid_t> {
|
||||||
self.processes()
|
self.processes().iter().filter_map(|proc| proc.pid()).last()
|
||||||
.iter()
|
|
||||||
.rev()
|
|
||||||
.find(|proc| proc.has_pid())
|
|
||||||
.map(|proc| proc.pid())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The id of this job.
|
/// The id of this job.
|
||||||
|
@ -1044,9 +1094,8 @@ impl Job {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// This job lives in fish's pgroup and we need to signal procs individually.
|
// This job lives in fish's pgroup and we need to signal procs individually.
|
||||||
for p in self.processes().iter() {
|
for p in self.external_procs() {
|
||||||
if !p.is_completed() && p.has_pid() && unsafe { libc::kill(p.pid(), signal) } == -1
|
if !p.is_completed() && unsafe { libc::kill(p.pid().unwrap(), signal) } == -1 {
|
||||||
{
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1182,7 +1231,13 @@ pub fn print_exit_warning_for_jobs(jobs: &JobList) {
|
||||||
printf!("%s", wgettext!("There are still jobs active:\n"));
|
printf!("%s", wgettext!("There are still jobs active:\n"));
|
||||||
printf!("%s", wgettext!("\n PID Command\n"));
|
printf!("%s", wgettext!("\n PID Command\n"));
|
||||||
for j in jobs {
|
for j in jobs {
|
||||||
printf!("%6d %ls\n", j.processes()[0].pid(), j.command());
|
// Unwrap safety: we can't have a background job that doesn't have an external process and
|
||||||
|
// external processes always have a pid set.
|
||||||
|
printf!(
|
||||||
|
"%6d %ls\n",
|
||||||
|
j.external_procs().next().unwrap().pid().unwrap(),
|
||||||
|
j.command()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
printf!("\n");
|
printf!("\n");
|
||||||
printf!(
|
printf!(
|
||||||
|
@ -1234,10 +1289,10 @@ pub fn proc_get_jiffies(inpid: libc::pid_t) -> ClockTicks {
|
||||||
/// process of every job.
|
/// process of every job.
|
||||||
pub fn proc_update_jiffies(parser: &Parser) {
|
pub fn proc_update_jiffies(parser: &Parser) {
|
||||||
for job in parser.jobs().iter() {
|
for job in parser.jobs().iter() {
|
||||||
for p in job.processes.iter() {
|
for p in job.external_procs() {
|
||||||
p.last_times.replace(ProcTimes {
|
p.last_times.replace(ProcTimes {
|
||||||
time: timef(),
|
time: timef(),
|
||||||
jiffies: proc_get_jiffies(p.pid.load(Ordering::Relaxed)),
|
jiffies: proc_get_jiffies(p.pid.load().map(|p| p.as_pid_t()).unwrap()),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1335,10 +1390,8 @@ pub fn hup_jobs(jobs: &JobList) {
|
||||||
/// jobs. Used to avoid zombie processes after disown.
|
/// jobs. Used to avoid zombie processes after disown.
|
||||||
pub fn add_disowned_job(j: &Job) {
|
pub fn add_disowned_job(j: &Job) {
|
||||||
let mut disowned_pids = DISOWNED_PIDS.get().borrow_mut();
|
let mut disowned_pids = DISOWNED_PIDS.get().borrow_mut();
|
||||||
for process in j.processes().iter() {
|
for process in j.external_procs() {
|
||||||
if process.has_pid() {
|
disowned_pids.push(process.pid().unwrap());
|
||||||
disowned_pids.push(process.pid());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1403,9 +1456,9 @@ fn process_mark_finished_children(parser: &Parser, block_ok: bool) {
|
||||||
// We structure this as two loops for some simplicity.
|
// We structure this as two loops for some simplicity.
|
||||||
// First reap all pids.
|
// First reap all pids.
|
||||||
for j in parser.jobs().iter() {
|
for j in parser.jobs().iter() {
|
||||||
for proc in j.processes.iter() {
|
for proc in j.external_procs() {
|
||||||
// Does this proc have a pid that is reapable?
|
// It's an external proc so it has a pid, but is it reapable?
|
||||||
if proc.pid.load(Ordering::Relaxed) <= 0 || !j.can_reap(proc) {
|
if !j.can_reap(proc) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1421,12 +1474,16 @@ fn process_mark_finished_children(parser: &Parser, block_ok: bool) {
|
||||||
// Ok, we are reapable. Run waitpid()!
|
// Ok, we are reapable. Run waitpid()!
|
||||||
let mut statusv: libc::c_int = -1;
|
let mut statusv: libc::c_int = -1;
|
||||||
let pid = unsafe {
|
let pid = unsafe {
|
||||||
libc::waitpid(proc.pid(), &mut statusv, WNOHANG | WUNTRACED | WCONTINUED)
|
libc::waitpid(
|
||||||
|
proc.pid().unwrap(),
|
||||||
|
&mut statusv,
|
||||||
|
WNOHANG | WUNTRACED | WCONTINUED,
|
||||||
|
)
|
||||||
};
|
};
|
||||||
assert!(pid <= 0 || pid == proc.pid(), "Unexpected waitpid() return");
|
|
||||||
if pid <= 0 {
|
if pid <= 0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
assert!(pid == proc.pid().unwrap(), "Unexpected waitpid() return");
|
||||||
|
|
||||||
// The process has stopped or exited! Update its status.
|
// The process has stopped or exited! Update its status.
|
||||||
let status = ProcStatus::from_waitpid(statusv);
|
let status = ProcStatus::from_waitpid(statusv);
|
||||||
|
@ -1451,7 +1508,7 @@ fn process_mark_finished_children(parser: &Parser, block_ok: bool) {
|
||||||
proc_reap_external,
|
proc_reap_external,
|
||||||
"External process '%ls' (pid %d, %s)",
|
"External process '%ls' (pid %d, %s)",
|
||||||
proc.argv0().unwrap(),
|
proc.argv0().unwrap(),
|
||||||
proc.pid(),
|
proc.pid().unwrap(),
|
||||||
if proc.status.stopped() {
|
if proc.status.stopped() {
|
||||||
"stopped"
|
"stopped"
|
||||||
} else {
|
} else {
|
||||||
|
@ -1511,10 +1568,13 @@ fn generate_process_exit_events(j: &Job, out_evts: &mut Vec<Event>) {
|
||||||
// Historically we have avoided generating events for foreground jobs from event handlers, as an
|
// Historically we have avoided generating events for foreground jobs from event handlers, as an
|
||||||
// event handler may itself produce a new event.
|
// event handler may itself produce a new event.
|
||||||
if !j.from_event_handler() || !j.is_foreground() {
|
if !j.from_event_handler() || !j.is_foreground() {
|
||||||
for p in j.processes().iter() {
|
for p in j.external_procs() {
|
||||||
if p.has_pid() && p.is_completed() && !p.posted_proc_exit.load() {
|
if p.is_completed() && !p.posted_proc_exit.load() {
|
||||||
p.posted_proc_exit.store(true);
|
p.posted_proc_exit.store(true);
|
||||||
out_evts.push(Event::process_exit(p.pid(), p.status.status_value()));
|
out_evts.push(Event::process_exit(
|
||||||
|
p.pid().unwrap(),
|
||||||
|
p.status.status_value(),
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1631,8 +1691,11 @@ fn summary_command(j: &Job, p: Option<&Process>) -> WString {
|
||||||
buffer += &escape(sig.desc())[..];
|
buffer += &escape(sig.desc())[..];
|
||||||
|
|
||||||
// If we have multiple processes, we also append the pid and argv.
|
// If we have multiple processes, we also append the pid and argv.
|
||||||
if j.processes().len() > 1 {
|
if j.external_procs().count() > 1 {
|
||||||
buffer += &sprintf!(" %d", p.pid())[..];
|
// I don't think it's safe to blindly unwrap here because even though we exited with
|
||||||
|
// a signal, the job could have contained a fish function?
|
||||||
|
let pid = p.pid().map(|p| p.to_string()).unwrap_or("-".to_string());
|
||||||
|
buffer += &sprintf!(" %s", pid)[..];
|
||||||
|
|
||||||
buffer.push(' ');
|
buffer.push(' ');
|
||||||
buffer += &escape(p.argv0().unwrap())[..];
|
buffer += &escape(p.argv0().unwrap())[..];
|
||||||
|
|
Loading…
Reference in a new issue