Skip to main content

rustc_mir_transform/
errors.rs

1use rustc_errors::codes::*;
2use rustc_errors::{
3    Applicability, Diag, EmissionGuarantee, LintDiagnostic, Subdiagnostic, inline_fluent,
4};
5use rustc_macros::{Diagnostic, LintDiagnostic, Subdiagnostic};
6use rustc_middle::mir::AssertKind;
7use rustc_middle::query::Key;
8use rustc_middle::ty::TyCtxt;
9use rustc_session::lint::{self, Lint};
10use rustc_span::def_id::DefId;
11use rustc_span::{Ident, Span, Symbol};
12
13/// Emit diagnostic for calls to `#[inline(always)]`-annotated functions with a
14/// `#[target_feature]` attribute where the caller enables a different set of target features.
15pub(crate) fn emit_inline_always_target_feature_diagnostic<'a, 'tcx>(
16    tcx: TyCtxt<'tcx>,
17    call_span: Span,
18    callee_def_id: DefId,
19    caller_def_id: DefId,
20    callee_only: &[&'a str],
21) {
22    tcx.node_span_lint(
23        lint::builtin::INLINE_ALWAYS_MISMATCHING_TARGET_FEATURES,
24        tcx.local_def_id_to_hir_id(caller_def_id.as_local().unwrap()),
25        call_span,
26        |lint| {
27            let callee = tcx.def_path_str(callee_def_id);
28            let caller = tcx.def_path_str(caller_def_id);
29
30            lint.primary_message(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("call to `#[inline(always)]`-annotated `{0}` requires the same target features to be inlined",
                callee))
    })format!(
31                "call to `#[inline(always)]`-annotated `{callee}` \
32                requires the same target features to be inlined"
33            ));
34            lint.note("function will not be inlined");
35
36            lint.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the following target features are on `{1}` but missing from `{2}`: {0}",
                callee_only.join(", "), callee, caller))
    })format!(
37                "the following target features are on `{callee}` but missing from `{caller}`: {}",
38                callee_only.join(", ")
39            ));
40            lint.span_note(callee_def_id.default_span(tcx), ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` is defined here", callee))
    })format!("`{callee}` is defined here"));
41
42            let feats = callee_only.join(",");
43            lint.span_suggestion(
44                tcx.def_span(caller_def_id).shrink_to_lo(),
45                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("add `#[target_feature]` attribute to `{0}`",
                caller))
    })format!("add `#[target_feature]` attribute to `{caller}`"),
46                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("#[target_feature(enable = \"{0}\")]\n",
                feats))
    })format!("#[target_feature(enable = \"{feats}\")]\n"),
47                lint::Applicability::MaybeIncorrect,
48            );
49        },
50    );
51}
52
53#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for
            UnconditionalRecursion {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnconditionalRecursion {
                        span: __binding_0, call_sites: __binding_1 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("function cannot return without recursing")));
                        diag.help(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("a `loop` may express intention better if this is on purpose")));
                        ;
                        diag.span_label(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("cannot return without recursing")));
                        for __binding_1 in __binding_1 {
                            diag.span_label(__binding_1,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("recursive call site")));
                        }
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
54#[diag("function cannot return without recursing")]
55#[help("a `loop` may express intention better if this is on purpose")]
56pub(crate) struct UnconditionalRecursion {
57    #[label("cannot return without recursing")]
58    pub(crate) span: Span,
59    #[label("recursive call site")]
60    pub(crate) call_sites: Vec<Span>,
61}
62
63#[derive(const _: () =
    {
        impl<'_sess, G> rustc_errors::Diagnostic<'_sess, G> for
            InvalidForceInline where G: rustc_errors::EmissionGuarantee {
            #[track_caller]
            fn into_diag(self, dcx: rustc_errors::DiagCtxtHandle<'_sess>,
                level: rustc_errors::Level) -> rustc_errors::Diag<'_sess, G> {
                match self {
                    InvalidForceInline {
                        attr_span: __binding_0,
                        callee_span: __binding_1,
                        callee: __binding_2,
                        reason: __binding_3 } => {
                        let mut diag =
                            rustc_errors::Diag::new(dcx, level,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$callee}` is incompatible with `#[rustc_force_inline]`")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("incompatible due to: {$reason}")));
                        ;
                        diag.arg("callee", __binding_2);
                        diag.arg("reason", __binding_3);
                        diag.span(__binding_0);
                        diag.span_label(__binding_1,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$callee}` defined here")));
                        diag
                    }
                }
            }
        }
    };Diagnostic)]
64#[diag("`{$callee}` is incompatible with `#[rustc_force_inline]`")]
65#[note("incompatible due to: {$reason}")]
66pub(crate) struct InvalidForceInline {
67    #[primary_span]
68    pub attr_span: Span,
69    #[label("`{$callee}` defined here")]
70    pub callee_span: Span,
71    pub callee: String,
72    pub reason: &'static str,
73}
74
75#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for ConstMutate {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    ConstMutate::Modify { konst: __binding_0 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("attempting to modify a `const` item")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("each usage of a `const` item creates a new temporary; the original `const` item will not be modified")));
                        ;
                        diag.span_note(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`const` item defined here")));
                        diag
                    }
                    ConstMutate::MutBorrow {
                        method_call: __binding_0, konst: __binding_1 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("taking a mutable reference to a `const` item")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("each usage of a `const` item creates a new temporary")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("the mutable reference will refer to this temporary, not the original `const` item")));
                        ;
                        if let Some(__binding_0) = __binding_0 {
                            diag.span_note(__binding_0,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("mutable reference created due to call to this method")));
                        }
                        diag.span_note(__binding_1,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`const` item defined here")));
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
76pub(crate) enum ConstMutate {
77    #[diag("attempting to modify a `const` item")]
78    #[note(
79        "each usage of a `const` item creates a new temporary; the original `const` item will not be modified"
80    )]
81    Modify {
82        #[note("`const` item defined here")]
83        konst: Span,
84    },
85    #[diag("taking a mutable reference to a `const` item")]
86    #[note("each usage of a `const` item creates a new temporary")]
87    #[note("the mutable reference will refer to this temporary, not the original `const` item")]
88    MutBorrow {
89        #[note("mutable reference created due to call to this method")]
90        method_call: Option<Span>,
91        #[note("`const` item defined here")]
92        konst: Span,
93    },
94}
95
96#[derive(const _: () =
    {
        impl<'_sess, G> rustc_errors::Diagnostic<'_sess, G> for
            UnalignedPackedRef where G: rustc_errors::EmissionGuarantee {
            #[track_caller]
            fn into_diag(self, dcx: rustc_errors::DiagCtxtHandle<'_sess>,
                level: rustc_errors::Level) -> rustc_errors::Diag<'_sess, G> {
                match self {
                    UnalignedPackedRef {
                        span: __binding_0, ty_descr: __binding_1, align: __binding_2
                        } => {
                        let mut diag =
                            rustc_errors::Diag::new(dcx, level,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("reference to field of packed {$ty_descr} is unaligned")));
                        diag.code(E0793);
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this {$ty_descr} is {$align ->\n        [one] {\"\"}\n        *[other] {\"at most \"}\n    }{$align}-byte aligned, but the type of this field may require higher alignment")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("creating a misaligned reference is undefined behavior (even if that reference is never dereferenced)")));
                        diag.help(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("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)")));
                        ;
                        diag.arg("ty_descr", __binding_1);
                        diag.arg("align", __binding_2);
                        diag.span(__binding_0);
                        diag
                    }
                }
            }
        }
    };Diagnostic)]
97#[diag("reference to field of packed {$ty_descr} is unaligned", code = E0793)]
98#[note(
99    "this {$ty_descr} is {$align ->
100        [one] {\"\"}
101        *[other] {\"at most \"}
102    }{$align}-byte aligned, but the type of this field may require higher alignment"
103)]
104#[note(
105    "creating a misaligned reference is undefined behavior (even if that reference is never dereferenced)"
106)]
107#[help(
108    "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)"
109)]
110pub(crate) struct UnalignedPackedRef {
111    #[primary_span]
112    pub span: Span,
113    pub ty_descr: &'static str,
114    pub align: u64,
115}
116
117#[derive(const _: () =
    {
        impl<'_sess, 'a, G> rustc_errors::Diagnostic<'_sess, G> for
            UnknownPassName<'a> where G: rustc_errors::EmissionGuarantee {
            #[track_caller]
            fn into_diag(self, dcx: rustc_errors::DiagCtxtHandle<'_sess>,
                level: rustc_errors::Level) -> rustc_errors::Diag<'_sess, G> {
                match self {
                    UnknownPassName { name: __binding_0 } => {
                        let mut diag =
                            rustc_errors::Diag::new(dcx, level,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("MIR pass `{$name}` is unknown and will be ignored")));
                        ;
                        diag.arg("name", __binding_0);
                        diag
                    }
                }
            }
        }
    };Diagnostic)]
118#[diag("MIR pass `{$name}` is unknown and will be ignored")]
119pub(crate) struct UnknownPassName<'a> {
120    pub(crate) name: &'a str,
121}
122
123pub(crate) struct AssertLint<P> {
124    pub span: Span,
125    pub assert_kind: AssertKind<P>,
126    pub lint_kind: AssertLintKind,
127}
128
129pub(crate) enum AssertLintKind {
130    ArithmeticOverflow,
131    UnconditionalPanic,
132}
133
134impl<'a, P: std::fmt::Debug> LintDiagnostic<'a, ()> for AssertLint<P> {
135    fn decorate_lint<'b>(self, diag: &'b mut Diag<'a, ()>) {
136        diag.primary_message(match self.lint_kind {
137            AssertLintKind::ArithmeticOverflow => {
138                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this arithmetic operation will overflow"))inline_fluent!("this arithmetic operation will overflow")
139            }
140            AssertLintKind::UnconditionalPanic => {
141                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this operation will panic at runtime"))inline_fluent!("this operation will panic at runtime")
142            }
143        });
144        let label = self.assert_kind.diagnostic_message();
145        self.assert_kind.add_args(&mut |name, value| {
146            diag.arg(name, value);
147        });
148        diag.span_label(self.span, label);
149    }
150}
151
152impl AssertLintKind {
153    pub(crate) fn lint(&self) -> &'static Lint {
154        match self {
155            AssertLintKind::ArithmeticOverflow => lint::builtin::ARITHMETIC_OVERFLOW,
156            AssertLintKind::UnconditionalPanic => lint::builtin::UNCONDITIONAL_PANIC,
157        }
158    }
159}
160
161#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for AsmUnwindCall {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    AsmUnwindCall { span: __binding_0 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("call to inline assembly that may unwind")));
                        ;
                        diag.span_label(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("call to inline assembly that may unwind")));
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
162#[diag("call to inline assembly that may unwind")]
163pub(crate) struct AsmUnwindCall {
164    #[label("call to inline assembly that may unwind")]
165    pub span: Span,
166}
167
168#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for FfiUnwindCall {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    FfiUnwindCall { span: __binding_0, foreign: __binding_1 } =>
                        {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("call to {$foreign ->\n        [true] foreign function\n        *[false] function pointer\n    } with FFI-unwind ABI")));
                        ;
                        diag.arg("foreign", __binding_1);
                        diag.span_label(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("call to {$foreign ->\n            [true] foreign function\n            *[false] function pointer\n        } with FFI-unwind ABI")));
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
169#[diag(
170    "call to {$foreign ->
171        [true] foreign function
172        *[false] function pointer
173    } with FFI-unwind ABI"
174)]
175pub(crate) struct FfiUnwindCall {
176    #[label(
177        "call to {$foreign ->
178            [true] foreign function
179            *[false] function pointer
180        } with FFI-unwind ABI"
181    )]
182    pub span: Span,
183    pub foreign: bool,
184}
185
186#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for FnItemRef {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    FnItemRef {
                        span: __binding_0, sugg: __binding_1, ident: __binding_2 }
                        => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("taking a reference to a function item does not give a function pointer")));
                        ;
                        let __code_0 =
                            [::alloc::__export::must_use({
                                                ::alloc::fmt::format(format_args!("{0}", __binding_1))
                                            })].into_iter();
                        diag.arg("sugg", __binding_1);
                        diag.arg("ident", __binding_2);
                        diag.span_suggestions_with_style(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("cast `{$ident}` to obtain a function pointer")),
                            __code_0, rustc_errors::Applicability::Unspecified,
                            rustc_errors::SuggestionStyle::ShowCode);
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
187#[diag("taking a reference to a function item does not give a function pointer")]
188pub(crate) struct FnItemRef {
189    #[suggestion(
190        "cast `{$ident}` to obtain a function pointer",
191        code = "{sugg}",
192        applicability = "unspecified"
193    )]
194    pub span: Span,
195    pub sugg: String,
196    pub ident: Ident,
197}
198
199#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for
            UnusedCaptureMaybeCaptureRef {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnusedCaptureMaybeCaptureRef { name: __binding_0 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("value captured by `{$name}` is never read")));
                        diag.help(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("did you mean to capture by reference instead?")));
                        ;
                        diag.arg("name", __binding_0);
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
200#[diag("value captured by `{$name}` is never read")]
201#[help("did you mean to capture by reference instead?")]
202pub(crate) struct UnusedCaptureMaybeCaptureRef {
203    pub name: Symbol,
204}
205
206#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for
            UnusedVarAssignedOnly {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnusedVarAssignedOnly { name: __binding_0, typo: __binding_1
                        } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("variable `{$name}` is assigned to, but never used")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using `_{$name}` instead")));
                        ;
                        diag.arg("name", __binding_0);
                        if let Some(__binding_1) = __binding_1 {
                            diag.subdiagnostic(__binding_1);
                        }
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
207#[diag("variable `{$name}` is assigned to, but never used")]
208#[note("consider using `_{$name}` instead")]
209pub(crate) struct UnusedVarAssignedOnly {
210    pub name: Symbol,
211    #[subdiagnostic]
212    pub typo: Option<PatternTypo>,
213}
214
215#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for UnusedAssign {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnusedAssign {
                        name: __binding_0,
                        suggestion: __binding_1,
                        help: __binding_2 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("value assigned to `{$name}` is never read")));
                        ;
                        diag.arg("name", __binding_0);
                        if let Some(__binding_1) = __binding_1 {
                            diag.subdiagnostic(__binding_1);
                        }
                        if __binding_2 {
                            diag.help(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("maybe it is overwritten before being read?")));
                        }
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
216#[diag("value assigned to `{$name}` is never read")]
217pub(crate) struct UnusedAssign {
218    pub name: Symbol,
219    #[subdiagnostic]
220    pub suggestion: Option<UnusedAssignSuggestion>,
221    #[help("maybe it is overwritten before being read?")]
222    pub help: bool,
223}
224
225#[derive(const _: () =
    {
        impl rustc_errors::Subdiagnostic for UnusedAssignSuggestion {
            fn add_to_diag<__G>(self, diag: &mut rustc_errors::Diag<'_, __G>)
                where __G: rustc_errors::EmissionGuarantee {
                match self {
                    UnusedAssignSuggestion {
                        pre: __binding_0,
                        ty_span: __binding_1,
                        ty_ref_span: __binding_2,
                        pre_lhs_span: __binding_3,
                        rhs_borrow_span: __binding_4 } => {
                        let mut suggestions = Vec::new();
                        let __code_1 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("{0}mut ", __binding_0))
                                });
                        let __code_2 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!(""))
                                });
                        let __code_3 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("*"))
                                });
                        let __code_4 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!(""))
                                });
                        if let Some(__binding_1) = __binding_1 {
                            suggestions.push((__binding_1, __code_1));
                        }
                        suggestions.push((__binding_2, __code_2));
                        suggestions.push((__binding_3, __code_3));
                        suggestions.push((__binding_4, __code_4));
                        diag.store_args();
                        diag.arg("pre", __binding_0);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("you might have meant to mutate the pointed at value being passed in, instead of changing the reference in the local binding")));
                        diag.multipart_suggestion_with_style(__message, suggestions,
                            rustc_errors::Applicability::MaybeIncorrect,
                            rustc_errors::SuggestionStyle::ShowCode);
                        diag.restore_args();
                    }
                }
            }
        }
    };Subdiagnostic)]
226#[multipart_suggestion(
227    "you might have meant to mutate the pointed at value being passed in, instead of changing the reference in the local binding",
228    applicability = "maybe-incorrect"
229)]
230pub(crate) struct UnusedAssignSuggestion {
231    pub pre: &'static str,
232    #[suggestion_part(code = "{pre}mut ")]
233    pub ty_span: Option<Span>,
234    #[suggestion_part(code = "")]
235    pub ty_ref_span: Span,
236    #[suggestion_part(code = "*")]
237    pub pre_lhs_span: Span,
238    #[suggestion_part(code = "")]
239    pub rhs_borrow_span: Span,
240}
241
242#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for
            UnusedAssignPassed {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnusedAssignPassed { name: __binding_0 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("value passed to `{$name}` is never read")));
                        diag.help(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("maybe it is overwritten before being read?")));
                        ;
                        diag.arg("name", __binding_0);
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
243#[diag("value passed to `{$name}` is never read")]
244#[help("maybe it is overwritten before being read?")]
245pub(crate) struct UnusedAssignPassed {
246    pub name: Symbol,
247}
248
249#[derive(const _: () =
    {
        impl<'__a> rustc_errors::LintDiagnostic<'__a, ()> for UnusedVariable {
            #[track_caller]
            fn decorate_lint<'__b>(self,
                diag: &'__b mut rustc_errors::Diag<'__a, ()>) {
                match self {
                    UnusedVariable {
                        name: __binding_0,
                        string_interp: __binding_1,
                        sugg: __binding_2 } => {
                        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("unused variable: `{$name}`")));
                        ;
                        diag.arg("name", __binding_0);
                        for __binding_1 in __binding_1 {
                            diag.subdiagnostic(__binding_1);
                        }
                        diag.subdiagnostic(__binding_2);
                        diag
                    }
                };
            }
        }
    };LintDiagnostic)]
250#[diag("unused variable: `{$name}`")]
251pub(crate) struct UnusedVariable {
252    pub name: Symbol,
253    #[subdiagnostic]
254    pub string_interp: Vec<UnusedVariableStringInterp>,
255    #[subdiagnostic]
256    pub sugg: UnusedVariableSugg,
257}
258
259#[derive(const _: () =
    {
        impl rustc_errors::Subdiagnostic for UnusedVariableSugg {
            fn add_to_diag<__G>(self, diag: &mut rustc_errors::Diag<'_, __G>)
                where __G: rustc_errors::EmissionGuarantee {
                match self {
                    UnusedVariableSugg::TryIgnore {
                        shorthands: __binding_0,
                        non_shorthands: __binding_1,
                        name: __binding_2 } => {
                        let mut suggestions = Vec::new();
                        let __code_5 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("{0}: _", __binding_2))
                                });
                        let __code_6 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("_"))
                                });
                        for __binding_0 in __binding_0 {
                            suggestions.push((__binding_0, __code_5.clone()));
                        }
                        for __binding_1 in __binding_1 {
                            suggestions.push((__binding_1, __code_6.clone()));
                        }
                        diag.store_args();
                        diag.arg("name", __binding_2);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("try ignoring the field")));
                        diag.multipart_suggestion_with_style(__message, suggestions,
                            rustc_errors::Applicability::MachineApplicable,
                            rustc_errors::SuggestionStyle::ShowCode);
                        diag.restore_args();
                    }
                    UnusedVariableSugg::TryPrefix {
                        spans: __binding_0, name: __binding_1, typo: __binding_2 }
                        => {
                        let mut suggestions = Vec::new();
                        let __code_7 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("_{0}", __binding_1))
                                });
                        for __binding_0 in __binding_0 {
                            suggestions.push((__binding_0, __code_7.clone()));
                        }
                        if let Some(__binding_2) = __binding_2 {
                            __binding_2.add_to_diag(diag);
                        }
                        diag.store_args();
                        diag.arg("name", __binding_1);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("if this is intentional, prefix it with an underscore")));
                        diag.multipart_suggestion_with_style(__message, suggestions,
                            rustc_errors::Applicability::MachineApplicable,
                            rustc_errors::SuggestionStyle::ShowCode);
                        diag.restore_args();
                    }
                    UnusedVariableSugg::NoSugg {
                        span: __binding_0, name: __binding_1 } => {
                        diag.store_args();
                        diag.arg("name", __binding_1);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$name}` is captured in macro and introduced a unused variable")));
                        diag.span_help(__binding_0, __message);
                        diag.restore_args();
                    }
                }
            }
        }
    };Subdiagnostic)]
260pub(crate) enum UnusedVariableSugg {
261    #[multipart_suggestion("try ignoring the field", applicability = "machine-applicable")]
262    TryIgnore {
263        #[suggestion_part(code = "{name}: _")]
264        shorthands: Vec<Span>,
265        #[suggestion_part(code = "_")]
266        non_shorthands: Vec<Span>,
267        name: Symbol,
268    },
269
270    #[multipart_suggestion(
271        "if this is intentional, prefix it with an underscore",
272        applicability = "machine-applicable"
273    )]
274    TryPrefix {
275        #[suggestion_part(code = "_{name}")]
276        spans: Vec<Span>,
277        name: Symbol,
278        #[subdiagnostic]
279        typo: Option<PatternTypo>,
280    },
281
282    #[help("`{$name}` is captured in macro and introduced a unused variable")]
283    NoSugg {
284        #[primary_span]
285        span: Span,
286        name: Symbol,
287    },
288}
289
290pub(crate) struct UnusedVariableStringInterp {
291    pub lit: Span,
292}
293
294impl Subdiagnostic for UnusedVariableStringInterp {
295    fn add_to_diag<G: EmissionGuarantee>(self, diag: &mut Diag<'_, G>) {
296        diag.span_label(
297            self.lit,
298            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("you might have meant to use string interpolation in this string literal"))inline_fluent!(
299                "you might have meant to use string interpolation in this string literal"
300            ),
301        );
302        diag.multipart_suggestion(
303            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("string interpolation only works in `format!` invocations"))inline_fluent!("string interpolation only works in `format!` invocations"),
304            <[_]>::into_vec(::alloc::boxed::box_new([(self.lit.shrink_to_lo(),
                    String::from("format!(")),
                (self.lit.shrink_to_hi(), String::from(")"))]))vec![
305                (self.lit.shrink_to_lo(), String::from("format!(")),
306                (self.lit.shrink_to_hi(), String::from(")")),
307            ],
308            Applicability::MachineApplicable,
309        );
310    }
311}
312
313#[derive(const _: () =
    {
        impl rustc_errors::Subdiagnostic for PatternTypo {
            fn add_to_diag<__G>(self, diag: &mut rustc_errors::Diag<'_, __G>)
                where __G: rustc_errors::EmissionGuarantee {
                match self {
                    PatternTypo {
                        span: __binding_0,
                        code: __binding_1,
                        item_name: __binding_2,
                        kind: __binding_3 } => {
                        let mut suggestions = Vec::new();
                        let __code_8 =
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("{0}", __binding_1))
                                });
                        suggestions.push((__binding_0, __code_8));
                        diag.store_args();
                        diag.arg("code", __binding_1);
                        diag.arg("item_name", __binding_2);
                        diag.arg("kind", __binding_3);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("you might have meant to pattern match on the similarly named {$kind} `{$item_name}`")));
                        diag.multipart_suggestion_with_style(__message, suggestions,
                            rustc_errors::Applicability::MaybeIncorrect,
                            rustc_errors::SuggestionStyle::ShowAlways);
                        diag.restore_args();
                    }
                }
            }
        }
    };Subdiagnostic)]
314#[multipart_suggestion(
315    "you might have meant to pattern match on the similarly named {$kind} `{$item_name}`",
316    style = "verbose",
317    applicability = "maybe-incorrect"
318)]
319pub(crate) struct PatternTypo {
320    #[suggestion_part(code = "{code}")]
321    pub span: Span,
322    pub code: String,
323    pub item_name: Symbol,
324    pub kind: &'static str,
325}
326
327pub(crate) struct MustNotSupend<'a, 'tcx> {
328    pub tcx: TyCtxt<'tcx>,
329    pub yield_sp: Span,
330    pub reason: Option<MustNotSuspendReason>,
331    pub src_sp: Span,
332    pub pre: &'a str,
333    pub def_id: DefId,
334    pub post: &'a str,
335}
336
337// Needed for def_path_str
338impl<'a> LintDiagnostic<'a, ()> for MustNotSupend<'_, '_> {
339    fn decorate_lint<'b>(self, diag: &'b mut rustc_errors::Diag<'a, ()>) {
340        diag.primary_message(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("{$pre}`{$def_path}`{$post} held across a suspend point, but should not be"))inline_fluent!(
341            "{$pre}`{$def_path}`{$post} held across a suspend point, but should not be"
342        ));
343        diag.span_label(
344            self.yield_sp,
345            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("the value is held across this suspend point"))inline_fluent!("the value is held across this suspend point"),
346        );
347        if let Some(reason) = self.reason {
348            diag.subdiagnostic(reason);
349        }
350        diag.span_help(self.src_sp, rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using a block (`{\"{ ... }\"}`) to shrink the value's scope, ending before the suspend point"))inline_fluent!("consider using a block (`{\"{ ... }\"}`) to shrink the value's scope, ending before the suspend point"));
351        diag.arg("pre", self.pre);
352        diag.arg("def_path", self.tcx.def_path_str(self.def_id));
353        diag.arg("post", self.post);
354    }
355}
356
357#[derive(const _: () =
    {
        impl rustc_errors::Subdiagnostic for MustNotSuspendReason {
            fn add_to_diag<__G>(self, diag: &mut rustc_errors::Diag<'_, __G>)
                where __G: rustc_errors::EmissionGuarantee {
                match self {
                    MustNotSuspendReason {
                        span: __binding_0, reason: __binding_1 } => {
                        diag.store_args();
                        diag.arg("reason", __binding_1);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("{$reason}")));
                        diag.span_note(__binding_0, __message);
                        diag.restore_args();
                    }
                }
            }
        }
    };Subdiagnostic)]
