Rollup merge of #152114 - JonathanBrouwer:convert_mir_transform, r=jdonszelmann
Convert to inline diagnostics in `rustc_mir_transform` For https://github.com/rust-lang/rust/issues/151366 r? @jdonszelmann
This commit is contained in:
commit
c2da69cd18
7 changed files with 121 additions and 187 deletions
|
|
@ -4340,7 +4340,6 @@ dependencies = [
|
||||||
"rustc_const_eval",
|
"rustc_const_eval",
|
||||||
"rustc_data_structures",
|
"rustc_data_structures",
|
||||||
"rustc_errors",
|
"rustc_errors",
|
||||||
"rustc_fluent_macro",
|
|
||||||
"rustc_hir",
|
"rustc_hir",
|
||||||
"rustc_index",
|
"rustc_index",
|
||||||
"rustc_infer",
|
"rustc_infer",
|
||||||
|
|
|
||||||
|
|
@ -125,7 +125,6 @@ pub static DEFAULT_LOCALE_RESOURCES: &[&str] = &[
|
||||||
rustc_metadata::DEFAULT_LOCALE_RESOURCE,
|
rustc_metadata::DEFAULT_LOCALE_RESOURCE,
|
||||||
rustc_middle::DEFAULT_LOCALE_RESOURCE,
|
rustc_middle::DEFAULT_LOCALE_RESOURCE,
|
||||||
rustc_mir_build::DEFAULT_LOCALE_RESOURCE,
|
rustc_mir_build::DEFAULT_LOCALE_RESOURCE,
|
||||||
rustc_mir_transform::DEFAULT_LOCALE_RESOURCE,
|
|
||||||
rustc_parse::DEFAULT_LOCALE_RESOURCE,
|
rustc_parse::DEFAULT_LOCALE_RESOURCE,
|
||||||
rustc_passes::DEFAULT_LOCALE_RESOURCE,
|
rustc_passes::DEFAULT_LOCALE_RESOURCE,
|
||||||
rustc_pattern_analysis::DEFAULT_LOCALE_RESOURCE,
|
rustc_pattern_analysis::DEFAULT_LOCALE_RESOURCE,
|
||||||
|
|
|
||||||
|
|
@ -13,7 +13,6 @@ rustc_ast = { path = "../rustc_ast" }
|
||||||
rustc_const_eval = { path = "../rustc_const_eval" }
|
rustc_const_eval = { path = "../rustc_const_eval" }
|
||||||
rustc_data_structures = { path = "../rustc_data_structures" }
|
rustc_data_structures = { path = "../rustc_data_structures" }
|
||||||
rustc_errors = { path = "../rustc_errors" }
|
rustc_errors = { path = "../rustc_errors" }
|
||||||
rustc_fluent_macro = { path = "../rustc_fluent_macro" }
|
|
||||||
rustc_hir = { path = "../rustc_hir" }
|
rustc_hir = { path = "../rustc_hir" }
|
||||||
rustc_index = { path = "../rustc_index" }
|
rustc_index = { path = "../rustc_index" }
|
||||||
rustc_infer = { path = "../rustc_infer" }
|
rustc_infer = { path = "../rustc_infer" }
|
||||||
|
|
|
||||||
|
|
@ -1,114 +0,0 @@
|
||||||
mir_transform_arithmetic_overflow = this arithmetic operation will overflow
|
|
||||||
|
|
||||||
mir_transform_asm_unwind_call = call to inline assembly that may unwind
|
|
||||||
|
|
||||||
mir_transform_const_defined_here = `const` item defined here
|
|
||||||
|
|
||||||
mir_transform_const_modify = attempting to modify a `const` item
|
|
||||||
.note = each usage of a `const` item creates a new temporary; the original `const` item will not be modified
|
|
||||||
|
|
||||||
mir_transform_const_mut_borrow = taking a mutable reference to a `const` item
|
|
||||||
.note = each usage of a `const` item creates a new temporary
|
|
||||||
.note2 = the mutable reference will refer to this temporary, not the original `const` item
|
|
||||||
.note3 = mutable reference created due to call to this method
|
|
||||||
|
|
||||||
mir_transform_ffi_unwind_call = call to {$foreign ->
|
|
||||||
[true] foreign function
|
|
||||||
*[false] function pointer
|
|
||||||
} with FFI-unwind ABI
|
|
||||||
|
|
||||||
mir_transform_fn_item_ref = taking a reference to a function item does not give a function pointer
|
|
||||||
.suggestion = cast `{$ident}` to obtain a function pointer
|
|
||||||
|
|
||||||
mir_transform_force_inline =
|
|
||||||
`{$callee}` could not be inlined into `{$caller}` but is required to be inlined
|
|
||||||
.call = ...`{$callee}` called here
|
|
||||||
.attr = inlining due to this annotation
|
|
||||||
.caller = within `{$caller}`...
|
|
||||||
.callee = `{$callee}` defined here
|
|
||||||
.note = could not be inlined due to: {$reason}
|
|
||||||
|
|
||||||
mir_transform_force_inline_attr =
|
|
||||||
`{$callee}` is incompatible with `#[rustc_force_inline]`
|
|
||||||
.attr = annotation here
|
|
||||||
.callee = `{$callee}` defined here
|
|
||||||
.note = incompatible due to: {$reason}
|
|
||||||
|
|
||||||
mir_transform_force_inline_justification =
|
|
||||||
`{$callee}` is required to be inlined to: {$sym}
|
|
||||||
|
|
||||||
mir_transform_maybe_string_interpolation = you might have meant to use string interpolation in this string literal
|
|
||||||
|
|
||||||
mir_transform_must_not_suspend = {$pre}`{$def_path}`{$post} held across a suspend point, but should not be
|
|
||||||
.label = the value is held across this suspend point
|
|
||||||
.note = {$reason}
|
|
||||||
.help = consider using a block (`{"{ ... }"}`) to shrink the value's scope, ending before the suspend point
|
|
||||||
mir_transform_operation_will_panic = this operation will panic at runtime
|
|
||||||
|
|
||||||
mir_transform_string_interpolation_only_works = string interpolation only works in `format!` invocations
|
|
||||||
|
|
||||||
mir_transform_tail_expr_drop_order = relative drop order changing in Rust 2024
|
|
||||||
.temporaries = in Rust 2024, this temporary value will be dropped first
|
|
||||||
.observers = in Rust 2024, this local variable or temporary value will be dropped second
|
|
||||||
.note_dtors =
|
|
||||||
dropping the temporary value runs this custom `Drop` impl, which we could not prove to be side-effect free
|
|
||||||
.note_observer_dtors =
|
|
||||||
dropping the local runs this custom `Drop` impl, which we could not prove to be side-effect free
|
|
||||||
.drop_location =
|
|
||||||
now the temporary value is dropped here, before the local variables in the block or statement
|
|
||||||
.note_epilogue = most of the time, changing drop order is harmless; inspect the `impl Drop`s for side effects like releasing locks or sending messages
|
|
||||||
.label_local_epilogue = {$is_dropped_first_edition_2024 ->
|
|
||||||
[true] up until Edition 2021 `{$name}` is dropped last but will be dropped earlier in Edition 2024
|
|
||||||
*[false] `{$name}` will be dropped later as of Edition 2024
|
|
||||||
}
|
|
||||||
|
|
||||||
mir_transform_tail_expr_dtor = {$dtor_kind ->
|
|
||||||
[dyn] `{$name}` may invoke a custom destructor because it contains a trait object
|
|
||||||
*[concrete] `{$name}` invokes this custom destructor
|
|
||||||
}
|
|
||||||
|
|
||||||
mir_transform_tail_expr_local = {$is_generated_name ->
|
|
||||||
[true] this value will be stored in a temporary; let us call it `{$name}`
|
|
||||||
*[false] `{$name}` calls a custom destructor
|
|
||||||
}
|
|
||||||
|
|
||||||
mir_transform_unaligned_packed_ref = reference to field of packed {$ty_descr} is unaligned
|
|
||||||
.note = this {$ty_descr} is {$align ->
|
|
||||||
[one] {""}
|
|
||||||
*[other] {"at most "}
|
|
||||||
}{$align}-byte aligned, but the type of this field may require higher alignment
|
|
||||||
.note_ub = creating a misaligned reference is undefined behavior (even if that reference is never dereferenced)
|
|
||||||
.help = copy the field contents to a local variable, or replace the reference with a raw pointer and use `read_unaligned`/`write_unaligned` (loads and stores via `*p` must be properly aligned even when using raw pointers)
|
|
||||||
|
|
||||||
mir_transform_unconditional_recursion = function cannot return without recursing
|
|
||||||
.label = cannot return without recursing
|
|
||||||
.help = a `loop` may express intention better if this is on purpose
|
|
||||||
|
|
||||||
mir_transform_unconditional_recursion_call_site_label = recursive call site
|
|
||||||
|
|
||||||
mir_transform_unknown_pass_name = MIR pass `{$name}` is unknown and will be ignored
|
|
||||||
|
|
||||||
mir_transform_unused_assign = value assigned to `{$name}` is never read
|
|
||||||
.help = maybe it is overwritten before being read?
|
|
||||||
|
|
||||||
mir_transform_unused_assign_passed = value passed to `{$name}` is never read
|
|
||||||
.help = maybe it is overwritten before being read?
|
|
||||||
|
|
||||||
mir_transform_unused_assign_suggestion =
|
|
||||||
you might have meant to mutate the pointed at value being passed in, instead of changing the reference in the local binding
|
|
||||||
|
|
||||||
mir_transform_unused_capture_maybe_capture_ref = value captured by `{$name}` is never read
|
|
||||||
.help = did you mean to capture by reference instead?
|
|
||||||
|
|
||||||
mir_transform_unused_var_assigned_only = variable `{$name}` is assigned to, but never used
|
|
||||||
.note = consider using `_{$name}` instead
|
|
||||||
|
|
||||||
mir_transform_unused_var_underscore = if this is intentional, prefix it with an underscore
|
|
||||||
|
|
||||||
mir_transform_unused_variable = unused variable: `{$name}`
|
|
||||||
|
|
||||||
mir_transform_unused_variable_args_in_macro = `{$name}` is captured in macro and introduced a unused variable
|
|
||||||
|
|
||||||
mir_transform_unused_variable_try_ignore = try ignoring the field
|
|
||||||
|
|
||||||
mir_transform_unused_variable_typo = you might have meant to pattern match on the similarly named {$kind} `{$item_name}`
|
|
||||||
|
|
@ -1,5 +1,7 @@
|
||||||
use rustc_errors::codes::*;
|
use rustc_errors::codes::*;
|
||||||
use rustc_errors::{Applicability, Diag, EmissionGuarantee, LintDiagnostic, Subdiagnostic};
|
use rustc_errors::{
|
||||||
|
Applicability, Diag, EmissionGuarantee, LintDiagnostic, Subdiagnostic, inline_fluent,
|
||||||
|
};
|
||||||
use rustc_macros::{Diagnostic, LintDiagnostic, Subdiagnostic};
|
use rustc_macros::{Diagnostic, LintDiagnostic, Subdiagnostic};
|
||||||
use rustc_middle::mir::AssertKind;
|
use rustc_middle::mir::AssertKind;
|
||||||
use rustc_middle::query::Key;
|
use rustc_middle::query::Key;
|
||||||
|
|
@ -8,8 +10,6 @@ use rustc_session::lint::{self, Lint};
|
||||||
use rustc_span::def_id::DefId;
|
use rustc_span::def_id::DefId;
|
||||||
use rustc_span::{Ident, Span, Symbol};
|
use rustc_span::{Ident, Span, Symbol};
|
||||||
|
|
||||||
use crate::fluent_generated as fluent;
|
|
||||||
|
|
||||||
/// Emit diagnostic for calls to `#[inline(always)]`-annotated functions with a
|
/// Emit diagnostic for calls to `#[inline(always)]`-annotated functions with a
|
||||||
/// `#[target_feature]` attribute where the caller enables a different set of target features.
|
/// `#[target_feature]` attribute where the caller enables a different set of target features.
|
||||||
pub(crate) fn emit_inline_always_target_feature_diagnostic<'a, 'tcx>(
|
pub(crate) fn emit_inline_always_target_feature_diagnostic<'a, 'tcx>(
|
||||||
|
|
@ -51,22 +51,22 @@ pub(crate) fn emit_inline_always_target_feature_diagnostic<'a, 'tcx>(
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unconditional_recursion)]
|
#[diag("function cannot return without recursing")]
|
||||||
#[help]
|
#[help("a `loop` may express intention better if this is on purpose")]
|
||||||
pub(crate) struct UnconditionalRecursion {
|
pub(crate) struct UnconditionalRecursion {
|
||||||
#[label]
|
#[label("cannot return without recursing")]
|
||||||
pub(crate) span: Span,
|
pub(crate) span: Span,
|
||||||
#[label(mir_transform_unconditional_recursion_call_site_label)]
|
#[label("recursive call site")]
|
||||||
pub(crate) call_sites: Vec<Span>,
|
pub(crate) call_sites: Vec<Span>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Diagnostic)]
|
#[derive(Diagnostic)]
|
||||||
#[diag(mir_transform_force_inline_attr)]
|
#[diag("`{$callee}` is incompatible with `#[rustc_force_inline]`")]
|
||||||
#[note]
|
#[note("incompatible due to: {$reason}")]
|
||||||
pub(crate) struct InvalidForceInline {
|
pub(crate) struct InvalidForceInline {
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
pub attr_span: Span,
|
pub attr_span: Span,
|
||||||
#[label(mir_transform_callee)]
|
#[label("`{$callee}` defined here")]
|
||||||
pub callee_span: Span,
|
pub callee_span: Span,
|
||||||
pub callee: String,
|
pub callee: String,
|
||||||
pub reason: &'static str,
|
pub reason: &'static str,
|
||||||
|
|
@ -74,28 +74,39 @@ pub(crate) struct InvalidForceInline {
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
pub(crate) enum ConstMutate {
|
pub(crate) enum ConstMutate {
|
||||||
#[diag(mir_transform_const_modify)]
|
#[diag("attempting to modify a `const` item")]
|
||||||
#[note]
|
#[note(
|
||||||
|
"each usage of a `const` item creates a new temporary; the original `const` item will not be modified"
|
||||||
|
)]
|
||||||
Modify {
|
Modify {
|
||||||
#[note(mir_transform_const_defined_here)]
|
#[note("`const` item defined here")]
|
||||||
konst: Span,
|
konst: Span,
|
||||||
},
|
},
|
||||||
#[diag(mir_transform_const_mut_borrow)]
|
#[diag("taking a mutable reference to a `const` item")]
|
||||||
#[note]
|
#[note("each usage of a `const` item creates a new temporary")]
|
||||||
#[note(mir_transform_note2)]
|
#[note("the mutable reference will refer to this temporary, not the original `const` item")]
|
||||||
MutBorrow {
|
MutBorrow {
|
||||||
#[note(mir_transform_note3)]
|
#[note("mutable reference created due to call to this method")]
|
||||||
method_call: Option<Span>,
|
method_call: Option<Span>,
|
||||||
#[note(mir_transform_const_defined_here)]
|
#[note("`const` item defined here")]
|
||||||
konst: Span,
|
konst: Span,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Diagnostic)]
|
#[derive(Diagnostic)]
|
||||||
#[diag(mir_transform_unaligned_packed_ref, code = E0793)]
|
#[diag("reference to field of packed {$ty_descr} is unaligned", code = E0793)]
|
||||||
#[note]
|
#[note(
|
||||||
#[note(mir_transform_note_ub)]
|
"this {$ty_descr} is {$align ->
|
||||||
#[help]
|
[one] {\"\"}
|
||||||
|
*[other] {\"at most \"}
|
||||||
|
}{$align}-byte aligned, but the type of this field may require higher alignment"
|
||||||
|
)]
|
||||||
|
#[note(
|
||||||
|
"creating a misaligned reference is undefined behavior (even if that reference is never dereferenced)"
|
||||||
|
)]
|
||||||
|
#[help(
|
||||||
|
"copy the field contents to a local variable, or replace the reference with a raw pointer and use `read_unaligned`/`write_unaligned` (loads and stores via `*p` must be properly aligned even when using raw pointers)"
|
||||||
|
)]
|
||||||
pub(crate) struct UnalignedPackedRef {
|
pub(crate) struct UnalignedPackedRef {
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
|
|
@ -104,7 +115,7 @@ pub(crate) struct UnalignedPackedRef {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Diagnostic)]
|
#[derive(Diagnostic)]
|
||||||
#[diag(mir_transform_unknown_pass_name)]
|
#[diag("MIR pass `{$name}` is unknown and will be ignored")]
|
||||||
pub(crate) struct UnknownPassName<'a> {
|
pub(crate) struct UnknownPassName<'a> {
|
||||||
pub(crate) name: &'a str,
|
pub(crate) name: &'a str,
|
||||||
}
|
}
|
||||||
|
|
@ -123,8 +134,12 @@ pub(crate) enum AssertLintKind {
|
||||||
impl<'a, P: std::fmt::Debug> LintDiagnostic<'a, ()> for AssertLint<P> {
|
impl<'a, P: std::fmt::Debug> LintDiagnostic<'a, ()> for AssertLint<P> {
|
||||||
fn decorate_lint<'b>(self, diag: &'b mut Diag<'a, ()>) {
|
fn decorate_lint<'b>(self, diag: &'b mut Diag<'a, ()>) {
|
||||||
diag.primary_message(match self.lint_kind {
|
diag.primary_message(match self.lint_kind {
|
||||||
AssertLintKind::ArithmeticOverflow => fluent::mir_transform_arithmetic_overflow,
|
AssertLintKind::ArithmeticOverflow => {
|
||||||
AssertLintKind::UnconditionalPanic => fluent::mir_transform_operation_will_panic,
|
inline_fluent!("this arithmetic operation will overflow")
|
||||||
|
}
|
||||||
|
AssertLintKind::UnconditionalPanic => {
|
||||||
|
inline_fluent!("this operation will panic at runtime")
|
||||||
|
}
|
||||||
});
|
});
|
||||||
let label = self.assert_kind.diagnostic_message();
|
let label = self.assert_kind.diagnostic_message();
|
||||||
self.assert_kind.add_args(&mut |name, value| {
|
self.assert_kind.add_args(&mut |name, value| {
|
||||||
|
|
@ -144,39 +159,53 @@ impl AssertLintKind {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_asm_unwind_call)]
|
#[diag("call to inline assembly that may unwind")]
|
||||||
pub(crate) struct AsmUnwindCall {
|
pub(crate) struct AsmUnwindCall {
|
||||||
#[label(mir_transform_asm_unwind_call)]
|
#[label("call to inline assembly that may unwind")]
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_ffi_unwind_call)]
|
#[diag(
|
||||||
|
"call to {$foreign ->
|
||||||
|
[true] foreign function
|
||||||
|
*[false] function pointer
|
||||||
|
} with FFI-unwind ABI"
|
||||||
|
)]
|
||||||
pub(crate) struct FfiUnwindCall {
|
pub(crate) struct FfiUnwindCall {
|
||||||
#[label(mir_transform_ffi_unwind_call)]
|
#[label(
|
||||||
|
"call to {$foreign ->
|
||||||
|
[true] foreign function
|
||||||
|
*[false] function pointer
|
||||||
|
} with FFI-unwind ABI"
|
||||||
|
)]
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
pub foreign: bool,
|
pub foreign: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_fn_item_ref)]
|
#[diag("taking a reference to a function item does not give a function pointer")]
|
||||||
pub(crate) struct FnItemRef {
|
pub(crate) struct FnItemRef {
|
||||||
#[suggestion(code = "{sugg}", applicability = "unspecified")]
|
#[suggestion(
|
||||||
|
"cast `{$ident}` to obtain a function pointer",
|
||||||
|
code = "{sugg}",
|
||||||
|
applicability = "unspecified"
|
||||||
|
)]
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
pub sugg: String,
|
pub sugg: String,
|
||||||
pub ident: Ident,
|
pub ident: Ident,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unused_capture_maybe_capture_ref)]
|
#[diag("value captured by `{$name}` is never read")]
|
||||||
#[help]
|
#[help("did you mean to capture by reference instead?")]
|
||||||
pub(crate) struct UnusedCaptureMaybeCaptureRef {
|
pub(crate) struct UnusedCaptureMaybeCaptureRef {
|
||||||
pub name: Symbol,
|
pub name: Symbol,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unused_var_assigned_only)]
|
#[diag("variable `{$name}` is assigned to, but never used")]
|
||||||
#[note]
|
#[note("consider using `_{$name}` instead")]
|
||||||
pub(crate) struct UnusedVarAssignedOnly {
|
pub(crate) struct UnusedVarAssignedOnly {
|
||||||
pub name: Symbol,
|
pub name: Symbol,
|
||||||
#[subdiagnostic]
|
#[subdiagnostic]
|
||||||
|
|
@ -184,17 +213,20 @@ pub(crate) struct UnusedVarAssignedOnly {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unused_assign)]
|
#[diag("value assigned to `{$name}` is never read")]
|
||||||
pub(crate) struct UnusedAssign {
|
pub(crate) struct UnusedAssign {
|
||||||
pub name: Symbol,
|
pub name: Symbol,
|
||||||
#[subdiagnostic]
|
#[subdiagnostic]
|
||||||
pub suggestion: Option<UnusedAssignSuggestion>,
|
pub suggestion: Option<UnusedAssignSuggestion>,
|
||||||
#[help]
|
#[help("maybe it is overwritten before being read?")]
|
||||||
pub help: bool,
|
pub help: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
#[multipart_suggestion(mir_transform_unused_assign_suggestion, applicability = "maybe-incorrect")]
|
#[multipart_suggestion(
|
||||||
|
"you might have meant to mutate the pointed at value being passed in, instead of changing the reference in the local binding",
|
||||||
|
applicability = "maybe-incorrect"
|
||||||
|
)]
|
||||||
pub(crate) struct UnusedAssignSuggestion {
|
pub(crate) struct UnusedAssignSuggestion {
|
||||||
pub pre: &'static str,
|
pub pre: &'static str,
|
||||||
#[suggestion_part(code = "{pre}mut ")]
|
#[suggestion_part(code = "{pre}mut ")]
|
||||||
|
|
@ -208,14 +240,14 @@ pub(crate) struct UnusedAssignSuggestion {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unused_assign_passed)]
|
#[diag("value passed to `{$name}` is never read")]
|
||||||
#[help]
|
#[help("maybe it is overwritten before being read?")]
|
||||||
pub(crate) struct UnusedAssignPassed {
|
pub(crate) struct UnusedAssignPassed {
|
||||||
pub name: Symbol,
|
pub name: Symbol,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_unused_variable)]
|
#[diag("unused variable: `{$name}`")]
|
||||||
pub(crate) struct UnusedVariable {
|
pub(crate) struct UnusedVariable {
|
||||||
pub name: Symbol,
|
pub name: Symbol,
|
||||||
#[subdiagnostic]
|
#[subdiagnostic]
|
||||||
|
|
@ -226,10 +258,7 @@ pub(crate) struct UnusedVariable {
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
pub(crate) enum UnusedVariableSugg {
|
pub(crate) enum UnusedVariableSugg {
|
||||||
#[multipart_suggestion(
|
#[multipart_suggestion("try ignoring the field", applicability = "machine-applicable")]
|
||||||
mir_transform_unused_variable_try_ignore,
|
|
||||||
applicability = "machine-applicable"
|
|
||||||
)]
|
|
||||||
TryIgnore {
|
TryIgnore {
|
||||||
#[suggestion_part(code = "{name}: _")]
|
#[suggestion_part(code = "{name}: _")]
|
||||||
shorthands: Vec<Span>,
|
shorthands: Vec<Span>,
|
||||||
|
|
@ -239,7 +268,7 @@ pub(crate) enum UnusedVariableSugg {
|
||||||
},
|
},
|
||||||
|
|
||||||
#[multipart_suggestion(
|
#[multipart_suggestion(
|
||||||
mir_transform_unused_var_underscore,
|
"if this is intentional, prefix it with an underscore",
|
||||||
applicability = "machine-applicable"
|
applicability = "machine-applicable"
|
||||||
)]
|
)]
|
||||||
TryPrefix {
|
TryPrefix {
|
||||||
|
|
@ -250,7 +279,7 @@ pub(crate) enum UnusedVariableSugg {
|
||||||
typo: Option<PatternTypo>,
|
typo: Option<PatternTypo>,
|
||||||
},
|
},
|
||||||
|
|
||||||
#[help(mir_transform_unused_variable_args_in_macro)]
|
#[help("`{$name}` is captured in macro and introduced a unused variable")]
|
||||||
NoSugg {
|
NoSugg {
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
span: Span,
|
span: Span,
|
||||||
|
|
@ -266,10 +295,12 @@ impl Subdiagnostic for UnusedVariableStringInterp {
|
||||||
fn add_to_diag<G: EmissionGuarantee>(self, diag: &mut Diag<'_, G>) {
|
fn add_to_diag<G: EmissionGuarantee>(self, diag: &mut Diag<'_, G>) {
|
||||||
diag.span_label(
|
diag.span_label(
|
||||||
self.lit,
|
self.lit,
|
||||||
crate::fluent_generated::mir_transform_maybe_string_interpolation,
|
inline_fluent!(
|
||||||
|
"you might have meant to use string interpolation in this string literal"
|
||||||
|
),
|
||||||
);
|
);
|
||||||
diag.multipart_suggestion(
|
diag.multipart_suggestion(
|
||||||
crate::fluent_generated::mir_transform_string_interpolation_only_works,
|
inline_fluent!("string interpolation only works in `format!` invocations"),
|
||||||
vec![
|
vec![
|
||||||
(self.lit.shrink_to_lo(), String::from("format!(")),
|
(self.lit.shrink_to_lo(), String::from("format!(")),
|
||||||
(self.lit.shrink_to_hi(), String::from(")")),
|
(self.lit.shrink_to_hi(), String::from(")")),
|
||||||
|
|
@ -281,7 +312,7 @@ impl Subdiagnostic for UnusedVariableStringInterp {
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
#[multipart_suggestion(
|
#[multipart_suggestion(
|
||||||
mir_transform_unused_variable_typo,
|
"you might have meant to pattern match on the similarly named {$kind} `{$item_name}`",
|
||||||
style = "verbose",
|
style = "verbose",
|
||||||
applicability = "maybe-incorrect"
|
applicability = "maybe-incorrect"
|
||||||
)]
|
)]
|
||||||
|
|
@ -306,12 +337,17 @@ pub(crate) struct MustNotSupend<'a, 'tcx> {
|
||||||
// Needed for def_path_str
|
// Needed for def_path_str
|
||||||
impl<'a> LintDiagnostic<'a, ()> for MustNotSupend<'_, '_> {
|
impl<'a> LintDiagnostic<'a, ()> for MustNotSupend<'_, '_> {
|
||||||
fn decorate_lint<'b>(self, diag: &'b mut rustc_errors::Diag<'a, ()>) {
|
fn decorate_lint<'b>(self, diag: &'b mut rustc_errors::Diag<'a, ()>) {
|
||||||
diag.primary_message(fluent::mir_transform_must_not_suspend);
|
diag.primary_message(inline_fluent!(
|
||||||
diag.span_label(self.yield_sp, fluent::_subdiag::label);
|
"{$pre}`{$def_path}`{$post} held across a suspend point, but should not be"
|
||||||
|
));
|
||||||
|
diag.span_label(
|
||||||
|
self.yield_sp,
|
||||||
|
inline_fluent!("the value is held across this suspend point"),
|
||||||
|
);
|
||||||
if let Some(reason) = self.reason {
|
if let Some(reason) = self.reason {
|
||||||
diag.subdiagnostic(reason);
|
diag.subdiagnostic(reason);
|
||||||
}
|
}
|
||||||
diag.span_help(self.src_sp, fluent::_subdiag::help);
|
diag.span_help(self.src_sp, inline_fluent!("consider using a block (`{\"{ ... }\"}`) to shrink the value's scope, ending before the suspend point"));
|
||||||
diag.arg("pre", self.pre);
|
diag.arg("pre", self.pre);
|
||||||
diag.arg("def_path", self.tcx.def_path_str(self.def_id));
|
diag.arg("def_path", self.tcx.def_path_str(self.def_id));
|
||||||
diag.arg("post", self.post);
|
diag.arg("post", self.post);
|
||||||
|
|
@ -319,7 +355,7 @@ impl<'a> LintDiagnostic<'a, ()> for MustNotSupend<'_, '_> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
#[note(mir_transform_note)]
|
#[note("{$reason}")]
|
||||||
pub(crate) struct MustNotSuspendReason {
|
pub(crate) struct MustNotSuspendReason {
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
|
|
@ -327,17 +363,17 @@ pub(crate) struct MustNotSuspendReason {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Diagnostic)]
|
#[derive(Diagnostic)]
|
||||||
#[diag(mir_transform_force_inline)]
|
#[diag("`{$callee}` could not be inlined into `{$caller}` but is required to be inlined")]
|
||||||
#[note]
|
#[note("could not be inlined due to: {$reason}")]
|
||||||
pub(crate) struct ForceInlineFailure {
|
pub(crate) struct ForceInlineFailure {
|
||||||
#[label(mir_transform_caller)]
|
#[label("within `{$caller}`...")]
|
||||||
pub caller_span: Span,
|
pub caller_span: Span,
|
||||||
#[label(mir_transform_callee)]
|
#[label("`{$callee}` defined here")]
|
||||||
pub callee_span: Span,
|
pub callee_span: Span,
|
||||||
#[label(mir_transform_attr)]
|
#[label("annotation here")]
|
||||||
pub attr_span: Span,
|
pub attr_span: Span,
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
#[label(mir_transform_call)]
|
#[label("...`{$callee}` called here")]
|
||||||
pub call_span: Span,
|
pub call_span: Span,
|
||||||
pub callee: String,
|
pub callee: String,
|
||||||
pub caller: String,
|
pub caller: String,
|
||||||
|
|
@ -347,7 +383,7 @@ pub(crate) struct ForceInlineFailure {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
#[note(mir_transform_force_inline_justification)]
|
#[note("`{$callee}` is required to be inlined to: {$sym}")]
|
||||||
pub(crate) struct ForceInlineJustification {
|
pub(crate) struct ForceInlineJustification {
|
||||||
pub sym: Symbol,
|
pub sym: Symbol,
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -213,8 +213,6 @@ declare_passes! {
|
||||||
mod validate : Validator;
|
mod validate : Validator;
|
||||||
}
|
}
|
||||||
|
|
||||||
rustc_fluent_macro::fluent_messages! { "../messages.ftl" }
|
|
||||||
|
|
||||||
pub fn provide(providers: &mut Providers) {
|
pub fn provide(providers: &mut Providers) {
|
||||||
coverage::query::provide(providers);
|
coverage::query::provide(providers);
|
||||||
ffi_unwind_calls::provide(&mut providers.queries);
|
ffi_unwind_calls::provide(&mut providers.queries);
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@ use std::rc::Rc;
|
||||||
use itertools::Itertools as _;
|
use itertools::Itertools as _;
|
||||||
use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexMap};
|
use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexMap};
|
||||||
use rustc_data_structures::unord::{UnordMap, UnordSet};
|
use rustc_data_structures::unord::{UnordMap, UnordSet};
|
||||||
use rustc_errors::Subdiagnostic;
|
use rustc_errors::{Subdiagnostic, inline_fluent};
|
||||||
use rustc_hir::CRATE_HIR_ID;
|
use rustc_hir::CRATE_HIR_ID;
|
||||||
use rustc_hir::def_id::LocalDefId;
|
use rustc_hir::def_id::LocalDefId;
|
||||||
use rustc_index::bit_set::MixedBitSet;
|
use rustc_index::bit_set::MixedBitSet;
|
||||||
|
|
@ -499,13 +499,17 @@ fn assign_observables_names(
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(mir_transform_tail_expr_drop_order)]
|
#[diag("relative drop order changing in Rust 2024")]
|
||||||
struct TailExprDropOrderLint<'a> {
|
struct TailExprDropOrderLint<'a> {
|
||||||
#[subdiagnostic]
|
#[subdiagnostic]
|
||||||
local_labels: Vec<LocalLabel<'a>>,
|
local_labels: Vec<LocalLabel<'a>>,
|
||||||
#[label(mir_transform_drop_location)]
|
#[label(
|
||||||
|
"now the temporary value is dropped here, before the local variables in the block or statement"
|
||||||
|
)]
|
||||||
drop_span: Option<Span>,
|
drop_span: Option<Span>,
|
||||||
#[note(mir_transform_note_epilogue)]
|
#[note(
|
||||||
|
"most of the time, changing drop order is harmless; inspect the `impl Drop`s for side effects like releasing locks or sending messages"
|
||||||
|
)]
|
||||||
_epilogue: (),
|
_epilogue: (),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -527,19 +531,32 @@ impl Subdiagnostic for LocalLabel<'_> {
|
||||||
diag.arg("is_generated_name", self.is_generated_name);
|
diag.arg("is_generated_name", self.is_generated_name);
|
||||||
diag.remove_arg("is_dropped_first_edition_2024");
|
diag.remove_arg("is_dropped_first_edition_2024");
|
||||||
diag.arg("is_dropped_first_edition_2024", self.is_dropped_first_edition_2024);
|
diag.arg("is_dropped_first_edition_2024", self.is_dropped_first_edition_2024);
|
||||||
let msg = diag.eagerly_translate(crate::fluent_generated::mir_transform_tail_expr_local);
|
let msg = diag.eagerly_translate(inline_fluent!(
|
||||||
|
"{$is_generated_name ->
|
||||||
|
[true] this value will be stored in a temporary; let us call it `{$name}`
|
||||||
|
*[false] `{$name}` calls a custom destructor
|
||||||
|
}"
|
||||||
|
));
|
||||||
diag.span_label(self.span, msg);
|
diag.span_label(self.span, msg);
|
||||||
for dtor in self.destructors {
|
for dtor in self.destructors {
|
||||||
dtor.add_to_diag(diag);
|
dtor.add_to_diag(diag);
|
||||||
}
|
}
|
||||||
let msg =
|
let msg =
|
||||||
diag.eagerly_translate(crate::fluent_generated::mir_transform_label_local_epilogue);
|
diag.eagerly_translate(inline_fluent!("{$is_dropped_first_edition_2024 ->
|
||||||
|
[true] up until Edition 2021 `{$name}` is dropped last but will be dropped earlier in Edition 2024
|
||||||
|
*[false] `{$name}` will be dropped later as of Edition 2024
|
||||||
|
}"));
|
||||||
diag.span_label(self.span, msg);
|
diag.span_label(self.span, msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Subdiagnostic)]
|
#[derive(Subdiagnostic)]
|
||||||
#[note(mir_transform_tail_expr_dtor)]
|
#[note(
|
||||||
|
"{$dtor_kind ->
|
||||||
|
[dyn] `{$name}` may invoke a custom destructor because it contains a trait object
|
||||||
|
*[concrete] `{$name}` invokes this custom destructor
|
||||||
|
}"
|
||||||
|
)]
|
||||||
struct DestructorLabel<'a> {
|
struct DestructorLabel<'a> {
|
||||||
#[primary_span]
|
#[primary_span]
|
||||||
span: Span,
|
span: Span,
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue