diff --git a/src/diagnostics.rs b/src/diagnostics.rs index 26442da6d65..3a0d17b22a7 100644 --- a/src/diagnostics.rs +++ b/src/diagnostics.rs @@ -70,7 +70,9 @@ pub enum NonHaltingDiagnostic { CreatedAlloc(AllocId, Size, Align, MemoryKind), FreedAlloc(AllocId), RejectedIsolatedOp(String), - ProgressReport, + ProgressReport { + block_count: u64, // how many basic blocks have been run so far + }, Int2Ptr { details: bool, }, @@ -261,6 +263,7 @@ pub fn report_error<'tcx, 'mir>( DiagLevel::Error, &if let Some(title) = title { format!("{}: {}", title, msg[0]) } else { msg[0].clone() }, msg, + vec![], helps, &stacktrace, ); @@ -307,6 +310,7 @@ fn report_msg<'mir, 'tcx>( diag_level: DiagLevel, title: &str, span_msg: Vec, + notes: Vec<(Option, String)>, helps: Vec<(Option, String)>, stacktrace: &[FrameInfo<'tcx>], ) { @@ -331,15 +335,22 @@ fn report_msg<'mir, 'tcx>( err.note("(no span available)"); } - // Show help messages. - if !helps.is_empty() { - for (span_data, help) in helps { - if let Some(span_data) = span_data { - err.span_help(span_data.span(), &help); - } else { - err.help(&help); - } + // Show note and help messages. + for (span_data, note) in ¬es { + if let Some(span_data) = span_data { + err.span_note(span_data.span(), note); + } else { + err.note(note); } + } + for (span_data, help) in &helps { + if let Some(span_data) = span_data { + err.span_help(span_data.span(), help); + } else { + err.help(help); + } + } + if notes.len() + helps.len() > 0 { // Add visual separator before backtrace. err.note("backtrace:"); } @@ -436,6 +447,21 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx // Show diagnostics. for e in diagnostics.drain(..) { use NonHaltingDiagnostic::*; + + let (title, diag_level) = match e { + RejectedIsolatedOp(_) => + ("operation rejected by isolation", DiagLevel::Warning), + Int2Ptr { .. } => ("integer-to-pointer cast", DiagLevel::Warning), + CreatedPointerTag(..) + | PoppedPointerTag(..) + | CreatedCallId(..) + | CreatedAlloc(..) + | FreedAlloc(..) + | ProgressReport { .. } + | WeakMemoryOutdatedLoad => + ("tracking was triggered", DiagLevel::Note), + }; + let msg = match e { CreatedPointerTag(tag, None) => format!("created tag {tag:?}"), @@ -465,7 +491,7 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx format!("freed allocation with id {id}"), RejectedIsolatedOp(ref op) => format!("{op} was made to return an error due to isolation"), - ProgressReport => + ProgressReport { .. } => format!("progress report: current operation being executed is here"), Int2Ptr { .. } => format!("integer-to-pointer cast"), @@ -473,18 +499,15 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx format!("weak memory emulation: outdated value returned from load"), }; - let (title, diag_level) = match e { - RejectedIsolatedOp(_) => - ("operation rejected by isolation", DiagLevel::Warning), - Int2Ptr { .. } => ("integer-to-pointer cast", DiagLevel::Warning), - CreatedPointerTag(..) - | PoppedPointerTag(..) - | CreatedCallId(..) - | CreatedAlloc(..) - | FreedAlloc(..) - | ProgressReport - | WeakMemoryOutdatedLoad => - ("tracking was triggered", DiagLevel::Note), + let notes = match e { + ProgressReport { block_count } => { + // It is important that each progress report is slightly different, since + // identical diagnostics are being deduplicated. + vec![ + (None, format!("so far, {block_count} basic blocks have been executed")), + ] + } + _ => vec![], }; let helps = match e { @@ -500,7 +523,7 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx _ => vec![], }; - report_msg(this, diag_level, title, vec![msg], helps, &stacktrace); + report_msg(this, diag_level, title, vec![msg], notes, helps, &stacktrace); } }); } @@ -519,6 +542,7 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx "the place in the program where the ICE was triggered", vec![], vec![], + vec![], &stacktrace, ); } diff --git a/src/machine.rs b/src/machine.rs index 4f7e3a6a71b..3be4d1d1b5b 100644 --- a/src/machine.rs +++ b/src/machine.rs @@ -374,8 +374,8 @@ pub struct Evaluator<'mir, 'tcx> { /// If `Some`, we will report the current stack every N basic blocks. pub(crate) report_progress: Option, - /// The number of blocks that passed since the last progress report. - pub(crate) since_progress_report: u32, + // The total number of blocks that have been executed. + pub(crate) basic_block_count: u64, /// Handle of the optional shared object file for external functions. pub external_so_lib: Option<(libloading::Library, std::path::PathBuf)>, @@ -433,7 +433,7 @@ impl<'mir, 'tcx> Evaluator<'mir, 'tcx> { weak_memory: config.weak_memory_emulation, preemption_rate: config.preemption_rate, report_progress: config.report_progress, - since_progress_report: 0, + basic_block_count: 0, external_so_lib: config.external_so_file.as_ref().map(|lib_file_path| { // Check if host target == the session target. if env!("TARGET") != target_triple { @@ -992,14 +992,14 @@ impl<'mir, 'tcx> Machine<'mir, 'tcx> for Evaluator<'mir, 'tcx> { } fn before_terminator(ecx: &mut InterpCx<'mir, 'tcx, Self>) -> InterpResult<'tcx> { + ecx.machine.basic_block_count += 1u64; // a u64 that is only incremented by 1 will "never" overflow // Possibly report our progress. if let Some(report_progress) = ecx.machine.report_progress { - if ecx.machine.since_progress_report >= report_progress { - register_diagnostic(NonHaltingDiagnostic::ProgressReport); - ecx.machine.since_progress_report = 0; + if ecx.machine.basic_block_count % u64::from(report_progress) == 0 { + register_diagnostic(NonHaltingDiagnostic::ProgressReport { + block_count: ecx.machine.basic_block_count, + }); } - // Cannot overflow, since it is strictly less than `report_progress`. - ecx.machine.since_progress_report += 1; } // These are our preemption points. ecx.maybe_preempt_active_thread();