Auto merge of #111769 - saethlin:ctfe-backtrace-ctrlc, r=RalfJung
Print a backtrace in const eval if interrupted Demo: ```rust #![feature(const_eval_limit)] #![const_eval_limit = "0"] const OW: u64 = { let mut res: u64 = 0; let mut i = 0; while i < u64::MAX { res = res.wrapping_add(i); i += 1; } res }; fn main() { println!("{}", OW); } ``` ``` ╭ ➜ ben@archlinux:~/rust ╰ ➤ rustc +stage1 spin.rs ^Cerror[E0080]: evaluation of constant value failed --> spin.rs:8:33 | 8 | res = res.wrapping_add(i); | ^ Compilation was interrupted note: erroneous constant used --> spin.rs:15:20 | 15 | println!("{}", OW); | ^^ note: erroneous constant used --> spin.rs:15:20 | 15 | println!("{}", OW); | ^^ | = note: this note originates in the macro `$crate::format_args_nl` which comes from the expansion of the macro `println` (in Nightly builds, run with -Z macro-backtrace for more info) error: aborting due to previous error For more information about this error, try `rustc --explain E0080`. ```
This commit is contained in:
commit
c98ea0d808
13 changed files with 56 additions and 18 deletions
|
@ -3784,6 +3784,7 @@ dependencies = [
|
||||||
name = "rustc_driver_impl"
|
name = "rustc_driver_impl"
|
||||||
version = "0.0.0"
|
version = "0.0.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"ctrlc",
|
||||||
"libc",
|
"libc",
|
||||||
"rustc_ast",
|
"rustc_ast",
|
||||||
"rustc_ast_lowering",
|
"rustc_ast_lowering",
|
||||||
|
|
|
@ -146,6 +146,8 @@ const_eval_intern_kind = {$kind ->
|
||||||
*[other] {""}
|
*[other] {""}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const_eval_interrupted = compilation was interrupted
|
||||||
|
|
||||||
const_eval_invalid_align_details =
|
const_eval_invalid_align_details =
|
||||||
invalid align passed to `{$name}`: {$align} is {$err_kind ->
|
invalid align passed to `{$name}`: {$align} is {$err_kind ->
|
||||||
[not_power_of_two] not a power of 2
|
[not_power_of_two] not a power of 2
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use std::sync::atomic::Ordering::Relaxed;
|
||||||
|
|
||||||
use either::{Left, Right};
|
use either::{Left, Right};
|
||||||
|
|
||||||
use rustc_hir::def::DefKind;
|
use rustc_hir::def::DefKind;
|
||||||
|
@ -22,6 +24,7 @@ use crate::interpret::{
|
||||||
InternKind, InterpCx, InterpError, InterpResult, MPlaceTy, MemoryKind, OpTy, RefTracking,
|
InternKind, InterpCx, InterpError, InterpResult, MPlaceTy, MemoryKind, OpTy, RefTracking,
|
||||||
StackPopCleanup,
|
StackPopCleanup,
|
||||||
};
|
};
|
||||||
|
use crate::CTRL_C_RECEIVED;
|
||||||
|
|
||||||
// Returns a pointer to where the result lives
|
// Returns a pointer to where the result lives
|
||||||
#[instrument(level = "trace", skip(ecx, body))]
|
#[instrument(level = "trace", skip(ecx, body))]
|
||||||
|
@ -79,7 +82,11 @@ fn eval_body_using_ecx<'mir, 'tcx, R: InterpretationResult<'tcx>>(
|
||||||
ecx.storage_live_for_always_live_locals()?;
|
ecx.storage_live_for_always_live_locals()?;
|
||||||
|
|
||||||
// The main interpreter loop.
|
// The main interpreter loop.
|
||||||
while ecx.step()? {}
|
while ecx.step()? {
|
||||||
|
if CTRL_C_RECEIVED.load(Relaxed) {
|
||||||
|
throw_exhaust!(Interrupted);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Intern the result
|
// Intern the result
|
||||||
intern_const_alloc_recursive(ecx, intern_kind, &ret)?;
|
intern_const_alloc_recursive(ecx, intern_kind, &ret)?;
|
||||||
|
|
|
@ -884,6 +884,7 @@ impl ReportErrorExt for ResourceExhaustionInfo {
|
||||||
ResourceExhaustionInfo::StackFrameLimitReached => const_eval_stack_frame_limit_reached,
|
ResourceExhaustionInfo::StackFrameLimitReached => const_eval_stack_frame_limit_reached,
|
||||||
ResourceExhaustionInfo::MemoryExhausted => const_eval_memory_exhausted,
|
ResourceExhaustionInfo::MemoryExhausted => const_eval_memory_exhausted,
|
||||||
ResourceExhaustionInfo::AddressSpaceFull => const_eval_address_space_full,
|
ResourceExhaustionInfo::AddressSpaceFull => const_eval_address_space_full,
|
||||||
|
ResourceExhaustionInfo::Interrupted => const_eval_interrupted,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn add_args<G: EmissionGuarantee>(self, _: &mut Diag<'_, G>) {}
|
fn add_args<G: EmissionGuarantee>(self, _: &mut Diag<'_, G>) {}
|
||||||
|
|
|
@ -32,6 +32,8 @@ pub mod interpret;
|
||||||
pub mod transform;
|
pub mod transform;
|
||||||
pub mod util;
|
pub mod util;
|
||||||
|
|
||||||
|
use std::sync::atomic::AtomicBool;
|
||||||
|
|
||||||
pub use errors::ReportErrorExt;
|
pub use errors::ReportErrorExt;
|
||||||
|
|
||||||
use rustc_middle::{ty, util::Providers};
|
use rustc_middle::{ty, util::Providers};
|
||||||
|
@ -58,3 +60,8 @@ pub fn provide(providers: &mut Providers) {
|
||||||
util::check_validity_requirement(tcx, init_kind, param_env_and_ty)
|
util::check_validity_requirement(tcx, init_kind, param_env_and_ty)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// `rustc_driver::main` installs a handler that will set this to `true` if
|
||||||
|
/// the compiler has been sent a request to shut down, such as by a Ctrl-C.
|
||||||
|
/// This static lives here because it is only read by the interpreter.
|
||||||
|
pub static CTRL_C_RECEIVED: AtomicBool = AtomicBool::new(false);
|
||||||
|
|
|
@ -5,6 +5,7 @@ edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
# tidy-alphabetical-start
|
# tidy-alphabetical-start
|
||||||
|
ctrlc = "3.4.4"
|
||||||
rustc_ast = { path = "../rustc_ast" }
|
rustc_ast = { path = "../rustc_ast" }
|
||||||
rustc_ast_lowering = { path = "../rustc_ast_lowering" }
|
rustc_ast_lowering = { path = "../rustc_ast_lowering" }
|
||||||
rustc_ast_passes = { path = "../rustc_ast_passes" }
|
rustc_ast_passes = { path = "../rustc_ast_passes" }
|
||||||
|
|
|
@ -19,6 +19,7 @@ extern crate tracing;
|
||||||
|
|
||||||
use rustc_ast as ast;
|
use rustc_ast as ast;
|
||||||
use rustc_codegen_ssa::{traits::CodegenBackend, CodegenErrors, CodegenResults};
|
use rustc_codegen_ssa::{traits::CodegenBackend, CodegenErrors, CodegenResults};
|
||||||
|
use rustc_const_eval::CTRL_C_RECEIVED;
|
||||||
use rustc_data_structures::profiling::{
|
use rustc_data_structures::profiling::{
|
||||||
get_resident_set_size, print_time_passes_entry, TimePassesFormat,
|
get_resident_set_size, print_time_passes_entry, TimePassesFormat,
|
||||||
};
|
};
|
||||||
|
@ -1518,6 +1519,22 @@ pub fn init_logger(early_dcx: &EarlyDiagCtxt, cfg: rustc_log::LoggerConfig) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Install our usual `ctrlc` handler, which sets [`rustc_const_eval::CTRL_C_RECEIVED`].
|
||||||
|
/// Making this handler optional lets tools can install a different handler, if they wish.
|
||||||
|
pub fn install_ctrlc_handler() {
|
||||||
|
ctrlc::set_handler(move || {
|
||||||
|
// Indicate that we have been signaled to stop. If we were already signaled, exit
|
||||||
|
// immediately. In our interpreter loop we try to consult this value often, but if for
|
||||||
|
// whatever reason we don't get to that check or the cleanup we do upon finding that
|
||||||
|
// this bool has become true takes a long time, the exit here will promptly exit the
|
||||||
|
// process on the second Ctrl-C.
|
||||||
|
if CTRL_C_RECEIVED.swap(true, Ordering::Relaxed) {
|
||||||
|
std::process::exit(1);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.expect("Unable to install ctrlc handler");
|
||||||
|
}
|
||||||
|
|
||||||
pub fn main() -> ! {
|
pub fn main() -> ! {
|
||||||
let start_time = Instant::now();
|
let start_time = Instant::now();
|
||||||
let start_rss = get_resident_set_size();
|
let start_rss = get_resident_set_size();
|
||||||
|
@ -1528,6 +1545,8 @@ pub fn main() -> ! {
|
||||||
signal_handler::install();
|
signal_handler::install();
|
||||||
let mut callbacks = TimePassesCallbacks::default();
|
let mut callbacks = TimePassesCallbacks::default();
|
||||||
let using_internal_features = install_ice_hook(DEFAULT_BUG_REPORT_URL, |_| ());
|
let using_internal_features = install_ice_hook(DEFAULT_BUG_REPORT_URL, |_| ());
|
||||||
|
install_ctrlc_handler();
|
||||||
|
|
||||||
let exit_code = catch_with_exit_code(|| {
|
let exit_code = catch_with_exit_code(|| {
|
||||||
RunCompiler::new(&args::raw_args(&early_dcx)?, &mut callbacks)
|
RunCompiler::new(&args::raw_args(&early_dcx)?, &mut callbacks)
|
||||||
.set_using_internal_features(using_internal_features)
|
.set_using_internal_features(using_internal_features)
|
||||||
|
|
|
@ -482,6 +482,8 @@ pub enum ResourceExhaustionInfo {
|
||||||
MemoryExhausted,
|
MemoryExhausted,
|
||||||
/// The address space (of the target) is full.
|
/// The address space (of the target) is full.
|
||||||
AddressSpaceFull,
|
AddressSpaceFull,
|
||||||
|
/// The compiler got an interrupt signal (a user ran out of patience).
|
||||||
|
Interrupted,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A trait for machine-specific errors (or other "machine stop" conditions).
|
/// A trait for machine-specific errors (or other "machine stop" conditions).
|
||||||
|
|
|
@ -364,6 +364,10 @@ fn main() {
|
||||||
let args = rustc_driver::args::raw_args(&early_dcx)
|
let args = rustc_driver::args::raw_args(&early_dcx)
|
||||||
.unwrap_or_else(|_| std::process::exit(rustc_driver::EXIT_FAILURE));
|
.unwrap_or_else(|_| std::process::exit(rustc_driver::EXIT_FAILURE));
|
||||||
|
|
||||||
|
// Install the ctrlc handler that sets `rustc_const_eval::CTRL_C_RECEIVED`, even if
|
||||||
|
// MIRI_BE_RUSTC is set.
|
||||||
|
rustc_driver::install_ctrlc_handler();
|
||||||
|
|
||||||
// If the environment asks us to actually be rustc, then do that.
|
// If the environment asks us to actually be rustc, then do that.
|
||||||
if let Some(crate_kind) = env::var_os("MIRI_BE_RUSTC") {
|
if let Some(crate_kind) = env::var_os("MIRI_BE_RUSTC") {
|
||||||
// Earliest rustc setup.
|
// Earliest rustc setup.
|
||||||
|
|
|
@ -3,12 +3,13 @@
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use std::collections::hash_map::Entry;
|
use std::collections::hash_map::Entry;
|
||||||
use std::num::TryFromIntError;
|
use std::num::TryFromIntError;
|
||||||
use std::sync::atomic::{AtomicBool, Ordering::Relaxed};
|
use std::sync::atomic::Ordering::Relaxed;
|
||||||
use std::task::Poll;
|
use std::task::Poll;
|
||||||
use std::time::{Duration, SystemTime};
|
use std::time::{Duration, SystemTime};
|
||||||
|
|
||||||
use either::Either;
|
use either::Either;
|
||||||
|
|
||||||
|
use rustc_const_eval::CTRL_C_RECEIVED;
|
||||||
use rustc_data_structures::fx::FxHashMap;
|
use rustc_data_structures::fx::FxHashMap;
|
||||||
use rustc_hir::def_id::DefId;
|
use rustc_hir::def_id::DefId;
|
||||||
use rustc_index::{Idx, IndexVec};
|
use rustc_index::{Idx, IndexVec};
|
||||||
|
@ -1045,21 +1046,9 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
|
||||||
/// Run the core interpreter loop. Returns only when an interrupt occurs (an error or program
|
/// Run the core interpreter loop. Returns only when an interrupt occurs (an error or program
|
||||||
/// termination).
|
/// termination).
|
||||||
fn run_threads(&mut self) -> InterpResult<'tcx, !> {
|
fn run_threads(&mut self) -> InterpResult<'tcx, !> {
|
||||||
static SIGNALED: AtomicBool = AtomicBool::new(false);
|
let this = self.eval_context_mut();
|
||||||
ctrlc::set_handler(move || {
|
|
||||||
// Indicate that we have ben signaled to stop. If we were already signaled, exit
|
|
||||||
// immediately. In our interpreter loop we try to consult this value often, but if for
|
|
||||||
// whatever reason we don't get to that check or the cleanup we do upon finding that
|
|
||||||
// this bool has become true takes a long time, the exit here will promptly exit the
|
|
||||||
// process on the second Ctrl-C.
|
|
||||||
if SIGNALED.swap(true, Relaxed) {
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.unwrap();
|
|
||||||
let this = self.eval_context_mut();
|
|
||||||
loop {
|
loop {
|
||||||
if SIGNALED.load(Relaxed) {
|
if CTRL_C_RECEIVED.load(Relaxed) {
|
||||||
this.machine.handle_abnormal_termination();
|
this.machine.handle_abnormal_termination();
|
||||||
std::process::exit(1);
|
std::process::exit(1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -208,6 +208,7 @@ const PERMITTED_RUSTC_DEPENDENCIES: &[&str] = &[
|
||||||
"byteorder", // via ruzstd in object in thorin-dwp
|
"byteorder", // via ruzstd in object in thorin-dwp
|
||||||
"cc",
|
"cc",
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
|
"cfg_aliases",
|
||||||
"compiler_builtins",
|
"compiler_builtins",
|
||||||
"cpufeatures",
|
"cpufeatures",
|
||||||
"crc32fast",
|
"crc32fast",
|
||||||
|
@ -216,6 +217,7 @@ const PERMITTED_RUSTC_DEPENDENCIES: &[&str] = &[
|
||||||
"crossbeam-epoch",
|
"crossbeam-epoch",
|
||||||
"crossbeam-utils",
|
"crossbeam-utils",
|
||||||
"crypto-common",
|
"crypto-common",
|
||||||
|
"ctrlc",
|
||||||
"darling",
|
"darling",
|
||||||
"darling_core",
|
"darling_core",
|
||||||
"darling_macro",
|
"darling_macro",
|
||||||
|
@ -281,6 +283,7 @@ const PERMITTED_RUSTC_DEPENDENCIES: &[&str] = &[
|
||||||
"memmap2",
|
"memmap2",
|
||||||
"memoffset",
|
"memoffset",
|
||||||
"miniz_oxide",
|
"miniz_oxide",
|
||||||
|
"nix",
|
||||||
"nu-ansi-term",
|
"nu-ansi-term",
|
||||||
"num-conv",
|
"num-conv",
|
||||||
"num_cpus",
|
"num_cpus",
|
||||||
|
|
|
@ -4,9 +4,11 @@ include ../tools.mk
|
||||||
# ignore-cross-compile
|
# ignore-cross-compile
|
||||||
|
|
||||||
# Test compiler behavior in case environment specifies wrong jobserver.
|
# Test compiler behavior in case environment specifies wrong jobserver.
|
||||||
|
# Note that by default, the compiler uses file descriptors 0 (stdin), 1 (stdout), 2 (stderr),
|
||||||
|
# but also 3 and 4 for either end of the ctrl-c signal handler self-pipe.
|
||||||
|
|
||||||
all:
|
all:
|
||||||
bash -c 'echo "fn main() {}" | MAKEFLAGS="--jobserver-auth=3,3" $(RUSTC)' 2>&1 | diff cannot_open_fd.stderr -
|
bash -c 'echo "fn main() {}" | MAKEFLAGS="--jobserver-auth=5,5" $(RUSTC)' 2>&1 | diff cannot_open_fd.stderr -
|
||||||
bash -c 'echo "fn main() {}" | MAKEFLAGS="--jobserver-auth=3,3" $(RUSTC) - 3</dev/null' 2>&1 | diff not_a_pipe.stderr -
|
bash -c 'echo "fn main() {}" | MAKEFLAGS="--jobserver-auth=3,3" $(RUSTC) - 3</dev/null' 2>&1 | diff not_a_pipe.stderr -
|
||||||
|
|
||||||
# This test randomly fails, see https://github.com/rust-lang/rust/issues/110321
|
# This test randomly fails, see https://github.com/rust-lang/rust/issues/110321
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
warning: failed to connect to jobserver from environment variable `MAKEFLAGS="--jobserver-auth=3,3"`: cannot open file descriptor 3 from the jobserver environment variable value: Bad file descriptor (os error 9)
|
warning: failed to connect to jobserver from environment variable `MAKEFLAGS="--jobserver-auth=5,5"`: cannot open file descriptor 5 from the jobserver environment variable value: Bad file descriptor (os error 9)
|
||||||
|
|
|
|
||||||
= note: the build environment is likely misconfigured
|
= note: the build environment is likely misconfigured
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue