Skip to main content

rustc_borrowck/diagnostics/
mod.rs

1//! Borrow checker diagnostics.
2
3use std::collections::BTreeMap;
4
5use rustc_abi::{FieldIdx, VariantIdx};
6use rustc_data_structures::fx::FxIndexMap;
7use rustc_errors::formatting::DiagMessageAddArg;
8use rustc_errors::{Applicability, Diag, DiagMessage, EmissionGuarantee, MultiSpan, listify, msg};
9use rustc_hir::def::{CtorKind, Namespace};
10use rustc_hir::{
11    self as hir, CoroutineKind, GenericBound, LangItem, WhereBoundPredicate, WherePredicateKind,
12};
13use rustc_index::{IndexSlice, IndexVec};
14use rustc_infer::infer::{BoundRegionConversionTime, NllRegionVariableOrigin};
15use rustc_infer::traits::SelectionError;
16use rustc_middle::mir::{
17    AggregateKind, CallSource, ConstOperand, ConstraintCategory, FakeReadCause, Local, LocalInfo,
18    LocalKind, Location, Operand, Place, PlaceRef, PlaceTy, ProjectionElem, Rvalue, Statement,
19    StatementKind, Terminator, TerminatorKind, VarDebugInfoContents, find_self_call,
20};
21use rustc_middle::ty::print::Print;
22use rustc_middle::ty::{self, Ty, TyCtxt};
23use rustc_middle::{bug, span_bug};
24use rustc_mir_dataflow::move_paths::{InitLocation, LookupResult, MoveOutIndex};
25use rustc_span::def_id::LocalDefId;
26use rustc_span::{DUMMY_SP, ErrorGuaranteed, Span, Spanned, Symbol, sym};
27use rustc_trait_selection::error_reporting::InferCtxtErrorExt;
28use rustc_trait_selection::error_reporting::traits::call_kind::{CallDesugaringKind, call_kind};
29use rustc_trait_selection::infer::InferCtxtExt;
30use rustc_trait_selection::traits::{
31    FulfillmentError, FulfillmentErrorCode, type_known_to_meet_bound_modulo_regions,
32};
33use tracing::debug;
34
35use super::MirBorrowckCtxt;
36use super::borrow_set::BorrowData;
37use crate::LocalMutationIsAllowed;
38use crate::constraints::OutlivesConstraint;
39use crate::nll::ConstraintDescription;
40use crate::session_diagnostics::{
41    CaptureArgLabel, CaptureReasonLabel, CaptureReasonNote, CaptureReasonSuggest, CaptureVarCause,
42    CaptureVarKind, CaptureVarPathUseCause, OnClosureNote,
43};
44
45mod find_all_local_uses;
46mod find_use;
47mod outlives_suggestion;
48mod region_name;
49mod var_name;
50
51mod bound_region_errors;
52mod conflict_errors;
53mod explain_borrow;
54mod move_errors;
55mod mutability_errors;
56mod opaque_types;
57mod region_errors;
58
59pub(crate) use bound_region_errors::{ToUniverseInfo, UniverseInfo};
60pub(crate) use move_errors::{IllegalMoveOriginKind, MoveError};
61pub(crate) use mutability_errors::AccessKind;
62pub(crate) use outlives_suggestion::OutlivesSuggestionBuilder;
63pub(crate) use region_errors::{ErrorConstraintInfo, RegionErrorKind, RegionErrors};
64pub(crate) use region_name::{RegionName, RegionNameSource};
65pub(crate) use rustc_trait_selection::error_reporting::traits::call_kind::CallKind;
66
67pub(super) struct DescribePlaceOpt {
68    including_downcast: bool,
69
70    /// Enable/Disable tuple fields.
71    /// For example `x` tuple. if it's `true` `x.0`. Otherwise `x`
72    including_tuple_field: bool,
73}
74
75pub(super) struct IncludingTupleField(pub(super) bool);
76
77enum BufferedDiag<'infcx> {
78    Error(Diag<'infcx>),
79    NonError(Diag<'infcx, ()>),
80}
81
82impl<'infcx> BufferedDiag<'infcx> {
83    fn sort_span(&self) -> Span {
84        match self {
85            BufferedDiag::Error(diag) => diag.sort_span,
86            BufferedDiag::NonError(diag) => diag.sort_span,
87        }
88    }
89}
90
91#[derive(#[automatically_derived]
impl<'infcx, 'tcx> ::core::default::Default for
    BorrowckDiagnosticsBuffer<'infcx, 'tcx> {
    #[inline]
    fn default() -> BorrowckDiagnosticsBuffer<'infcx, 'tcx> {
        BorrowckDiagnosticsBuffer {
            buffered_move_errors: ::core::default::Default::default(),
            buffered_mut_errors: ::core::default::Default::default(),
            buffered_diags: ::core::default::Default::default(),
        }
    }
}Default)]
92pub(crate) struct BorrowckDiagnosticsBuffer<'infcx, 'tcx> {
93    /// This field keeps track of move errors that are to be reported for given move indices.
94    ///
95    /// There are situations where many errors can be reported for a single move out (see
96    /// #53807) and we want only the best of those errors.
97    ///
98    /// The `report_use_of_moved_or_uninitialized` function checks this map and replaces the
99    /// diagnostic (if there is one) if the `Place` of the error being reported is a prefix of
100    /// the `Place` of the previous most diagnostic. This happens instead of buffering the
101    /// error. Once all move errors have been reported, any diagnostics in this map are added
102    /// to the buffer to be emitted.
103    ///
104    /// `BTreeMap` is used to preserve the order of insertions when iterating. This is necessary
105    /// when errors in the map are being re-added to the error buffer so that errors with the
106    /// same primary span come out in a consistent order.
107    buffered_move_errors: BTreeMap<Vec<MoveOutIndex>, (PlaceRef<'tcx>, Diag<'infcx>)>,
108
109    buffered_mut_errors: FxIndexMap<Span, (Diag<'infcx>, usize)>,
110
111    /// Buffer of diagnostics to be reported. A mixture of error and non-error diagnostics.
112    buffered_diags: Vec<BufferedDiag<'infcx>>,
113}
114
115impl<'infcx, 'tcx> BorrowckDiagnosticsBuffer<'infcx, 'tcx> {
116    pub(crate) fn buffer_non_error(&mut self, diag: Diag<'infcx, ()>) {
117        self.buffered_diags.push(BufferedDiag::NonError(diag));
118    }
119    pub(crate) fn buffer_error(&mut self, diag: Diag<'infcx>) {
120        self.buffered_diags.push(BufferedDiag::Error(diag));
121    }
122
123    pub(crate) fn emit_errors(&mut self) -> Option<ErrorGuaranteed> {
124        let mut res = None;
125
126        // Buffer any move errors that we collected and de-duplicated.
127        for (_, (_, diag)) in std::mem::take(&mut self.buffered_move_errors) {
128            // We have already set tainted for this error, so just buffer it.
129            self.buffer_error(diag);
130        }
131        for (_, (mut diag, count)) in std::mem::take(&mut self.buffered_mut_errors) {
132            if count > 10 {
133                diag.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("...and {0} other attempted mutable borrows",
                count - 10))
    })format!("...and {} other attempted mutable borrows", count - 10));
134            }
135            self.buffer_error(diag);
136        }
137
138        if !self.buffered_diags.is_empty() {
139            self.buffered_diags.sort_by_key(|buffered_diag| buffered_diag.sort_span());
140            for buffered_diag in self.buffered_diags.drain(..) {
141                match buffered_diag {
142                    BufferedDiag::Error(diag) => res = Some(diag.emit()),
143                    BufferedDiag::NonError(diag) => diag.emit(),
144                }
145            }
146        }
147
148        res
149    }
150}
151
152impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
153    pub(crate) fn buffer_error(&mut self, diag: Diag<'infcx>) {
154        self.diags_buffer.buffer_error(diag);
155    }
156
157    pub(crate) fn buffer_non_error(&mut self, diag: Diag<'infcx, ()>) {
158        self.diags_buffer.buffer_non_error(diag);
159    }
160
161    pub(crate) fn buffer_move_error(
162        &mut self,
163        move_out_indices: Vec<MoveOutIndex>,
164        place_and_err: (PlaceRef<'tcx>, Diag<'infcx>),
165    ) -> bool {
166        if let Some((_, diag)) =
167            self.diags_buffer.buffered_move_errors.insert(move_out_indices, place_and_err)
168        {
169            // Cancel the old diagnostic so we don't ICE
170            diag.cancel();
171            false
172        } else {
173            true
174        }
175    }
176
177    pub(crate) fn get_buffered_mut_error(&mut self, span: Span) -> Option<(Diag<'infcx>, usize)> {
178        // FIXME(#120456) - is `swap_remove` correct?
179        self.diags_buffer.buffered_mut_errors.swap_remove(&span)
180    }
181
182    pub(crate) fn buffer_mut_error(&mut self, span: Span, diag: Diag<'infcx>, count: usize) {
183        self.diags_buffer.buffered_mut_errors.insert(span, (diag, count));
184    }
185
186    pub(crate) fn has_buffered_diags(&self) -> bool {
187        self.diags_buffer.buffered_diags.is_empty()
188    }
189
190    pub(crate) fn has_move_error(
191        &self,
192        move_out_indices: &[MoveOutIndex],
193    ) -> Option<&(PlaceRef<'tcx>, Diag<'infcx>)> {
194        self.diags_buffer.buffered_move_errors.get(move_out_indices)
195    }
196
197    /// Uses `body.var_debug_info` to find the symbol
198    fn local_name(&self, index: Local) -> Option<Symbol> {
199        *self.local_names().get(index)?
200    }
201
202    fn local_names(&self) -> &IndexSlice<Local, Option<Symbol>> {
203        self.local_names.get_or_init(|| {
204            let mut local_names = IndexVec::from_elem(None, &self.body.local_decls);
205            for var_debug_info in &self.body.var_debug_info {
206                if let VarDebugInfoContents::Place(place) = var_debug_info.value {
207                    if let Some(local) = place.as_local() {
208                        if let Some(prev_name) = local_names[local]
209                            && var_debug_info.name != prev_name
210                        {
211                            ::rustc_middle::util::bug::span_bug_fmt(var_debug_info.source_info.span,
    format_args!("local {0:?} has many names (`{1}` vs `{2}`)", local,
        prev_name, var_debug_info.name));span_bug!(
212                                var_debug_info.source_info.span,
213                                "local {:?} has many names (`{}` vs `{}`)",
214                                local,
215                                prev_name,
216                                var_debug_info.name
217                            );
218                        }
219                        local_names[local] = Some(var_debug_info.name);
220                    }
221                }
222            }
223            local_names
224        })
225    }
226}
227
228impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
229    /// Adds a suggestion when a closure is invoked twice with a moved variable or when a closure
230    /// is moved after being invoked.
231    ///
232    /// ```text
233    /// note: closure cannot be invoked more than once because it moves the variable `dict` out of
234    ///       its environment
235    ///   --> $DIR/issue-42065.rs:16:29
236    ///    |
237    /// LL |         for (key, value) in dict {
238    ///    |                             ^^^^
239    /// ```
240    pub(super) fn add_moved_or_invoked_closure_note(
241        &self,
242        location: Location,
243        place: PlaceRef<'tcx>,
244        diag: &mut Diag<'infcx>,
245    ) -> bool {
246        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:246",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(246u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_moved_or_invoked_closure_note: location={0:?} place={1:?}",
                                                    location, place) as &dyn Value))])
            });
    } else { ; }
};debug!("add_moved_or_invoked_closure_note: location={:?} place={:?}", location, place);
247        let mut target = place.local_or_deref_local();
248        for stmt in &self.body[location.block].statements[location.statement_index..] {
249            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:249",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(249u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_moved_or_invoked_closure_note: stmt={0:?} target={1:?}",
                                                    stmt, target) as &dyn Value))])
            });
    } else { ; }
};debug!("add_moved_or_invoked_closure_note: stmt={:?} target={:?}", stmt, target);
250            if let StatementKind::Assign((into, Rvalue::Use(from, _))) = &stmt.kind {
251                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:251",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(251u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_fnonce_closure_note: into={0:?} from={1:?}",
                                                    into, from) as &dyn Value))])
            });
    } else { ; }
};debug!("add_fnonce_closure_note: into={:?} from={:?}", into, from);
252                match from {
253                    Operand::Copy(place) | Operand::Move(place)
254                        if target == place.local_or_deref_local() =>
255                    {
256                        target = into.local_or_deref_local()
257                    }
258                    _ => {}
259                }
260            }
261        }
262
263        // Check if we are attempting to call a closure after it has been invoked.
264        let terminator = self.body[location.block].terminator();
265        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:265",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(265u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_moved_or_invoked_closure_note: terminator={0:?}",
                                                    terminator) as &dyn Value))])
            });
    } else { ; }
};debug!("add_moved_or_invoked_closure_note: terminator={:?}", terminator);
266        if let TerminatorKind::Call {
267            func: Operand::Constant(ConstOperand { const_, .. }),
268            args,
269            ..
270        } = &terminator.kind
271            && let ty::FnDef(id, _) = *const_.ty().kind()
272        {
273            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:273",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(273u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_moved_or_invoked_closure_note: id={0:?}",
                                                    id) as &dyn Value))])
            });
    } else { ; }
};debug!("add_moved_or_invoked_closure_note: id={:?}", id);
274            if self.infcx.tcx.is_lang_item(self.infcx.tcx.parent(id), LangItem::FnOnce) {
275                let closure = match args.first() {
276                    Some(Spanned { node: Operand::Copy(place) | Operand::Move(place), .. })
277                        if target == place.local_or_deref_local() =>
278                    {
279                        place.local_or_deref_local().unwrap()
280                    }
281                    _ => return false,
282                };
283
284                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:284",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(284u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("add_moved_or_invoked_closure_note: closure={0:?}",
                                                    closure) as &dyn Value))])
            });
    } else { ; }
};debug!("add_moved_or_invoked_closure_note: closure={:?}", closure);
285                if let ty::Closure(did, _) = self.body.local_decls[closure].ty.kind() {
286                    let did = did.expect_local();
287                    if let Some((span, hir_place)) = self.infcx.tcx.closure_kind_origin(did) {
288                        diag.subdiagnostic(OnClosureNote::InvokedTwice {
289                            place_name: &ty::place_to_string_for_capture(self.infcx.tcx, hir_place),
290                            span: *span,
291                        });
292                        return true;
293                    }
294                }
295            }
296        }
297
298        // Check if we are just moving a closure after it has been invoked.
299        if let Some(target) = target
300            && let ty::Closure(did, _) = self.body.local_decls[target].ty.kind()
301        {
302            let did = did.expect_local();
303            if let Some((span, hir_place)) = self.infcx.tcx.closure_kind_origin(did) {
304                diag.subdiagnostic(OnClosureNote::MovedTwice {
305                    place_name: &ty::place_to_string_for_capture(self.infcx.tcx, hir_place),
306                    span: *span,
307                });
308                return true;
309            }
310        }
311        false
312    }
313
314    /// End-user visible description of `place` if one can be found.
315    /// If the place is a temporary for instance, `"value"` will be returned.
316    pub(super) fn describe_any_place(&self, place_ref: PlaceRef<'tcx>) -> String {
317        match self.describe_place(place_ref) {
318            Some(mut descr) => {
319                // Surround descr with `backticks`.
320                descr.reserve(2);
321                descr.insert(0, '`');
322                descr.push('`');
323                descr
324            }
325            None => "value".to_string(),
326        }
327    }
328
329    /// End-user visible description of `place` if one can be found.
330    /// If the place is a temporary for instance, `None` will be returned.
331    pub(super) fn describe_place(&self, place_ref: PlaceRef<'tcx>) -> Option<String> {
332        self.describe_place_with_options(
333            place_ref,
334            DescribePlaceOpt { including_downcast: false, including_tuple_field: true },
335        )
336    }
337
338    /// End-user visible description of `place` if one can be found. If the place is a temporary
339    /// for instance, `None` will be returned.
340    /// `IncludingDowncast` parameter makes the function return `None` if `ProjectionElem` is
341    /// `Downcast` and `IncludingDowncast` is true
342    pub(super) fn describe_place_with_options(
343        &self,
344        place: PlaceRef<'tcx>,
345        opt: DescribePlaceOpt,
346    ) -> Option<String> {
347        let local = place.local;
348        if self.body.local_decls[local]
349            .source_info
350            .span
351            .in_external_macro(self.infcx.tcx.sess.source_map())
352        {
353            return None;
354        }
355
356        let mut autoderef_index = None;
357        let mut buf = String::new();
358        let mut ok = self.append_local_to_string(local, &mut buf);
359
360        for (index, elem) in place.projection.into_iter().enumerate() {
361            match elem {
362                ProjectionElem::Deref => {
363                    if index == 0 {
364                        if self.body.local_decls[local].is_ref_for_guard() {
365                            continue;
366                        }
367                        if let LocalInfo::StaticRef { def_id, .. } =
368                            *self.body.local_decls[local].local_info()
369                        {
370                            buf.push_str(self.infcx.tcx.item_name(def_id).as_str());
371                            ok = Ok(());
372                            continue;
373                        }
374                    }
375                    if let Some(field) = self.is_upvar_field_projection(PlaceRef {
376                        local,
377                        projection: place.projection.split_at(index + 1).0,
378                    }) {
379                        let var_index = field.index();
380                        buf = self.upvars[var_index].to_string(self.infcx.tcx);
381                        ok = Ok(());
382                        if !self.upvars[var_index].is_by_ref() {
383                            buf.insert(0, '*');
384                        }
385                    } else {
386                        if autoderef_index.is_none() {
387                            autoderef_index = match place.projection.iter().rposition(|elem| {
388                                !#[allow(non_exhaustive_omitted_patterns)] match elem {
    ProjectionElem::Deref | ProjectionElem::Downcast(..) => true,
    _ => false,
}matches!(
389                                    elem,
390                                    ProjectionElem::Deref | ProjectionElem::Downcast(..)
391                                )
392                            }) {
393                                Some(index) => Some(index + 1),
394                                None => Some(0),
395                            };
396                        }
397                        if index >= autoderef_index.unwrap() {
398                            buf.insert(0, '*');
399                        }
400                    }
401                }
402                ProjectionElem::Downcast(..) if opt.including_downcast => return None,
403                ProjectionElem::Downcast(..) => (),
404                ProjectionElem::OpaqueCast(..) => (),
405                ProjectionElem::UnwrapUnsafeBinder(_) => (),
406                ProjectionElem::Field(field, _ty) => {
407                    // FIXME(project-rfc_2229#36): print capture precisely here.
408                    if let Some(field) = self.is_upvar_field_projection(PlaceRef {
409                        local,
410                        projection: place.projection.split_at(index + 1).0,
411                    }) {
412                        buf = self.upvars[field.index()].to_string(self.infcx.tcx);
413                        ok = Ok(());
414                    } else {
415                        let field_name = self.describe_field(
416                            PlaceRef { local, projection: place.projection.split_at(index).0 },
417                            *field,
418                            IncludingTupleField(opt.including_tuple_field),
419                        );
420                        if let Some(field_name_str) = field_name {
421                            buf.push('.');
422                            buf.push_str(&field_name_str);
423                        }
424                    }
425                }
426                ProjectionElem::Index(index) => {
427                    buf.push('[');
428                    if self.append_local_to_string(*index, &mut buf).is_err() {
429                        buf.push('_');
430                    }
431                    buf.push(']');
432                }
433                ProjectionElem::ConstantIndex { .. } | ProjectionElem::Subslice { .. } => {
434                    // Since it isn't possible to borrow an element on a particular index and
435                    // then use another while the borrow is held, don't output indices details
436                    // to avoid confusing the end-user
437                    buf.push_str("[..]");
438                }
439            }
440        }
441        ok.ok().map(|_| buf)
442    }
443
444    fn describe_name(&self, place: PlaceRef<'tcx>) -> Option<Symbol> {
445        for elem in place.projection.into_iter() {
446            match elem {
447                ProjectionElem::Downcast(Some(name), _) => {
448                    return Some(*name);
449                }
450                _ => {}
451            }
452        }
453        None
454    }
455
456    /// Appends end-user visible description of the `local` place to `buf`. If `local` doesn't have
457    /// a name, or its name was generated by the compiler, then `Err` is returned
458    fn append_local_to_string(&self, local: Local, buf: &mut String) -> Result<(), ()> {
459        let decl = &self.body.local_decls[local];
460        match self.local_name(local) {
461            Some(name) if !decl.from_compiler_desugaring() => {
462                buf.push_str(name.as_str());
463                Ok(())
464            }
465            _ => Err(()),
466        }
467    }
468
469    /// End-user visible description of the `field`nth field of `base`
470    fn describe_field(
471        &self,
472        place: PlaceRef<'tcx>,
473        field: FieldIdx,
474        including_tuple_field: IncludingTupleField,
475    ) -> Option<String> {
476        let place_ty = match place {
477            PlaceRef { local, projection: [] } => PlaceTy::from_ty(self.body.local_decls[local].ty),
478            PlaceRef { local, projection: [proj_base @ .., elem] } => match elem {
479                ProjectionElem::Deref
480                | ProjectionElem::Index(..)
481                | ProjectionElem::ConstantIndex { .. }
482                | ProjectionElem::Subslice { .. } => {
483                    PlaceRef { local, projection: proj_base }.ty(self.body, self.infcx.tcx)
484                }
485                ProjectionElem::Downcast(..) => place.ty(self.body, self.infcx.tcx),
486                ProjectionElem::OpaqueCast(ty) | ProjectionElem::UnwrapUnsafeBinder(ty) => {
487                    PlaceTy::from_ty(*ty)
488                }
489                ProjectionElem::Field(_, field_type) => PlaceTy::from_ty(*field_type),
490            },
491        };
492        self.describe_field_from_ty(
493            place_ty.ty,
494            field,
495            place_ty.variant_index,
496            including_tuple_field,
497        )
498    }
499
500    /// End-user visible description of the `field_index`nth field of `ty`
501    fn describe_field_from_ty(
502        &self,
503        ty: Ty<'_>,
504        field: FieldIdx,
505        variant_index: Option<VariantIdx>,
506        including_tuple_field: IncludingTupleField,
507    ) -> Option<String> {
508        if let Some(boxed_ty) = ty.boxed_ty() {
509            // If the type is a box, the field is described from the boxed type
510            self.describe_field_from_ty(boxed_ty, field, variant_index, including_tuple_field)
511        } else {
512            match *ty.kind() {
513                ty::Adt(def, _) => {
514                    let variant = if let Some(idx) = variant_index {
515                        if !def.is_enum() {
    ::core::panicking::panic("assertion failed: def.is_enum()")
};assert!(def.is_enum());
516                        def.variant(idx)
517                    } else {
518                        def.non_enum_variant()
519                    };
520                    if !including_tuple_field.0 && variant.ctor_kind() == Some(CtorKind::Fn) {
521                        return None;
522                    }
523                    Some(variant.fields[field].name.to_string())
524                }
525                ty::Tuple(_) => Some(field.index().to_string()),
526                ty::Ref(_, ty, _) | ty::RawPtr(ty, _) => {
527                    self.describe_field_from_ty(ty, field, variant_index, including_tuple_field)
528                }
529                ty::Array(ty, _) | ty::Slice(ty) => {
530                    self.describe_field_from_ty(ty, field, variant_index, including_tuple_field)
531                }
532                ty::Closure(def_id, _) | ty::Coroutine(def_id, _) => {
533                    // We won't be borrowck'ing here if the closure came from another crate,
534                    // so it's safe to call `expect_local`.
535                    //
536                    // We know the field exists so it's safe to call operator[] and `unwrap` here.
537                    let def_id = def_id.expect_local();
538                    let var_id =
539                        self.infcx.tcx.closure_captures(def_id)[field.index()].get_root_variable();
540
541                    Some(self.infcx.tcx.hir_name(var_id).to_string())
542                }
543                _ => {
544                    // This can happen for field accesses on `Box<T>`: the field is
545                    // described from the boxed type, which may have no named fields
546                    Some(field.index().to_string())
547                }
548            }
549        }
550    }
551
552    pub(super) fn borrowed_content_source(
553        &self,
554        deref_base: PlaceRef<'tcx>,
555    ) -> BorrowedContentSource<'tcx> {
556        let tcx = self.infcx.tcx;
557
558        // Look up the provided place and work out the move path index for it,
559        // we'll use this to check whether it was originally from an overloaded
560        // operator.
561        match self.move_data.rev_lookup.find(deref_base) {
562            LookupResult::Exact(mpi) | LookupResult::Parent(Some(mpi)) => {
563                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:563",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(563u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("borrowed_content_source: mpi={0:?}",
                                                    mpi) as &dyn Value))])
            });
    } else { ; }
};debug!("borrowed_content_source: mpi={:?}", mpi);
564
565                for i in &self.move_data.init_path_map[mpi] {
566                    let init = &self.move_data.inits[*i];
567                    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:567",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(567u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("borrowed_content_source: init={0:?}",
                                                    init) as &dyn Value))])
            });
    } else { ; }
};debug!("borrowed_content_source: init={:?}", init);
568                    // We're only interested in statements that initialized a value, not the
569                    // initializations from arguments.
570                    let InitLocation::Statement(loc) = init.location else { continue };
571
572                    let bbd = &self.body[loc.block];
573                    let is_terminator = bbd.statements.len() == loc.statement_index;
574                    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:574",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(574u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("borrowed_content_source: loc={0:?} is_terminator={1:?}",
                                                    loc, is_terminator) as &dyn Value))])
            });
    } else { ; }
};debug!(
575                        "borrowed_content_source: loc={:?} is_terminator={:?}",
576                        loc, is_terminator,
577                    );
578                    if !is_terminator {
579                        continue;
580                    } else if let Some(Terminator {
581                        kind:
582                            TerminatorKind::Call {
583                                func,
584                                call_source: CallSource::OverloadedOperator,
585                                ..
586                            },
587                        ..
588                    }) = &bbd.terminator
589                    {
590                        if let Some(source) =
591                            BorrowedContentSource::from_call(func.ty(self.body, tcx), tcx)
592                        {
593                            return source;
594                        }
595                    }
596                }
597            }
598            // Base is a `static` so won't be from an overloaded operator
599            _ => (),
600        };
601
602        // If we didn't find an overloaded deref or index, then assume it's a
603        // built in deref and check the type of the base.
604        let base_ty = deref_base.ty(self.body, tcx).ty;
605        if base_ty.is_raw_ptr() {
606            BorrowedContentSource::DerefRawPointer
607        } else if base_ty.is_mutable_ptr() {
608            BorrowedContentSource::DerefMutableRef
609        } else if base_ty.is_ref() {
610            BorrowedContentSource::DerefSharedRef
611        } else {
612            // Custom type implementing `Deref` (e.g. `MyBox<T>`, `Rc<T>`, `Arc<T>`)
613            // that wasn't detected via the MIR init trace above. This can happen
614            // when the deref base is initialized by a regular statement rather than
615            // a `TerminatorKind::Call` with `CallSource::OverloadedOperator`.
616            BorrowedContentSource::OverloadedDeref(base_ty)
617        }
618    }
619
620    /// Return the name of the provided `Ty` (that must be a reference) with a synthesized lifetime
621    /// name where required.
622    pub(super) fn get_name_for_ty(&self, ty: Ty<'tcx>, counter: usize) -> String {
623        let mut p = ty::print::FmtPrinter::new(self.infcx.tcx, Namespace::TypeNS);
624
625        // We need to add synthesized lifetimes where appropriate. We do
626        // this by hooking into the pretty printer and telling it to label the
627        // lifetimes without names with the value `'0`.
628        if let ty::Ref(region, ..) = ty.kind() {
629            match region.kind() {
630                ty::ReBound(_, ty::BoundRegion { kind: br, .. })
631                | ty::RePlaceholder(ty::PlaceholderRegion {
632                    bound: ty::BoundRegion { kind: br, .. },
633                    ..
634                }) => p.region_highlight_mode.highlighting_bound_region(br, counter),
635                _ => {}
636            }
637        }
638
639        ty.print(&mut p).unwrap();
640        p.into_buffer()
641    }
642
643    /// Returns the name of the provided `Ty` (that must be a reference)'s region with a
644    /// synthesized lifetime name where required.
645    pub(super) fn get_region_name_for_ty(&self, ty: Ty<'tcx>, counter: usize) -> String {
646        let mut p = ty::print::FmtPrinter::new(self.infcx.tcx, Namespace::TypeNS);
647
648        let region = if let ty::Ref(region, ..) = ty.kind() {
649            match region.kind() {
650                ty::ReBound(_, ty::BoundRegion { kind: br, .. })
651                | ty::RePlaceholder(ty::PlaceholderRegion {
652                    bound: ty::BoundRegion { kind: br, .. },
653                    ..
654                }) => p.region_highlight_mode.highlighting_bound_region(br, counter),
655                _ => {}
656            }
657            region
658        } else {
659            ::rustc_middle::util::bug::bug_fmt(format_args!("ty for annotation of borrow region is not a reference"));bug!("ty for annotation of borrow region is not a reference");
660        };
661
662        region.print(&mut p).unwrap();
663        p.into_buffer()
664    }
665
666    /// Add a note to region errors and borrow explanations when higher-ranked regions in predicates
667    /// implicitly introduce an "outlives `'static`" constraint.
668    ///
669    /// This is very similar to `fn suggest_static_lifetime_for_gat_from_hrtb` which handles this
670    /// note for failed type tests instead of outlives errors.
671    fn add_placeholder_from_predicate_note<G: EmissionGuarantee>(
672        &self,
673        diag: &mut Diag<'_, G>,
674        path: &[OutlivesConstraint<'tcx>],
675    ) {
676        let tcx = self.infcx.tcx;
677        let Some((gat_hir_id, generics)) = path.iter().find_map(|constraint| {
678            let outlived = constraint.sub;
679            if let Some(origin) = self.regioncx.definitions.get(outlived)
680                && let NllRegionVariableOrigin::Placeholder(placeholder) = origin.origin
681                && let Some(id) = placeholder.bound.kind.get_id()
682                && let Some(placeholder_id) = id.as_local()
683                && let gat_hir_id = tcx.local_def_id_to_hir_id(placeholder_id)
684                && let Some(generics_impl) =
685                    tcx.parent_hir_node(tcx.parent_hir_id(gat_hir_id)).generics()
686            {
687                Some((gat_hir_id, generics_impl))
688            } else {
689                None
690            }
691        }) else {
692            return;
693        };
694
695        // Look for the where-bound which introduces the placeholder.
696        // As we're using the HIR, we need to handle both `for<'a> T: Trait<'a>`
697        // and `T: for<'a> Trait`<'a>.
698        for pred in generics.predicates {
699            let WherePredicateKind::BoundPredicate(WhereBoundPredicate {
700                bound_generic_params,
701                bounds,
702                ..
703            }) = pred.kind
704            else {
705                continue;
706            };
707            if bound_generic_params
708                .iter()
709                .rfind(|bgp| tcx.local_def_id_to_hir_id(bgp.def_id) == gat_hir_id)
710                .is_some()
711            {
712                diag.span_note(pred.span, LIMITATION_NOTE);
713                return;
714            }
715            for bound in bounds.iter() {
716                if let GenericBound::Trait(bound) = bound {
717                    if bound
718                        .bound_generic_params
719                        .iter()
720                        .rfind(|bgp| tcx.local_def_id_to_hir_id(bgp.def_id) == gat_hir_id)
721                        .is_some()
722                    {
723                        diag.span_note(bound.span, LIMITATION_NOTE);
724                        return;
725                    }
726                }
727            }
728        }
729    }
730
731    /// Add a label to region errors and borrow explanations when outlives constraints arise from
732    /// proving a type implements `Sized` or `Copy`.
733    fn add_sized_or_copy_bound_info<G: EmissionGuarantee>(
734        &self,
735        err: &mut Diag<'_, G>,
736        blamed_category: ConstraintCategory<'tcx>,
737        path: &[OutlivesConstraint<'tcx>],
738    ) {
739        for sought_category in [ConstraintCategory::SizedBound, ConstraintCategory::CopyBound] {
740            if sought_category != blamed_category
741                && let Some(sought_constraint) = path.iter().find(|c| c.category == sought_category)
742            {
743                let label = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("requirement occurs due to {0}",
                sought_category.description().trim_end()))
    })format!(
744                    "requirement occurs due to {}",
745                    sought_category.description().trim_end()
746                );
747                err.span_label(sought_constraint.span, label);
748            }
749        }
750    }
751}
752
753/// The span(s) associated to a use of a place.
754#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for UseSpans<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for UseSpans<'tcx> {
    #[inline]
    fn clone(&self) -> UseSpans<'tcx> {
        let _: ::core::clone::AssertParamIsClone<hir::ClosureKind>;
        let _: ::core::clone::AssertParamIsClone<Span>;
        let _: ::core::clone::AssertParamIsClone<CallKind<'tcx>>;
        *self
    }
}Clone, #[automatically_derived]
impl<'tcx> ::core::cmp::PartialEq for UseSpans<'tcx> {
    #[inline]
    fn eq(&self, other: &UseSpans<'tcx>) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr &&
            match (self, other) {
                (UseSpans::ClosureUse {
                    closure_kind: __self_0,
                    args_span: __self_1,
                    capture_kind_span: __self_2,
                    path_span: __self_3 }, UseSpans::ClosureUse {
                    closure_kind: __arg1_0,
                    args_span: __arg1_1,
                    capture_kind_span: __arg1_2,
                    path_span: __arg1_3 }) =>
                    __self_0 == __arg1_0 && __self_1 == __arg1_1 &&
                            __self_2 == __arg1_2 && __self_3 == __arg1_3,
                (UseSpans::FnSelfUse {
                    var_span: __self_0,
                    fn_call_span: __self_1,
                    fn_span: __self_2,
                    kind: __self_3 }, UseSpans::FnSelfUse {
                    var_span: __arg1_0,
                    fn_call_span: __arg1_1,
                    fn_span: __arg1_2,
                    kind: __arg1_3 }) =>
                    __self_0 == __arg1_0 && __self_1 == __arg1_1 &&
                            __self_2 == __arg1_2 && __self_3 == __arg1_3,
                (UseSpans::PatUse(__self_0), UseSpans::PatUse(__arg1_0)) =>
                    __self_0 == __arg1_0,
                (UseSpans::OtherUse(__self_0), UseSpans::OtherUse(__arg1_0))
                    => __self_0 == __arg1_0,
                _ => unsafe { ::core::intrinsics::unreachable() }
            }
    }
}PartialEq, #[automatically_derived]
impl<'tcx> ::core::cmp::Eq for UseSpans<'tcx> {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<hir::ClosureKind>;
        let _: ::core::cmp::AssertParamIsEq<Span>;
        let _: ::core::cmp::AssertParamIsEq<CallKind<'tcx>>;
    }
}Eq, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for UseSpans<'tcx> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            UseSpans::ClosureUse {
                closure_kind: __self_0,
                args_span: __self_1,
                capture_kind_span: __self_2,
                path_span: __self_3 } =>
                ::core::fmt::Formatter::debug_struct_field4_finish(f,
                    "ClosureUse", "closure_kind", __self_0, "args_span",
                    __self_1, "capture_kind_span", __self_2, "path_span",
                    &__self_3),
            UseSpans::FnSelfUse {
                var_span: __self_0,
                fn_call_span: __self_1,
                fn_span: __self_2,
                kind: __self_3 } =>
                ::core::fmt::Formatter::debug_struct_field4_finish(f,
                    "FnSelfUse", "var_span", __self_0, "fn_call_span", __self_1,
                    "fn_span", __self_2, "kind", &__self_3),
            UseSpans::PatUse(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "PatUse",
                    &__self_0),
            UseSpans::OtherUse(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "OtherUse", &__self_0),
        }
    }
}Debug)]
755pub(super) enum UseSpans<'tcx> {
756    /// The access is caused by capturing a variable for a closure.
757    ClosureUse {
758        /// This is true if the captured variable was from a coroutine.
759        closure_kind: hir::ClosureKind,
760        /// The span of the args of the closure, including the `move` keyword if
761        /// it's present.
762        args_span: Span,
763        /// The span of the use resulting in capture kind
764        /// Check `ty::CaptureInfo` for more details
765        capture_kind_span: Span,
766        /// The span of the use resulting in the captured path
767        /// Check `ty::CaptureInfo` for more details
768        path_span: Span,
769    },
770    /// The access is caused by using a variable as the receiver of a method
771    /// that takes 'self'
772    FnSelfUse {
773        /// The span of the variable being moved
774        var_span: Span,
775        /// The span of the method call on the variable
776        fn_call_span: Span,
777        /// The definition span of the method being called
778        fn_span: Span,
779        kind: CallKind<'tcx>,
780    },
781    /// This access is caused by a `match` or `if let` pattern.
782    PatUse(Span),
783    /// This access has a single span associated to it: common case.
784    OtherUse(Span),
785}
786
787impl UseSpans<'_> {
788    pub(super) fn args_or_use(self) -> Span {
789        match self {
790            UseSpans::ClosureUse { args_span: span, .. }
791            | UseSpans::PatUse(span)
792            | UseSpans::OtherUse(span) => span,
793            UseSpans::FnSelfUse { var_span, .. } => var_span,
794        }
795    }
796
797    /// Returns the span of `self`, in the case of a `ClosureUse` returns the `path_span`
798    pub(super) fn var_or_use_path_span(self) -> Span {
799        match self {
800            UseSpans::ClosureUse { path_span: span, .. }
801            | UseSpans::PatUse(span)
802            | UseSpans::OtherUse(span) => span,
803            UseSpans::FnSelfUse { var_span, .. } => var_span,
804        }
805    }
806
807    /// Returns the span of `self`, in the case of a `ClosureUse` returns the `capture_kind_span`
808    pub(super) fn var_or_use(self) -> Span {
809        match self {
810            UseSpans::ClosureUse { capture_kind_span: span, .. }
811            | UseSpans::PatUse(span)
812            | UseSpans::OtherUse(span) => span,
813            UseSpans::FnSelfUse { var_span, .. } => var_span,
814        }
815    }
816
817    // FIXME(coroutines): Make this just return the `ClosureKind` directly?
818    pub(super) fn coroutine_kind(self) -> Option<CoroutineKind> {
819        match self {
820            UseSpans::ClosureUse {
821                closure_kind: hir::ClosureKind::Coroutine(coroutine_kind),
822                ..
823            } => Some(coroutine_kind),
824            _ => None,
825        }
826    }
827
828    /// Add a span label to the arguments of the closure, if it exists.
829    pub(super) fn args_subdiag(self, err: &mut Diag<'_>, f: impl FnOnce(Span) -> CaptureArgLabel) {
830        if let UseSpans::ClosureUse { args_span, .. } = self {
831            err.subdiagnostic(f(args_span));
832        }
833    }
834
835    /// Add a span label to the use of the captured variable, if it exists.
836    /// only adds label to the `path_span`
837    pub(super) fn var_path_only_subdiag(
838        self,
839        err: &mut Diag<'_>,
840        action: crate::InitializationRequiringAction,
841    ) {
842        use CaptureVarPathUseCause::*;
843
844        use crate::InitializationRequiringAction::*;
845        if let UseSpans::ClosureUse { closure_kind, path_span, .. } = self {
846            match closure_kind {
847                hir::ClosureKind::Coroutine(_) => {
848                    err.subdiagnostic(match action {
849                        Borrow => BorrowInCoroutine { path_span },
850                        MatchOn | Use => UseInCoroutine { path_span },
851                        Assignment => AssignInCoroutine { path_span },
852                        PartialAssignment => AssignPartInCoroutine { path_span },
853                    });
854                }
855                hir::ClosureKind::Closure | hir::ClosureKind::CoroutineClosure(_) => {
856                    err.subdiagnostic(match action {
857                        Borrow => BorrowInClosure { path_span },
858                        MatchOn | Use => UseInClosure { path_span },
859                        Assignment => AssignInClosure { path_span },
860                        PartialAssignment => AssignPartInClosure { path_span },
861                    });
862                }
863            }
864        }
865    }
866
867    /// Add a subdiagnostic to the use of the captured variable, if it exists.
868    pub(super) fn var_subdiag(
869        self,
870        err: &mut Diag<'_>,
871        kind: Option<rustc_middle::mir::BorrowKind>,
872        f: impl FnOnce(hir::ClosureKind, Span) -> CaptureVarCause,
873    ) {
874        if let UseSpans::ClosureUse { closure_kind, capture_kind_span, path_span, .. } = self {
875            if capture_kind_span != path_span {
876                err.subdiagnostic(match kind {
877                    Some(kd) => match kd {
878                        rustc_middle::mir::BorrowKind::Shared
879                        | rustc_middle::mir::BorrowKind::Fake(_) => {
880                            CaptureVarKind::Immut { kind_span: capture_kind_span }
881                        }
882
883                        rustc_middle::mir::BorrowKind::Mut { .. } => {
884                            CaptureVarKind::Mut { kind_span: capture_kind_span }
885                        }
886                    },
887                    None => CaptureVarKind::Move { kind_span: capture_kind_span },
888                });
889            };
890            let diag = f(closure_kind, path_span);
891            err.subdiagnostic(diag);
892        }
893    }
894
895    /// Returns `false` if this place is not used in a closure.
896    pub(super) fn for_closure(&self) -> bool {
897        match *self {
898            UseSpans::ClosureUse { closure_kind, .. } => {
899                #[allow(non_exhaustive_omitted_patterns)] match closure_kind {
    hir::ClosureKind::Closure => true,
    _ => false,
}matches!(closure_kind, hir::ClosureKind::Closure)
900            }
901            _ => false,
902        }
903    }
904
905    /// Returns `false` if this place is not used in a coroutine.
906    pub(super) fn for_coroutine(&self) -> bool {
907        match *self {
908            // FIXME(coroutines): Do we want this to apply to synthetic coroutines?
909            UseSpans::ClosureUse { closure_kind, .. } => {
910                #[allow(non_exhaustive_omitted_patterns)] match closure_kind {
    hir::ClosureKind::Coroutine(..) => true,
    _ => false,
}matches!(closure_kind, hir::ClosureKind::Coroutine(..))
911            }
912            _ => false,
913        }
914    }
915
916    pub(super) fn or_else<F>(self, if_other: F) -> Self
917    where
918        F: FnOnce() -> Self,
919    {
920        match self {
921            closure @ UseSpans::ClosureUse { .. } => closure,
922            UseSpans::PatUse(_) | UseSpans::OtherUse(_) => if_other(),
923            fn_self @ UseSpans::FnSelfUse { .. } => fn_self,
924        }
925    }
926}
927
928pub(super) enum BorrowedContentSource<'tcx> {
929    DerefRawPointer,
930    DerefMutableRef,
931    DerefSharedRef,
932    OverloadedDeref(Ty<'tcx>),
933    OverloadedIndex(Ty<'tcx>),
934}
935
936impl<'tcx> BorrowedContentSource<'tcx> {
937    pub(super) fn describe_for_unnamed_place(&self, tcx: TyCtxt<'_>) -> String {
938        match *self {
939            BorrowedContentSource::DerefRawPointer => "a raw pointer".to_string(),
940            BorrowedContentSource::DerefSharedRef => "a shared reference".to_string(),
941            BorrowedContentSource::DerefMutableRef => "a mutable reference".to_string(),
942            BorrowedContentSource::OverloadedDeref(ty) => ty
943                .ty_adt_def()
944                .and_then(|adt| match tcx.get_diagnostic_name(adt.did())? {
945                    name @ (sym::Rc | sym::Arc) => Some(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("an `{0}`", name))
    })format!("an `{name}`")),
946                    _ => None,
947                })
948                .unwrap_or_else(|| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("dereference of `{0}`", ty))
    })format!("dereference of `{ty}`")),
949            BorrowedContentSource::OverloadedIndex(ty) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("index of `{0}`", ty))
    })format!("index of `{ty}`"),
950        }
951    }
952
953    pub(super) fn describe_for_named_place(&self) -> Option<&'static str> {
954        match *self {
955            BorrowedContentSource::DerefRawPointer => Some("raw pointer"),
956            BorrowedContentSource::DerefSharedRef => Some("shared reference"),
957            BorrowedContentSource::DerefMutableRef => Some("mutable reference"),
958            // Overloaded deref and index operators should be evaluated into a
959            // temporary. So we don't need a description here.
960            BorrowedContentSource::OverloadedDeref(_)
961            | BorrowedContentSource::OverloadedIndex(_) => None,
962        }
963    }
964
965    pub(super) fn describe_for_immutable_place(&self, tcx: TyCtxt<'_>) -> String {
966        match *self {
967            BorrowedContentSource::DerefRawPointer => "a `*const` pointer".to_string(),
968            BorrowedContentSource::DerefSharedRef => "a `&` reference".to_string(),
969            BorrowedContentSource::DerefMutableRef => {
970                ::rustc_middle::util::bug::bug_fmt(format_args!("describe_for_immutable_place: DerefMutableRef isn\'t immutable"))bug!("describe_for_immutable_place: DerefMutableRef isn't immutable")
971            }
972            BorrowedContentSource::OverloadedDeref(ty) => ty
973                .ty_adt_def()
974                .and_then(|adt| match tcx.get_diagnostic_name(adt.did())? {
975                    name @ (sym::Rc | sym::Arc) => Some(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("an `{0}`", name))
    })format!("an `{name}`")),
976                    _ => None,
977                })
978                .unwrap_or_else(|| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("dereference of `{0}`", ty))
    })format!("dereference of `{ty}`")),
979            BorrowedContentSource::OverloadedIndex(ty) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("an index of `{0}`", ty))
    })format!("an index of `{ty}`"),
980        }
981    }
982
983    fn from_call(func: Ty<'tcx>, tcx: TyCtxt<'tcx>) -> Option<Self> {
984        match *func.kind() {
985            ty::FnDef(def_id, args) => {
986                let trait_id = tcx.trait_of_assoc(def_id)?;
987
988                if tcx.is_lang_item(trait_id, LangItem::Deref)
989                    || tcx.is_lang_item(trait_id, LangItem::DerefMut)
990                {
991                    Some(BorrowedContentSource::OverloadedDeref(args.type_at(0)))
992                } else if tcx.is_lang_item(trait_id, LangItem::Index)
993                    || tcx.is_lang_item(trait_id, LangItem::IndexMut)
994                {
995                    Some(BorrowedContentSource::OverloadedIndex(args.type_at(0)))
996                } else {
997                    None
998                }
999            }
1000            _ => None,
1001        }
1002    }
1003}
1004
1005/// Helper struct for `explain_captures`.
1006struct CapturedMessageOpt {
1007    is_partial_move: bool,
1008    is_loop_message: bool,
1009    is_move_msg: bool,
1010    is_loop_move: bool,
1011    has_suggest_reborrow: bool,
1012    maybe_reinitialized_locations_is_empty: bool,
1013}
1014
1015/// Tracks whether [`MirBorrowckCtxt::explain_captures`] emitted a clone
1016/// suggestion, so callers can avoid emitting redundant suggestions downstream.
1017#[derive(#[automatically_derived]
impl ::core::marker::Copy for CloneSuggestion { }Copy, #[automatically_derived]
impl ::core::clone::Clone for CloneSuggestion {
    #[inline]
    fn clone(&self) -> CloneSuggestion { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for CloneSuggestion {
    #[inline]
    fn eq(&self, other: &CloneSuggestion) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for CloneSuggestion {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq)]
1018pub(super) enum CloneSuggestion {
1019    Emitted,
1020    NotEmitted,
1021}
1022
1023impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1024    /// Finds the spans associated to a move or copy of move_place at location.
1025    pub(super) fn move_spans(
1026        &self,
1027        moved_place: PlaceRef<'tcx>, // Could also be an upvar.
1028        location: Location,
1029    ) -> UseSpans<'tcx> {
1030        use self::UseSpans::*;
1031
1032        let Some(stmt) = self.body[location.block].statements.get(location.statement_index) else {
1033            return OtherUse(self.body.source_info(location).span);
1034        };
1035
1036        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1036",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1036u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("move_spans: moved_place={0:?} location={1:?} stmt={2:?}",
                                                    moved_place, location, stmt) as &dyn Value))])
            });
    } else { ; }
};debug!("move_spans: moved_place={:?} location={:?} stmt={:?}", moved_place, location, stmt);
1037        if let StatementKind::Assign((_, Rvalue::Aggregate(kind, places))) = &stmt.kind
1038            && let AggregateKind::Closure(def_id, _) | AggregateKind::Coroutine(def_id, _) = **kind
1039        {
1040            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1040",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1040u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("move_spans: def_id={0:?} places={1:?}",
                                                    def_id, places) as &dyn Value))])
            });
    } else { ; }
};debug!("move_spans: def_id={:?} places={:?}", def_id, places);
1041            let def_id = def_id.expect_local();
1042            if let Some((args_span, closure_kind, capture_kind_span, path_span)) =
1043                self.closure_span(def_id, moved_place, places)
1044            {
1045                return ClosureUse { closure_kind, args_span, capture_kind_span, path_span };
1046            }
1047        }
1048
1049        // StatementKind::FakeRead only contains a def_id if they are introduced as a result
1050        // of pattern matching within a closure.
1051        if let StatementKind::FakeRead((cause, place)) = stmt.kind {
1052            match cause {
1053                FakeReadCause::ForMatchedPlace(Some(closure_def_id))
1054                | FakeReadCause::ForLet(Some(closure_def_id)) => {
1055                    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1055",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1055u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("move_spans: def_id={0:?} place={1:?}",
                                                    closure_def_id, place) as &dyn Value))])
            });
    } else { ; }
};debug!("move_spans: def_id={:?} place={:?}", closure_def_id, place);
1056                    let places = &[Operand::Move(place)];
1057                    if let Some((args_span, closure_kind, capture_kind_span, path_span)) =
1058                        self.closure_span(closure_def_id, moved_place, IndexSlice::from_raw(places))
1059                    {
1060                        return ClosureUse {
1061                            closure_kind,
1062                            args_span,
1063                            capture_kind_span,
1064                            path_span,
1065                        };
1066                    }
1067                }
1068                _ => {}
1069            }
1070        }
1071
1072        let normal_ret =
1073            if moved_place.projection.iter().any(|p| #[allow(non_exhaustive_omitted_patterns)] match p {
    ProjectionElem::Downcast(..) => true,
    _ => false,
}matches!(p, ProjectionElem::Downcast(..))) {
1074                PatUse(stmt.source_info.span)
1075            } else {
1076                OtherUse(stmt.source_info.span)
1077            };
1078
1079        // We are trying to find MIR of the form:
1080        // ```
1081        // _temp = _moved_val;
1082        // ...
1083        // FnSelfCall(_temp, ...)
1084        // ```
1085        //
1086        // where `_moved_val` is the place we generated the move error for,
1087        // `_temp` is some other local, and `FnSelfCall` is a function
1088        // that has a `self` parameter.
1089
1090        let target_temp = match stmt.kind {
1091            StatementKind::Assign((temp, _)) if temp.as_local().is_some() => {
1092                temp.as_local().unwrap()
1093            }
1094            _ => return normal_ret,
1095        };
1096
1097        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1097",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1097u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("move_spans: target_temp = {0:?}",
                                                    target_temp) as &dyn Value))])
            });
    } else { ; }
};debug!("move_spans: target_temp = {:?}", target_temp);
1098
1099        if let Some(Terminator {
1100            kind: TerminatorKind::Call { fn_span, call_source, .. }, ..
1101        }) = &self.body[location.block].terminator
1102        {
1103            let Some((method_did, method_args)) =
1104                find_self_call(self.infcx.tcx, self.body, target_temp, location.block)
1105            else {
1106                return normal_ret;
1107            };
1108
1109            let kind = call_kind(
1110                self.infcx.tcx,
1111                self.infcx.typing_env(self.infcx.param_env),
1112                method_did,
1113                method_args,
1114                *fn_span,
1115                call_source.from_hir_call(),
1116                self.infcx.tcx.fn_arg_idents(method_did)[0],
1117            );
1118
1119            return FnSelfUse {
1120                var_span: stmt.source_info.span,
1121                fn_call_span: *fn_span,
1122                fn_span: self.infcx.tcx.def_span(method_did),
1123                kind,
1124            };
1125        }
1126
1127        normal_ret
1128    }
1129
1130    /// Finds the span of arguments of a closure (within `maybe_closure_span`)
1131    /// and its usage of the local assigned at `location`.
1132    /// This is done by searching in statements succeeding `location`
1133    /// and originating from `maybe_closure_span`.
1134    pub(super) fn borrow_spans(&self, use_span: Span, location: Location) -> UseSpans<'tcx> {
1135        use self::UseSpans::*;
1136        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1136",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1136u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("borrow_spans: use_span={0:?} location={1:?}",
                                                    use_span, location) as &dyn Value))])
            });
    } else { ; }
};debug!("borrow_spans: use_span={:?} location={:?}", use_span, location);
1137
1138        let Some(Statement { kind: StatementKind::Assign((place, _)), .. }) =
1139            self.body[location.block].statements.get(location.statement_index)
1140        else {
1141            return OtherUse(use_span);
1142        };
1143        let Some(target) = place.as_local() else { return OtherUse(use_span) };
1144
1145        if self.body.local_kind(target) != LocalKind::Temp {
1146            // operands are always temporaries.
1147            return OtherUse(use_span);
1148        }
1149
1150        // drop and replace might have moved the assignment to the next block
1151        let maybe_additional_statement =
1152            if let TerminatorKind::Drop { target: drop_target, .. } =
1153                self.body[location.block].terminator().kind
1154            {
1155                self.body[drop_target].statements.first()
1156            } else {
1157                None
1158            };
1159
1160        let statements =
1161            self.body[location.block].statements[location.statement_index + 1..].iter();
1162
1163        for stmt in statements.chain(maybe_additional_statement) {
1164            if let StatementKind::Assign((_, Rvalue::Aggregate(kind, places))) = &stmt.kind {
1165                let (&def_id, is_coroutine) = match kind {
1166                    AggregateKind::Closure(def_id, _) => (def_id, false),
1167                    AggregateKind::Coroutine(def_id, _) => (def_id, true),
1168                    _ => continue,
1169                };
1170                let def_id = def_id.expect_local();
1171
1172                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1172",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1172u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("borrow_spans: def_id={0:?} is_coroutine={1:?} places={2:?}",
                                                    def_id, is_coroutine, places) as &dyn Value))])
            });
    } else { ; }
};debug!(
1173                    "borrow_spans: def_id={:?} is_coroutine={:?} places={:?}",
1174                    def_id, is_coroutine, places
1175                );
1176                if let Some((args_span, closure_kind, capture_kind_span, path_span)) =
1177                    self.closure_span(def_id, Place::from(target).as_ref(), places)
1178                {
1179                    return ClosureUse { closure_kind, args_span, capture_kind_span, path_span };
1180                } else {
1181                    return OtherUse(use_span);
1182                }
1183            }
1184
1185            if use_span != stmt.source_info.span {
1186                break;
1187            }
1188        }
1189
1190        OtherUse(use_span)
1191    }
1192
1193    /// Finds the spans of a captured place within a closure or coroutine.
1194    /// The first span is the location of the use resulting in the capture kind of the capture
1195    /// The second span is the location the use resulting in the captured path of the capture
1196    fn closure_span(
1197        &self,
1198        def_id: LocalDefId,
1199        target_place: PlaceRef<'tcx>,
1200        places: &IndexSlice<FieldIdx, Operand<'tcx>>,
1201    ) -> Option<(Span, hir::ClosureKind, Span, Span)> {
1202        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1202",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1202u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("closure_span: def_id={0:?} target_place={1:?} places={2:?}",
                                                    def_id, target_place, places) as &dyn Value))])
            });
    } else { ; }
};debug!(
1203            "closure_span: def_id={:?} target_place={:?} places={:?}",
1204            def_id, target_place, places
1205        );
1206        let hir_id = self.infcx.tcx.local_def_id_to_hir_id(def_id);
1207        let expr = &self.infcx.tcx.hir_expect_expr(hir_id).kind;
1208        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1208",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1208u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("closure_span: hir_id={0:?} expr={1:?}",
                                                    hir_id, expr) as &dyn Value))])
            });
    } else { ; }
};debug!("closure_span: hir_id={:?} expr={:?}", hir_id, expr);
1209        if let &hir::ExprKind::Closure(&hir::Closure { kind, fn_decl_span, .. }) = expr {
1210            for (captured_place, place) in
1211                self.infcx.tcx.closure_captures(def_id).iter().zip(places)
1212            {
1213                match place {
1214                    Operand::Copy(place) | Operand::Move(place)
1215                        if target_place == place.as_ref() =>
1216                    {
1217                        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_borrowck/src/diagnostics/mod.rs:1217",
                        "rustc_borrowck::diagnostics", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(1217u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("closure_span: found captured local {0:?}",
                                                    place) as &dyn Value))])
            });
    } else { ; }
};debug!("closure_span: found captured local {:?}", place);
1218                        return Some((
1219                            fn_decl_span,
1220                            kind,
1221                            captured_place.get_capture_kind_span(self.infcx.tcx),
1222                            captured_place.get_path_span(self.infcx.tcx),
1223                        ));
1224                    }
1225                    _ => {}
1226                }
1227            }
1228        }
1229        None
1230    }
1231
1232    /// Helper to retrieve span(s) of given borrow from the current MIR
1233    /// representation
1234    pub(super) fn retrieve_borrow_spans(&self, borrow: &BorrowData<'_>) -> UseSpans<'tcx> {
1235        let span = self.body.source_info(borrow.reserve_location).span;
1236        self.borrow_spans(span, borrow.reserve_location)
1237    }
1238
1239    fn explain_captures(
1240        &mut self,
1241        err: &mut Diag<'infcx>,
1242        span: Span,
1243        move_span: Span,
1244        move_spans: UseSpans<'tcx>,
1245        moved_place: Place<'tcx>,
1246        msg_opt: CapturedMessageOpt,
1247    ) -> CloneSuggestion {
1248        let CapturedMessageOpt {
1249            is_partial_move: is_partial,
1250            is_loop_message,
1251            is_move_msg,
1252            is_loop_move,
1253            has_suggest_reborrow,
1254            maybe_reinitialized_locations_is_empty,
1255        } = msg_opt;
1256        let mut suggested_cloning = false;
1257        if let UseSpans::FnSelfUse { var_span, fn_call_span, fn_span, kind } = move_spans {
1258            let place_name = self
1259                .describe_place(moved_place.as_ref())
1260                .map(|n| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", n))
    })format!("`{n}`"))
1261                .unwrap_or_else(|| "value".to_owned());
1262            match kind {
1263                CallKind::FnCall { fn_trait_id, self_ty }
1264                    if self.infcx.tcx.is_lang_item(fn_trait_id, LangItem::FnOnce) =>
1265                {
1266                    err.subdiagnostic(CaptureReasonLabel::Call {
1267                        fn_call_span,
1268                        place_name: &place_name,
1269                        is_partial,
1270                        is_loop_message,
1271                    });
1272                    // Check if the move occurs on a value because of a call on a closure that comes
1273                    // from a type parameter `F: FnOnce()`. If so, we provide a targeted `note`:
1274                    // ```
1275                    // error[E0382]: use of moved value: `blk`
1276                    //   --> $DIR/once-cant-call-twice-on-heap.rs:8:5
1277                    //    |
1278                    // LL | fn foo<F:FnOnce()>(blk: F) {
1279                    //    |                    --- move occurs because `blk` has type `F`, which does not implement the `Copy` trait
1280                    // LL | blk();
1281                    //    | ----- `blk` moved due to this call
1282                    // LL | blk();
1283                    //    | ^^^ value used here after move
1284                    //    |
1285                    // note: `FnOnce` closures can only be called once
1286                    //   --> $DIR/once-cant-call-twice-on-heap.rs:6:10
1287                    //    |
1288                    // LL | fn foo<F:FnOnce()>(blk: F) {
1289                    //    |        ^^^^^^^^ `F` is made to be an `FnOnce` closure here
1290                    // LL | blk();
1291                    //    | ----- this value implements `FnOnce`, which causes it to be moved when called
1292                    // ```
1293                    if let ty::Param(param_ty) = *self_ty.kind()
1294                        && let generics = self.infcx.tcx.generics_of(self.mir_def_id())
1295                        && let param = generics.type_param(param_ty, self.infcx.tcx)
1296                        && let Some(hir_generics) = self.infcx.tcx.hir_get_generics(
1297                            self.infcx.tcx.typeck_root_def_id_local(self.mir_def_id()),
1298                        )
1299                        && let spans = hir_generics
1300                            .predicates
1301                            .iter()
1302                            .filter_map(|pred| match pred.kind {
1303                                hir::WherePredicateKind::BoundPredicate(pred) => Some(pred),
1304                                _ => None,
1305                            })
1306                            .filter(|pred| {
1307                                if let Some((id, _)) = pred.bounded_ty.as_generic_param() {
1308                                    id == param.def_id
1309                                } else {
1310                                    false
1311                                }
1312                            })
1313                            .flat_map(|pred| pred.bounds)
1314                            .filter_map(|bound| {
1315                                if let Some(trait_ref) = bound.trait_ref()
1316                                    && let Some(trait_def_id) = trait_ref.trait_def_id()
1317                                    && trait_def_id == fn_trait_id
1318                                {
1319                                    Some(bound.span())
1320                                } else {
1321                                    None
1322                                }
1323                            })
1324                            .collect::<Vec<Span>>()
1325                        && !spans.is_empty()
1326                    {
1327                        let mut span: MultiSpan = spans.clone().into();
1328                        let msg = rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`{$ty}` is made to be an `FnOnce` closure here"))msg!("`{$ty}` is made to be an `FnOnce` closure here")
1329                            .arg("ty", param_ty.to_string())
1330                            .format();
1331                        for sp in spans {
1332                            span.push_span_label(sp, msg.clone());
1333                        }
1334                        span.push_span_label(
1335                            fn_call_span,
1336                            rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this value implements `FnOnce`, which causes it to be moved when called"))msg!("this value implements `FnOnce`, which causes it to be moved when called"),
1337                        );
1338                        err.span_note(span, rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`FnOnce` closures can only be called once"))msg!("`FnOnce` closures can only be called once"));
1339                    } else {
1340                        err.subdiagnostic(CaptureReasonNote::FnOnceMoveInCall { var_span });
1341                    }
1342                }
1343                CallKind::Operator { self_arg, trait_id, .. } => {
1344                    let self_arg = self_arg.unwrap();
1345                    err.subdiagnostic(CaptureReasonLabel::OperatorUse {
1346                        fn_call_span,
1347                        place_name: &place_name,
1348                        is_partial,
1349                        is_loop_message,
1350                    });
1351                    if self.fn_self_span_reported.insert(fn_span) {
1352                        let lang = self.infcx.tcx.lang_items();
1353                        err.subdiagnostic(
1354                            if [lang.not_trait(), lang.deref_trait(), lang.neg_trait()]
1355                                .contains(&Some(trait_id))
1356                            {
1357                                CaptureReasonNote::UnOpMoveByOperator { span: self_arg.span }
1358                            } else {
1359                                CaptureReasonNote::LhsMoveByOperator { span: self_arg.span }
1360                            },
1361                        );
1362                    }
1363                }
1364                CallKind::Normal { self_arg, desugaring, method_did, method_args } => {
1365                    let self_arg = self_arg.unwrap();
1366                    let mut has_sugg = false;
1367                    let tcx = self.infcx.tcx;
1368                    // Avoid pointing to the same function in multiple different
1369                    // error messages.
1370                    if span != DUMMY_SP && self.fn_self_span_reported.insert(self_arg.span) {
1371                        self.explain_iterator_advancement_in_for_loop_if_applicable(
1372                            err,
1373                            span,
1374                            &move_spans,
1375                        );
1376
1377                        let func = tcx.def_path_str(method_did);
1378                        err.subdiagnostic(CaptureReasonNote::FuncTakeSelf {
1379                            func,
1380                            place_name: place_name.clone(),
1381                            span: self_arg.span,
1382                        });
1383                    }
1384                    let parent_did = tcx.parent(method_did);
1385                    let parent_self_ty =
1386                        #[allow(non_exhaustive_omitted_patterns)] match tcx.def_kind(parent_did) {
    rustc_hir::def::DefKind::Impl { .. } => true,
    _ => false,
}matches!(tcx.def_kind(parent_did), rustc_hir::def::DefKind::Impl { .. })
1387                            .then_some(parent_did)
1388                            .and_then(|did| {
1389                                match tcx.type_of(did).instantiate_identity().skip_norm_wip().kind()
1390                                {
1391                                    ty::Adt(def, ..) => Some(def.did()),
1392                                    _ => None,
1393                                }
1394                            });
1395                    let is_option_or_result = parent_self_ty.is_some_and(|def_id| {
1396                        #[allow(non_exhaustive_omitted_patterns)] match tcx.get_diagnostic_name(def_id)
    {
    Some(sym::Option | sym::Result) => true,
    _ => false,
}matches!(tcx.get_diagnostic_name(def_id), Some(sym::Option | sym::Result))
1397                    });
1398                    if is_option_or_result && maybe_reinitialized_locations_is_empty {
1399                        err.subdiagnostic(CaptureReasonLabel::BorrowContent {
1400                            var_span: var_span.shrink_to_hi(),
1401                        });
1402                    }
1403                    if let Some((CallDesugaringKind::ForLoopIntoIter, _)) = desugaring {
1404                        let ty = moved_place.ty(self.body, tcx).ty;
1405                        let suggest = match tcx.get_diagnostic_item(sym::IntoIterator) {
1406                            Some(def_id) => type_known_to_meet_bound_modulo_regions(
1407                                self.infcx,
1408                                self.infcx.param_env,
1409                                Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, ty),
1410                                def_id,
1411                            ),
1412                            _ => false,
1413                        };
1414                        if suggest {
1415                            err.subdiagnostic(CaptureReasonSuggest::IterateSlice {
1416                                ty,
1417                                span: move_span.shrink_to_lo(),
1418                            });
1419                        }
1420
1421                        err.subdiagnostic(CaptureReasonLabel::ImplicitCall {
1422                            fn_call_span,
1423                            place_name: &place_name,
1424                            is_partial,
1425                            is_loop_message,
1426                        });
1427                        // If the moved place was a `&mut` ref, then we can
1428                        // suggest to reborrow it where it was moved, so it
1429                        // will still be valid by the time we get to the usage.
1430                        if let ty::Ref(_, _, hir::Mutability::Mut) =
1431                            moved_place.ty(self.body, self.infcx.tcx).ty.kind()
1432                        {
1433                            // The `&mut *place` reborrow suggestion is `MachineApplicable`, so
1434                            // only offer it where `*place` can be borrowed mutably: a value
1435                            // captured by an `Fn` closure (held via `&self`) cannot, and the
1436                            // suggestion would otherwise fail to compile with E0596.
1437                            let reborrow_place = self.infcx.tcx.mk_place_deref(moved_place);
1438                            let reborrow_is_valid = self
1439                                .is_mutable(reborrow_place.as_ref(), LocalMutationIsAllowed::No)
1440                                .is_ok();
1441                            // Suggest `reborrow` in other place for following situations:
1442                            // 1. If we are in a loop this will be suggested later.
1443                            // 2. If the moved value is a mut reference, it is used in a
1444                            // generic function and the corresponding arg's type is generic param.
1445                            if !is_loop_move && !has_suggest_reborrow && reborrow_is_valid {
1446                                self.suggest_reborrow(
1447                                    err,
1448                                    move_span.shrink_to_lo(),
1449                                    moved_place.as_ref(),
1450                                );
1451                            }
1452                        }
1453                    } else {
1454                        if let Some((CallDesugaringKind::Await, _)) = desugaring {
1455                            err.subdiagnostic(CaptureReasonLabel::Await {
1456                                fn_call_span,
1457                                place_name: &place_name,
1458                                is_partial,
1459                                is_loop_message,
1460                            });
1461                        } else {
1462                            err.subdiagnostic(CaptureReasonLabel::MethodCall {
1463                                fn_call_span,
1464                                place_name: &place_name,
1465                                is_partial,
1466                                is_loop_message,
1467                            });
1468                        }
1469                        // Erase and shadow everything that could be passed to the new infcx.
1470                        let ty = moved_place.ty(self.body, tcx).ty;
1471
1472                        if let ty::Adt(def, args) = ty.peel_refs().kind()
1473                            && tcx.is_lang_item(def.did(), LangItem::Pin)
1474                            && let ty::Ref(_, _, hir::Mutability::Mut) = args.type_at(0).kind()
1475                            && let self_ty = self.infcx.instantiate_binder_with_fresh_vars(
1476                                fn_call_span,
1477                                BoundRegionConversionTime::FnCall,
1478                                tcx.fn_sig(method_did)
1479                                    .instantiate(tcx, method_args)
1480                                    .skip_norm_wip()
1481                                    .input(0),
1482                            )
1483                            && self.infcx.can_eq(self.infcx.param_env, ty, self_ty)
1484                        {
1485                            err.subdiagnostic(CaptureReasonSuggest::FreshReborrow {
1486                                span: move_span.shrink_to_hi(),
1487                            });
1488                            has_sugg = true;
1489                        }
1490                        if let Some(clone_trait) = tcx.lang_items().clone_trait() {
1491                            // Check whether the deref is from a custom Deref impl
1492                            // (e.g. Rc, Box) or a built-in reference deref.
1493                            // For built-in derefs with Clone fully satisfied, we skip
1494                            // the UFCS suggestion here and let `suggest_cloning`
1495                            // downstream emit a simpler `.clone()` suggestion instead.
1496                            let has_overloaded_deref =
1497                                moved_place.iter_projections().any(|(place, elem)| {
1498                                    #[allow(non_exhaustive_omitted_patterns)] match elem {
    ProjectionElem::Deref => true,
    _ => false,
}matches!(elem, ProjectionElem::Deref)
1499                                        && #[allow(non_exhaustive_omitted_patterns)] match self.borrowed_content_source(place)
    {
    BorrowedContentSource::OverloadedDeref(_) |
        BorrowedContentSource::OverloadedIndex(_) => true,
    _ => false,
}matches!(
1500                                            self.borrowed_content_source(place),
1501                                            BorrowedContentSource::OverloadedDeref(_)
1502                                                | BorrowedContentSource::OverloadedIndex(_)
1503                                        )
1504                                });
1505
1506                            let has_deref = moved_place
1507                                .iter_projections()
1508                                .any(|(_, elem)| #[allow(non_exhaustive_omitted_patterns)] match elem {
    ProjectionElem::Deref => true,
    _ => false,
}matches!(elem, ProjectionElem::Deref));
1509
1510                            let sugg = if has_deref {
1511                                let (start, end) = if let Some(expr) = self.find_expr(move_span)
1512                                    && let Some(_) = self.clone_on_reference(expr)
1513                                    && let hir::ExprKind::MethodCall(_, rcvr, _, _) = expr.kind
1514                                {
1515                                    (move_span.shrink_to_lo(), move_span.with_lo(rcvr.span.hi()))
1516                                } else {
1517                                    (move_span.shrink_to_lo(), move_span.shrink_to_hi())
1518                                };
1519                                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(start,
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("<{0} as Clone>::clone(&",
                                    ty))
                        })), (end, ")".to_string())]))vec![
1520                                    // We use the fully-qualified path because `.clone()` can
1521                                    // sometimes choose `<&T as Clone>` instead of `<T as Clone>`
1522                                    // when going through auto-deref, so this ensures that doesn't
1523                                    // happen, causing suggestions for `.clone().clone()`.
1524                                    (start, format!("<{ty} as Clone>::clone(&")),
1525                                    (end, ")".to_string()),
1526                                ]
1527                            } else {
1528                                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(move_span.shrink_to_hi(), ".clone()".to_string())]))vec![(move_span.shrink_to_hi(), ".clone()".to_string())]
1529                            };
1530                            if let Some(errors) = self.infcx.type_implements_trait_shallow(
1531                                clone_trait,
1532                                ty,
1533                                self.infcx.param_env,
1534                            ) && !has_sugg
1535                            {
1536                                let skip_for_simple_clone =
1537                                    has_deref && !has_overloaded_deref && errors.is_empty();
1538                                if !skip_for_simple_clone {
1539                                    let msg = match &errors[..] {
1540                                        [] => "you can `clone` the value and consume it, but \
1541                                               this might not be your desired behavior"
1542                                            .to_string(),
1543                                        [error] => {
1544                                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("you could `clone` the value and consume it, if the `{0}` trait bound could be satisfied",
                error.obligation.predicate))
    })format!(
1545                                                "you could `clone` the value and consume it, if \
1546                                                 the `{}` trait bound could be satisfied",
1547                                                error.obligation.predicate,
1548                                            )
1549                                        }
1550                                        _ => {
1551                                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("you could `clone` the value and consume it, if the following trait bounds could be satisfied: {0}",
                listify(&errors,
                        |e: &FulfillmentError<'tcx>|
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("`{0}`",
                                            e.obligation.predicate))
                                })).unwrap()))
    })format!(
1552                                                "you could `clone` the value and consume it, if \
1553                                                 the following trait bounds could be satisfied: \
1554                                                 {}",
1555                                                listify(
1556                                                    &errors,
1557                                                    |e: &FulfillmentError<'tcx>| format!(
1558                                                        "`{}`",
1559                                                        e.obligation.predicate
1560                                                    )
1561                                                )
1562                                                .unwrap(),
1563                                            )
1564                                        }
1565                                    };
1566                                    err.multipart_suggestion(
1567                                        msg,
1568                                        sugg,
1569                                        Applicability::MaybeIncorrect,
1570                                    );
1571
1572                                    suggested_cloning = errors.is_empty();
1573
1574                                    for error in errors {
1575                                        if let FulfillmentErrorCode::Select(
1576                                            SelectionError::Unimplemented,
1577                                        ) = error.code
1578                                            && let ty::PredicateKind::Clause(ty::ClauseKind::Trait(
1579                                                pred,
1580                                            )) = error.obligation.predicate.kind().skip_binder()
1581                                        {
1582                                            self.infcx.err_ctxt().suggest_derive(
1583                                                &error.obligation,
1584                                                err,
1585                                                error.obligation.predicate.kind().rebind(pred),
1586                                            );
1587                                        }
1588                                    }
1589                                }
1590                            }
1591                        }
1592                    }
1593                }
1594                // Other desugarings takes &self, which cannot cause a move
1595                _ => {}
1596            }
1597        } else {
1598            if move_span != span || is_loop_message {
1599                err.subdiagnostic(CaptureReasonLabel::MovedHere {
1600                    move_span,
1601                    is_partial,
1602                    is_move_msg,
1603                    is_loop_message,
1604                });
1605            }
1606            // If the move error occurs due to a loop, don't show
1607            // another message for the same span
1608            if !is_loop_message {
1609                move_spans.var_subdiag(err, None, |kind, var_span| match kind {
1610                    hir::ClosureKind::Coroutine(_) => {
1611                        CaptureVarCause::PartialMoveUseInCoroutine { var_span, is_partial }
1612                    }
1613                    hir::ClosureKind::Closure | hir::ClosureKind::CoroutineClosure(_) => {
1614                        CaptureVarCause::PartialMoveUseInClosure { var_span, is_partial }
1615                    }
1616                })
1617            }
1618        }
1619        if suggested_cloning { CloneSuggestion::Emitted } else { CloneSuggestion::NotEmitted }
1620    }
1621
1622    /// Skip over locals that begin with an underscore or have no name
1623    pub(crate) fn local_excluded_from_unused_mut_lint(&self, index: Local) -> bool {
1624        self.local_name(index).is_none_or(|name| name.as_str().starts_with('_'))
1625    }
1626}
1627
1628const LIMITATION_NOTE: DiagMessage =
1629    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("due to a current limitation of the type system, this implies a `'static` lifetime"))msg!("due to a current limitation of the type system, this implies a `'static` lifetime");