Skip to content

Commit 9857952

Browse files
committedOct 13, 2023
Auto merge of #115108 - ijackson:broken-wait-status, r=dtolnay
Fix exit status / wait status on non-Unix cfg(unix) platforms Fixes #114593 Needs FCP due to behavioural changes (NB only on non-Unix `#[cfg(unix)]` platforms). Also, I think this is likely to break in CI. I have not been yet able to compile the new bits of `process_unsupported.rs`, although I have compiled the new module. I'd like some help from people familiar with eg emscripten and fuchsia (which are going to be affected, I think).
2 parents 193e8a1 + f625528 commit 9857952

File tree

5 files changed

+144
-50
lines changed

5 files changed

+144
-50
lines changed
 

‎library/std/src/sys/unix/process/process_common/tests.rs

+33
Original file line numberDiff line numberDiff line change
@@ -159,3 +159,36 @@ fn test_program_kind() {
159159
);
160160
}
161161
}
162+
163+
// Test that Rust std handles wait status values (`ExitStatus`) the way that Unix does,
164+
// at least for the values which represent a Unix exit status (`ExitCode`).
165+
// Should work on every #[cfg(unix)] platform. However:
166+
#[cfg(not(any(
167+
// Fuchsia is not Unix and has totally broken std::os::unix.
168+
// https://github.com/rust-lang/rust/issues/58590#issuecomment-836535609
169+
target_os = "fuchsia",
170+
)))]
171+
#[test]
172+
fn unix_exit_statuses() {
173+
use crate::num::NonZeroI32;
174+
use crate::os::unix::process::ExitStatusExt;
175+
use crate::process::*;
176+
177+
for exit_code in 0..=0xff {
178+
// FIXME impl From<ExitCode> for ExitStatus and then test that here too;
179+
// the two ExitStatus values should be the same
180+
let raw_wait_status = exit_code << 8;
181+
let exit_status = ExitStatus::from_raw(raw_wait_status);
182+
183+
assert_eq!(exit_status.code(), Some(exit_code));
184+
185+
if let Ok(nz) = NonZeroI32::try_from(exit_code) {
186+
assert!(!exit_status.success());
187+
let es_error = exit_status.exit_ok().unwrap_err();
188+
assert_eq!(es_error.code().unwrap(), i32::from(nz));
189+
} else {
190+
assert!(exit_status.success());
191+
assert_eq!(exit_status.exit_ok(), Ok(()));
192+
}
193+
}
194+
}

‎library/std/src/sys/unix/process/process_unix.rs

+5
Original file line numberDiff line numberDiff line change
@@ -1074,3 +1074,8 @@ impl crate::os::linux::process::ChildExt for crate::process::Child {
10741074
#[cfg(test)]
10751075
#[path = "process_unix/tests.rs"]
10761076
mod tests;
1077+
1078+
// See [`process_unsupported_wait_status::compare_with_linux`];
1079+
#[cfg(all(test, target_os = "linux"))]
1080+
#[path = "process_unsupported/wait_status.rs"]
1081+
mod process_unsupported_wait_status;

‎library/std/src/sys/unix/process/process_unsupported.rs

+2-50
Original file line numberDiff line numberDiff line change
@@ -55,56 +55,8 @@ impl Process {
5555
}
5656
}
5757

58-
#[derive(PartialEq, Eq, Clone, Copy, Debug, Default)]
59-
pub struct ExitStatus(c_int);
60-
61-
impl ExitStatus {
62-
#[cfg_attr(target_os = "horizon", allow(unused))]
63-
pub fn success(&self) -> bool {
64-
self.code() == Some(0)
65-
}
66-
67-
pub fn exit_ok(&self) -> Result<(), ExitStatusError> {
68-
Err(ExitStatusError(1.try_into().unwrap()))
69-
}
70-
71-
pub fn code(&self) -> Option<i32> {
72-
None
73-
}
74-
75-
pub fn signal(&self) -> Option<i32> {
76-
None
77-
}
78-
79-
pub fn core_dumped(&self) -> bool {
80-
false
81-
}
82-
83-
pub fn stopped_signal(&self) -> Option<i32> {
84-
None
85-
}
86-
87-
pub fn continued(&self) -> bool {
88-
false
89-
}
90-
91-
pub fn into_raw(&self) -> c_int {
92-
0
93-
}
94-
}
95-
96-
/// Converts a raw `c_int` to a type-safe `ExitStatus` by wrapping it without copying.
97-
impl From<c_int> for ExitStatus {
98-
fn from(a: c_int) -> ExitStatus {
99-
ExitStatus(a as i32)
100-
}
101-
}
102-
103-
impl fmt::Display for ExitStatus {
104-
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
105-
write!(f, "exit code: {}", self.0)
106-
}
107-
}
58+
mod wait_status;
59+
pub use wait_status::ExitStatus;
10860

10961
#[derive(PartialEq, Eq, Clone, Copy, Debug)]
11062
pub struct ExitStatusError(NonZero_c_int);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
//! Emulated wait status for non-Unix #[cfg(unix) platforms
2+
//!
3+
//! Separate module to facilitate testing against a real Unix implementation.
4+
5+
use crate::ffi::c_int;
6+
use crate::fmt;
7+
8+
/// Emulated wait status for use by `process_unsupported.rs`
9+
///
10+
/// Uses the "traditional unix" encoding. For use on platfors which are `#[cfg(unix)]`
11+
/// but do not actually support subprocesses at all.
12+
///
13+
/// These platforms aren't Unix, but are simply pretending to be for porting convenience.
14+
/// So, we provide a faithful pretence here.
15+
#[derive(PartialEq, Eq, Clone, Copy, Debug, Default)]
16+
pub struct ExitStatus {
17+
wait_status: c_int,
18+
}
19+
20+
/// Converts a raw `c_int` to a type-safe `ExitStatus` by wrapping it
21+
impl From<c_int> for ExitStatus {
22+
fn from(wait_status: c_int) -> ExitStatus {
23+
ExitStatus { wait_status }
24+
}
25+
}
26+
27+
impl fmt::Display for ExitStatus {
28+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
29+
write!(f, "emulated wait status: {}", self.wait_status)
30+
}
31+
}
32+
33+
impl ExitStatus {
34+
pub fn code(&self) -> Option<i32> {
35+
// Linux and FreeBSD both agree that values linux 0x80
36+
// count as "WIFEXITED" even though this is quite mad.
37+
// Likewise the macros disregard all the high bits, so are happy to declare
38+
// out-of-range values to be WIFEXITED, WIFSTOPPED, etc.
39+
let w = self.wait_status;
40+
if (w & 0x7f) == 0 { Some((w & 0xff00) >> 8) } else { None }
41+
}
42+
43+
pub fn signal(&self) -> Option<i32> {
44+
let signal = self.wait_status & 0x007f;
45+
if signal > 0 && signal < 0x7f { Some(signal) } else { None }
46+
}
47+
48+
pub fn core_dumped(&self) -> bool {
49+
self.signal().is_some() && (self.wait_status & 0x80) != 0
50+
}
51+
52+
pub fn stopped_signal(&self) -> Option<i32> {
53+
let w = self.wait_status;
54+
if (w & 0xff) == 0x7f { Some((w & 0xff00) >> 8) } else { None }
55+
}
56+
57+
pub fn continued(&self) -> bool {
58+
self.wait_status == 0xffff
59+
}
60+
61+
pub fn into_raw(&self) -> c_int {
62+
self.wait_status
63+
}
64+
}
65+
66+
#[cfg(test)]
67+
#[path = "wait_status/tests.rs"] // needed because of strange layout of process_unsupported
68+
mod tests;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
// Note that tests in this file are run on Linux as well as on platforms using process_unsupported
2+
3+
// Test that our emulation exactly matches Linux
4+
//
5+
// This test runs *on Linux* but it tests
6+
// the implementation used on non-Unix `#[cfg(unix)]` platforms.
7+
//
8+
// I.e. we're using Linux as a proxy for "trad unix".
9+
#[cfg(target_os = "linux")]
10+
#[test]
11+
fn compare_with_linux() {
12+
use super::ExitStatus as Emulated;
13+
use crate::os::unix::process::ExitStatusExt as _;
14+
use crate::process::ExitStatus as Real;
15+
16+
// Check that we handle out-of-range values similarly, too.
17+
for wstatus in -0xf_ffff..0xf_ffff {
18+
let emulated = Emulated::from(wstatus);
19+
let real = Real::from_raw(wstatus);
20+
21+
macro_rules! compare { { $method:ident } => {
22+
assert_eq!(
23+
emulated.$method(),
24+
real.$method(),
25+
"{wstatus:#x}.{}()",
26+
stringify!($method),
27+
);
28+
} }
29+
compare!(code);
30+
compare!(signal);
31+
compare!(core_dumped);
32+
compare!(stopped_signal);
33+
compare!(continued);
34+
compare!(into_raw);
35+
}
36+
}

0 commit comments

Comments
 (0)
Please sign in to comment.