Skip to content

Commit b12ed75

Browse files
authored
Unrolled build for rust-lang#119510
Rollup merge of rust-lang#119510 - saethlin:fatal-io-errors, r=WaffleLapkin,Nilstrieb Report I/O errors from rmeta encoding with emit_fatal rust-lang#119456 reminded me that I never did systematic testing to provoke the out-of-disk ICEs so I grepped through a recent crater run (rust-lang#119440 (comment)) for more out-of-disk ICEs on current master and yep there's 2 in there. So I finally cooked up a way to provoke for these crashes. I wrote a little `cdylib` crate that has a `#[no_mangle] pub extern "C" fn write` which occasionally reports `ENOSPC`, and prints a backtrace when it does. <details><summary><strong>code for the dylib</strong></summary> <p> ```rust // cargo add libc rand backtrace use rand::Rng; #[no_mangle] pub extern "C" fn write( fd: libc::c_int, buf: *const libc::c_void, count: libc::size_t, ) -> libc::ssize_t { if fd > 2 && rand::thread_rng().gen::<u8>() == 0 { let mut count = 0; backtrace::trace(|frame| { backtrace::resolve_frame(frame, |symbol| { if let Some(name) = symbol.name() { if count > 3 { eprintln!("{}", name); } } count += 1; }); true }); unsafe { *libc::__errno_location() = libc::ENOSPC; } return -1; } else { unsafe { let res = libc::syscall(libc::SYS_write, fd as usize, buf as usize, count as usize) as isize; if res < 0 { *libc::__errno_location() = -res as i32; -1 } else { res } } } } ``` </p> </details> Then `LD_PRELOAD` that dylib and repeatedly build a big project until it ICEs, such as with this: ```bash while true; do cargo clean LD_PRELOAD=/home/ben/evil/target/release/libevil.so cargo +stage1 check 2> errors if grep "thread 'rustc' panicked" errors; then break fi done ``` My "big project" for testing was an otherwise-empty project with `cargo add axum`. Before this PR, the above procedure finds a crash in between 1 and 15 minutes. With this PR, I have not found a crash in 30 minutes, and I'll be leaving this to run overnight (starting now). (A night has now passed, no crashes were found) I believe the problem is that even though since rust-lang#117301 we correctly check `FileEncoder` for errors on all paths, we use `emit_err`, so there is a window of time between the call to `emit_err` and the full error reporting where rustc believes it has emitted a valid rmeta file and will permit Cargo to launch a build for a dependent crate. Changing these calls to `emit_fatal` closes that window. I think there are a number of other cases where `emit_err` has been used instead of the more-correct `emit_fatal` such as https://github.com/rust-lang/rust/blob/e51e98dde6a60637b6a71b8105245b629ac3fe77/compiler/rustc_codegen_ssa/src/back/write.rs#L542 but unlike rmeta encoding I am not aware of those cases of those causing problems. r? ``@WaffleLapkin``
2 parents 1a47f5b + 94c43cc commit b12ed75

File tree

3 files changed

+6
-14
lines changed

3 files changed

+6
-14
lines changed

compiler/rustc_incremental/src/persist/file_format.rs

+3-11
Original file line numberDiff line numberDiff line change
@@ -55,18 +55,12 @@ where
5555
debug!("save: remove old file");
5656
}
5757
Err(err) if err.kind() == io::ErrorKind::NotFound => (),
58-
Err(err) => {
59-
sess.dcx().emit_err(errors::DeleteOld { name, path: path_buf, err });
60-
return;
61-
}
58+
Err(err) => sess.dcx().emit_fatal(errors::DeleteOld { name, path: path_buf, err }),
6259
}
6360

6461
let mut encoder = match FileEncoder::new(&path_buf) {
6562
Ok(encoder) => encoder,
66-
Err(err) => {
67-
sess.dcx().emit_err(errors::CreateNew { name, path: path_buf, err });
68-
return;
69-
}
63+
Err(err) => sess.dcx().emit_fatal(errors::CreateNew { name, path: path_buf, err }),
7064
};
7165

7266
write_file_header(&mut encoder, sess);
@@ -80,9 +74,7 @@ where
8074
);
8175
debug!("save: data written to disk successfully");
8276
}
83-
Err((path, err)) => {
84-
sess.dcx().emit_err(errors::WriteNew { name, path, err });
85-
}
77+
Err((path, err)) => sess.dcx().emit_fatal(errors::WriteNew { name, path, err }),
8678
}
8779
}
8880

compiler/rustc_interface/src/queries.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -332,7 +332,7 @@ impl Compiler {
332332
// the global context.
333333
_timer = Some(self.sess.timer("free_global_ctxt"));
334334
if let Err((path, error)) = queries.finish() {
335-
self.sess.dcx().emit_err(errors::FailedWritingFile { path: &path, error });
335+
self.sess.dcx().emit_fatal(errors::FailedWritingFile { path: &path, error });
336336
}
337337

338338
ret

compiler/rustc_metadata/src/rmeta/encoder.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -2255,12 +2255,12 @@ pub fn encode_metadata(tcx: TyCtxt<'_>, path: &Path) {
22552255
// If we forget this, compilation can succeed with an incomplete rmeta file,
22562256
// causing an ICE when the rmeta file is read by another compilation.
22572257
if let Err((path, err)) = ecx.opaque.finish() {
2258-
tcx.dcx().emit_err(FailWriteFile { path: &path, err });
2258+
tcx.dcx().emit_fatal(FailWriteFile { path: &path, err });
22592259
}
22602260

22612261
let file = ecx.opaque.file();
22622262
if let Err(err) = encode_root_position(file, root.position.get()) {
2263-
tcx.dcx().emit_err(FailWriteFile { path: ecx.opaque.path(), err });
2263+
tcx.dcx().emit_fatal(FailWriteFile { path: ecx.opaque.path(), err });
22642264
}
22652265

22662266
// Record metadata size for self-profiling

0 commit comments

Comments
 (0)