2019-06-10 10:59:03 +02:00
|
|
|
//! Diagnostics creation and emission for `rustc`.
|
2019-06-09 12:04:40 +02:00
|
|
|
//!
|
|
|
|
//! This module contains the code for creating and emitting diagnostics.
|
|
|
|
|
2020-09-23 21:51:56 +02:00
|
|
|
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/nightly-rustc/")]
|
2023-11-13 07:39:17 -05:00
|
|
|
#![doc(rust_logo)]
|
|
|
|
#![feature(rustdoc_internals)]
|
2023-01-15 14:29:20 +01:00
|
|
|
#![feature(array_windows)]
|
2023-12-18 16:31:15 +11:00
|
|
|
#![feature(associated_type_defaults)]
|
2023-04-09 00:37:21 +02:00
|
|
|
#![feature(extract_if)]
|
2021-08-16 17:29:49 +02:00
|
|
|
#![feature(if_let_guard)]
|
2022-08-20 20:40:08 +02:00
|
|
|
#![feature(let_chains)]
|
2022-03-09 16:11:28 -08:00
|
|
|
#![feature(never_type)]
|
2022-06-10 15:50:06 +01:00
|
|
|
#![feature(rustc_attrs)]
|
2023-01-05 03:39:07 +01:00
|
|
|
#![feature(yeet_expr)]
|
|
|
|
#![feature(try_blocks)]
|
|
|
|
#![feature(box_patterns)]
|
|
|
|
#![feature(error_reporter)]
|
2022-01-27 09:44:25 +00:00
|
|
|
#![allow(incomplete_features)]
|
2023-08-22 07:06:38 -04:00
|
|
|
#![allow(internal_features)]
|
2016-06-21 18:08:13 -04:00
|
|
|
|
2020-06-11 15:49:57 +01:00
|
|
|
#[macro_use]
|
|
|
|
extern crate rustc_macros;
|
|
|
|
|
2021-09-20 15:24:47 +00:00
|
|
|
#[macro_use]
|
|
|
|
extern crate tracing;
|
|
|
|
|
2023-07-03 18:08:49 +09:00
|
|
|
extern crate self as rustc_errors;
|
|
|
|
|
2023-12-19 15:11:38 +11:00
|
|
|
pub use diagnostic::{
|
|
|
|
AddToDiagnostic, DecorateLint, Diagnostic, DiagnosticArg, DiagnosticArgValue, DiagnosticId,
|
|
|
|
DiagnosticStyledString, IntoDiagnosticArg, SubDiagnostic,
|
|
|
|
};
|
|
|
|
pub use diagnostic_builder::{
|
|
|
|
BugAbort, DiagnosticBuilder, EmissionGuarantee, FatalAbort, IntoDiagnostic,
|
|
|
|
};
|
|
|
|
pub use diagnostic_impls::{
|
|
|
|
DiagnosticArgFromDisplay, DiagnosticSymbolList, ExpectedLifetimeParameter,
|
|
|
|
IndicateAnonymousLifetime, InvalidFlushedDelayedDiagnosticLevel, LabelKind,
|
|
|
|
SingleLabelManySpans,
|
|
|
|
};
|
2016-06-21 18:08:13 -04:00
|
|
|
pub use emitter::ColorConfig;
|
2023-12-19 15:11:38 +11:00
|
|
|
pub use rustc_error_messages::{
|
|
|
|
fallback_fluent_bundle, fluent_bundle, DelayDm, DiagnosticMessage, FluentBundle,
|
|
|
|
LanguageIdentifier, LazyFallbackBundle, MultiSpan, SpanLabel, SubdiagnosticMessage,
|
|
|
|
};
|
|
|
|
pub use rustc_lint_defs::{pluralize, Applicability};
|
|
|
|
pub use rustc_span::fatal_error::{FatalError, FatalErrorMarker};
|
|
|
|
pub use rustc_span::ErrorGuaranteed;
|
|
|
|
pub use snippet::Style;
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2023-12-19 15:11:38 +11:00
|
|
|
// Used by external projects such as `rust-gpu`.
|
|
|
|
// See https://github.com/rust-lang/rust/pull/115393.
|
|
|
|
pub use termcolor::{Color, ColorSpec, WriteColor};
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2023-12-19 15:11:38 +11:00
|
|
|
use crate::diagnostic_impls::{DelayedAtWithNewline, DelayedAtWithoutNewline};
|
2020-10-31 03:14:32 +01:00
|
|
|
use emitter::{is_case_difference, DynEmitter, Emitter, EmitterWriter};
|
2019-04-17 13:26:38 -04:00
|
|
|
use registry::Registry;
|
2022-07-16 15:16:57 +02:00
|
|
|
use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexMap, FxIndexSet};
|
2023-04-07 23:11:20 -04:00
|
|
|
use rustc_data_structures::stable_hasher::{Hash128, StableHasher};
|
2020-10-31 03:14:32 +01:00
|
|
|
use rustc_data_structures::sync::{Lock, Lrc};
|
2019-12-25 13:38:57 -05:00
|
|
|
use rustc_data_structures::AtomicRef;
|
2023-12-19 15:11:38 +11:00
|
|
|
use rustc_lint_defs::LintExpectationId;
|
2019-12-31 20:15:40 +03:00
|
|
|
use rustc_span::source_map::SourceMap;
|
2023-09-02 04:02:11 +00:00
|
|
|
use rustc_span::{Loc, Span, DUMMY_SP};
|
2023-12-19 15:11:38 +11:00
|
|
|
use std::backtrace::{Backtrace, BacktraceStatus};
|
2017-06-11 13:31:40 +02:00
|
|
|
use std::borrow::Cow;
|
2023-01-08 23:35:43 +01:00
|
|
|
use std::error::Report;
|
2022-12-31 01:20:59 +00:00
|
|
|
use std::fmt;
|
2022-04-14 23:07:57 -07:00
|
|
|
use std::hash::Hash;
|
2023-03-03 22:25:18 +00:00
|
|
|
use std::io::Write;
|
2021-02-18 05:25:45 -06:00
|
|
|
use std::num::NonZeroUsize;
|
2018-01-21 12:47:58 +01:00
|
|
|
use std::panic;
|
2023-03-03 22:25:18 +00:00
|
|
|
use std::path::{Path, PathBuf};
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2023-12-19 15:11:38 +11:00
|
|
|
use Level::*;
|
2018-02-27 10:33:02 -08:00
|
|
|
|
2019-06-05 21:13:56 +02:00
|
|
|
pub mod annotate_snippet_emitter_writer;
|
2017-08-19 03:09:55 +03:00
|
|
|
mod diagnostic;
|
|
|
|
mod diagnostic_builder;
|
2022-10-04 20:56:05 -04:00
|
|
|
mod diagnostic_impls;
|
2015-12-15 14:11:27 +13:00
|
|
|
pub mod emitter;
|
2023-01-05 03:39:07 +01:00
|
|
|
pub mod error;
|
2019-06-05 21:13:56 +02:00
|
|
|
pub mod json;
|
2017-07-02 13:46:38 -07:00
|
|
|
mod lock;
|
Add a simple markdown parser for formatting `rustc --explain`
Currently, the output of `rustc --explain foo` displays the raw markdown in a
pager. This is acceptable, but using actual formatting makes it easier to
understand.
This patch consists of three major components:
1. A markdown parser. This is an extremely simple non-backtracking recursive
implementation that requires normalization of the final token stream
2. A utility to write the token stream to an output buffer
3. Configuration within rustc_driver_impl to invoke this combination for
`--explain`. Like the current implementation, it first attempts to print to
a pager with a fallback colorized terminal, and standard print as a last
resort.
If color is disabled, or if the output does not support it, or if printing
with color fails, it will write the raw markdown (which matches current
behavior).
Pagers known to support color are: `less` (with `-r`), `bat` (aka `catbat`),
and `delta`.
The markdown parser does not support the entire markdown specification, but
should support the following with reasonable accuracy:
- Headings, including formatting
- Comments
- Code, inline and fenced block (no indented block)
- Strong, emphasis, and strikethrough formatted text
- Links, anchor, inline, and reference-style
- Horizontal rules
- Unordered and ordered list items, including formatting
This parser and writer should be reusable by other systems if ever needed.
2022-12-19 12:09:40 -06:00
|
|
|
pub mod markdown;
|
2016-06-21 18:08:13 -04:00
|
|
|
pub mod registry;
|
2019-12-22 17:42:04 -05:00
|
|
|
mod snippet;
|
2017-07-02 13:46:38 -07:00
|
|
|
mod styled_buffer;
|
2023-01-08 23:35:43 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests;
|
2022-08-10 17:30:47 +01:00
|
|
|
pub mod translation;
|
2022-03-24 02:03:04 +00:00
|
|
|
|
2023-12-19 15:26:24 +11:00
|
|
|
pub type PErr<'a> = DiagnosticBuilder<'a>;
|
2022-09-21 14:01:39 +10:00
|
|
|
pub type PResult<'a, T> = Result<T, PErr<'a>>;
|
2019-10-11 13:06:36 +02:00
|
|
|
|
2023-11-22 09:53:07 +11:00
|
|
|
rustc_fluent_macro::fluent_messages! { "../messages.ftl" }
|
2022-10-13 10:13:02 +01:00
|
|
|
|
2019-10-11 13:06:36 +02:00
|
|
|
// `PResult` is used a lot. Make sure it doesn't unintentionally get bigger.
|
2022-09-30 09:23:55 +10:00
|
|
|
// (See also the comment on `DiagnosticBuilderInner`'s `diagnostic` field.)
|
2021-03-06 16:02:48 +00:00
|
|
|
#[cfg(all(target_arch = "x86_64", target_pointer_width = "64"))]
|
2022-01-24 15:14:40 +00:00
|
|
|
rustc_data_structures::static_assert_size!(PResult<'_, ()>, 16);
|
2022-09-20 15:41:42 +02:00
|
|
|
#[cfg(all(target_arch = "x86_64", target_pointer_width = "64"))]
|
2022-03-08 19:07:01 +00:00
|
|
|
rustc_data_structures::static_assert_size!(PResult<'_, bool>, 16);
|
2019-10-11 13:06:36 +02:00
|
|
|
|
2020-06-11 15:49:57 +01:00
|
|
|
#[derive(Debug, PartialEq, Eq, Clone, Copy, Hash, Encodable, Decodable)]
|
2019-02-08 02:45:53 -08:00
|
|
|
pub enum SuggestionStyle {
|
|
|
|
/// Hide the suggested code when displaying this suggestion inline.
|
|
|
|
HideCodeInline,
|
2019-02-09 03:39:08 -08:00
|
|
|
/// Always hide the suggested code but display the message.
|
2019-02-08 02:45:53 -08:00
|
|
|
HideCodeAlways,
|
2019-02-09 03:39:08 -08:00
|
|
|
/// Do not display this suggestion in the cli output, it is only meant for tools.
|
|
|
|
CompletelyHidden,
|
2019-02-08 02:45:53 -08:00
|
|
|
/// Always show the suggested code.
|
|
|
|
/// This will *not* show the code if the suggestion is inline *and* the suggested code is
|
|
|
|
/// empty.
|
|
|
|
ShowCode,
|
2019-10-03 13:22:18 -07:00
|
|
|
/// Always show the suggested code independently.
|
|
|
|
ShowAlways,
|
2019-02-08 02:45:53 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SuggestionStyle {
|
|
|
|
fn hide_inline(&self) -> bool {
|
2020-10-26 21:02:48 -04:00
|
|
|
!matches!(*self, SuggestionStyle::ShowCode)
|
2019-02-08 02:45:53 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-11 15:49:57 +01:00
|
|
|
#[derive(Clone, Debug, PartialEq, Hash, Encodable, Decodable)]
|
2015-12-13 13:12:47 +01:00
|
|
|
pub struct CodeSuggestion {
|
2017-05-09 10:04:24 +02:00
|
|
|
/// Each substitute can have multiple variants due to multiple
|
|
|
|
/// applicable suggestions
|
|
|
|
///
|
|
|
|
/// `foo.bar` might be replaced with `a.b` or `x.y` by replacing
|
|
|
|
/// `foo` and `bar` on their own:
|
|
|
|
///
|
2022-04-15 15:04:34 -07:00
|
|
|
/// ```ignore (illustrative)
|
2017-05-09 10:04:24 +02:00
|
|
|
/// vec![
|
2017-11-03 16:17:33 +01:00
|
|
|
/// Substitution { parts: vec![(0..3, "a"), (4..7, "b")] },
|
|
|
|
/// Substitution { parts: vec![(0..3, "x"), (4..7, "y")] },
|
2017-05-09 10:04:24 +02:00
|
|
|
/// ]
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// or by replacing the entire span:
|
|
|
|
///
|
2022-04-15 15:04:34 -07:00
|
|
|
/// ```ignore (illustrative)
|
2017-11-03 16:17:33 +01:00
|
|
|
/// vec![
|
|
|
|
/// Substitution { parts: vec![(0..7, "a.b")] },
|
|
|
|
/// Substitution { parts: vec![(0..7, "x.y")] },
|
|
|
|
/// ]
|
2017-05-09 10:04:24 +02:00
|
|
|
/// ```
|
2017-11-03 16:17:33 +01:00
|
|
|
pub substitutions: Vec<Substitution>,
|
2022-03-23 07:34:20 +00:00
|
|
|
pub msg: DiagnosticMessage,
|
2019-02-08 02:45:53 -08:00
|
|
|
/// Visual representation of this suggestion.
|
|
|
|
pub style: SuggestionStyle,
|
2018-01-18 17:17:46 +05:30
|
|
|
/// Whether or not the suggestion is approximate
|
|
|
|
///
|
|
|
|
/// Sometimes we may show suggestions with placeholders,
|
|
|
|
/// which are useful for users but not useful for
|
|
|
|
/// tools like rustfix
|
2018-04-25 14:51:06 -07:00
|
|
|
pub applicability: Applicability,
|
2016-06-21 18:08:13 -04:00
|
|
|
}
|
|
|
|
|
2020-06-11 15:49:57 +01:00
|
|
|
#[derive(Clone, Debug, PartialEq, Hash, Encodable, Decodable)]
|
2017-05-11 15:26:22 +02:00
|
|
|
/// See the docs on `CodeSuggestion::substitutions`
|
|
|
|
pub struct Substitution {
|
2017-11-03 16:17:33 +01:00
|
|
|
pub parts: Vec<SubstitutionPart>,
|
|
|
|
}
|
|
|
|
|
2020-06-11 15:49:57 +01:00
|
|
|
#[derive(Clone, Debug, PartialEq, Hash, Encodable, Decodable)]
|
2017-11-03 16:17:33 +01:00
|
|
|
pub struct SubstitutionPart {
|
2017-05-11 15:26:22 +02:00
|
|
|
pub span: Span,
|
2017-11-03 16:17:33 +01:00
|
|
|
pub snippet: String,
|
2017-05-11 15:26:22 +02:00
|
|
|
}
|
|
|
|
|
2021-06-21 19:07:19 -07:00
|
|
|
/// Used to translate between `Span`s and byte positions within a single output line in highlighted
|
|
|
|
/// code of structured suggestions.
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub struct SubstitutionHighlight {
|
|
|
|
start: usize,
|
|
|
|
end: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SubstitutionPart {
|
|
|
|
pub fn is_addition(&self, sm: &SourceMap) -> bool {
|
2022-09-11 19:19:07 +00:00
|
|
|
!self.snippet.is_empty() && !self.replaces_meaningful_content(sm)
|
2021-06-21 19:07:19 -07:00
|
|
|
}
|
|
|
|
|
2022-09-11 19:19:07 +00:00
|
|
|
pub fn is_deletion(&self, sm: &SourceMap) -> bool {
|
|
|
|
self.snippet.trim().is_empty() && self.replaces_meaningful_content(sm)
|
2021-06-21 19:07:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_replacement(&self, sm: &SourceMap) -> bool {
|
2022-09-11 19:19:07 +00:00
|
|
|
!self.snippet.is_empty() && self.replaces_meaningful_content(sm)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn replaces_meaningful_content(&self, sm: &SourceMap) -> bool {
|
|
|
|
sm.span_to_snippet(self.span)
|
|
|
|
.map_or(!self.span.is_empty(), |snippet| !snippet.trim().is_empty())
|
2021-06-21 19:07:19 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-13 13:12:47 +01:00
|
|
|
impl CodeSuggestion {
|
2019-10-13 21:48:39 -07:00
|
|
|
/// Returns the assembled code suggestions, whether they should be shown with an underline
|
|
|
|
/// and whether the substitution only differs in capitalization.
|
2021-06-21 19:07:19 -07:00
|
|
|
pub fn splice_lines(
|
|
|
|
&self,
|
|
|
|
sm: &SourceMap,
|
|
|
|
) -> Vec<(String, Vec<SubstitutionPart>, Vec<Vec<SubstitutionHighlight>>, bool)> {
|
|
|
|
// For the `Vec<Vec<SubstitutionHighlight>>` value, the first level of the vector
|
|
|
|
// corresponds to the output snippet's lines, while the second level corresponds to the
|
|
|
|
// substrings within that line that should be highlighted.
|
|
|
|
|
2019-12-31 20:15:40 +03:00
|
|
|
use rustc_span::{CharPos, Pos};
|
2015-12-13 13:12:47 +01:00
|
|
|
|
2023-08-24 17:09:14 -04:00
|
|
|
/// Extracts a substring from the provided `line_opt` based on the specified low and high indices,
|
|
|
|
/// appends it to the given buffer `buf`, and returns the count of newline characters in the substring
|
|
|
|
/// for accurate highlighting.
|
|
|
|
/// If `line_opt` is `None`, a newline character is appended to the buffer, and 0 is returned.
|
|
|
|
///
|
|
|
|
/// ## Returns
|
|
|
|
///
|
|
|
|
/// The count of newline characters in the extracted substring.
|
2016-10-18 23:13:02 +05:30
|
|
|
fn push_trailing(
|
|
|
|
buf: &mut String,
|
2019-02-07 03:53:01 +09:00
|
|
|
line_opt: Option<&Cow<'_, str>>,
|
2016-10-18 23:13:02 +05:30
|
|
|
lo: &Loc,
|
|
|
|
hi_opt: Option<&Loc>,
|
2021-06-21 19:07:19 -07:00
|
|
|
) -> usize {
|
|
|
|
let mut line_count = 0;
|
2023-08-24 17:09:14 -04:00
|
|
|
// Convert CharPos to Usize, as CharPose is character offset
|
|
|
|
// Extract low index and high index
|
2016-10-18 23:13:02 +05:30
|
|
|
let (lo, hi_opt) = (lo.col.to_usize(), hi_opt.map(|hi| hi.col.to_usize()));
|
2015-12-13 13:12:47 +01:00
|
|
|
if let Some(line) = line_opt {
|
2017-02-25 22:05:30 +09:00
|
|
|
if let Some(lo) = line.char_indices().map(|(i, _)| i).nth(lo) {
|
2023-08-24 17:09:14 -04:00
|
|
|
// Get high index while account for rare unicode and emoji with char_indices
|
2017-02-25 22:05:30 +09:00
|
|
|
let hi_opt = hi_opt.and_then(|hi| line.char_indices().map(|(i, _)| i).nth(hi));
|
2019-01-17 13:53:21 +00:00
|
|
|
match hi_opt {
|
2023-08-24 17:09:14 -04:00
|
|
|
// If high index exist, take string from low to high index
|
2021-06-21 19:07:19 -07:00
|
|
|
Some(hi) if hi > lo => {
|
2023-08-24 17:09:14 -04:00
|
|
|
// count how many '\n' exist
|
2021-06-21 19:07:19 -07:00
|
|
|
line_count = line[lo..hi].matches('\n').count();
|
|
|
|
buf.push_str(&line[lo..hi])
|
|
|
|
}
|
2019-01-17 13:53:21 +00:00
|
|
|
Some(_) => (),
|
2023-08-24 17:09:14 -04:00
|
|
|
// If high index absence, take string from low index till end string.len
|
2021-06-21 19:07:19 -07:00
|
|
|
None => {
|
2023-08-24 17:09:14 -04:00
|
|
|
// count how many '\n' exist
|
2021-06-21 19:07:19 -07:00
|
|
|
line_count = line[lo..].matches('\n').count();
|
|
|
|
buf.push_str(&line[lo..])
|
|
|
|
}
|
2019-01-17 13:53:21 +00:00
|
|
|
}
|
2015-12-13 13:12:47 +01:00
|
|
|
}
|
2023-08-24 17:09:14 -04:00
|
|
|
// If high index is None
|
2020-03-04 15:53:14 +01:00
|
|
|
if hi_opt.is_none() {
|
2015-12-13 13:12:47 +01:00
|
|
|
buf.push('\n');
|
|
|
|
}
|
|
|
|
}
|
2021-06-21 19:07:19 -07:00
|
|
|
line_count
|
2015-12-13 13:12:47 +01:00
|
|
|
}
|
|
|
|
|
2017-11-03 16:17:33 +01:00
|
|
|
assert!(!self.substitutions.is_empty());
|
|
|
|
|
|
|
|
self.substitutions
|
|
|
|
.iter()
|
2020-01-16 18:27:18 -08:00
|
|
|
.filter(|subst| {
|
|
|
|
// Suggestions coming from macros can have malformed spans. This is a heavy
|
|
|
|
// handed approach to avoid ICEs by ignoring the suggestion outright.
|
2020-02-22 16:07:05 +02:00
|
|
|
let invalid = subst.parts.iter().any(|item| sm.is_valid_span(item.span).is_err());
|
2020-01-16 18:27:18 -08:00
|
|
|
if invalid {
|
|
|
|
debug!("splice_lines: suggestion contains an invalid span: {:?}", subst);
|
|
|
|
}
|
|
|
|
!invalid
|
|
|
|
})
|
2017-11-03 16:17:33 +01:00
|
|
|
.cloned()
|
2020-01-24 18:03:09 -08:00
|
|
|
.filter_map(|mut substitution| {
|
2017-11-03 16:17:33 +01:00
|
|
|
// Assumption: all spans are in the same file, and all spans
|
|
|
|
// are disjoint. Sort in ascending order.
|
|
|
|
substitution.parts.sort_by_key(|part| part.span.lo());
|
2019-12-22 17:42:04 -05:00
|
|
|
|
2017-11-03 16:17:33 +01:00
|
|
|
// Find the bounding span.
|
2020-01-24 18:03:09 -08:00
|
|
|
let lo = substitution.parts.iter().map(|part| part.span.lo()).min()?;
|
|
|
|
let hi = substitution.parts.iter().map(|part| part.span.hi()).max()?;
|
2019-08-11 01:44:55 +03:00
|
|
|
let bounding_span = Span::with_root_ctxt(lo, hi);
|
2020-01-24 18:03:09 -08:00
|
|
|
// The different spans might belong to different contexts, if so ignore suggestion.
|
2020-02-22 16:07:05 +02:00
|
|
|
let lines = sm.span_to_lines(bounding_span).ok()?;
|
2020-03-13 17:01:35 -04:00
|
|
|
assert!(!lines.lines.is_empty() || bounding_span.is_dummy());
|
2019-12-22 17:42:04 -05:00
|
|
|
|
2020-03-05 15:31:11 +00:00
|
|
|
// We can't splice anything if the source is unavailable.
|
2023-08-31 12:50:44 +02:00
|
|
|
if !sm.ensure_source_file_source_present(&lines.file) {
|
2020-03-05 15:31:11 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2021-06-21 19:07:19 -07:00
|
|
|
let mut highlights = vec![];
|
2017-11-03 16:17:33 +01:00
|
|
|
// To build up the result, we do this for each span:
|
|
|
|
// - push the line segment trailing the previous span
|
|
|
|
// (at the beginning a "phantom" span pointing at the start of the line)
|
|
|
|
// - push lines between the previous and current span (if any)
|
|
|
|
// - if the previous and current span are not on the same line
|
|
|
|
// push the line segment leading up to the current span
|
|
|
|
// - splice in the span substitution
|
|
|
|
//
|
|
|
|
// Finally push the trailing line segment of the last span
|
2020-02-22 16:07:05 +02:00
|
|
|
let sf = &lines.file;
|
|
|
|
let mut prev_hi = sm.lookup_char_pos(bounding_span.lo());
|
2017-11-03 16:17:33 +01:00
|
|
|
prev_hi.col = CharPos::from_usize(0);
|
2020-03-13 17:01:35 -04:00
|
|
|
let mut prev_line =
|
|
|
|
lines.lines.get(0).and_then(|line0| sf.get_line(line0.line_index));
|
2017-11-03 16:17:33 +01:00
|
|
|
let mut buf = String::new();
|
2019-12-22 17:42:04 -05:00
|
|
|
|
2021-06-21 19:07:19 -07:00
|
|
|
let mut line_highlight = vec![];
|
2021-08-12 19:33:19 +00:00
|
|
|
// We need to keep track of the difference between the existing code and the added
|
|
|
|
// or deleted code in order to point at the correct column *after* substitution.
|
|
|
|
let mut acc = 0;
|
2017-11-03 16:17:33 +01:00
|
|
|
for part in &substitution.parts {
|
2020-02-22 16:07:05 +02:00
|
|
|
let cur_lo = sm.lookup_char_pos(part.span.lo());
|
2017-05-09 10:04:24 +02:00
|
|
|
if prev_hi.line == cur_lo.line {
|
2021-06-21 19:07:19 -07:00
|
|
|
let mut count =
|
|
|
|
push_trailing(&mut buf, prev_line.as_ref(), &prev_hi, Some(&cur_lo));
|
|
|
|
while count > 0 {
|
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
2021-08-12 19:33:19 +00:00
|
|
|
acc = 0;
|
2021-06-21 19:07:19 -07:00
|
|
|
count -= 1;
|
|
|
|
}
|
2017-05-09 10:04:24 +02:00
|
|
|
} else {
|
2021-08-17 15:46:42 +00:00
|
|
|
acc = 0;
|
2021-06-21 19:07:19 -07:00
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
|
|
|
let mut count = push_trailing(&mut buf, prev_line.as_ref(), &prev_hi, None);
|
|
|
|
while count > 0 {
|
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
|
|
|
count -= 1;
|
|
|
|
}
|
2017-05-09 10:04:24 +02:00
|
|
|
// push lines between the previous and current span (if any)
|
|
|
|
for idx in prev_hi.line..(cur_lo.line - 1) {
|
2020-02-22 16:07:05 +02:00
|
|
|
if let Some(line) = sf.get_line(idx) {
|
2017-06-11 13:31:40 +02:00
|
|
|
buf.push_str(line.as_ref());
|
2017-05-09 10:04:24 +02:00
|
|
|
buf.push('\n');
|
2021-06-21 19:07:19 -07:00
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
2019-12-22 17:42:04 -05:00
|
|
|
}
|
|
|
|
}
|
2020-02-22 16:07:05 +02:00
|
|
|
if let Some(cur_line) = sf.get_line(cur_lo.line - 1) {
|
2020-04-12 11:36:37 -04:00
|
|
|
let end = match cur_line.char_indices().nth(cur_lo.col.to_usize()) {
|
|
|
|
Some((i, _)) => i,
|
|
|
|
None => cur_line.len(),
|
|
|
|
};
|
2019-07-25 15:59:38 -07:00
|
|
|
buf.push_str(&cur_line[..end]);
|
2017-05-09 10:04:24 +02:00
|
|
|
}
|
|
|
|
}
|
2021-06-21 19:07:19 -07:00
|
|
|
// Add a whole line highlight per line in the snippet.
|
2021-08-23 12:42:08 +00:00
|
|
|
let len: isize = part
|
|
|
|
.snippet
|
|
|
|
.split('\n')
|
|
|
|
.next()
|
|
|
|
.unwrap_or(&part.snippet)
|
|
|
|
.chars()
|
|
|
|
.map(|c| match c {
|
|
|
|
'\t' => 4,
|
|
|
|
_ => 1,
|
|
|
|
})
|
|
|
|
.sum();
|
2021-06-21 19:07:19 -07:00
|
|
|
line_highlight.push(SubstitutionHighlight {
|
2021-08-12 19:33:19 +00:00
|
|
|
start: (cur_lo.col.0 as isize + acc) as usize,
|
2021-08-23 12:42:08 +00:00
|
|
|
end: (cur_lo.col.0 as isize + acc + len) as usize,
|
2021-06-21 19:07:19 -07:00
|
|
|
});
|
2021-08-12 19:33:19 +00:00
|
|
|
buf.push_str(&part.snippet);
|
2021-08-23 12:42:08 +00:00
|
|
|
let cur_hi = sm.lookup_char_pos(part.span.hi());
|
2023-05-19 00:44:14 +02:00
|
|
|
// Account for the difference between the width of the current code and the
|
|
|
|
// snippet being suggested, so that the *later* suggestions are correctly
|
|
|
|
// aligned on the screen. Note that cur_hi and cur_lo can be on different
|
|
|
|
// lines, so cur_hi.col can be smaller than cur_lo.col
|
|
|
|
acc += len - (cur_hi.col.0 as isize - cur_lo.col.0 as isize);
|
2021-08-23 12:42:08 +00:00
|
|
|
prev_hi = cur_hi;
|
2021-08-12 19:33:19 +00:00
|
|
|
prev_line = sf.get_line(prev_hi.line - 1);
|
2021-06-21 19:07:19 -07:00
|
|
|
for line in part.snippet.split('\n').skip(1) {
|
2021-08-12 19:33:19 +00:00
|
|
|
acc = 0;
|
2021-06-21 19:07:19 -07:00
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
2021-08-23 12:42:08 +00:00
|
|
|
let end: usize = line
|
|
|
|
.chars()
|
|
|
|
.map(|c| match c {
|
|
|
|
'\t' => 4,
|
|
|
|
_ => 1,
|
|
|
|
})
|
|
|
|
.sum();
|
|
|
|
line_highlight.push(SubstitutionHighlight { start: 0, end });
|
2021-06-21 19:07:19 -07:00
|
|
|
}
|
2015-12-13 13:12:47 +01:00
|
|
|
}
|
2021-06-21 19:07:19 -07:00
|
|
|
highlights.push(std::mem::take(&mut line_highlight));
|
2020-02-22 16:07:05 +02:00
|
|
|
let only_capitalization = is_case_difference(sm, &buf, bounding_span);
|
2017-05-10 13:19:29 +02:00
|
|
|
// if the replacement already ends with a newline, don't print the next line
|
|
|
|
if !buf.ends_with('\n') {
|
2017-11-03 16:17:33 +01:00
|
|
|
push_trailing(&mut buf, prev_line.as_ref(), &prev_hi, None);
|
2017-05-10 13:19:29 +02:00
|
|
|
}
|
2017-08-18 12:46:28 +02:00
|
|
|
// remove trailing newlines
|
|
|
|
while buf.ends_with('\n') {
|
|
|
|
buf.pop();
|
|
|
|
}
|
2021-06-21 19:07:19 -07:00
|
|
|
Some((buf, substitution.parts, highlights, only_capitalization))
|
2017-11-03 16:17:33 +01:00
|
|
|
})
|
|
|
|
.collect()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Signifies that the compiler died with an explicit call to `.bug`
|
|
|
|
/// or `.span_bug` rather than a failed assertion, etc.
|
|
|
|
pub struct ExplicitBug;
|
|
|
|
|
2022-12-31 01:56:59 +00:00
|
|
|
/// Signifies that the compiler died with an explicit call to `.delay_*_bug`
|
2022-12-31 01:20:59 +00:00
|
|
|
/// rather than a failed assertion, etc.
|
2022-12-31 01:56:59 +00:00
|
|
|
pub struct DelayedBugPanic;
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2023-12-20 10:43:41 +11:00
|
|
|
/// A `DiagCtxt` deals with errors and other compiler output.
|
2019-05-02 05:06:33 +03:00
|
|
|
/// Certain errors (fatal, bug, unimpl) may cause immediate exit,
|
|
|
|
/// others log errors for later reporting.
|
2023-12-17 21:48:57 +11:00
|
|
|
pub struct DiagCtxt {
|
2023-12-17 21:58:27 +11:00
|
|
|
inner: Lock<DiagCtxtInner>,
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
2017-11-20 18:03:20 +00:00
|
|
|
|
2019-09-23 22:28:14 +02:00
|
|
|
/// This inner struct exists to keep it all behind a single lock;
|
|
|
|
/// this is done to prevent possible deadlocks in a multi-threaded compiler,
|
|
|
|
/// as well as inconsistent state observation.
|
2023-12-17 21:58:27 +11:00
|
|
|
struct DiagCtxtInner {
|
2023-12-18 08:47:03 +11:00
|
|
|
flags: DiagCtxtFlags,
|
2021-07-20 23:23:22 -04:00
|
|
|
/// The number of lint errors that have been emitted.
|
|
|
|
lint_err_count: usize,
|
2019-06-22 12:46:48 +01:00
|
|
|
/// The number of errors that have been emitted, including duplicates.
|
|
|
|
///
|
|
|
|
/// This is not necessarily the count that's reported to the user once
|
|
|
|
/// compilation ends.
|
2019-09-07 12:09:52 -04:00
|
|
|
err_count: usize,
|
2020-08-22 22:24:48 +03:00
|
|
|
warn_count: usize,
|
2019-09-07 12:09:52 -04:00
|
|
|
deduplicated_err_count: usize,
|
2020-10-31 03:14:32 +01:00
|
|
|
emitter: Box<DynEmitter>,
|
2023-11-30 15:57:21 +11:00
|
|
|
span_delayed_bugs: Vec<DelayedDiagnostic>,
|
2023-11-30 16:05:50 +11:00
|
|
|
good_path_delayed_bugs: Vec<DelayedDiagnostic>,
|
2022-06-04 00:43:24 +02:00
|
|
|
/// This flag indicates that an expected diagnostic was emitted and suppressed.
|
2023-11-30 16:05:50 +11:00
|
|
|
/// This is used for the `good_path_delayed_bugs` check.
|
2022-06-04 00:43:24 +02:00
|
|
|
suppressed_expected_diag: bool,
|
2017-10-25 15:01:06 +02:00
|
|
|
|
2019-04-17 12:03:39 +03:00
|
|
|
/// This set contains the `DiagnosticId` of all emitted diagnostics to avoid
|
|
|
|
/// emitting the same diagnostic with extended help (`--teach`) twice, which
|
2021-04-19 15:57:08 +03:00
|
|
|
/// would be unnecessary repetition.
|
2019-09-07 12:09:52 -04:00
|
|
|
taught_diagnostics: FxHashSet<DiagnosticId>,
|
2018-03-03 06:20:26 +01:00
|
|
|
|
2022-10-08 14:42:38 +02:00
|
|
|
/// Used to suggest rustc --explain `<error code>`
|
2022-07-16 15:16:57 +02:00
|
|
|
emitted_diagnostic_codes: FxIndexSet<DiagnosticId>,
|
2017-10-25 15:01:06 +02:00
|
|
|
|
2019-04-17 12:03:39 +03:00
|
|
|
/// This set contains a hash of every diagnostic that has been emitted by
|
2023-12-20 10:43:41 +11:00
|
|
|
/// this `DiagCtxt`. These hashes is used to avoid emitting the same error
|
2019-04-17 12:03:39 +03:00
|
|
|
/// twice.
|
2023-04-07 23:11:20 -04:00
|
|
|
emitted_diagnostics: FxHashSet<Hash128>,
|
2019-09-23 04:45:21 +02:00
|
|
|
|
|
|
|
/// Stashed diagnostics emitted in one stage of the compiler that may be
|
|
|
|
/// stolen by other stages (e.g. to improve them and add more information).
|
|
|
|
/// The stashed diagnostics count towards the total error count.
|
|
|
|
/// When `.abort_if_errors()` is called, these are also emitted.
|
|
|
|
stashed_diagnostics: FxIndexMap<(Span, StashKey), Diagnostic>,
|
2020-03-11 16:30:09 +01:00
|
|
|
|
|
|
|
/// The warning count, used for a recap upon finishing
|
|
|
|
deduplicated_warn_count: usize,
|
2020-08-13 15:41:52 -04:00
|
|
|
|
|
|
|
future_breakage_diagnostics: Vec<Diagnostic>,
|
2021-08-06 23:28:58 +02:00
|
|
|
|
2022-04-05 18:23:38 +02:00
|
|
|
/// The [`Self::unstable_expect_diagnostics`] should be empty when this struct is
|
2022-03-29 00:42:41 +02:00
|
|
|
/// dropped. However, it can have values if the compilation is stopped early
|
|
|
|
/// or is only partially executed. To avoid ICEs, like in rust#94953 we only
|
2022-04-05 18:23:38 +02:00
|
|
|
/// check if [`Self::unstable_expect_diagnostics`] is empty, if the expectation ids
|
2022-03-29 00:42:41 +02:00
|
|
|
/// have been converted.
|
|
|
|
check_unstable_expect_diagnostics: bool,
|
|
|
|
|
2023-07-03 08:39:44 +08:00
|
|
|
/// Expected [`Diagnostic`][struct@diagnostic::Diagnostic]s store a [`LintExpectationId`] as part of
|
2021-11-20 20:45:27 +01:00
|
|
|
/// the lint level. [`LintExpectationId`]s created early during the compilation
|
|
|
|
/// (before `HirId`s have been defined) are not stable and can therefore not be
|
|
|
|
/// stored on disk. This buffer stores these diagnostics until the ID has been
|
2023-07-03 08:39:44 +08:00
|
|
|
/// replaced by a stable [`LintExpectationId`]. The [`Diagnostic`][struct@diagnostic::Diagnostic]s are the
|
2021-11-20 20:45:27 +01:00
|
|
|
/// submitted for storage and added to the list of fulfilled expectations.
|
|
|
|
unstable_expect_diagnostics: Vec<Diagnostic>,
|
|
|
|
|
2021-08-06 23:28:58 +02:00
|
|
|
/// expected diagnostic will have the level `Expect` which additionally
|
|
|
|
/// carries the [`LintExpectationId`] of the expectation that can be
|
|
|
|
/// marked as fulfilled. This is a collection of all [`LintExpectationId`]s
|
|
|
|
/// that have been marked as fulfilled this way.
|
|
|
|
///
|
|
|
|
/// [RFC-2383]: https://rust-lang.github.io/rfcs/2383-lint-reasons.html
|
|
|
|
fulfilled_expectations: FxHashSet<LintExpectationId>,
|
2023-03-03 22:25:18 +00:00
|
|
|
|
|
|
|
/// The file where the ICE information is stored. This allows delayed_span_bug backtraces to be
|
|
|
|
/// stored along side the main panic backtrace.
|
|
|
|
ice_file: Option<PathBuf>,
|
2019-09-23 04:45:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A key denoting where from a diagnostic was stashed.
|
2022-09-08 06:15:33 +09:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
|
2019-09-23 04:45:21 +02:00
|
|
|
pub enum StashKey {
|
|
|
|
ItemNoType,
|
2022-08-16 00:16:14 +09:00
|
|
|
UnderscoreForArrayLengths,
|
2022-08-17 06:52:47 -05:00
|
|
|
EarlySyntaxWarning,
|
2022-10-05 05:35:34 +00:00
|
|
|
CallIntoMethod,
|
2022-09-01 18:48:09 +00:00
|
|
|
/// When an invalid lifetime e.g. `'2` should be reinterpreted
|
|
|
|
/// as a char literal in the parser
|
|
|
|
LifetimeIsChar,
|
2022-11-16 19:10:38 +00:00
|
|
|
/// Maybe there was a typo where a comma was forgotten before
|
|
|
|
/// FRU syntax
|
|
|
|
MaybeFruTypo,
|
2022-09-18 15:35:21 +08:00
|
|
|
CallAssocMethod,
|
2023-05-17 16:59:39 +08:00
|
|
|
TraitMissingMethod,
|
2023-05-22 23:33:34 +00:00
|
|
|
OpaqueHiddenTypeMismatch,
|
2023-08-20 00:56:22 +08:00
|
|
|
MaybeForgetReturn,
|
2023-10-26 17:30:53 +00:00
|
|
|
/// Query cycle detected, stashing in favor of a better error.
|
|
|
|
Cycle,
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
|
2021-04-30 19:38:06 +02:00
|
|
|
fn default_track_diagnostic(d: &mut Diagnostic, f: &mut dyn FnMut(&mut Diagnostic)) {
|
|
|
|
(*f)(d)
|
|
|
|
}
|
2018-03-15 10:03:36 +01:00
|
|
|
|
2021-04-30 19:38:06 +02:00
|
|
|
pub static TRACK_DIAGNOSTICS: AtomicRef<fn(&mut Diagnostic, &mut dyn FnMut(&mut Diagnostic))> =
|
|
|
|
AtomicRef::new(&(default_track_diagnostic as _));
|
2018-03-15 10:03:36 +01:00
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
#[derive(Copy, Clone, Default)]
|
2023-12-18 08:47:03 +11:00
|
|
|
pub struct DiagCtxtFlags {
|
2018-09-15 06:27:55 +02:00
|
|
|
/// If false, warning-level lints are suppressed.
|
|
|
|
/// (rustc: see `--allow warnings` and `--cap-lints`)
|
2017-11-20 18:03:20 +00:00
|
|
|
pub can_emit_warnings: bool,
|
2023-09-22 13:33:55 +02:00
|
|
|
/// If Some, the Nth error-level diagnostic is upgraded to bug-level.
|
2018-09-15 06:27:55 +02:00
|
|
|
/// (rustc: see `-Z treat-err-as-bug`)
|
2021-02-18 05:25:45 -06:00
|
|
|
pub treat_err_as_bug: Option<NonZeroUsize>,
|
2018-09-15 06:27:55 +02:00
|
|
|
/// If true, immediately emit diagnostics that would otherwise be buffered.
|
|
|
|
/// (rustc: see `-Z dont-buffer-diagnostics` and `-Z treat-err-as-bug`)
|
|
|
|
pub dont_buffer_diagnostics: bool,
|
2023-11-30 15:01:11 +11:00
|
|
|
/// If true, immediately print bugs registered with `span_delayed_bug`.
|
2018-09-15 06:27:55 +02:00
|
|
|
/// (rustc: see `-Z report-delayed-bugs`)
|
2018-07-19 17:53:44 +02:00
|
|
|
pub report_delayed_bugs: bool,
|
2019-12-15 17:12:30 +02:00
|
|
|
/// Show macro backtraces.
|
|
|
|
/// (rustc: see `-Z macro-backtrace`)
|
|
|
|
pub macro_backtrace: bool,
|
2019-12-29 22:10:47 +03:00
|
|
|
/// If true, identical diagnostics are reported only once.
|
|
|
|
pub deduplicate_diagnostics: bool,
|
2022-10-19 00:08:20 +02:00
|
|
|
/// Track where errors are created. Enabled with `-Ztrack-diagnostics`.
|
|
|
|
pub track_diagnostics: bool,
|
2017-11-20 18:03:20 +00:00
|
|
|
}
|
|
|
|
|
2023-12-17 21:58:27 +11:00
|
|
|
impl Drop for DiagCtxtInner {
|
2018-07-19 17:53:44 +02:00
|
|
|
fn drop(&mut self) {
|
2019-09-23 04:45:21 +02:00
|
|
|
self.emit_stashed_diagnostics();
|
|
|
|
|
|
|
|
if !self.has_errors() {
|
2023-11-30 15:57:21 +11:00
|
|
|
let bugs = std::mem::replace(&mut self.span_delayed_bugs, Vec::new());
|
2023-11-30 15:01:11 +11:00
|
|
|
self.flush_delayed(bugs, "no errors encountered even though `span_delayed_bug` issued");
|
2020-08-22 22:24:48 +03:00
|
|
|
}
|
|
|
|
|
2023-11-30 16:05:50 +11:00
|
|
|
// FIXME(eddyb) this explains what `good_path_delayed_bugs` are!
|
2023-11-30 15:57:21 +11:00
|
|
|
// They're `span_delayed_bugs` but for "require some diagnostic happened"
|
2022-01-23 23:11:37 +00:00
|
|
|
// instead of "require some error happened". Sadly that isn't ideal, as
|
|
|
|
// lints can be `#[allow]`'d, potentially leading to this triggering.
|
|
|
|
// Also, "good path" should be replaced with a better naming.
|
2023-12-04 14:06:28 +11:00
|
|
|
let has_any_message = self.err_count > 0 || self.lint_err_count > 0 || self.warn_count > 0;
|
|
|
|
if !has_any_message && !self.suppressed_expected_diag && !std::thread::panicking() {
|
2023-11-30 16:05:50 +11:00
|
|
|
let bugs = std::mem::replace(&mut self.good_path_delayed_bugs, Vec::new());
|
2020-08-22 22:24:48 +03:00
|
|
|
self.flush_delayed(
|
2022-12-31 01:56:59 +00:00
|
|
|
bugs,
|
2023-11-30 16:05:50 +11:00
|
|
|
"no warnings or errors encountered even though `good_path_delayed_bugs` issued",
|
2020-08-22 22:24:48 +03:00
|
|
|
);
|
2018-07-19 17:53:44 +02:00
|
|
|
}
|
2022-03-05 21:54:49 +01:00
|
|
|
|
2022-03-29 00:42:41 +02:00
|
|
|
if self.check_unstable_expect_diagnostics {
|
|
|
|
assert!(
|
|
|
|
self.unstable_expect_diagnostics.is_empty(),
|
|
|
|
"all diagnostics with unstable expectations should have been converted",
|
|
|
|
);
|
|
|
|
}
|
2018-07-19 17:53:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-17 21:48:57 +11:00
|
|
|
impl DiagCtxt {
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn with_tty_emitter(
|
2020-02-22 16:07:05 +02:00
|
|
|
sm: Option<Lrc<SourceMap>>,
|
2022-04-12 09:34:40 +01:00
|
|
|
fallback_bundle: LazyFallbackBundle,
|
2019-09-23 22:28:14 +02:00
|
|
|
) -> Self {
|
2023-07-25 13:09:53 +00:00
|
|
|
let emitter = Box::new(EmitterWriter::stderr(ColorConfig::Auto, fallback_bundle).sm(sm));
|
2023-07-25 10:27:34 +00:00
|
|
|
Self::with_emitter(emitter)
|
|
|
|
}
|
|
|
|
pub fn disable_warnings(mut self) -> Self {
|
|
|
|
self.inner.get_mut().flags.can_emit_warnings = false;
|
|
|
|
self
|
2019-09-23 22:28:14 +02:00
|
|
|
}
|
|
|
|
|
2023-12-18 08:47:03 +11:00
|
|
|
pub fn with_flags(mut self, flags: DiagCtxtFlags) -> Self {
|
2023-07-25 10:27:34 +00:00
|
|
|
self.inner.get_mut().flags = flags;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn with_ice_file(mut self, ice_file: PathBuf) -> Self {
|
|
|
|
self.inner.get_mut().ice_file = Some(ice_file);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-10-31 03:14:32 +01:00
|
|
|
pub fn with_emitter(emitter: Box<DynEmitter>) -> Self {
|
2019-09-23 22:28:14 +02:00
|
|
|
Self {
|
2023-12-17 21:58:27 +11:00
|
|
|
inner: Lock::new(DiagCtxtInner {
|
2023-12-18 08:47:03 +11:00
|
|
|
flags: DiagCtxtFlags { can_emit_warnings: true, ..Default::default() },
|
2021-07-20 23:23:22 -04:00
|
|
|
lint_err_count: 0,
|
2019-09-07 12:09:52 -04:00
|
|
|
err_count: 0,
|
2020-08-22 22:24:48 +03:00
|
|
|
warn_count: 0,
|
2019-09-07 12:09:52 -04:00
|
|
|
deduplicated_err_count: 0,
|
2020-03-11 16:30:09 +01:00
|
|
|
deduplicated_warn_count: 0,
|
2020-10-31 03:14:32 +01:00
|
|
|
emitter,
|
2023-11-30 15:57:21 +11:00
|
|
|
span_delayed_bugs: Vec::new(),
|
2023-11-30 16:05:50 +11:00
|
|
|
good_path_delayed_bugs: Vec::new(),
|
2022-06-04 00:43:24 +02:00
|
|
|
suppressed_expected_diag: false,
|
2019-09-07 12:09:52 -04:00
|
|
|
taught_diagnostics: Default::default(),
|
|
|
|
emitted_diagnostic_codes: Default::default(),
|
|
|
|
emitted_diagnostics: Default::default(),
|
2019-09-23 04:45:21 +02:00
|
|
|
stashed_diagnostics: Default::default(),
|
2020-08-13 15:41:52 -04:00
|
|
|
future_breakage_diagnostics: Vec::new(),
|
2022-03-29 00:42:41 +02:00
|
|
|
check_unstable_expect_diagnostics: false,
|
2021-11-20 20:45:27 +01:00
|
|
|
unstable_expect_diagnostics: Vec::new(),
|
2021-08-06 23:28:58 +02:00
|
|
|
fulfilled_expectations: Default::default(),
|
2023-07-25 10:27:34 +00:00
|
|
|
ice_file: None,
|
2019-09-07 12:09:52 -04:00
|
|
|
}),
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
/// Translate `message` eagerly with `args` to `SubdiagnosticMessage::Eager`.
|
2022-10-03 14:14:51 +01:00
|
|
|
pub fn eagerly_translate<'a>(
|
|
|
|
&self,
|
|
|
|
message: DiagnosticMessage,
|
|
|
|
args: impl Iterator<Item = DiagnosticArg<'a, 'static>>,
|
|
|
|
) -> SubdiagnosticMessage {
|
Use `Cow` in `{D,Subd}iagnosticMessage`.
Each of `{D,Subd}iagnosticMessage::{Str,Eager}` has a comment:
```
// FIXME(davidtwco): can a `Cow<'static, str>` be used here?
```
This commit answers that question in the affirmative. It's not the most
compelling change ever, but it might be worth merging.
This requires changing the `impl<'a> From<&'a str>` impls to `impl
From<&'static str>`, which involves a bunch of knock-on changes that
require/result in call sites being a little more precise about exactly
what kind of string they use to create errors, and not just `&str`. This
will result in fewer unnecessary allocations, though this will not have
any notable perf effects given that these are error paths.
Note that I was lazy within Clippy, using `to_string` in a few places to
preserve the existing string imprecision. I could have used `impl
Into<{D,Subd}iagnosticMessage>` in various places as is done in the
compiler, but that would have required changes to *many* call sites
(mostly changing `&format("...")` to `format!("...")`) which didn't seem
worthwhile.
2023-05-04 10:55:21 +10:00
|
|
|
SubdiagnosticMessage::Eager(Cow::from(self.eagerly_translate_to_string(message, args)))
|
2022-08-19 14:48:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Translate `message` eagerly with `args` to `String`.
|
|
|
|
pub fn eagerly_translate_to_string<'a>(
|
|
|
|
&self,
|
|
|
|
message: DiagnosticMessage,
|
|
|
|
args: impl Iterator<Item = DiagnosticArg<'a, 'static>>,
|
|
|
|
) -> String {
|
2022-10-03 14:14:51 +01:00
|
|
|
let inner = self.inner.borrow();
|
|
|
|
let args = crate::translation::to_fluent_args(args);
|
2022-08-19 14:48:15 +01:00
|
|
|
inner.emitter.translate_message(&message, &args).map_err(Report::new).unwrap().to_string()
|
2022-10-03 14:14:51 +01:00
|
|
|
}
|
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
// This is here to not allow mutation of flags;
|
2020-03-29 15:24:45 +02:00
|
|
|
// as of this writing it's only used in tests in librustc_middle.
|
2019-09-07 12:09:52 -04:00
|
|
|
pub fn can_emit_warnings(&self) -> bool {
|
2023-12-01 08:23:34 +11:00
|
|
|
self.inner.borrow_mut().flags.can_emit_warnings
|
2016-03-25 18:17:04 +01:00
|
|
|
}
|
|
|
|
|
2018-01-06 13:33:20 +01:00
|
|
|
/// Resets the diagnostic error count as well as the cached emitted diagnostics.
|
|
|
|
///
|
2019-02-08 14:53:55 +01:00
|
|
|
/// NOTE: *do not* call this function from rustc. It is only meant to be called from external
|
2018-01-06 13:33:20 +01:00
|
|
|
/// tools that want to reuse a `Parser` cleaning the previously emitted diagnostics as well as
|
|
|
|
/// the overall count of emitted error diagnostics.
|
2017-08-10 09:17:03 +09:00
|
|
|
pub fn reset_err_count(&self) {
|
2019-09-07 12:09:52 -04:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
inner.err_count = 0;
|
2020-08-22 22:24:48 +03:00
|
|
|
inner.warn_count = 0;
|
2019-10-02 04:13:02 +03:00
|
|
|
inner.deduplicated_err_count = 0;
|
2020-03-11 16:30:09 +01:00
|
|
|
inner.deduplicated_warn_count = 0;
|
2019-10-02 04:13:02 +03:00
|
|
|
|
|
|
|
// actually free the underlying memory (which `clear` would not do)
|
2023-11-30 15:57:21 +11:00
|
|
|
inner.span_delayed_bugs = Default::default();
|
2023-11-30 16:05:50 +11:00
|
|
|
inner.good_path_delayed_bugs = Default::default();
|
2019-10-02 04:13:02 +03:00
|
|
|
inner.taught_diagnostics = Default::default();
|
|
|
|
inner.emitted_diagnostic_codes = Default::default();
|
|
|
|
inner.emitted_diagnostics = Default::default();
|
|
|
|
inner.stashed_diagnostics = Default::default();
|
2019-09-23 04:45:21 +02:00
|
|
|
}
|
|
|
|
|
2022-11-16 12:21:15 -06:00
|
|
|
/// Stash a given diagnostic with the given `Span` and [`StashKey`] as the key.
|
|
|
|
/// Retrieve a stashed diagnostic with `steal_diagnostic`.
|
2019-09-23 04:45:21 +02:00
|
|
|
pub fn stash_diagnostic(&self, span: Span, key: StashKey, diag: Diagnostic) {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
|
|
|
|
let key = (span.with_parent(None), key);
|
|
|
|
|
|
|
|
if diag.is_error() {
|
|
|
|
if matches!(diag.level, Level::Error { lint: true }) {
|
|
|
|
inner.lint_err_count += 1;
|
|
|
|
} else {
|
|
|
|
inner.err_count += 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Warnings are only automatically flushed if they're forced.
|
|
|
|
if diag.is_force_warn() {
|
|
|
|
inner.warn_count += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME(Centril, #69537): Consider reintroducing panic on overwriting a stashed diagnostic
|
|
|
|
// if/when we have a more robust macro-friendly replacement for `(span, key)` as a key.
|
|
|
|
// See the PR for a discussion.
|
|
|
|
inner.stashed_diagnostics.insert(key, diag);
|
2019-09-23 04:45:21 +02:00
|
|
|
}
|
|
|
|
|
2022-11-16 12:21:15 -06:00
|
|
|
/// Steal a previously stashed diagnostic with the given `Span` and [`StashKey`] as the key.
|
2022-01-27 09:44:25 +00:00
|
|
|
pub fn steal_diagnostic(&self, span: Span, key: StashKey) -> Option<DiagnosticBuilder<'_, ()>> {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
let key = (span.with_parent(None), key);
|
|
|
|
let diag = inner.stashed_diagnostics.remove(&key)?;
|
|
|
|
if diag.is_error() {
|
|
|
|
if matches!(diag.level, Level::Error { lint: true }) {
|
|
|
|
inner.lint_err_count -= 1;
|
|
|
|
} else {
|
|
|
|
inner.err_count -= 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if diag.is_force_warn() {
|
|
|
|
inner.warn_count -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(DiagnosticBuilder::new_diagnostic(self, diag))
|
2019-09-23 04:45:21 +02:00
|
|
|
}
|
|
|
|
|
2022-09-16 11:24:14 +09:00
|
|
|
pub fn has_stashed_diagnostic(&self, span: Span, key: StashKey) -> bool {
|
2021-04-30 23:57:02 +02:00
|
|
|
self.inner.borrow().stashed_diagnostics.get(&(span.with_parent(None), key)).is_some()
|
2022-09-16 11:24:14 +09:00
|
|
|
}
|
|
|
|
|
2019-09-23 04:45:21 +02:00
|
|
|
/// Emit all stashed diagnostics.
|
2022-01-22 18:49:12 -06:00
|
|
|
pub fn emit_stashed_diagnostics(&self) -> Option<ErrorGuaranteed> {
|
|
|
|
self.inner.borrow_mut().emit_stashed_diagnostics()
|
2017-08-10 09:17:03 +09:00
|
|
|
}
|
|
|
|
|
2019-09-23 22:28:14 +02:00
|
|
|
/// Construct a builder at the `Warning` level at the given `span` and with the `msg`.
|
2021-07-06 13:47:03 +02:00
|
|
|
///
|
2022-01-26 03:46:56 +00:00
|
|
|
/// Attempting to `.emit()` the builder will only emit if either:
|
|
|
|
/// * `can_emit_warnings` is `true`
|
|
|
|
/// * `is_force_warn` was set in `DiagnosticId::Lint`
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-01-27 09:44:25 +00:00
|
|
|
pub fn struct_span_warn(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2022-01-27 09:44:25 +00:00
|
|
|
) -> DiagnosticBuilder<'_, ()> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_warn(msg);
|
|
|
|
result.set_span(span);
|
2015-12-18 16:15:53 +13:00
|
|
|
result
|
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
|
|
|
/// Construct a builder at the `Warning` level at the given `span` and with the `msg`.
|
|
|
|
/// Also include a code.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn struct_span_warn_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2019-09-23 22:28:14 +02:00
|
|
|
code: DiagnosticId,
|
2022-01-27 09:44:25 +00:00
|
|
|
) -> DiagnosticBuilder<'_, ()> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_span_warn(span, msg);
|
2017-10-27 08:21:22 +02:00
|
|
|
result.code(code);
|
2015-12-18 16:15:53 +13:00
|
|
|
result
|
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
|
|
|
/// Construct a builder at the `Warning` level with the `msg`.
|
2021-07-06 13:47:03 +02:00
|
|
|
///
|
2022-01-26 03:46:56 +00:00
|
|
|
/// Attempting to `.emit()` the builder will only emit if either:
|
|
|
|
/// * `can_emit_warnings` is `true`
|
|
|
|
/// * `is_force_warn` was set in `DiagnosticId::Lint`
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn struct_warn(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_, ()> {
|
2023-12-04 10:44:57 +11:00
|
|
|
DiagnosticBuilder::new(self, Level::Warning(None), msg)
|
2022-06-05 12:33:45 +02:00
|
|
|
}
|
|
|
|
|
2020-08-13 15:41:52 -04:00
|
|
|
/// Construct a builder at the `Allow` level with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-31 16:14:29 +01:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn struct_allow(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_, ()> {
|
2020-08-13 15:41:52 -04:00
|
|
|
DiagnosticBuilder::new(self, Level::Allow, msg)
|
|
|
|
}
|
|
|
|
|
2021-08-06 23:18:16 +02:00
|
|
|
/// Construct a builder at the `Expect` level with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn struct_expect(
|
|
|
|
&self,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
id: LintExpectationId,
|
|
|
|
) -> DiagnosticBuilder<'_, ()> {
|
2021-08-06 23:18:16 +02:00
|
|
|
DiagnosticBuilder::new(self, Level::Expect(id), msg)
|
|
|
|
}
|
|
|
|
|
2019-09-23 22:28:14 +02:00
|
|
|
/// Construct a builder at the `Error` level at the given `span` and with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-01-27 09:44:25 +00:00
|
|
|
pub fn struct_span_err(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2023-12-19 15:26:24 +11:00
|
|
|
) -> DiagnosticBuilder<'_> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_err(msg);
|
|
|
|
result.set_span(span);
|
2015-12-23 19:27:20 +13:00
|
|
|
result
|
2015-12-18 16:15:53 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
|
|
|
/// Construct a builder at the `Error` level at the given `span`, with the `msg`, and `code`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn struct_span_err_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2019-09-23 22:28:14 +02:00
|
|
|
code: DiagnosticId,
|
2023-12-19 15:26:24 +11:00
|
|
|
) -> DiagnosticBuilder<'_> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_span_err(span, msg);
|
2017-10-27 08:21:22 +02:00
|
|
|
result.code(code);
|
2015-12-23 19:27:20 +13:00
|
|
|
result
|
2015-12-18 16:15:53 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
|
|
|
/// Construct a builder at the `Error` level with the `msg`.
|
2017-05-29 18:46:29 +02:00
|
|
|
// FIXME: This method should be removed (every error should have an associated error code).
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2023-12-19 15:26:24 +11:00
|
|
|
pub fn struct_err(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_> {
|
2023-12-04 10:44:57 +11:00
|
|
|
DiagnosticBuilder::new(self, Level::Error { lint: false }, msg)
|
2021-07-20 23:23:22 -04:00
|
|
|
}
|
|
|
|
|
2019-09-23 22:28:14 +02:00
|
|
|
/// Construct a builder at the `Error` level with the `msg` and the `code`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-01-27 09:44:25 +00:00
|
|
|
pub fn struct_err_with_code(
|
|
|
|
&self,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2022-01-27 09:44:25 +00:00
|
|
|
code: DiagnosticId,
|
2023-12-19 15:26:24 +11:00
|
|
|
) -> DiagnosticBuilder<'_> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_err(msg);
|
2017-10-27 08:21:22 +02:00
|
|
|
result.code(code);
|
2017-05-29 18:46:29 +02:00
|
|
|
result
|
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-03-31 08:35:17 +01:00
|
|
|
/// Construct a builder at the `Warn` level with the `msg` and the `code`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-31 08:35:17 +01:00
|
|
|
pub fn struct_warn_with_code(
|
|
|
|
&self,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
code: DiagnosticId,
|
|
|
|
) -> DiagnosticBuilder<'_, ()> {
|
|
|
|
let mut result = self.struct_warn(msg);
|
|
|
|
result.code(code);
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2019-09-23 22:28:14 +02:00
|
|
|
/// Construct a builder at the `Fatal` level at the given `span` and with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn struct_span_fatal(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2023-12-18 16:31:15 +11:00
|
|
|
) -> DiagnosticBuilder<'_, FatalAbort> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_fatal(msg);
|
|
|
|
result.set_span(span);
|
2015-12-23 19:27:20 +13:00
|
|
|
result
|
2015-12-18 16:15:53 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
|
|
|
/// Construct a builder at the `Fatal` level at the given `span`, with the `msg`, and `code`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-31 16:14:29 +01:00
|
|
|
#[track_caller]
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn struct_span_fatal_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2019-09-23 22:28:14 +02:00
|
|
|
code: DiagnosticId,
|
2023-12-18 16:31:15 +11:00
|
|
|
) -> DiagnosticBuilder<'_, FatalAbort> {
|
2019-09-23 22:28:14 +02:00
|
|
|
let mut result = self.struct_span_fatal(span, msg);
|
2017-10-27 08:21:22 +02:00
|
|
|
result.code(code);
|
2015-12-23 19:27:20 +13:00
|
|
|
result
|
2015-12-18 16:15:53 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2023-12-13 15:19:34 +11:00
|
|
|
/// Construct a builder at the `Fatal` level with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-31 16:14:29 +01:00
|
|
|
#[track_caller]
|
2023-12-18 16:31:15 +11:00
|
|
|
pub fn struct_fatal(
|
|
|
|
&self,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
) -> DiagnosticBuilder<'_, FatalAbort> {
|
2023-12-04 10:44:57 +11:00
|
|
|
DiagnosticBuilder::new(self, Level::Fatal, msg)
|
2015-12-23 19:27:20 +13:00
|
|
|
}
|
|
|
|
|
2023-12-13 15:19:34 +11:00
|
|
|
/// Construct a builder at the `Fatal` level with the `msg`, that doesn't abort.
|
|
|
|
#[rustc_lint_diagnostics]
|
|
|
|
#[track_caller]
|
|
|
|
pub fn struct_almost_fatal(
|
|
|
|
&self,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
) -> DiagnosticBuilder<'_, FatalError> {
|
|
|
|
DiagnosticBuilder::new(self, Level::Fatal, msg)
|
|
|
|
}
|
|
|
|
|
2019-10-03 18:14:25 -05:00
|
|
|
/// Construct a builder at the `Help` level with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn struct_help(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_, ()> {
|
2019-10-03 18:14:25 -05:00
|
|
|
DiagnosticBuilder::new(self, Level::Help, msg)
|
|
|
|
}
|
|
|
|
|
2020-06-09 14:37:59 +01:00
|
|
|
/// Construct a builder at the `Note` level with the `msg`.
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2023-11-30 14:01:35 +11:00
|
|
|
pub fn struct_note(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_, ()> {
|
2020-06-09 14:37:59 +01:00
|
|
|
DiagnosticBuilder::new(self, Level::Note, msg)
|
|
|
|
}
|
|
|
|
|
2023-12-18 17:05:01 +11:00
|
|
|
/// Construct a builder at the `Bug` level with the `msg`.
|
|
|
|
#[rustc_lint_diagnostics]
|
|
|
|
#[track_caller]
|
|
|
|
pub fn struct_bug(&self, msg: impl Into<DiagnosticMessage>) -> DiagnosticBuilder<'_, BugAbort> {
|
|
|
|
DiagnosticBuilder::new(self, Level::Bug, msg)
|
|
|
|
}
|
|
|
|
|
2023-12-19 07:47:39 +11:00
|
|
|
/// Construct a builder at the `Bug` level at the given `span` with the `msg`.
|
|
|
|
#[rustc_lint_diagnostics]
|
|
|
|
#[track_caller]
|
|
|
|
pub fn struct_span_bug(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
) -> DiagnosticBuilder<'_, BugAbort> {
|
|
|
|
let mut result = self.struct_bug(msg);
|
|
|
|
result.set_span(span);
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-31 16:14:29 +01:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn span_fatal(&self, span: impl Into<MultiSpan>, msg: impl Into<DiagnosticMessage>) -> ! {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_fatal(span, msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2019-09-23 22:28:14 +02:00
|
|
|
pub fn span_fatal_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2019-09-23 22:28:14 +02:00
|
|
|
code: DiagnosticId,
|
2021-03-27 22:45:01 -04:00
|
|
|
) -> ! {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_fatal_with_code(span, msg, code).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn span_err(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
) -> ErrorGuaranteed {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_err(span, msg).emit()
|
2016-05-27 19:05:22 -07:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn span_err_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
code: DiagnosticId,
|
2023-11-30 13:57:35 +11:00
|
|
|
) -> ErrorGuaranteed {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_err_with_code(span, msg, code).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn span_warn(&self, span: impl Into<MultiSpan>, msg: impl Into<DiagnosticMessage>) {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_warn(span, msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-07-01 15:48:23 +02:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn span_warn_with_code(
|
|
|
|
&self,
|
|
|
|
span: impl Into<MultiSpan>,
|
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
code: DiagnosticId,
|
|
|
|
) {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_warn_with_code(span, msg, code).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2023-12-14 12:26:15 +11:00
|
|
|
pub fn span_bug(&self, span: impl Into<MultiSpan>, msg: impl Into<DiagnosticMessage>) -> ! {
|
2019-09-23 22:28:14 +02:00
|
|
|
self.inner.borrow_mut().span_bug(span, msg)
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2023-11-30 15:01:11 +11:00
|
|
|
/// For documentation on this, see `Session::span_delayed_bug`.
|
2023-12-04 14:06:28 +11:00
|
|
|
///
|
|
|
|
/// Note: this function used to be called `delay_span_bug`. It was renamed
|
|
|
|
/// to match similar functions like `span_bug`, `span_err`, etc.
|
2020-05-26 12:49:11 -05:00
|
|
|
#[track_caller]
|
2023-11-30 15:01:11 +11:00
|
|
|
pub fn span_delayed_bug(
|
2022-03-26 07:27:43 +00:00
|
|
|
&self,
|
2023-12-04 14:06:28 +11:00
|
|
|
sp: impl Into<MultiSpan>,
|
2023-12-14 12:26:15 +11:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
2022-03-26 07:27:43 +00:00
|
|
|
) -> ErrorGuaranteed {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
|
|
|
|
// This is technically `self.treat_err_as_bug()` but `span_delayed_bug` is called before
|
|
|
|
// incrementing `err_count` by one, so we need to +1 the comparing.
|
|
|
|
// FIXME: Would be nice to increment err_count in a more coherent way.
|
|
|
|
if inner.flags.treat_err_as_bug.is_some_and(|c| {
|
|
|
|
inner.err_count + inner.lint_err_count + inner.delayed_bug_count() + 1 >= c.get()
|
|
|
|
}) {
|
|
|
|
// FIXME: don't abort here if report_delayed_bugs is off
|
2023-12-14 12:26:15 +11:00
|
|
|
inner.span_bug(sp, msg);
|
2023-12-04 14:06:28 +11:00
|
|
|
}
|
2023-12-14 12:26:15 +11:00
|
|
|
let mut diagnostic = Diagnostic::new(Level::DelayedBug, msg);
|
|
|
|
diagnostic.set_span(sp);
|
2023-12-14 14:13:35 +11:00
|
|
|
inner.emit_diagnostic(diagnostic).unwrap()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2023-12-17 21:58:27 +11:00
|
|
|
// FIXME(eddyb) note the comment inside `impl Drop for DiagCtxtInner`, that's
|
2022-01-23 23:11:37 +00:00
|
|
|
// where the explanation of what "good path" is (also, it should be renamed).
|
2023-11-30 16:05:50 +11:00
|
|
|
pub fn good_path_delayed_bug(&self, msg: impl Into<DiagnosticMessage>) {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
|
|
|
|
let mut diagnostic = Diagnostic::new(Level::DelayedBug, msg);
|
|
|
|
if inner.flags.report_delayed_bugs {
|
2023-12-14 14:13:35 +11:00
|
|
|
inner.emit_diagnostic_without_consuming(&mut diagnostic);
|
2023-12-04 14:06:28 +11:00
|
|
|
}
|
|
|
|
let backtrace = std::backtrace::Backtrace::capture();
|
|
|
|
inner.good_path_delayed_bugs.push(DelayedDiagnostic::with_backtrace(diagnostic, backtrace));
|
2020-08-22 22:24:48 +03:00
|
|
|
}
|
|
|
|
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2023-11-30 14:01:35 +11:00
|
|
|
pub fn span_note(&self, span: impl Into<MultiSpan>, msg: impl Into<DiagnosticMessage>) {
|
2023-12-13 17:15:58 +11:00
|
|
|
self.struct_span_note(span, msg).emit()
|
2015-12-18 16:15:53 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2023-11-30 14:04:52 +11:00
|
|
|
pub fn struct_span_note(
|
2022-03-26 07:27:43 +00:00
|
|
|
&self,
|
2023-12-13 17:15:58 +11:00
|
|
|
span: impl Into<MultiSpan>,
|
2022-03-26 07:27:43 +00:00
|
|
|
msg: impl Into<DiagnosticMessage>,
|
|
|
|
) -> DiagnosticBuilder<'_, ()> {
|
2017-04-24 16:27:07 -07:00
|
|
|
let mut db = DiagnosticBuilder::new(self, Note, msg);
|
2019-09-23 22:28:14 +02:00
|
|
|
db.set_span(span);
|
2017-05-05 21:49:59 -07:00
|
|
|
db
|
2017-04-24 16:27:07 -07:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2023-12-01 14:08:10 +11:00
|
|
|
pub fn fatal(&self, msg: impl Into<DiagnosticMessage>) -> ! {
|
2023-12-13 16:36:57 +11:00
|
|
|
self.struct_fatal(msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn err(&self, msg: impl Into<DiagnosticMessage>) -> ErrorGuaranteed {
|
2023-12-13 16:36:57 +11:00
|
|
|
self.struct_err(msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn warn(&self, msg: impl Into<DiagnosticMessage>) {
|
2023-12-13 16:36:57 +11:00
|
|
|
self.struct_warn(msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
#[rustc_lint_diagnostics]
|
2023-11-30 14:01:35 +11:00
|
|
|
pub fn note(&self, msg: impl Into<DiagnosticMessage>) {
|
2023-12-13 16:36:57 +11:00
|
|
|
self.struct_note(msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-23 22:28:14 +02:00
|
|
|
|
2023-12-18 17:05:01 +11:00
|
|
|
#[rustc_lint_diagnostics]
|
2022-03-26 07:27:43 +00:00
|
|
|
pub fn bug(&self, msg: impl Into<DiagnosticMessage>) -> ! {
|
2023-12-18 17:05:01 +11:00
|
|
|
self.struct_bug(msg).emit()
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
|
2021-06-01 00:00:00 +00:00
|
|
|
#[inline]
|
2015-12-15 14:11:27 +13:00
|
|
|
pub fn err_count(&self) -> usize {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.inner.borrow().err_count
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
|
2022-01-22 18:49:12 -06:00
|
|
|
pub fn has_errors(&self) -> Option<ErrorGuaranteed> {
|
2023-05-05 17:31:54 +00:00
|
|
|
self.inner.borrow().has_errors().then(|| {
|
|
|
|
#[allow(deprecated)]
|
|
|
|
ErrorGuaranteed::unchecked_claim_error_was_emitted()
|
|
|
|
})
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2022-08-19 14:48:15 +01:00
|
|
|
|
2022-01-22 18:49:12 -06:00
|
|
|
pub fn has_errors_or_lint_errors(&self) -> Option<ErrorGuaranteed> {
|
2023-12-04 14:27:43 +11:00
|
|
|
let inner = self.inner.borrow();
|
2023-12-04 14:06:28 +11:00
|
|
|
let has_errors_or_lint_errors = inner.has_errors() || inner.lint_err_count > 0;
|
|
|
|
has_errors_or_lint_errors.then(|| {
|
2023-05-05 17:31:54 +00:00
|
|
|
#[allow(deprecated)]
|
|
|
|
ErrorGuaranteed::unchecked_claim_error_was_emitted()
|
|
|
|
})
|
2022-11-24 11:12:34 +00:00
|
|
|
}
|
2023-12-04 14:06:28 +11:00
|
|
|
|
2023-11-30 15:57:21 +11:00
|
|
|
pub fn has_errors_or_span_delayed_bugs(&self) -> Option<ErrorGuaranteed> {
|
2023-12-04 14:27:43 +11:00
|
|
|
let inner = self.inner.borrow();
|
2023-12-04 14:06:28 +11:00
|
|
|
let has_errors_or_span_delayed_bugs =
|
|
|
|
inner.has_errors() || !inner.span_delayed_bugs.is_empty();
|
|
|
|
has_errors_or_span_delayed_bugs.then(|| {
|
2023-05-05 17:31:54 +00:00
|
|
|
#[allow(deprecated)]
|
|
|
|
ErrorGuaranteed::unchecked_claim_error_was_emitted()
|
|
|
|
})
|
2021-07-20 23:23:22 -04:00
|
|
|
}
|
2023-12-04 14:06:28 +11:00
|
|
|
|
2022-11-24 11:12:34 +00:00
|
|
|
pub fn is_compilation_going_to_fail(&self) -> Option<ErrorGuaranteed> {
|
2023-12-04 14:06:28 +11:00
|
|
|
let inner = self.inner.borrow();
|
|
|
|
let will_fail =
|
|
|
|
inner.has_errors() || inner.lint_err_count > 0 || !inner.span_delayed_bugs.is_empty();
|
|
|
|
will_fail.then(|| {
|
2023-05-05 17:31:54 +00:00
|
|
|
#[allow(deprecated)]
|
|
|
|
ErrorGuaranteed::unchecked_claim_error_was_emitted()
|
|
|
|
})
|
2019-10-16 13:13:13 +02:00
|
|
|
}
|
2018-03-15 10:09:20 +01:00
|
|
|
|
2019-04-17 13:26:38 -04:00
|
|
|
pub fn print_error_count(&self, registry: &Registry) {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
|
|
|
|
inner.emit_stashed_diagnostics();
|
|
|
|
|
|
|
|
let warnings = match inner.deduplicated_warn_count {
|
|
|
|
0 => Cow::from(""),
|
|
|
|
1 => Cow::from("1 warning emitted"),
|
|
|
|
count => Cow::from(format!("{count} warnings emitted")),
|
|
|
|
};
|
|
|
|
let errors = match inner.deduplicated_err_count {
|
|
|
|
0 => Cow::from(""),
|
|
|
|
1 => Cow::from("aborting due to 1 previous error"),
|
|
|
|
count => Cow::from(format!("aborting due to {count} previous errors")),
|
|
|
|
};
|
|
|
|
if inner.treat_err_as_bug() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
match (errors.len(), warnings.len()) {
|
|
|
|
(0, 0) => return,
|
|
|
|
(0, _) => inner.emitter.emit_diagnostic(&Diagnostic::new(
|
|
|
|
Level::Warning(None),
|
|
|
|
DiagnosticMessage::Str(warnings),
|
|
|
|
)),
|
|
|
|
(_, 0) => {
|
2023-12-14 14:13:35 +11:00
|
|
|
inner.emit_diagnostic(Diagnostic::new(Fatal, errors));
|
2023-12-04 14:06:28 +11:00
|
|
|
}
|
|
|
|
(_, _) => {
|
2023-12-14 14:13:35 +11:00
|
|
|
inner.emit_diagnostic(Diagnostic::new(Fatal, format!("{errors}; {warnings}")));
|
2023-12-04 14:06:28 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let can_show_explain = inner.emitter.should_show_explain();
|
|
|
|
let are_there_diagnostics = !inner.emitted_diagnostic_codes.is_empty();
|
|
|
|
if can_show_explain && are_there_diagnostics {
|
|
|
|
let mut error_codes = inner
|
|
|
|
.emitted_diagnostic_codes
|
|
|
|
.iter()
|
|
|
|
.filter_map(|x| match &x {
|
|
|
|
DiagnosticId::Error(s) if registry.try_find_description(s).is_ok() => {
|
|
|
|
Some(s.clone())
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
if !error_codes.is_empty() {
|
|
|
|
error_codes.sort();
|
|
|
|
if error_codes.len() > 1 {
|
|
|
|
let limit = if error_codes.len() > 9 { 9 } else { error_codes.len() };
|
|
|
|
inner.failure_note(format!(
|
|
|
|
"Some errors have detailed explanations: {}{}",
|
|
|
|
error_codes[..limit].join(", "),
|
|
|
|
if error_codes.len() > 9 { "..." } else { "." }
|
|
|
|
));
|
|
|
|
inner.failure_note(format!(
|
2023-12-20 10:55:18 +11:00
|
|
|
"For more information about an error, try `rustc --explain {}`.",
|
2023-12-04 14:06:28 +11:00
|
|
|
&error_codes[0]
|
|
|
|
));
|
|
|
|
} else {
|
|
|
|
inner.failure_note(format!(
|
2023-12-20 10:55:18 +11:00
|
|
|
"For more information about this error, try `rustc --explain {}`.",
|
2023-12-04 14:06:28 +11:00
|
|
|
&error_codes[0]
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2020-08-13 15:41:52 -04:00
|
|
|
pub fn take_future_breakage_diagnostics(&self) -> Vec<Diagnostic> {
|
|
|
|
std::mem::take(&mut self.inner.borrow_mut().future_breakage_diagnostics)
|
|
|
|
}
|
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
pub fn abort_if_errors(&self) {
|
2023-12-04 14:06:28 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
inner.emit_stashed_diagnostics();
|
|
|
|
if inner.has_errors() {
|
|
|
|
FatalError.raise();
|
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2019-09-23 19:29:02 +02:00
|
|
|
/// `true` if we haven't taught a diagnostic with this code already.
|
|
|
|
/// The caller must then teach the user about such a diagnostic.
|
|
|
|
///
|
|
|
|
/// Used to suppress emitting the same error multiple times with extended explanation when
|
|
|
|
/// calling `-Zteach`.
|
2019-09-07 12:09:52 -04:00
|
|
|
pub fn must_teach(&self, code: &DiagnosticId) -> bool {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.inner.borrow_mut().taught_diagnostics.insert(code.clone())
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn force_print_diagnostic(&self, db: Diagnostic) {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.inner.borrow_mut().emitter.emit_diagnostic(&db);
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2023-12-14 14:13:35 +11:00
|
|
|
pub fn emit_diagnostic(&self, mut diagnostic: Diagnostic) -> Option<ErrorGuaranteed> {
|
|
|
|
self.emit_diagnostic_without_consuming(&mut diagnostic)
|
|
|
|
}
|
|
|
|
|
|
|
|
// It's unfortunate this exists. `emit_diagnostic` is preferred, because it
|
|
|
|
// consumes the diagnostic, thus ensuring it is emitted just once.
|
|
|
|
pub(crate) fn emit_diagnostic_without_consuming(
|
|
|
|
&self,
|
|
|
|
diagnostic: &mut Diagnostic,
|
|
|
|
) -> Option<ErrorGuaranteed> {
|
|
|
|
self.inner.borrow_mut().emit_diagnostic_without_consuming(diagnostic)
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2022-09-18 11:45:41 -04:00
|
|
|
pub fn emit_err<'a>(&'a self, err: impl IntoDiagnostic<'a>) -> ErrorGuaranteed {
|
|
|
|
self.create_err(err).emit()
|
|
|
|
}
|
|
|
|
|
2023-12-18 14:00:17 +11:00
|
|
|
#[track_caller]
|
2023-12-19 15:26:24 +11:00
|
|
|
pub fn create_err<'a>(&'a self, err: impl IntoDiagnostic<'a>) -> DiagnosticBuilder<'a> {
|
2023-12-18 14:12:39 +11:00
|
|
|
err.into_diagnostic(self, Level::Error { lint: false })
|
2022-09-18 11:45:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_warning<'a>(
|
|
|
|
&'a self,
|
|
|
|
warning: impl IntoDiagnostic<'a, ()>,
|
|
|
|
) -> DiagnosticBuilder<'a, ()> {
|
2023-12-18 14:12:39 +11:00
|
|
|
warning.into_diagnostic(self, Level::Warning(None))
|
2022-09-18 11:45:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn emit_warning<'a>(&'a self, warning: impl IntoDiagnostic<'a, ()>) {
|
|
|
|
self.create_warning(warning).emit()
|
|
|
|
}
|
|
|
|
|
2022-08-19 14:48:15 +01:00
|
|
|
pub fn create_almost_fatal<'a>(
|
|
|
|
&'a self,
|
|
|
|
fatal: impl IntoDiagnostic<'a, FatalError>,
|
|
|
|
) -> DiagnosticBuilder<'a, FatalError> {
|
2023-12-18 14:12:39 +11:00
|
|
|
fatal.into_diagnostic(self, Level::Fatal)
|
2022-08-19 14:48:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn emit_almost_fatal<'a>(
|
|
|
|
&'a self,
|
|
|
|
fatal: impl IntoDiagnostic<'a, FatalError>,
|
|
|
|
) -> FatalError {
|
|
|
|
self.create_almost_fatal(fatal).emit()
|
|
|
|
}
|
|
|
|
|
2022-09-18 11:45:41 -04:00
|
|
|
pub fn create_fatal<'a>(
|
|
|
|
&'a self,
|
2023-12-18 16:31:15 +11:00
|
|
|
fatal: impl IntoDiagnostic<'a, FatalAbort>,
|
|
|
|
) -> DiagnosticBuilder<'a, FatalAbort> {
|
2023-12-18 14:12:39 +11:00
|
|
|
fatal.into_diagnostic(self, Level::Fatal)
|
2022-09-18 11:45:41 -04:00
|
|
|
}
|
|
|
|
|
2023-12-18 16:31:15 +11:00
|
|
|
pub fn emit_fatal<'a>(&'a self, fatal: impl IntoDiagnostic<'a, FatalAbort>) -> ! {
|
2022-09-18 11:45:41 -04:00
|
|
|
self.create_fatal(fatal).emit()
|
|
|
|
}
|
|
|
|
|
2022-11-23 01:07:36 -05:00
|
|
|
pub fn create_bug<'a>(
|
|
|
|
&'a self,
|
2023-12-18 16:31:15 +11:00
|
|
|
bug: impl IntoDiagnostic<'a, BugAbort>,
|
|
|
|
) -> DiagnosticBuilder<'a, BugAbort> {
|
2023-12-18 14:12:39 +11:00
|
|
|
bug.into_diagnostic(self, Level::Bug)
|
2022-11-23 01:07:36 -05:00
|
|
|
}
|
|
|
|
|
2023-12-18 16:31:15 +11:00
|
|
|
pub fn emit_bug<'a>(&'a self, bug: impl IntoDiagnostic<'a, BugAbort>) -> ! {
|
2022-11-23 01:07:36 -05:00
|
|
|
self.create_bug(bug).emit()
|
|
|
|
}
|
|
|
|
|
2023-12-18 16:31:15 +11:00
|
|
|
pub fn emit_note<'a>(&'a self, note: impl IntoDiagnostic<'a, ()>) {
|
2022-10-14 14:00:06 +01:00
|
|
|
self.create_note(note).emit()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_note<'a>(
|
|
|
|
&'a self,
|
2023-12-18 16:31:15 +11:00
|
|
|
note: impl IntoDiagnostic<'a, ()>,
|
|
|
|
) -> DiagnosticBuilder<'a, ()> {
|
2023-12-18 14:12:39 +11:00
|
|
|
note.into_diagnostic(self, Level::Note)
|
2022-10-14 14:00:06 +01:00
|
|
|
}
|
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
pub fn emit_artifact_notification(&self, path: &Path, artifact_type: &str) {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.inner.borrow_mut().emitter.emit_artifact_notification(path, artifact_type);
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2021-07-11 13:08:58 -07:00
|
|
|
pub fn emit_future_breakage_report(&self, diags: Vec<Diagnostic>) {
|
2020-08-13 15:41:52 -04:00
|
|
|
self.inner.borrow_mut().emitter.emit_future_breakage_report(diags)
|
|
|
|
}
|
|
|
|
|
2022-04-16 17:11:33 -07:00
|
|
|
pub fn emit_unused_externs(
|
|
|
|
&self,
|
|
|
|
lint_level: rustc_lint_defs::Level,
|
|
|
|
loud: bool,
|
|
|
|
unused_externs: &[&str],
|
|
|
|
) {
|
2022-04-15 00:37:40 -07:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
|
2022-04-16 17:11:33 -07:00
|
|
|
if loud && lint_level.is_error() {
|
2022-04-15 00:37:40 -07:00
|
|
|
inner.bump_err_count();
|
|
|
|
}
|
|
|
|
|
2023-12-04 14:06:28 +11:00
|
|
|
inner.emitter.emit_unused_externs(lint_level, unused_externs)
|
2020-06-30 18:58:15 +02:00
|
|
|
}
|
2021-08-06 23:28:58 +02:00
|
|
|
|
2021-11-20 20:45:27 +01:00
|
|
|
pub fn update_unstable_expectation_id(
|
|
|
|
&self,
|
|
|
|
unstable_to_stable: &FxHashMap<LintExpectationId, LintExpectationId>,
|
|
|
|
) {
|
2022-03-29 00:42:41 +02:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
|
|
|
let diags = std::mem::take(&mut inner.unstable_expect_diagnostics);
|
|
|
|
inner.check_unstable_expect_diagnostics = true;
|
2021-11-20 20:45:27 +01:00
|
|
|
|
2022-06-04 00:43:24 +02:00
|
|
|
if !diags.is_empty() {
|
|
|
|
inner.suppressed_expected_diag = true;
|
|
|
|
for mut diag in diags.into_iter() {
|
|
|
|
diag.update_unstable_expectation_id(unstable_to_stable);
|
2022-03-05 21:54:49 +01:00
|
|
|
|
2022-06-05 12:33:45 +02:00
|
|
|
// Here the diagnostic is given back to `emit_diagnostic` where it was first
|
|
|
|
// intercepted. Now it should be processed as usual, since the unstable expectation
|
|
|
|
// id is now stable.
|
2023-12-14 14:13:35 +11:00
|
|
|
inner.emit_diagnostic(diag);
|
2022-06-04 00:43:24 +02:00
|
|
|
}
|
2021-11-20 20:45:27 +01:00
|
|
|
}
|
2022-03-05 21:54:49 +01:00
|
|
|
|
|
|
|
inner
|
|
|
|
.stashed_diagnostics
|
|
|
|
.values_mut()
|
|
|
|
.for_each(|diag| diag.update_unstable_expectation_id(unstable_to_stable));
|
|
|
|
inner
|
|
|
|
.future_breakage_diagnostics
|
|
|
|
.iter_mut()
|
|
|
|
.for_each(|diag| diag.update_unstable_expectation_id(unstable_to_stable));
|
2021-11-20 20:45:27 +01:00
|
|
|
}
|
|
|
|
|
2021-08-06 23:28:58 +02:00
|
|
|
/// This methods steals all [`LintExpectationId`]s that are stored inside
|
2023-12-17 21:58:27 +11:00
|
|
|
/// [`DiagCtxtInner`] and indicate that the linked expectation has been fulfilled.
|
2022-03-06 14:18:28 +01:00
|
|
|
#[must_use]
|
2021-08-06 23:28:58 +02:00
|
|
|
pub fn steal_fulfilled_expectation_ids(&self) -> FxHashSet<LintExpectationId> {
|
2022-02-18 12:00:16 +01:00
|
|
|
assert!(
|
|
|
|
self.inner.borrow().unstable_expect_diagnostics.is_empty(),
|
2023-12-17 21:58:27 +11:00
|
|
|
"`DiagCtxtInner::unstable_expect_diagnostics` should be empty at this point",
|
2022-02-18 12:00:16 +01:00
|
|
|
);
|
2021-08-06 23:28:58 +02:00
|
|
|
std::mem::take(&mut self.inner.borrow_mut().fulfilled_expectations)
|
|
|
|
}
|
2022-09-27 20:56:05 +02:00
|
|
|
|
|
|
|
pub fn flush_delayed(&self) {
|
2023-12-01 08:23:34 +11:00
|
|
|
let mut inner = self.inner.borrow_mut();
|
2023-11-30 15:57:21 +11:00
|
|
|
let bugs = std::mem::replace(&mut inner.span_delayed_bugs, Vec::new());
|
2023-11-30 15:01:11 +11:00
|
|
|
inner.flush_delayed(bugs, "no errors encountered even though `span_delayed_bug` issued");
|
2022-09-27 20:56:05 +02:00
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2023-12-17 21:48:57 +11:00
|
|
|
// Note: we prefer implementing operations on `DiagCtxt`, rather than
|
2023-12-17 21:58:27 +11:00
|
|
|
// `DiagCtxtInner`, whenever possible. This minimizes functions where
|
2023-12-17 21:48:57 +11:00
|
|
|
// `DiagCtxt::foo()` just borrows `inner` and forwards a call to
|
2023-12-04 14:06:28 +11:00
|
|
|
// `HanderInner::foo`.
|
2023-12-17 21:58:27 +11:00
|
|
|
impl DiagCtxtInner {
|
2019-09-23 04:45:21 +02:00
|
|
|
/// Emit all stashed diagnostics.
|
2022-01-22 18:49:12 -06:00
|
|
|
fn emit_stashed_diagnostics(&mut self) -> Option<ErrorGuaranteed> {
|
2022-08-17 06:07:33 -05:00
|
|
|
let has_errors = self.has_errors();
|
2019-09-23 04:45:21 +02:00
|
|
|
let diags = self.stashed_diagnostics.drain(..).map(|x| x.1).collect::<Vec<_>>();
|
2022-01-22 18:49:12 -06:00
|
|
|
let mut reported = None;
|
2023-12-14 14:13:35 +11:00
|
|
|
for diag in diags {
|
2022-08-17 06:07:33 -05:00
|
|
|
// Decrement the count tracking the stash; emitting will increment it.
|
2022-01-22 18:49:12 -06:00
|
|
|
if diag.is_error() {
|
2022-08-17 06:07:33 -05:00
|
|
|
if matches!(diag.level, Level::Error { lint: true }) {
|
|
|
|
self.lint_err_count -= 1;
|
|
|
|
} else {
|
|
|
|
self.err_count -= 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if diag.is_force_warn() {
|
|
|
|
self.warn_count -= 1;
|
|
|
|
} else {
|
|
|
|
// Unless they're forced, don't flush stashed warnings when
|
|
|
|
// there are errors, to avoid causing warning overload. The
|
|
|
|
// stash would've been stolen already if it were important.
|
|
|
|
if has_errors {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2022-01-22 18:49:12 -06:00
|
|
|
}
|
2023-12-14 14:13:35 +11:00
|
|
|
let reported_this = self.emit_diagnostic(diag);
|
2022-08-17 06:07:33 -05:00
|
|
|
reported = reported.or(reported_this);
|
2022-03-20 18:26:09 +01:00
|
|
|
}
|
2022-01-22 18:49:12 -06:00
|
|
|
reported
|
2019-09-23 04:45:21 +02:00
|
|
|
}
|
|
|
|
|
2023-12-14 14:13:35 +11:00
|
|
|
fn emit_diagnostic(&mut self, mut diagnostic: Diagnostic) -> Option<ErrorGuaranteed> {
|
|
|
|
self.emit_diagnostic_without_consuming(&mut diagnostic)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn emit_diagnostic_without_consuming(
|
|
|
|
&mut self,
|
|
|
|
diagnostic: &mut Diagnostic,
|
|
|
|
) -> Option<ErrorGuaranteed> {
|
2023-12-04 14:27:43 +11:00
|
|
|
if matches!(diagnostic.level, Level::Error { .. } | Level::Fatal) && self.treat_err_as_bug()
|
|
|
|
{
|
|
|
|
diagnostic.level = Level::Bug;
|
|
|
|
}
|
|
|
|
|
2022-06-05 12:33:45 +02:00
|
|
|
// The `LintExpectationId` can be stable or unstable depending on when it was created.
|
|
|
|
// Diagnostics created before the definition of `HirId`s are unstable and can not yet
|
|
|
|
// be stored. Instead, they are buffered until the `LintExpectationId` is replaced by
|
|
|
|
// a stable one by the `LintLevelsBuilder`.
|
|
|
|
if let Some(LintExpectationId::Unstable { .. }) = diagnostic.level.get_expectation_id() {
|
|
|
|
self.unstable_expect_diagnostics.push(diagnostic.clone());
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2022-01-23 23:11:37 +00:00
|
|
|
if diagnostic.level == Level::DelayedBug {
|
|
|
|
// FIXME(eddyb) this should check for `has_errors` and stop pushing
|
2023-11-30 15:57:21 +11:00
|
|
|
// once *any* errors were emitted (and truncate `span_delayed_bugs`
|
2022-01-23 23:11:37 +00:00
|
|
|
// when an error is first emitted, also), but maybe there's a case
|
|
|
|
// in which that's not sound? otherwise this is really inefficient.
|
2023-06-06 19:24:33 +00:00
|
|
|
let backtrace = std::backtrace::Backtrace::capture();
|
2023-11-30 15:57:21 +11:00
|
|
|
self.span_delayed_bugs
|
2022-12-31 01:56:59 +00:00
|
|
|
.push(DelayedDiagnostic::with_backtrace(diagnostic.clone(), backtrace));
|
2022-01-23 23:11:37 +00:00
|
|
|
|
|
|
|
if !self.flags.report_delayed_bugs {
|
2023-05-05 17:31:54 +00:00
|
|
|
#[allow(deprecated)]
|
2022-01-22 18:49:12 -06:00
|
|
|
return Some(ErrorGuaranteed::unchecked_claim_error_was_emitted());
|
2022-01-23 23:11:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-13 15:41:52 -04:00
|
|
|
if diagnostic.has_future_breakage() {
|
2022-11-10 04:21:11 +00:00
|
|
|
// Future breakages aren't emitted if they're Level::Allowed,
|
|
|
|
// but they still need to be constructed and stashed below,
|
|
|
|
// so they'll trigger the good-path bug check.
|
|
|
|
self.suppressed_expected_diag = true;
|
2020-08-13 15:41:52 -04:00
|
|
|
self.future_breakage_diagnostics.push(diagnostic.clone());
|
|
|
|
}
|
|
|
|
|
2022-06-05 12:33:45 +02:00
|
|
|
if let Some(expectation_id) = diagnostic.level.get_expectation_id() {
|
|
|
|
self.suppressed_expected_diag = true;
|
2022-07-22 16:48:36 +00:00
|
|
|
self.fulfilled_expectations.insert(expectation_id.normalize());
|
2022-06-05 12:33:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if matches!(diagnostic.level, Warning(_))
|
2021-06-04 14:37:20 +02:00
|
|
|
&& !self.flags.can_emit_warnings
|
|
|
|
&& !diagnostic.is_force_warn()
|
|
|
|
{
|
2020-08-13 15:41:52 -04:00
|
|
|
if diagnostic.has_future_breakage() {
|
2021-04-30 19:38:06 +02:00
|
|
|
(*TRACK_DIAGNOSTICS)(diagnostic, &mut |_| {});
|
2020-08-13 15:41:52 -04:00
|
|
|
}
|
2022-01-22 18:49:12 -06:00
|
|
|
return None;
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2022-06-05 12:33:45 +02:00
|
|
|
if matches!(diagnostic.level, Level::Expect(_) | Level::Allow) {
|
2021-04-30 19:38:06 +02:00
|
|
|
(*TRACK_DIAGNOSTICS)(diagnostic, &mut |_| {});
|
2022-01-22 18:49:12 -06:00
|
|
|
return None;
|
2020-08-13 15:41:52 -04:00
|
|
|
}
|
|
|
|
|
2021-04-30 19:38:06 +02:00
|
|
|
let mut guaranteed = None;
|
|
|
|
(*TRACK_DIAGNOSTICS)(diagnostic, &mut |diagnostic| {
|
|
|
|
if let Some(ref code) = diagnostic.code {
|
|
|
|
self.emitted_diagnostic_codes.insert(code.clone());
|
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
|
2023-10-04 18:34:50 +00:00
|
|
|
let already_emitted = {
|
2022-03-20 20:02:18 +01:00
|
|
|
let mut hasher = StableHasher::new();
|
2021-04-30 19:38:06 +02:00
|
|
|
diagnostic.hash(&mut hasher);
|
2022-03-20 20:02:18 +01:00
|
|
|
let diagnostic_hash = hasher.finish();
|
2023-10-04 18:34:50 +00:00
|
|
|
!self.emitted_diagnostics.insert(diagnostic_hash)
|
2022-03-20 20:02:18 +01:00
|
|
|
};
|
|
|
|
|
2021-04-30 19:38:06 +02:00
|
|
|
// Only emit the diagnostic if we've been asked to deduplicate or
|
|
|
|
// haven't already emitted an equivalent diagnostic.
|
2023-10-04 18:34:50 +00:00
|
|
|
if !(self.flags.deduplicate_diagnostics && already_emitted) {
|
2021-04-30 19:38:06 +02:00
|
|
|
debug!(?diagnostic);
|
|
|
|
debug!(?self.emitted_diagnostics);
|
|
|
|
let already_emitted_sub = |sub: &mut SubDiagnostic| {
|
|
|
|
debug!(?sub);
|
2023-07-25 19:37:45 +02:00
|
|
|
if sub.level != Level::OnceNote && sub.level != Level::OnceHelp {
|
2021-04-30 19:38:06 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
let mut hasher = StableHasher::new();
|
|
|
|
sub.hash(&mut hasher);
|
|
|
|
let diagnostic_hash = hasher.finish();
|
|
|
|
debug!(?diagnostic_hash);
|
|
|
|
!self.emitted_diagnostics.insert(diagnostic_hash)
|
|
|
|
};
|
|
|
|
|
2023-07-03 08:40:34 +08:00
|
|
|
diagnostic.children.extract_if(already_emitted_sub).for_each(|_| {});
|
2023-10-04 18:34:50 +00:00
|
|
|
if already_emitted {
|
|
|
|
diagnostic.note(
|
|
|
|
"duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`",
|
|
|
|
);
|
|
|
|
}
|
2021-04-30 19:38:06 +02:00
|
|
|
|
|
|
|
self.emitter.emit_diagnostic(diagnostic);
|
|
|
|
if diagnostic.is_error() {
|
|
|
|
self.deduplicated_err_count += 1;
|
|
|
|
} else if let Warning(_) = diagnostic.level {
|
|
|
|
self.deduplicated_warn_count += 1;
|
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
2021-04-30 19:38:06 +02:00
|
|
|
if diagnostic.is_error() {
|
|
|
|
if matches!(diagnostic.level, Level::Error { lint: true }) {
|
|
|
|
self.bump_lint_err_count();
|
|
|
|
} else {
|
|
|
|
self.bump_err_count();
|
|
|
|
}
|
|
|
|
|
2023-05-05 17:31:54 +00:00
|
|
|
#[allow(deprecated)]
|
|
|
|
{
|
|
|
|
guaranteed = Some(ErrorGuaranteed::unchecked_claim_error_was_emitted());
|
|
|
|
}
|
2021-07-20 23:23:22 -04:00
|
|
|
} else {
|
2021-04-30 19:38:06 +02:00
|
|
|
self.bump_warn_count();
|
2021-07-20 23:23:22 -04:00
|
|
|
}
|
2021-04-30 19:38:06 +02:00
|
|
|
});
|
2022-01-22 18:49:12 -06:00
|
|
|
|
2021-04-30 19:38:06 +02:00
|
|
|
guaranteed
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn treat_err_as_bug(&self) -> bool {
|
2023-05-24 14:19:22 +00:00
|
|
|
self.flags.treat_err_as_bug.is_some_and(|c| {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.err_count + self.lint_err_count + self.delayed_bug_count() >= c.get()
|
2022-08-28 01:08:24 +00:00
|
|
|
})
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
|
|
|
|
2022-09-06 07:08:12 +00:00
|
|
|
fn delayed_bug_count(&self) -> usize {
|
2023-11-30 16:05:50 +11:00
|
|
|
self.span_delayed_bugs.len() + self.good_path_delayed_bugs.len()
|
2022-09-06 07:08:12 +00:00
|
|
|
}
|
|
|
|
|
2019-09-23 04:45:21 +02:00
|
|
|
fn has_errors(&self) -> bool {
|
2023-12-04 14:06:28 +11:00
|
|
|
self.err_count > 0
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2017-08-12 15:37:28 -07:00
|
|
|
|
2022-10-19 00:08:20 +02:00
|
|
|
#[track_caller]
|
2023-12-14 12:26:15 +11:00
|
|
|
fn span_bug(&mut self, sp: impl Into<MultiSpan>, msg: impl Into<DiagnosticMessage>) -> ! {
|
2023-12-14 14:13:35 +11:00
|
|
|
let mut diag = Diagnostic::new(Bug, msg);
|
|
|
|
diag.set_span(sp);
|
|
|
|
self.emit_diagnostic(diag);
|
2021-02-02 00:17:51 +01:00
|
|
|
panic::panic_any(ExplicitBug);
|
2019-09-23 19:29:02 +02:00
|
|
|
}
|
|
|
|
|
2023-11-30 14:40:26 +11:00
|
|
|
fn failure_note(&mut self, msg: impl Into<DiagnosticMessage>) {
|
2023-12-14 14:13:35 +11:00
|
|
|
self.emit_diagnostic(Diagnostic::new(FailureNote, msg));
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
2017-10-25 15:01:06 +02:00
|
|
|
|
2022-03-26 07:27:43 +00:00
|
|
|
fn flush_delayed(
|
|
|
|
&mut self,
|
2023-12-19 17:59:04 +11:00
|
|
|
bugs: Vec<DelayedDiagnostic>,
|
2022-03-26 07:27:43 +00:00
|
|
|
explanation: impl Into<DiagnosticMessage> + Copy,
|
|
|
|
) {
|
2023-12-19 17:59:04 +11:00
|
|
|
if bugs.is_empty() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-03 22:25:18 +00:00
|
|
|
// If backtraces are enabled, also print the query stack
|
|
|
|
let backtrace = std::env::var_os("RUST_BACKTRACE").map_or(true, |x| &x != "0");
|
2023-12-19 17:59:04 +11:00
|
|
|
for (i, bug) in bugs.into_iter().enumerate() {
|
2023-03-03 22:25:18 +00:00
|
|
|
if let Some(file) = self.ice_file.as_ref()
|
2023-07-27 12:32:22 +00:00
|
|
|
&& let Ok(mut out) = std::fs::File::options().create(true).append(true).open(file)
|
2023-03-03 22:25:18 +00:00
|
|
|
{
|
|
|
|
let _ = write!(
|
|
|
|
&mut out,
|
2023-07-27 12:32:22 +00:00
|
|
|
"delayed span bug: {}\n{}\n",
|
2023-03-03 22:25:18 +00:00
|
|
|
bug.inner
|
|
|
|
.styled_message()
|
|
|
|
.iter()
|
|
|
|
.filter_map(|(msg, _)| msg.as_str())
|
|
|
|
.collect::<String>(),
|
|
|
|
&bug.note
|
|
|
|
);
|
|
|
|
}
|
2022-12-31 01:56:59 +00:00
|
|
|
|
2023-12-19 17:59:04 +11:00
|
|
|
if i == 0 {
|
2022-01-23 23:11:37 +00:00
|
|
|
// Put the overall explanation before the `DelayedBug`s, to
|
|
|
|
// frame them better (e.g. separate warnings from them).
|
2023-12-14 14:13:35 +11:00
|
|
|
self.emit_diagnostic(Diagnostic::new(Bug, explanation));
|
2022-01-23 23:11:37 +00:00
|
|
|
}
|
|
|
|
|
2023-12-19 17:59:04 +11:00
|
|
|
let mut bug =
|
|
|
|
if backtrace || self.ice_file.is_none() { bug.decorate() } else { bug.inner };
|
|
|
|
|
2022-01-23 23:11:37 +00:00
|
|
|
// "Undelay" the `DelayedBug`s (into plain `Bug`s).
|
|
|
|
if bug.level != Level::DelayedBug {
|
|
|
|
// NOTE(eddyb) not panicking here because we're already producing
|
|
|
|
// an ICE, and the more information the merrier.
|
2023-07-03 18:08:49 +09:00
|
|
|
bug.subdiagnostic(InvalidFlushedDelayedDiagnosticLevel {
|
|
|
|
span: bug.span.primary_span().unwrap(),
|
|
|
|
level: bug.level,
|
|
|
|
});
|
2022-01-23 23:11:37 +00:00
|
|
|
}
|
|
|
|
bug.level = Level::Bug;
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2023-12-14 14:13:35 +11:00
|
|
|
self.emit_diagnostic(bug);
|
2020-08-22 22:24:48 +03:00
|
|
|
}
|
2022-01-23 23:11:37 +00:00
|
|
|
|
2022-12-31 01:56:59 +00:00
|
|
|
// Panic with `DelayedBugPanic` to avoid "unexpected panic" messages.
|
2023-12-19 17:59:04 +11:00
|
|
|
panic::panic_any(DelayedBugPanic);
|
2020-08-22 22:24:48 +03:00
|
|
|
}
|
|
|
|
|
2021-07-20 23:23:22 -04:00
|
|
|
fn bump_lint_err_count(&mut self) {
|
|
|
|
self.lint_err_count += 1;
|
|
|
|
self.panic_if_treat_err_as_bug();
|
|
|
|
}
|
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
fn bump_err_count(&mut self) {
|
|
|
|
self.err_count += 1;
|
|
|
|
self.panic_if_treat_err_as_bug();
|
|
|
|
}
|
|
|
|
|
2020-08-22 22:24:48 +03:00
|
|
|
fn bump_warn_count(&mut self) {
|
|
|
|
self.warn_count += 1;
|
|
|
|
}
|
|
|
|
|
2019-09-07 12:09:52 -04:00
|
|
|
fn panic_if_treat_err_as_bug(&self) {
|
|
|
|
if self.treat_err_as_bug() {
|
2021-12-01 16:32:55 +00:00
|
|
|
match (
|
2023-12-04 14:06:28 +11:00
|
|
|
self.err_count + self.lint_err_count,
|
2022-09-06 07:08:12 +00:00
|
|
|
self.delayed_bug_count(),
|
2023-09-22 13:33:55 +02:00
|
|
|
self.flags.treat_err_as_bug.map(|c| c.get()).unwrap(),
|
2021-12-01 16:32:55 +00:00
|
|
|
) {
|
2022-09-06 07:08:12 +00:00
|
|
|
(1, 0, 1) => panic!("aborting due to `-Z treat-err-as-bug=1`"),
|
|
|
|
(0, 1, 1) => panic!("aborting due delayed bug with `-Z treat-err-as-bug=1`"),
|
2023-09-22 13:33:55 +02:00
|
|
|
(count, delayed_count, val) => {
|
2022-09-06 07:08:12 +00:00
|
|
|
if delayed_count > 0 {
|
|
|
|
panic!(
|
2023-09-22 13:33:55 +02:00
|
|
|
"aborting after {count} errors and {delayed_count} delayed bugs due to `-Z treat-err-as-bug={val}`",
|
2022-09-06 07:08:12 +00:00
|
|
|
)
|
|
|
|
} else {
|
2023-09-22 13:33:55 +02:00
|
|
|
panic!("aborting after {count} errors due to `-Z treat-err-as-bug={val}`")
|
2022-09-06 07:08:12 +00:00
|
|
|
}
|
|
|
|
}
|
2020-09-18 05:57:01 +00:00
|
|
|
}
|
2019-09-07 12:09:52 -04:00
|
|
|
}
|
2019-04-15 08:26:08 +10:00
|
|
|
}
|
|
|
|
}
|
2015-12-15 14:11:27 +13:00
|
|
|
|
2021-05-05 22:52:58 +02:00
|
|
|
struct DelayedDiagnostic {
|
|
|
|
inner: Diagnostic,
|
|
|
|
note: Backtrace,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DelayedDiagnostic {
|
|
|
|
fn with_backtrace(diagnostic: Diagnostic, backtrace: Backtrace) -> Self {
|
|
|
|
DelayedDiagnostic { inner: diagnostic, note: backtrace }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn decorate(mut self) -> Diagnostic {
|
2023-06-06 19:24:33 +00:00
|
|
|
match self.note.status() {
|
|
|
|
BacktraceStatus::Captured => {
|
2023-07-03 18:08:49 +09:00
|
|
|
let inner = &self.inner;
|
|
|
|
self.inner.subdiagnostic(DelayedAtWithNewline {
|
2023-09-02 04:02:11 +00:00
|
|
|
span: inner.span.primary_span().unwrap_or(DUMMY_SP),
|
2023-07-03 18:08:49 +09:00
|
|
|
emitted_at: inner.emitted_at.clone(),
|
|
|
|
note: self.note,
|
|
|
|
});
|
2023-06-06 19:24:33 +00:00
|
|
|
}
|
|
|
|
// Avoid the needless newline when no backtrace has been captured,
|
|
|
|
// the display impl should just be a single line.
|
|
|
|
_ => {
|
2023-07-03 18:08:49 +09:00
|
|
|
let inner = &self.inner;
|
|
|
|
self.inner.subdiagnostic(DelayedAtWithoutNewline {
|
2023-09-02 04:02:11 +00:00
|
|
|
span: inner.span.primary_span().unwrap_or(DUMMY_SP),
|
2023-07-03 18:08:49 +09:00
|
|
|
emitted_at: inner.emitted_at.clone(),
|
|
|
|
note: self.note,
|
|
|
|
});
|
2023-06-06 19:24:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-05 22:52:58 +02:00
|
|
|
self.inner
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-27 09:44:25 +00:00
|
|
|
#[derive(Copy, PartialEq, Eq, Clone, Hash, Debug, Encodable, Decodable)]
|
2015-12-15 14:11:27 +13:00
|
|
|
pub enum Level {
|
2023-12-19 13:24:34 +11:00
|
|
|
/// For bugs in the compiler. Manifests as an ICE (internal compiler error) panic.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `BugAbort`.
|
2015-12-15 14:11:27 +13:00
|
|
|
Bug,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// This is a strange one: lets you register an error without emitting it. If compilation ends
|
|
|
|
/// without any other errors occurring, this will be emitted as a bug. Otherwise, it will be
|
|
|
|
/// silently dropped. I.e. "expect other errors are emitted" semantics. Useful on code paths
|
|
|
|
/// that should only be reached when compiling erroneous code.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `ErrorGuaranteed`.
|
2022-01-23 23:11:37 +00:00
|
|
|
DelayedBug,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// An error that causes an immediate abort. Used for things like configuration errors,
|
|
|
|
/// internal overflows, some file operation errors.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `FatalAbort`, except in the non-aborting "almost fatal" case
|
|
|
|
/// that is occasionaly used, where it is `FatalError`.
|
2015-12-15 14:11:27 +13:00
|
|
|
Fatal,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// An error in the code being compiled, which prevents compilation from finishing. This is the
|
|
|
|
/// most common case.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `ErrorGuaranteed`.
|
2021-07-20 23:23:22 -04:00
|
|
|
Error {
|
2023-12-19 13:24:34 +11:00
|
|
|
/// If this error comes from a lint, don't abort compilation even when abort_if_errors() is
|
|
|
|
/// called.
|
2021-07-20 23:23:22 -04:00
|
|
|
lint: bool,
|
|
|
|
},
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// A warning about the code being compiled. Does not prevent compilation from finishing.
|
|
|
|
///
|
2022-06-05 12:33:45 +02:00
|
|
|
/// This [`LintExpectationId`] is used for expected lint diagnostics, which should
|
|
|
|
/// also emit a warning due to the `force-warn` flag. In all other cases this should
|
|
|
|
/// be `None`.
|
2023-12-19 13:24:34 +11:00
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2022-06-05 12:33:45 +02:00
|
|
|
Warning(Option<LintExpectationId>),
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// A message giving additional context. Rare, because notes are more commonly attached to other
|
|
|
|
/// diagnostics such as errors.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2015-12-15 14:11:27 +13:00
|
|
|
Note,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// A note that is only emitted once. Rare, mostly used in circumstances relating to lints.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2022-03-20 20:02:18 +01:00
|
|
|
OnceNote,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// A message suggesting how to fix something. Rare, because help messages are more commonly
|
|
|
|
/// attached to other diagnostics such as errors.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2015-12-15 14:11:27 +13:00
|
|
|
Help,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// A help that is only emitted once. Rare.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2023-07-25 19:37:45 +02:00
|
|
|
OnceHelp,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// Similar to `Note`, but used in cases where compilation has failed. Rare.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2018-02-28 16:17:44 +01:00
|
|
|
FailureNote,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// Only used for lints.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2020-08-13 15:41:52 -04:00
|
|
|
Allow,
|
2023-12-19 13:24:34 +11:00
|
|
|
|
|
|
|
/// Only used for lints.
|
|
|
|
///
|
|
|
|
/// Its `EmissionGuarantee` is `()`.
|
2021-08-06 23:18:16 +02:00
|
|
|
Expect(LintExpectationId),
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Level {
|
2019-02-07 03:53:01 +09:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2015-12-31 18:47:14 +13:00
|
|
|
self.to_str().fmt(f)
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Level {
|
2018-02-27 10:33:02 -08:00
|
|
|
fn color(self) -> ColorSpec {
|
|
|
|
let mut spec = ColorSpec::new();
|
2015-12-15 14:11:27 +13:00
|
|
|
match self {
|
2022-01-23 23:11:37 +00:00
|
|
|
Bug | DelayedBug | Fatal | Error { .. } => {
|
2018-02-27 10:33:02 -08:00
|
|
|
spec.set_fg(Some(Color::Red)).set_intense(true);
|
|
|
|
}
|
2022-06-05 12:33:45 +02:00
|
|
|
Warning(_) => {
|
2018-02-27 10:33:02 -08:00
|
|
|
spec.set_fg(Some(Color::Yellow)).set_intense(cfg!(windows));
|
|
|
|
}
|
2022-03-20 20:02:18 +01:00
|
|
|
Note | OnceNote => {
|
2018-02-27 10:33:02 -08:00
|
|
|
spec.set_fg(Some(Color::Green)).set_intense(true);
|
|
|
|
}
|
2023-07-25 19:37:45 +02:00
|
|
|
Help | OnceHelp => {
|
2018-02-27 10:33:02 -08:00
|
|
|
spec.set_fg(Some(Color::Cyan)).set_intense(true);
|
2016-10-18 23:13:02 +05:30
|
|
|
}
|
2018-02-28 16:17:44 +01:00
|
|
|
FailureNote => {}
|
2021-08-06 23:18:16 +02:00
|
|
|
Allow | Expect(_) => unreachable!(),
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2018-02-28 16:17:44 +01:00
|
|
|
spec
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2015-12-31 18:47:14 +13:00
|
|
|
|
2016-06-21 18:08:13 -04:00
|
|
|
pub fn to_str(self) -> &'static str {
|
2015-12-31 18:47:14 +13:00
|
|
|
match self {
|
2022-01-23 23:11:37 +00:00
|
|
|
Bug | DelayedBug => "error: internal compiler error",
|
2021-07-20 23:23:22 -04:00
|
|
|
Fatal | Error { .. } => "error",
|
2022-06-05 12:33:45 +02:00
|
|
|
Warning(_) => "warning",
|
2022-03-20 20:02:18 +01:00
|
|
|
Note | OnceNote => "note",
|
2023-07-25 19:37:45 +02:00
|
|
|
Help | OnceHelp => "help",
|
2019-09-13 06:48:47 -07:00
|
|
|
FailureNote => "failure-note",
|
2023-12-19 13:24:34 +11:00
|
|
|
Allow | Expect(_) => unreachable!(),
|
2015-12-31 18:47:14 +13:00
|
|
|
}
|
|
|
|
}
|
2018-02-28 16:17:44 +01:00
|
|
|
|
|
|
|
pub fn is_failure_note(&self) -> bool {
|
2020-10-26 21:02:48 -04:00
|
|
|
matches!(*self, FailureNote)
|
2018-02-28 16:17:44 +01:00
|
|
|
}
|
2021-11-20 20:45:27 +01:00
|
|
|
|
|
|
|
pub fn get_expectation_id(&self) -> Option<LintExpectationId> {
|
|
|
|
match self {
|
2022-06-05 12:33:45 +02:00
|
|
|
Level::Expect(id) | Level::Warning(Some(id)) => Some(*id),
|
2021-11-20 20:45:27 +01:00
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
2015-12-15 14:11:27 +13:00
|
|
|
}
|
2019-09-08 14:57:03 +05:30
|
|
|
|
2022-01-23 20:41:46 +00:00
|
|
|
// FIXME(eddyb) this doesn't belong here AFAICT, should be moved to callsite.
|
2020-08-13 15:41:52 -04:00
|
|
|
pub fn add_elided_lifetime_in_path_suggestion(
|
|
|
|
source_map: &SourceMap,
|
2022-01-23 20:41:46 +00:00
|
|
|
diag: &mut Diagnostic,
|
2020-08-13 15:41:52 -04:00
|
|
|
n: usize,
|
|
|
|
path_span: Span,
|
|
|
|
incl_angl_brckt: bool,
|
|
|
|
insertion_span: Span,
|
|
|
|
) {
|
2023-07-03 18:08:49 +09:00
|
|
|
diag.subdiagnostic(ExpectedLifetimeParameter { span: path_span, count: n });
|
2022-07-11 16:51:19 +09:00
|
|
|
if !source_map.is_span_accessible(insertion_span) {
|
2022-03-10 23:12:35 +01:00
|
|
|
// Do not try to suggest anything if generated by a proc-macro.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let anon_lts = vec!["'_"; n].join(", ");
|
|
|
|
let suggestion =
|
2023-07-25 22:00:13 +02:00
|
|
|
if incl_angl_brckt { format!("<{anon_lts}>") } else { format!("{anon_lts}, ") };
|
2023-07-03 18:08:49 +09:00
|
|
|
|
|
|
|
diag.subdiagnostic(IndicateAnonymousLifetime {
|
|
|
|
span: insertion_span.shrink_to_hi(),
|
|
|
|
count: n,
|
2020-08-13 15:41:52 -04:00
|
|
|
suggestion,
|
2023-07-03 18:08:49 +09:00
|
|
|
});
|
2019-09-08 14:57:03 +05:30
|
|
|
}
|
2019-11-24 16:59:56 -05:00
|
|
|
|
2023-07-26 22:46:49 +08:00
|
|
|
pub fn report_ambiguity_error<'a, G: EmissionGuarantee>(
|
|
|
|
db: &mut DiagnosticBuilder<'a, G>,
|
|
|
|
ambiguity: rustc_lint_defs::AmbiguityErrorDiag,
|
|
|
|
) {
|
|
|
|
db.span_label(ambiguity.label_span, ambiguity.label_msg);
|
|
|
|
db.note(ambiguity.note_msg);
|
|
|
|
db.span_note(ambiguity.b1_span, ambiguity.b1_note_msg);
|
|
|
|
for help_msg in ambiguity.b1_help_msgs {
|
|
|
|
db.help(help_msg);
|
|
|
|
}
|
|
|
|
db.span_note(ambiguity.b2_span, ambiguity.b2_note_msg);
|
|
|
|
for help_msg in ambiguity.b2_help_msgs {
|
|
|
|
db.help(help_msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-09 10:16:00 +00:00
|
|
|
#[derive(Clone, Copy, PartialEq, Hash, Debug)]
|
|
|
|
pub enum TerminalUrl {
|
|
|
|
No,
|
|
|
|
Yes,
|
|
|
|
Auto,
|
|
|
|
}
|