Auto merge of #2426 - saethlin:unix-exec, r=RalfJung

Use real exec on cfg(unix) targets

Closes https://github.com/rust-lang/miri/issues/2421

The standard library has a platform extension trait that lets us get the behavior we want on cfg(unix), so why not use it?

I tried this out and it produces the correct behavior in concert with nextest.
This commit is contained in:
bors 2022-07-28 22:05:42 +00:00
commit a719c05816

View File

@ -51,7 +51,7 @@ enum MiriCommand {
} }
/// The information to run a crate with the given environment. /// The information to run a crate with the given environment.
#[derive(Serialize, Deserialize)] #[derive(Clone, Serialize, Deserialize)]
struct CrateRunEnv { struct CrateRunEnv {
/// The command-line arguments. /// The command-line arguments.
args: Vec<String>, args: Vec<String>,
@ -250,19 +250,49 @@ fn xargo_check() -> Command {
Command::new(env::var_os("XARGO_CHECK").unwrap_or_else(|| OsString::from("xargo-check"))) Command::new(env::var_os("XARGO_CHECK").unwrap_or_else(|| OsString::from("xargo-check")))
} }
/// Execute the command. If it fails, fail this process with the same exit code. /// Execute the `Command`, where possible by replacing the current process with a new process
/// Otherwise, continue. /// described by the `Command`. Then exit this process with the exit code of the new process.
fn exec(mut cmd: Command) { fn exec(mut cmd: Command) -> ! {
// On non-Unix imitate POSIX exec as closely as we can
#[cfg(not(unix))]
{
let exit_status = cmd.status().expect("failed to run command"); let exit_status = cmd.status().expect("failed to run command");
if exit_status.success().not() {
std::process::exit(exit_status.code().unwrap_or(-1)) std::process::exit(exit_status.code().unwrap_or(-1))
} }
// On Unix targets, actually exec.
// If exec returns, process setup has failed. This is the same error condition as the expect in
// the non-Unix case.
#[cfg(unix)]
{
use std::os::unix::process::CommandExt;
let error = cmd.exec();
Err(error).expect("failed to run command")
}
} }
/// Execute the command and pipe `input` into its stdin. /// Execute the `Command`, where possible by replacing the current process with a new process
/// If it fails, fail this process with the same exit code. /// described by the `Command`. Then exit this process with the exit code of the new process.
/// Otherwise, continue. /// `input` is also piped to the new process's stdin, on cfg(unix) platforms by writing its
fn exec_with_pipe(mut cmd: Command, input: &[u8]) { /// contents to `path` first, then setting stdin to that file.
fn exec_with_pipe<P>(mut cmd: Command, input: &[u8], path: P) -> !
where
P: AsRef<Path>,
{
#[cfg(unix)]
{
// Write the bytes we want to send to stdin out to a file
std::fs::write(&path, input).unwrap();
// Open the file for reading, and set our new stdin to it
let stdin = File::open(&path).unwrap();
cmd.stdin(stdin);
// Unlink the file so that it is fully cleaned up as soon as the new process exits
std::fs::remove_file(&path).unwrap();
// Finally, we can hand off control.
exec(cmd)
}
#[cfg(not(unix))]
{
drop(path); // We don't need the path, we can pipe the bytes directly
cmd.stdin(process::Stdio::piped()); cmd.stdin(process::Stdio::piped());
let mut child = cmd.spawn().expect("failed to spawn process"); let mut child = cmd.spawn().expect("failed to spawn process");
{ {
@ -270,7 +300,6 @@ fn exec_with_pipe(mut cmd: Command, input: &[u8]) {
stdin.write_all(input).expect("failed to write out test source"); stdin.write_all(input).expect("failed to write out test source");
} }
let exit_status = child.wait().expect("failed to run command"); let exit_status = child.wait().expect("failed to run command");
if exit_status.success().not() {
std::process::exit(exit_status.code().unwrap_or(-1)) std::process::exit(exit_status.code().unwrap_or(-1))
} }
} }
@ -890,6 +919,8 @@ fn out_filename(prefix: &str, suffix: &str) -> PathBuf {
// and environment variables; this is used when cargo calls us again in the CARGO_TARGET_RUNNER phase. // and environment variables; this is used when cargo calls us again in the CARGO_TARGET_RUNNER phase.
let env = CrateRunEnv::collect(args, inside_rustdoc); let env = CrateRunEnv::collect(args, inside_rustdoc);
store_json(CrateRunInfo::RunWith(env.clone()));
// Rustdoc expects us to exit with an error code if the test is marked as `compile_fail`, // Rustdoc expects us to exit with an error code if the test is marked as `compile_fail`,
// just creating the JSON file is not enough: we need to detect syntax errors, // just creating the JSON file is not enough: we need to detect syntax errors,
// so we need to run Miri with `MIRI_BE_RUSTC` for a check-only build. // so we need to run Miri with `MIRI_BE_RUSTC` for a check-only build.
@ -906,7 +937,15 @@ fn out_filename(prefix: &str, suffix: &str) -> PathBuf {
cmd.arg("--emit=metadata"); cmd.arg("--emit=metadata");
} }
cmd.args(&env.args); // Alter the `-o` parameter so that it does not overwrite the JSON file we stored above.
let mut args = env.args.clone();
for i in 0..args.len() {
if args[i] == "-o" {
args[i + 1].push_str(".miri");
}
}
cmd.args(&args);
cmd.env("MIRI_BE_RUSTC", "target"); cmd.env("MIRI_BE_RUSTC", "target");
if verbose > 0 { if verbose > 0 {
@ -917,11 +956,9 @@ fn out_filename(prefix: &str, suffix: &str) -> PathBuf {
eprintln!("[cargo-miri rustc inside rustdoc] going to run:\n{:?}", cmd); eprintln!("[cargo-miri rustc inside rustdoc] going to run:\n{:?}", cmd);
} }
exec_with_pipe(cmd, &env.stdin); exec_with_pipe(cmd, &env.stdin, format!("{}.stdin", out_filename("", "").display()));
} }
store_json(CrateRunInfo::RunWith(env));
return; return;
} }
@ -997,8 +1034,6 @@ fn out_filename(prefix: &str, suffix: &str) -> PathBuf {
"[cargo-miri rustc] target_crate={target_crate} runnable_crate={runnable_crate} info_query={info_query}" "[cargo-miri rustc] target_crate={target_crate} runnable_crate={runnable_crate} info_query={info_query}"
); );
} }
debug_cmd("[cargo-miri rustc]", verbose, &cmd);
exec(cmd);
// Create a stub .rlib file if "link" was requested by cargo. // Create a stub .rlib file if "link" was requested by cargo.
// This is necessary to prevent cargo from doing rebuilds all the time. // This is necessary to prevent cargo from doing rebuilds all the time.
@ -1013,6 +1048,9 @@ fn out_filename(prefix: &str, suffix: &str) -> PathBuf {
File::create(out_filename("", ".dll")).expect("failed to create fake .dll file"); File::create(out_filename("", ".dll")).expect("failed to create fake .dll file");
File::create(out_filename("", ".lib")).expect("failed to create fake .lib file"); File::create(out_filename("", ".lib")).expect("failed to create fake .lib file");
} }
debug_cmd("[cargo-miri rustc]", verbose, &cmd);
exec(cmd);
} }
#[derive(Debug, Copy, Clone, PartialEq)] #[derive(Debug, Copy, Clone, PartialEq)]
@ -1117,7 +1155,7 @@ fn phase_runner(mut binary_args: impl Iterator<Item = String>, phase: RunnerPhas
// Run it. // Run it.
debug_cmd("[cargo-miri runner]", verbose, &cmd); debug_cmd("[cargo-miri runner]", verbose, &cmd);
match phase { match phase {
RunnerPhase::Rustdoc => exec_with_pipe(cmd, &info.stdin), RunnerPhase::Rustdoc => exec_with_pipe(cmd, &info.stdin, format!("{}.stdin", binary)),
RunnerPhase::Cargo => exec(cmd), RunnerPhase::Cargo => exec(cmd),
} }
} }