358#[note("{$reason}")]
359pub(crate) struct MustNotSuspendReason {
360    #[primary_span]
361    pub span: Span,
362    pub reason: Symbol,
363}
364
365#[derive(const _: () =
    {
        impl<'_sess, G> rustc_errors::Diagnostic<'_sess, G> for
            ForceInlineFailure where G: rustc_errors::EmissionGuarantee {
            #[track_caller]
            fn into_diag(self, dcx: rustc_errors::DiagCtxtHandle<'_sess>,
                level: rustc_errors::Level) -> rustc_errors::Diag<'_sess, G> {
                match self {
                    ForceInlineFailure {
                        caller_span: __binding_0,
                        callee_span: __binding_1,
                        attr_span: __binding_2,
                        call_span: __binding_3,
                        callee: __binding_4,
                        caller: __binding_5,
                        reason: __binding_6,
                        justification: __binding_7 } => {
                        let mut diag =
                            rustc_errors::Diag::new(dcx, level,
                                rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$callee}` could not be inlined into `{$caller}` but is required to be inlined")));
                        diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("could not be inlined due to: {$reason}")));
                        ;
                        diag.arg("callee", __binding_4);
                        diag.arg("caller", __binding_5);
                        diag.arg("reason", __binding_6);
                        diag.span_label(__binding_0,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("within `{$caller}`...")));
                        diag.span_label(__binding_1,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$callee}` defined here")));
                        diag.span_label(__binding_2,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("annotation here")));
                        diag.span(__binding_3);
                        diag.span_label(__binding_3,
                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("...`{$callee}` called here")));
                        if let Some(__binding_7) = __binding_7 {
                            diag.subdiagnostic(__binding_7);
                        }
                        diag
                    }
                }
            }
        }
    };Diagnostic)]
366#[diag("`{$callee}` could not be inlined into `{$caller}` but is required to be inlined")]
367#[note("could not be inlined due to: {$reason}")]
368pub(crate) struct ForceInlineFailure {
369    #[label("within `{$caller}`...")]
370    pub caller_span: Span,
371    #[label("`{$callee}` defined here")]
372    pub callee_span: Span,
373    #[label("annotation here")]
374    pub attr_span: Span,
375    #[primary_span]
376    #[label("...`{$callee}` called here")]
377    pub call_span: Span,
378    pub callee: String,
379    pub caller: String,
380    pub reason: &'static str,
381    #[subdiagnostic]
382    pub justification: Option<ForceInlineJustification>,
383}
384
385#[derive(const _: () =
    {
        impl rustc_errors::Subdiagnostic for ForceInlineJustification {
            fn add_to_diag<__G>(self, diag: &mut rustc_errors::Diag<'_, __G>)
                where __G: rustc_errors::EmissionGuarantee {
                match self {
                    ForceInlineJustification { sym: __binding_0 } => {
                        diag.store_args();
                        diag.arg("sym", __binding_0);
                        let __message =
                            diag.eagerly_translate(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$callee}` is required to be inlined to: {$sym}")));
                        diag.note(__message);
                        diag.restore_args();
                    }
                }
            }
        }
    };Subdiagnostic)]
386#[note("`{$callee}` is required to be inlined to: {$sym}")]
387pub(crate) struct ForceInlineJustification {
388    pub sym: Symbol,
389}