Skip to main content

rustc_borrowck/diagnostics/
mutability_errors.rs

1use core::ops::ControlFlow;
2
3use either::Either;
4use hir::{ExprKind, Param};
5use rustc_abi::FieldIdx;
6use rustc_errors::{Applicability, Diag};
7use rustc_hir::def_id::DefId;
8use rustc_hir::intravisit::Visitor;
9use rustc_hir::{self as hir, BindingMode, ByRef, Expr, Node};
10use rustc_middle::bug;
11use rustc_middle::hir::place::PlaceBase;
12use rustc_middle::mir::visit::PlaceContext;
13use rustc_middle::mir::{
14    self, BindingForm, Body, BorrowKind, Local, LocalDecl, LocalInfo, LocalKind, Location,
15    Mutability, Operand, Place, PlaceRef, ProjectionElem, RawPtrKind, Rvalue, Statement,
16    StatementKind, TerminatorKind,
17};
18use rustc_middle::ty::{self, InstanceKind, Ty, TyCtxt, Upcast};
19use rustc_span::{BytePos, DesugaringKind, Span, Symbol, kw, sym};
20use rustc_trait_selection::error_reporting::InferCtxtErrorExt;
21use rustc_trait_selection::infer::InferCtxtExt;
22use rustc_trait_selection::traits;
23use tracing::{debug, trace};
24
25use crate::diagnostics::BorrowedContentSource;
26use crate::{MirBorrowckCtxt, session_diagnostics};
27
28#[derive(#[automatically_derived]
impl ::core::marker::Copy for AccessKind { }Copy, #[automatically_derived]
impl ::core::clone::Clone for AccessKind {
    #[inline]
    fn clone(&self) -> AccessKind { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for AccessKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                AccessKind::MutableBorrow => "MutableBorrow",
                AccessKind::Mutate => "Mutate",
            })
    }
}Debug, #[automatically_derived]
impl ::core::cmp::Eq for AccessKind {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq, #[automatically_derived]
impl ::core::cmp::PartialEq for AccessKind {
    #[inline]
    fn eq(&self, other: &AccessKind) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq)]
29pub(crate) enum AccessKind {
30    MutableBorrow,
31    Mutate,
32}
33
34/// Finds all statements that assign directly to local (i.e., X = ...) and returns their
35/// locations.
36fn find_assignments(body: &Body<'_>, local: Local) -> Vec<Location> {
37    use rustc_middle::mir::visit::Visitor;
38
39    struct FindLocalAssignmentVisitor {
40        needle: Local,
41        locations: Vec<Location>,
42    }
43
44    impl<'tcx> Visitor<'tcx> for FindLocalAssignmentVisitor {
45        fn visit_local(&mut self, local: Local, place_context: PlaceContext, location: Location) {
46            if self.needle != local {
47                return;
48            }
49
50            if place_context.is_place_assignment() {
51                self.locations.push(location);
52            }
53        }
54    }
55
56    let mut visitor = FindLocalAssignmentVisitor { needle: local, locations: ::alloc::vec::Vec::new()vec![] };
57    visitor.visit_body(body);
58    visitor.locations
59}
60
61impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
62    pub(crate) fn report_mutability_error(
63        &mut self,
64        access_place: Place<'tcx>,
65        span: Span,
66        the_place_err: PlaceRef<'tcx>,
67        error_access: AccessKind,
68        location: Location,
69    ) {
70        {
    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/mutability_errors.rs:70",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(70u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("report_mutability_error(access_place={0:?}, span={1:?}, the_place_err={2:?}, error_access={3:?}, location={4:?},)",
                                                    access_place, span, the_place_err, error_access, location)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(
71            "report_mutability_error(\
72                access_place={:?}, span={:?}, the_place_err={:?}, error_access={:?}, location={:?},\
73            )",
74            access_place, span, the_place_err, error_access, location,
75        );
76
77        let mut err;
78        let item_msg;
79        let reason;
80        let mut opt_source = None;
81        let access_place_desc = self.describe_any_place(access_place.as_ref());
82        {
    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/mutability_errors.rs:82",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(82u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("report_mutability_error: access_place_desc={0:?}",
                                                    access_place_desc) as &dyn Value))])
            });
    } else { ; }
};debug!("report_mutability_error: access_place_desc={:?}", access_place_desc);
83
84        match the_place_err {
85            PlaceRef { local, projection: [] } => {
86                item_msg = access_place_desc;
87                if access_place.as_local().is_some() {
88                    reason = ", as it is not declared as mutable".to_string();
89                } else {
90                    let name = self.local_name(local).expect("immutable unnamed local");
91                    reason = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(", as `{0}` is not declared as mutable",
                name))
    })format!(", as `{name}` is not declared as mutable");
92                }
93            }
94
95            PlaceRef {
96                local,
97                projection: [proj_base @ .., ProjectionElem::Field(upvar_index, _)],
98            } => {
99                if true {
    if !is_closure_like(Place::ty_from(local, proj_base, self.body,
                        self.infcx.tcx).ty) {
        ::core::panicking::panic("assertion failed: is_closure_like(Place::ty_from(local, proj_base, self.body,\n            self.infcx.tcx).ty)")
    };
};debug_assert!(is_closure_like(
100                    Place::ty_from(local, proj_base, self.body, self.infcx.tcx).ty
101                ));
102
103                let imm_borrow_derefed = self.upvars[upvar_index.index()]
104                    .place
105                    .deref_tys()
106                    .any(|ty| #[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
    ty::Ref(.., hir::Mutability::Not) => true,
    _ => false,
}matches!(ty.kind(), ty::Ref(.., hir::Mutability::Not)));
107
108                // If the place is immutable then:
109                //
110                // - Either we deref an immutable ref to get to our final place.
111                //    - We don't capture derefs of raw ptrs
112                // - Or the final place is immut because the root variable of the capture
113                //   isn't marked mut and we should suggest that to the user.
114                if imm_borrow_derefed {
115                    // If we deref an immutable ref then the suggestion here doesn't help.
116                    return;
117                } else {
118                    item_msg = access_place_desc;
119                    if self.is_upvar_field_projection(access_place.as_ref()).is_some() {
120                        reason = ", as it is not declared as mutable".to_string();
121                    } else {
122                        let name = self.upvars[upvar_index.index()].to_string(self.infcx.tcx);
123                        reason = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(", as `{0}` is not declared as mutable",
                name))
    })format!(", as `{name}` is not declared as mutable");
124                    }
125                }
126            }
127
128            PlaceRef { local, projection: [ProjectionElem::Deref] }
129                if self.body.local_decls[local].is_ref_for_guard() =>
130            {
131                item_msg = access_place_desc;
132                reason = ", as it is immutable for the pattern guard".to_string();
133            }
134            PlaceRef { local, projection: [ProjectionElem::Deref] }
135                if self.body.local_decls[local].is_ref_to_static() =>
136            {
137                if access_place.projection.len() == 1 {
138                    item_msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("immutable static item {0}",
                access_place_desc))
    })format!("immutable static item {access_place_desc}");
139                    reason = String::new();
140                } else {
141                    item_msg = access_place_desc;
142                    let local_info = self.body.local_decls[local].local_info();
143                    let LocalInfo::StaticRef { def_id, .. } = *local_info else {
144                        ::rustc_middle::util::bug::bug_fmt(format_args!("is_ref_to_static return true, but not ref to static?"));bug!("is_ref_to_static return true, but not ref to static?");
145                    };
146                    let static_name = &self.infcx.tcx.item_name(def_id);
147                    reason = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(", as `{0}` is an immutable static item",
                static_name))
    })format!(", as `{static_name}` is an immutable static item");
148                }
149            }
150            PlaceRef { local, projection: [proj_base @ .., ProjectionElem::Deref] } => {
151                if local == ty::CAPTURE_STRUCT_LOCAL
152                    && proj_base.is_empty()
153                    && !self.upvars.is_empty()
154                {
155                    item_msg = access_place_desc;
156                    if true {
    if !self.body.local_decls[ty::CAPTURE_STRUCT_LOCAL].ty.is_ref() {
        ::core::panicking::panic("assertion failed: self.body.local_decls[ty::CAPTURE_STRUCT_LOCAL].ty.is_ref()")
    };
};debug_assert!(self.body.local_decls[ty::CAPTURE_STRUCT_LOCAL].ty.is_ref());
157                    if true {
    if !is_closure_like(the_place_err.ty(self.body, self.infcx.tcx).ty) {
        ::core::panicking::panic("assertion failed: is_closure_like(the_place_err.ty(self.body, self.infcx.tcx).ty)")
    };
};debug_assert!(is_closure_like(the_place_err.ty(self.body, self.infcx.tcx).ty));
158
159                    reason = if self.is_upvar_field_projection(access_place.as_ref()).is_some() {
160                        ", as it is a captured variable in a `Fn` closure".to_string()
161                    } else {
162                        ", as `Fn` closures cannot mutate their captured variables".to_string()
163                    }
164                } else {
165                    let source =
166                        self.borrowed_content_source(PlaceRef { local, projection: proj_base });
167                    let pointer_type = source.describe_for_immutable_place(self.infcx.tcx);
168                    opt_source = Some(source);
169                    if let Some(desc) = self.describe_place(access_place.as_ref()) {
170                        item_msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", desc))
    })format!("`{desc}`");
171                        reason = match error_access {
172                            AccessKind::Mutate => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(", which is behind {0}",
                pointer_type))
    })format!(", which is behind {pointer_type}"),
173                            AccessKind::MutableBorrow => {
174                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(", as it is behind {0}",
                pointer_type))
    })format!(", as it is behind {pointer_type}")
175                            }
176                        }
177                    } else {
178                        item_msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("data in {0}", pointer_type))
    })format!("data in {pointer_type}");
179                        reason = String::new();
180                    }
181                }
182            }
183
184            PlaceRef {
185                local: _,
186                projection:
187                    [
188                        ..,
189                        ProjectionElem::Index(_)
190                        | ProjectionElem::ConstantIndex { .. }
191                        | ProjectionElem::OpaqueCast { .. }
192                        | ProjectionElem::Subslice { .. }
193                        | ProjectionElem::Downcast(..)
194                        | ProjectionElem::UnwrapUnsafeBinder(_),
195                    ],
196            } => ::rustc_middle::util::bug::bug_fmt(format_args!("Unexpected immutable place."))bug!("Unexpected immutable place."),
197        }
198
199        {
    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/mutability_errors.rs:199",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(199u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("report_mutability_error: item_msg={0:?}, reason={1:?}",
                                                    item_msg, reason) as &dyn Value))])
            });
    } else { ; }
};debug!("report_mutability_error: item_msg={:?}, reason={:?}", item_msg, reason);
200
201        // `act` and `acted_on` are strings that let us abstract over
202        // the verbs used in some diagnostic messages.
203        let act;
204        let acted_on;
205        let mut suggest = true;
206        let mut mut_error = None;
207        let mut count = 1;
208
209        let span = match error_access {
210            AccessKind::Mutate => {
211                err = self.cannot_assign(span, &(item_msg + &reason));
212                act = "assign";
213                acted_on = "written to";
214                span
215            }
216            AccessKind::MutableBorrow => {
217                act = "borrow as mutable";
218                acted_on = "borrowed as mutable";
219
220                let borrow_spans = self.borrow_spans(span, location);
221                let borrow_span = borrow_spans.args_or_use();
222                match the_place_err {
223                    PlaceRef { local, projection: [] }
224                        if self.body.local_decls[local].can_be_made_mutable() =>
225                    {
226                        let span = self.body.local_decls[local].source_info.span;
227                        mut_error = Some(span);
228                        if let Some((buffered_err, c)) = self.get_buffered_mut_error(span) {
229                            // We've encountered a second (or more) attempt to mutably borrow an
230                            // immutable binding, so the likely problem is with the binding
231                            // declaration, not the use. We collect these in a single diagnostic
232                            // and make the binding the primary span of the error.
233                            err = buffered_err;
234                            count = c + 1;
235                            if count == 2 {
236                                err.replace_span_with(span, false);
237                                err.span_label(span, "not mutable");
238                            }
239                            suggest = false;
240                        } else {
241                            err = self.cannot_borrow_path_as_mutable_because(
242                                borrow_span,
243                                &item_msg,
244                                &reason,
245                            );
246                        }
247                    }
248                    _ => {
249                        err = self.cannot_borrow_path_as_mutable_because(
250                            borrow_span,
251                            &item_msg,
252                            &reason,
253                        );
254                    }
255                }
256                if suggest {
257                    borrow_spans.var_subdiag(
258                        &mut err,
259                        Some(mir::BorrowKind::Mut { kind: mir::MutBorrowKind::Default }),
260                        |_kind, var_span| {
261                            let place = self.describe_any_place(access_place.as_ref());
262                            session_diagnostics::CaptureVarCause::MutableBorrowUsePlaceClosure {
263                                place,
264                                var_span,
265                            }
266                        },
267                    );
268                }
269                borrow_span
270            }
271        };
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/mutability_errors.rs:273",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(273u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("report_mutability_error: act={0:?}, acted_on={1:?}",
                                                    act, acted_on) as &dyn Value))])
            });
    } else { ; }
};debug!("report_mutability_error: act={:?}, acted_on={:?}", act, acted_on);
274
275        match the_place_err {
276            // Suggest making an existing shared borrow in a struct definition a mutable borrow.
277            //
278            // This is applicable when we have a deref of a field access to a deref of a local -
279            // something like `*((*_1).0`. The local that we get will be a reference to the
280            // struct we've got a field access of (it must be a reference since there's a deref
281            // after the field access).
282            PlaceRef {
283                local,
284                projection:
285                    [
286                        proj_base @ ..,
287                        ProjectionElem::Deref,
288                        ProjectionElem::Field(field, _),
289                        ProjectionElem::Deref,
290                    ],
291            } => {
292                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
293
294                let place = Place::ty_from(local, proj_base, self.body, self.infcx.tcx);
295                if let Some(span) = get_mut_span_in_struct_field(self.infcx.tcx, place.ty, *field) {
296                    err.span_suggestion_verbose(
297                        span,
298                        "consider changing this to be mutable",
299                        " mut ",
300                        Applicability::MaybeIncorrect,
301                    );
302                }
303            }
304
305            // Suggest removing a `&mut` from the use of a mutable reference.
306            PlaceRef { local, projection: [] }
307                if self
308                    .body
309                    .local_decls
310                    .get(local)
311                    .is_some_and(|l| mut_borrow_of_mutable_ref(l, self.local_name(local))) =>
312            {
313                let decl = &self.body.local_decls[local];
314                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
315                if let Some(mir::Statement {
316                    source_info,
317                    kind:
318                        mir::StatementKind::Assign(box (
319                            _,
320                            mir::Rvalue::Ref(
321                                _,
322                                mir::BorrowKind::Mut { kind: mir::MutBorrowKind::Default },
323                                _,
324                            ),
325                        )),
326                    ..
327                }) = &self.body[location.block].statements.get(location.statement_index)
328                {
329                    match *decl.local_info() {
330                        LocalInfo::User(BindingForm::Var(mir::VarBindingForm {
331                            binding_mode: BindingMode(ByRef::No, Mutability::Not),
332                            opt_ty_info: Some(sp),
333                            pat_span,
334                            ..
335                        })) => {
336                            if suggest {
337                                err.span_note(sp, "the binding is already a mutable borrow");
338                                err.span_suggestion_verbose(
339                                    pat_span.shrink_to_lo(),
340                                    "consider making the binding mutable if you need to reborrow \
341                                     multiple times",
342                                    "mut ".to_string(),
343                                    Applicability::MaybeIncorrect,
344                                );
345                            }
346                        }
347                        _ => {
348                            err.span_note(
349                                decl.source_info.span,
350                                "the binding is already a mutable borrow",
351                            );
352                        }
353                    }
354                    if let Ok(snippet) =
355                        self.infcx.tcx.sess.source_map().span_to_snippet(source_info.span)
356                    {
357                        if snippet.starts_with("&mut ") {
358                            // In calls, `&mut &mut T` may be deref-coerced to `&mut T`, and
359                            // removing the extra `&mut` is the most direct suggestion. But for
360                            // pattern-matching expressions (`match`, `if let`, `while let`), that
361                            // can easily turn into a move, so prefer suggesting an explicit
362                            // reborrow via `&mut *x` instead.
363                            let mut in_pat_scrutinee = false;
364                            let mut is_deref_coerced = false;
365                            if let Some(expr) = self.find_expr(source_info.span) {
366                                let tcx = self.infcx.tcx;
367                                let span = expr.span.source_callsite();
368                                for (_, node) in tcx.hir_parent_iter(expr.hir_id) {
369                                    if let Node::Expr(parent_expr) = node {
370                                        match parent_expr.kind {
371                                            ExprKind::Match(scrutinee, ..)
372                                                if scrutinee
373                                                    .span
374                                                    .source_callsite()
375                                                    .contains(span) =>
376                                            {
377                                                in_pat_scrutinee = true;
378                                                break;
379                                            }
380                                            ExprKind::Let(let_expr)
381                                                if let_expr
382                                                    .init
383                                                    .span
384                                                    .source_callsite()
385                                                    .contains(span) =>
386                                            {
387                                                in_pat_scrutinee = true;
388                                                break;
389                                            }
390                                            _ => {}
391                                        }
392                                    }
393                                }
394
395                                let typeck = tcx.typeck(expr.hir_id.owner.def_id);
396                                is_deref_coerced =
397                                    typeck.expr_adjustments(expr).iter().any(|adj| {
398                                        #[allow(non_exhaustive_omitted_patterns)] match adj.kind {
    ty::adjustment::Adjust::Deref(_) => true,
    _ => false,
}matches!(adj.kind, ty::adjustment::Adjust::Deref(_))
399                                    });
400                            }
401
402                            if in_pat_scrutinee {
403                                // Best-effort structured suggestion: insert `*` after `&mut `.
404                                err.span_suggestion_verbose(
405                                    source_info
406                                        .span
407                                        .with_lo(source_info.span.lo() + BytePos(5))
408                                        .shrink_to_lo(),
409                                    "to reborrow the mutable reference, add `*`",
410                                    "*",
411                                    Applicability::MaybeIncorrect,
412                                );
413                            } else if is_deref_coerced {
414                                // We don't have access to the HIR to get accurate spans, but we
415                                // can give a best effort structured suggestion.
416                                err.span_suggestion_verbose(
417                                    source_info.span.with_hi(source_info.span.lo() + BytePos(5)),
418                                    "if there is only one mutable reborrow, remove the `&mut`",
419                                    "",
420                                    Applicability::MaybeIncorrect,
421                                );
422                            }
423                        } else {
424                            // This can occur with things like `(&mut self).foo()`.
425                            err.span_help(source_info.span, "try removing `&mut` here");
426                        }
427                    } else {
428                        err.span_help(source_info.span, "try removing `&mut` here");
429                    }
430                } else if decl.mutability.is_not() {
431                    if #[allow(non_exhaustive_omitted_patterns)] match decl.local_info() {
    LocalInfo::User(BindingForm::ImplicitSelf(hir::ImplicitSelfKind::RefMut))
        => true,
    _ => false,
}matches!(
432                        decl.local_info(),
433                        LocalInfo::User(BindingForm::ImplicitSelf(hir::ImplicitSelfKind::RefMut))
434                    ) {
435                        err.note(
436                            "as `Self` may be unsized, this call attempts to take `&mut &mut self`",
437                        );
438                        err.note("however, `&mut self` expands to `self: &mut Self`, therefore `self` cannot be borrowed mutably");
439                    } else {
440                        err.span_suggestion_verbose(
441                            decl.source_info.span.shrink_to_lo(),
442                            "consider making the binding mutable",
443                            "mut ",
444                            Applicability::MachineApplicable,
445                        );
446                    };
447                }
448            }
449
450            // We want to suggest users use `let mut` for local (user
451            // variable) mutations...
452            PlaceRef { local, projection: [] }
453                if self.body.local_decls[local].can_be_made_mutable() =>
454            {
455                // ... but it doesn't make sense to suggest it on
456                // variables that are `ref x`, `ref mut x`, `&self`,
457                // or `&mut self` (such variables are simply not
458                // mutable).
459                let local_decl = &self.body.local_decls[local];
460                match (&local_decl.mutability, &Mutability::Not) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(local_decl.mutability, Mutability::Not);
461
462                if count < 10 {
463                    err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
464                }
465                if suggest {
466                    self.construct_mut_suggestion_for_local_binding_patterns(&mut err, local);
467                    let tcx = self.infcx.tcx;
468                    if let ty::Closure(id, _) = *the_place_err.ty(self.body, tcx).ty.kind() {
469                        self.show_mutating_upvar(tcx, id.expect_local(), the_place_err, &mut err);
470                    }
471                }
472            }
473
474            // Also suggest adding mut for upvars.
475            PlaceRef {
476                local,
477                projection: [proj_base @ .., ProjectionElem::Field(upvar_index, _)],
478            } => {
479                if true {
    if !is_closure_like(Place::ty_from(local, proj_base, self.body,
                        self.infcx.tcx).ty) {
        ::core::panicking::panic("assertion failed: is_closure_like(Place::ty_from(local, proj_base, self.body,\n            self.infcx.tcx).ty)")
    };
};debug_assert!(is_closure_like(
480                    Place::ty_from(local, proj_base, self.body, self.infcx.tcx).ty
481                ));
482
483                let captured_place = self.upvars[upvar_index.index()];
484
485                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
486
487                let upvar_hir_id = captured_place.get_root_variable();
488
489                if let Node::Pat(pat) = self.infcx.tcx.hir_node(upvar_hir_id)
490                    && let hir::PatKind::Binding(hir::BindingMode::NONE, _, upvar_ident, _) =
491                        pat.kind
492                {
493                    if upvar_ident.name == kw::SelfLower {
494                        for (_, node) in self.infcx.tcx.hir_parent_iter(upvar_hir_id) {
495                            if let Some(fn_decl) = node.fn_decl() {
496                                if !#[allow(non_exhaustive_omitted_patterns)] match fn_decl.implicit_self() {
    hir::ImplicitSelfKind::RefImm | hir::ImplicitSelfKind::RefMut => true,
    _ => false,
}matches!(
497                                    fn_decl.implicit_self(),
498                                    hir::ImplicitSelfKind::RefImm | hir::ImplicitSelfKind::RefMut
499                                ) {
500                                    err.span_suggestion_verbose(
501                                        upvar_ident.span.shrink_to_lo(),
502                                        "consider changing this to be mutable",
503                                        "mut ",
504                                        Applicability::MachineApplicable,
505                                    );
506                                    break;
507                                }
508                            }
509                        }
510                    } else {
511                        err.span_suggestion_verbose(
512                            upvar_ident.span.shrink_to_lo(),
513                            "consider changing this to be mutable",
514                            "mut ",
515                            Applicability::MachineApplicable,
516                        );
517                    }
518                }
519
520                let tcx = self.infcx.tcx;
521                if let ty::Ref(_, ty, Mutability::Mut) = the_place_err.ty(self.body, tcx).ty.kind()
522                    && let ty::Closure(id, _) = *ty.kind()
523                {
524                    self.show_mutating_upvar(tcx, id.expect_local(), the_place_err, &mut err);
525                }
526            }
527
528            // Complete hack to approximate old AST-borrowck diagnostic: if the span starts
529            // with a mutable borrow of a local variable, then just suggest the user remove it.
530            PlaceRef { local: _, projection: [] }
531                if self
532                    .infcx
533                    .tcx
534                    .sess
535                    .source_map()
536                    .span_to_snippet(span)
537                    .is_ok_and(|snippet| snippet.starts_with("&mut ")) =>
538            {
539                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
540                err.span_suggestion_verbose(
541                    span.with_hi(span.lo() + BytePos(5)),
542                    "try removing `&mut` here",
543                    "",
544                    Applicability::MaybeIncorrect,
545                );
546            }
547
548            PlaceRef { local, projection: [ProjectionElem::Deref] }
549                if self.body.local_decls[local].is_ref_for_guard() =>
550            {
551                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
552                err.note(
553                    "variables bound in patterns are immutable until the end of the pattern guard",
554                );
555            }
556
557            // We want to point out when a `&` can be readily replaced
558            // with an `&mut`.
559            //
560            // FIXME: can this case be generalized to work for an
561            // arbitrary base for the projection?
562            PlaceRef { local, projection: [ProjectionElem::Deref] }
563                if self.body.local_decls[local].is_user_variable() =>
564            {
565                let local_decl = &self.body.local_decls[local];
566
567                let (pointer_sigil, pointer_desc) =
568                    if local_decl.ty.is_ref() { ("&", "reference") } else { ("*const", "pointer") };
569
570                match self.local_name(local) {
571                    Some(name) if !local_decl.from_compiler_desugaring() => {
572                        err.span_label(
573                            span,
574                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` is a `{1}` {2}, so it cannot be {3}",
                name, pointer_sigil, pointer_desc, acted_on))
    })format!(
575                                "`{name}` is a `{pointer_sigil}` {pointer_desc}, so it cannot be \
576                                 {acted_on}",
577                            ),
578                        );
579
580                        self.suggest_using_iter_mut(&mut err);
581                        self.suggest_make_local_mut(&mut err, local, name);
582                    }
583                    _ => {
584                        err.span_label(
585                            span,
586                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0} through `{1}` {2}", act,
                pointer_sigil, pointer_desc))
    })format!("cannot {act} through `{pointer_sigil}` {pointer_desc}"),
587                        );
588                    }
589                }
590            }
591
592            PlaceRef { local, projection: [ProjectionElem::Deref] }
593                if local == ty::CAPTURE_STRUCT_LOCAL && !self.upvars.is_empty() =>
594            {
595                self.point_at_binding_outside_closure(&mut err, local, access_place);
596                self.expected_fn_found_fn_mut_call(&mut err, span, act);
597            }
598
599            PlaceRef { local, projection: [.., ProjectionElem::Deref] } => {
600                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
601
602                match opt_source {
603                    Some(BorrowedContentSource::OverloadedDeref(ty)) => {
604                        err.help(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("trait `DerefMut` is required to modify through a dereference, but it is not implemented for `{0}`",
                ty))
    })format!(
605                            "trait `DerefMut` is required to modify through a dereference, \
606                             but it is not implemented for `{ty}`",
607                        ));
608                    }
609                    Some(BorrowedContentSource::OverloadedIndex(ty)) => {
610                        err.help(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("trait `IndexMut` is required to modify indexed content, but it is not implemented for `{0}`",
                ty))
    })format!(
611                            "trait `IndexMut` is required to modify indexed content, \
612                             but it is not implemented for `{ty}`",
613                        ));
614                        self.suggest_map_index_mut_alternatives(ty, &mut err, span);
615                    }
616                    _ => {
617                        let local = &self.body.local_decls[local];
618                        match *local.local_info() {
619                            LocalInfo::StaticRef { def_id, .. } => {
620                                let span = self.infcx.tcx.def_span(def_id);
621                                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this `static` cannot be {0}",
                acted_on))
    })format!("this `static` cannot be {acted_on}"));
622                            }
623                            LocalInfo::ConstRef { def_id } => {
624                                let span = self.infcx.tcx.def_span(def_id);
625                                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this `const` cannot be {0}",
                acted_on))
    })format!("this `const` cannot be {acted_on}"));
626                            }
627                            LocalInfo::BlockTailTemp(_) | LocalInfo::Boring
628                                if !local.source_info.span.overlaps(span) =>
629                            {
630                                err.span_label(
631                                    local.source_info.span,
632                                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this cannot be {0}", acted_on))
    })format!("this cannot be {acted_on}"),
633                                );
634                            }
635                            _ => {}
636                        }
637                    }
638                }
639            }
640
641            PlaceRef { local, .. } => {
642                let local = &self.body.local_decls[local];
643                if !local.source_info.span.overlaps(span) {
644                    err.span_label(local.source_info.span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this cannot be {0}", acted_on))
    })format!("this cannot be {acted_on}"));
645                }
646                err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
647            }
648        }
649
650        if let Some(span) = mut_error {
651            self.buffer_mut_error(span, err, count);
652        } else {
653            self.buffer_error(err);
654        }
655    }
656
657    /// Suggest `map[k] = v` => `map.insert(k, v)` and the like.
658    fn suggest_map_index_mut_alternatives(&self, ty: Ty<'tcx>, err: &mut Diag<'infcx>, span: Span) {
659        let Some(adt) = ty.ty_adt_def() else { return };
660        let did = adt.did();
661        if self.infcx.tcx.is_diagnostic_item(sym::HashMap, did)
662            || self.infcx.tcx.is_diagnostic_item(sym::BTreeMap, did)
663        {
664            /// Walks through the HIR, looking for the corresponding span for this error.
665            /// When it finds it, see if it corresponds to assignment operator whose LHS
666            /// is an index expr.
667            struct SuggestIndexOperatorAlternativeVisitor<'a, 'infcx, 'tcx> {
668                assign_span: Span,
669                err: &'a mut Diag<'infcx>,
670                ty: Ty<'tcx>,
671                suggested: bool,
672                infcx: &'a rustc_infer::infer::InferCtxt<'tcx>,
673            }
674
675            impl<'a, 'infcx, 'tcx> Visitor<'tcx> for SuggestIndexOperatorAlternativeVisitor<'a, 'infcx, 'tcx> {
676                fn visit_stmt(&mut self, stmt: &'tcx hir::Stmt<'tcx>) {
677                    hir::intravisit::walk_stmt(self, stmt);
678                    let expr = match stmt.kind {
679                        hir::StmtKind::Semi(expr) | hir::StmtKind::Expr(expr) => expr,
680                        hir::StmtKind::Let(hir::LetStmt { init: Some(expr), .. }) => expr,
681                        _ => {
682                            return;
683                        }
684                    };
685
686                    // Because of TypeChecking and indexing, we know: index is &Q
687                    // with K: Eq + Hash + Borrow<Q>,
688                    // with Q: Eq + Hash + ?Sized,
689                    //
690                    // which fulfill the requirements of `get_mut`. If Q=K or Q=&{n}K, the requirements
691                    // of `entry` and `insert` are fulfilled too after dereferencing. If K is not
692                    // copy, a subsequent `clone` call may be needed.
693
694                    /// Taken straight from https://doc.rust-lang.org/nightly/nightly-rustc/clippy_utils/fn.peel_hir_ty_refs.html
695                    /// Adapted to mid using https://doc.rust-lang.org/nightly/nightly-rustc/rustc_middle/ty/struct.Ty.html#method.peel_refs
696                    /// Simplified to counting only
697                    /// Peels off all references on the type. Returns the number of references
698                    /// removed.
699                    fn count_ty_refs<'tcx>(mut ty: Ty<'tcx>) -> usize {
700                        let mut count = 0;
701                        while let ty::Ref(_, inner_ty, _) = ty.kind() {
702                            ty = *inner_ty;
703                            count += 1;
704                        }
705                        count
706                    }
707
708                    /// Try to strip `n` `&` reference from an expression.
709                    /// If the expression does not have enough leading `&`, return an Error
710                    /// containing a count of the successfully stripped ones and the stripped
711                    /// expression.
712                    fn strip_n_refs<'a, 'b>(
713                        mut expr: &'a Expr<'b>,
714                        n: usize,
715                    ) -> Result<&'a Expr<'b>, (usize, &'a Expr<'b>)> {
716                        for count in 0..n {
717                            match expr {
718                                Expr {
719                                    kind: ExprKind::AddrOf(hir::BorrowKind::Ref, _, inner),
720                                    ..
721                                } => expr = inner,
722                                _ => return Err((count, expr)),
723                            }
724                        }
725                        Ok(expr)
726                    }
727
728                    // we know ty is a map, with a key type at walk distance 2.
729                    let key_ty = self.ty.walk().nth(1).unwrap().expect_ty();
730
731                    if let hir::ExprKind::Assign(place, rv, _sp) = expr.kind
732                        && let hir::ExprKind::Index(val, index, _) = place.kind
733                        && (expr.span == self.assign_span || place.span == self.assign_span)
734                    {
735                        // val[index] = rv;
736                        let index_ty =
737                            self.infcx.tcx.typeck(val.hir_id.owner.def_id).expr_ty(index);
738
739                        let (borrowed_prefix, borrowed_index);
740
741                        // only suggest `insert` and `entry` if index is of type K or &{n}K or *{n}K (when there is a Borrow impl for this case).
742                        // We use `peel_refs` because borrow lifetimes may differ in both index and
743                        // key. I.e, if they are of the same base type:
744                        if index_ty.peel_refs() == key_ty.peel_refs() {
745                            let (index_refs, key_refs) =
746                                (count_ty_refs(index_ty), count_ty_refs(key_ty));
747
748                            let (deref_prefix, deref_index) = if index_refs >= key_refs {
749                                // index is &{n}K
750                                strip_n_refs(index, index_refs - key_refs)
751                                    .map(|val| ("".to_string(), val))
752                                    .unwrap_or_else(|(depth, val)| {
753                                        (
754                                            if key_refs == 0 {
755                                                "*".repeat(
756                                                    (index_refs-key_refs).checked_sub(depth).expect("return depth from strip_n_refs should be smaller than the input")
757                                                )
758                                            } else {
759                                                String::new() //if key K is a ref, autoderef finish this for us.
760                                            },
761                                            val,
762                                        )
763                                    })
764                            } else {
765                                // in this case the minimal ref addition works for all subcases
766                                ("&".repeat(key_refs - index_refs), index)
767                            };
768
769                            self.err.multipart_suggestion(
770                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("use `.insert()` to insert a value into a `{0}`",
                self.ty))
    })format!("use `.insert()` to insert a value into a `{}`", self.ty),
771                                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(val.span.shrink_to_hi().with_hi(deref_index.span.lo()),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!(".insert({0}",
                                    deref_prefix))
                        })),
                (deref_index.span.shrink_to_hi().with_hi(rv.span.lo()),
                    ", ".to_string()),
                (rv.span.shrink_to_hi(), ")".to_string())]))vec![
772                                    // val.insert({deref_prefix}{deref_index}, rv);
773                                    (
774                                        val.span.shrink_to_hi().with_hi(deref_index.span.lo()),
775                                        format!(".insert({deref_prefix}"),
776                                    ),
777                                    (
778                                        deref_index.span.shrink_to_hi().with_hi(rv.span.lo()),
779                                        ", ".to_string(),
780                                    ),
781                                    (rv.span.shrink_to_hi(), ")".to_string()),
782                                ],
783                                Applicability::MaybeIncorrect,
784                            );
785                            self.err.multipart_suggestion(
786                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("use the entry API to modify a `{0}` for more flexibility",
                self.ty))
    })format!(
787                                    "use the entry API to modify a `{}` for more flexibility",
788                                    self.ty
789                                ),
790                                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(val.span.shrink_to_lo(), "let val = ".to_string()),
                (val.span.shrink_to_hi().with_hi(deref_index.span.lo()),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!(".entry({0}",
                                    deref_prefix))
                        })),
                (deref_index.span.shrink_to_hi().with_hi(rv.span.lo()),
                    ").insert_entry(".to_string()),
                (rv.span.shrink_to_hi(), ")".to_string())]))vec![
791                                    // let x = val.entry({deref_prefix}{deref_index}).insert_entry(rv);
792                                    (val.span.shrink_to_lo(), "let val = ".to_string()),
793                                    (
794                                        val.span.shrink_to_hi().with_hi(deref_index.span.lo()),
795                                        format!(".entry({deref_prefix}"),
796                                    ),
797                                    (
798                                        deref_index.span.shrink_to_hi().with_hi(rv.span.lo()),
799                                        ").insert_entry(".to_string(),
800                                    ),
801                                    (rv.span.shrink_to_hi(), ")".to_string()),
802                                ],
803                                Applicability::MaybeIncorrect,
804                            );
805
806                            // we can make the next suggestions nicer by stripping as many leading `&` as
807                            // we can, autoderef will do the rest
808                            (borrowed_prefix, borrowed_index) = (
809                                String::new(),
810                                if index_refs > key_refs {
811                                    strip_n_refs(index, index_refs - key_refs - 1)
812                                        .unwrap_or_else(|(_depth, val)| val)
813                                    // even if we tried to strip more, we can stop there thanks to autoderef
814                                } else {
815                                    // when the diff is negative or zero, we already are in the index=&Q case.
816                                    index
817                                },
818                            );
819                        } else {
820                            (borrowed_prefix, borrowed_index) = (String::new(), index)
821                        }
822                        // in all cases, suggest get_mut because K:Borrow<K> or Q:Borrow<K> as a
823                        // requirement of indexing.
824                        self.err.multipart_suggestion(
825                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("use `.get_mut()` to modify an existing key in a `{0}`",
                self.ty))
    })format!(
826                                "use `.get_mut()` to modify an existing key in a `{}`",
827                                self.ty,
828                            ),
829                            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(val.span.shrink_to_lo(), "if let Some(val) = ".to_string()),
                (val.span.shrink_to_hi().with_hi(borrowed_index.span.lo()),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!(".get_mut({0}",
                                    borrowed_prefix))
                        })),
                (borrowed_index.span.shrink_to_hi().with_hi(place.span.hi()),
                    ") { *val".to_string()),
                (rv.span.shrink_to_hi(), "; }".to_string())]))vec![
830                                // if let Some(v) = val.get_mut({borrowed_prefix}{borrowed_index}) { *v = rv; }
831                                (val.span.shrink_to_lo(), "if let Some(val) = ".to_string()),
832                                (
833                                    val.span.shrink_to_hi().with_hi(borrowed_index.span.lo()),
834                                    format!(".get_mut({borrowed_prefix}"),
835                                ),
836                                (
837                                    borrowed_index.span.shrink_to_hi().with_hi(place.span.hi()),
838                                    ") { *val".to_string(),
839                                ),
840                                (rv.span.shrink_to_hi(), "; }".to_string()),
841                            ],
842                            Applicability::MaybeIncorrect,
843                        );
844
845                        self.suggested = true;
846                    } else if let hir::ExprKind::MethodCall(_path, receiver, _, sp) = expr.kind
847                        && let hir::ExprKind::Index(val, index, _) = receiver.kind
848                        && receiver.span == self.assign_span
849                    {
850                        // val[index].path(args..);
851                        self.err.multipart_suggestion(
852                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("to modify a `{0}` use `.get_mut()`",
                self.ty))
    })format!("to modify a `{}` use `.get_mut()`", self.ty),
853                            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(val.span.shrink_to_lo(), "if let Some(val) = ".to_string()),
                (val.span.shrink_to_hi().with_hi(index.span.lo()),
                    ".get_mut(".to_string()),
                (index.span.shrink_to_hi().with_hi(receiver.span.hi()),
                    ") { val".to_string()),
                (sp.shrink_to_hi(), "; }".to_string())]))vec![
854                                (val.span.shrink_to_lo(), "if let Some(val) = ".to_string()),
855                                (
856                                    val.span.shrink_to_hi().with_hi(index.span.lo()),
857                                    ".get_mut(".to_string(),
858                                ),
859                                (
860                                    index.span.shrink_to_hi().with_hi(receiver.span.hi()),
861                                    ") { val".to_string(),
862                                ),
863                                (sp.shrink_to_hi(), "; }".to_string()),
864                            ],
865                            Applicability::MachineApplicable,
866                        );
867                        self.suggested = true;
868                    }
869                }
870            }
871            let def_id = self.body.source.def_id();
872            let Some(local_def_id) = def_id.as_local() else { return };
873            let Some(body) = self.infcx.tcx.hir_maybe_body_owned_by(local_def_id) else { return };
874
875            let mut v = SuggestIndexOperatorAlternativeVisitor {
876                assign_span: span,
877                err,
878                ty,
879                suggested: false,
880                infcx: self.infcx,
881            };
882            v.visit_body(&body);
883            if !v.suggested {
884                err.help(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("to modify a `{0}`, use `.get_mut()`, `.insert()` or the entry API",
                ty))
    })format!(
885                    "to modify a `{ty}`, use `.get_mut()`, `.insert()` or the entry API",
886                ));
887            }
888        }
889    }
890
891    /// User cannot make signature of a trait mutable without changing the
892    /// trait. So we find if this error belongs to a trait and if so we move
893    /// suggestion to the trait or disable it if it is out of scope of this crate
894    ///
895    /// The returned values are:
896    ///  - is the current item an assoc `fn` of an impl that corresponds to a trait def? if so, we
897    ///    have to suggest changing both the impl `fn` arg and the trait `fn` arg
898    ///  - is the trait from the local crate? If not, we can't suggest changing signatures
899    ///  - `Span` of the argument in the trait definition
900    fn is_error_in_trait(&self, local: Local) -> (bool, bool, Option<Span>) {
901        let tcx = self.infcx.tcx;
902        if self.body.local_kind(local) != LocalKind::Arg {
903            return (false, false, None);
904        }
905        let my_def = self.body.source.def_id();
906        let Some(td) = tcx.trait_impl_of_assoc(my_def).map(|id| self.infcx.tcx.impl_trait_id(id))
907        else {
908            return (false, false, None);
909        };
910
911        let implemented_trait_item = self.infcx.tcx.trait_item_of(my_def);
912
913        (
914            true,
915            td.is_local(),
916            implemented_trait_item.and_then(|f_in_trait| {
917                let f_in_trait = f_in_trait.as_local()?;
918                if let Node::TraitItem(ti) = self.infcx.tcx.hir_node_by_def_id(f_in_trait)
919                    && let hir::TraitItemKind::Fn(sig, _) = ti.kind
920                    && let Some(ty) = sig.decl.inputs.get(local.index() - 1)
921                    && let hir::TyKind::Ref(_, mut_ty) = ty.kind
922                    && let hir::Mutability::Not = mut_ty.mutbl
923                    && sig.decl.implicit_self().has_implicit_self()
924                {
925                    Some(ty.span)
926                } else {
927                    None
928                }
929            }),
930        )
931    }
932
933    fn construct_mut_suggestion_for_local_binding_patterns(
934        &self,
935        err: &mut Diag<'_>,
936        local: Local,
937    ) {
938        let local_decl = &self.body.local_decls[local];
939        {
    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/mutability_errors.rs:939",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(939u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("local_decl: {0:?}",
                                                    local_decl) as &dyn Value))])
            });
    } else { ; }
};debug!("local_decl: {:?}", local_decl);
940        let pat_span = match *local_decl.local_info() {
941            LocalInfo::User(BindingForm::Var(mir::VarBindingForm {
942                binding_mode: BindingMode(ByRef::No, Mutability::Not),
943                opt_ty_info: _,
944                opt_match_place: _,
945                pat_span,
946                introductions: _,
947            })) => pat_span,
948            _ => local_decl.source_info.span,
949        };
950
951        // With ref-binding patterns, the mutability suggestion has to apply to
952        // the binding, not the reference (which would be a type error):
953        //
954        // `let &b = a;` -> `let &(mut b) = a;`
955        // or
956        // `fn foo(&x: &i32)` -> `fn foo(&(mut x): &i32)`
957        let def_id = self.body.source.def_id();
958        if let Some(local_def_id) = def_id.as_local()
959            && let Some(body) = self.infcx.tcx.hir_maybe_body_owned_by(local_def_id)
960            && let Some(hir_id) = (BindingFinder { span: pat_span }).visit_body(&body).break_value()
961            && let node = self.infcx.tcx.hir_node(hir_id)
962            && let hir::Node::LetStmt(hir::LetStmt {
963                pat: hir::Pat { kind: hir::PatKind::Ref(_, _, _), .. },
964                ..
965            })
966            | hir::Node::Param(Param {
967                pat: hir::Pat { kind: hir::PatKind::Ref(_, _, _), .. },
968                ..
969            }) = node
970        {
971            err.multipart_suggestion(
972                "consider changing this to be mutable",
973                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(pat_span.until(local_decl.source_info.span), "&(mut ".to_string()),
                (local_decl.source_info.span.shrink_to_hi().with_hi(pat_span.hi()),
                    ")".to_string())]))vec![
974                    (pat_span.until(local_decl.source_info.span), "&(mut ".to_string()),
975                    (
976                        local_decl.source_info.span.shrink_to_hi().with_hi(pat_span.hi()),
977                        ")".to_string(),
978                    ),
979                ],
980                Applicability::MachineApplicable,
981            );
982            return;
983        }
984
985        err.span_suggestion_verbose(
986            local_decl.source_info.span.shrink_to_lo(),
987            "consider changing this to be mutable",
988            "mut ",
989            Applicability::MachineApplicable,
990        );
991    }
992
993    // Point to span of upvar making closure call that requires a mutable borrow
994    fn show_mutating_upvar(
995        &self,
996        tcx: TyCtxt<'_>,
997        closure_local_def_id: hir::def_id::LocalDefId,
998        the_place_err: PlaceRef<'tcx>,
999        err: &mut Diag<'_>,
1000    ) {
1001        let tables = tcx.typeck(closure_local_def_id);
1002        if let Some((span, closure_kind_origin)) = tcx.closure_kind_origin(closure_local_def_id) {
1003            let reason = if let PlaceBase::Upvar(upvar_id) = closure_kind_origin.base {
1004                let upvar = ty::place_to_string_for_capture(tcx, closure_kind_origin);
1005                let root_hir_id = upvar_id.var_path.hir_id;
1006                // We have an origin for this closure kind starting at this root variable so it's
1007                // safe to unwrap here.
1008                let captured_places =
1009                    tables.closure_min_captures[&closure_local_def_id].get(&root_hir_id).unwrap();
1010
1011                let origin_projection = closure_kind_origin
1012                    .projections
1013                    .iter()
1014                    .map(|proj| proj.kind)
1015                    .collect::<Vec<_>>();
1016                let mut capture_reason = String::new();
1017                for captured_place in captured_places {
1018                    let captured_place_kinds = captured_place
1019                        .place
1020                        .projections
1021                        .iter()
1022                        .map(|proj| proj.kind)
1023                        .collect::<Vec<_>>();
1024                    if rustc_middle::ty::is_ancestor_or_same_capture(
1025                        &captured_place_kinds,
1026                        &origin_projection,
1027                    ) {
1028                        match captured_place.info.capture_kind {
1029                            ty::UpvarCapture::ByRef(
1030                                ty::BorrowKind::Mutable | ty::BorrowKind::UniqueImmutable,
1031                            ) => {
1032                                capture_reason = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("mutable borrow of `{0}`", upvar))
    })format!("mutable borrow of `{upvar}`");
1033                            }
1034                            ty::UpvarCapture::ByValue | ty::UpvarCapture::ByUse => {
1035                                capture_reason = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("possible mutation of `{0}`",
                upvar))
    })format!("possible mutation of `{upvar}`");
1036                            }
1037                            _ => ::rustc_middle::util::bug::bug_fmt(format_args!("upvar `{0}` borrowed, but not mutably",
        upvar))bug!("upvar `{upvar}` borrowed, but not mutably"),
1038                        }
1039                        break;
1040                    }
1041                }
1042                if capture_reason.is_empty() {
1043                    ::rustc_middle::util::bug::bug_fmt(format_args!("upvar `{0}` borrowed, but cannot find reason",
        upvar));bug!("upvar `{upvar}` borrowed, but cannot find reason");
1044                }
1045                capture_reason
1046            } else {
1047                ::rustc_middle::util::bug::bug_fmt(format_args!("not an upvar"))bug!("not an upvar")
1048            };
1049            // Sometimes we deliberately don't store the name of a place when coming from a macro in
1050            // another crate. We generally want to limit those diagnostics a little, to hide
1051            // implementation details (such as those from pin!() or format!()). In that case show a
1052            // slightly different error message, or none at all if something else happened. In other
1053            // cases the message is likely not useful.
1054            if let Some(place_name) = self.describe_place(the_place_err) {
1055                err.span_label(
1056                    *span,
1057                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("calling `{0}` requires mutable binding due to {1}",
                place_name, reason))
    })format!("calling `{place_name}` requires mutable binding due to {reason}"),
1058                );
1059            } else if span.from_expansion() {
1060                err.span_label(
1061                    *span,
1062                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("a call in this macro requires a mutable binding due to {0}",
                reason))
    })format!("a call in this macro requires a mutable binding due to {reason}",),
1063                );
1064            }
1065        }
1066    }
1067
1068    // Attempt to search similar mutable associated items for suggestion.
1069    // In the future, attempt in all path but initially for RHS of for_loop
1070    fn suggest_similar_mut_method_for_for_loop(&self, err: &mut Diag<'_>, span: Span) {
1071        use hir::ExprKind::{AddrOf, Block, Call, MethodCall};
1072        use hir::{BorrowKind, Expr};
1073
1074        let tcx = self.infcx.tcx;
1075        struct Finder {
1076            span: Span,
1077        }
1078
1079        impl<'tcx> Visitor<'tcx> for Finder {
1080            type Result = ControlFlow<&'tcx Expr<'tcx>>;
1081            fn visit_expr(&mut self, e: &'tcx hir::Expr<'tcx>) -> Self::Result {
1082                if e.span == self.span {
1083                    ControlFlow::Break(e)
1084                } else {
1085                    hir::intravisit::walk_expr(self, e)
1086                }
1087            }
1088        }
1089        if let Some(body) = tcx.hir_maybe_body_owned_by(self.mir_def_id())
1090            && let Block(block, _) = body.value.kind
1091        {
1092            // `span` corresponds to the expression being iterated, find the `for`-loop desugared
1093            // expression with that span in order to identify potential fixes when encountering a
1094            // read-only iterator that should be mutable.
1095            if let ControlFlow::Break(expr) = (Finder { span }).visit_block(block)
1096                && let Call(_, [expr]) = expr.kind
1097            {
1098                match expr.kind {
1099                    MethodCall(path_segment, _, _, span) => {
1100                        // We have `for _ in iter.read_only_iter()`, try to
1101                        // suggest `for _ in iter.mutable_iter()` instead.
1102                        let opt_suggestions = tcx
1103                            .typeck(path_segment.hir_id.owner.def_id)
1104                            .type_dependent_def_id(expr.hir_id)
1105                            .and_then(|def_id| tcx.impl_of_assoc(def_id))
1106                            .map(|def_id| tcx.associated_items(def_id))
1107                            .map(|assoc_items| {
1108                                assoc_items
1109                                    .in_definition_order()
1110                                    .map(|assoc_item_def| assoc_item_def.ident(tcx))
1111                                    .filter(|&ident| {
1112                                        let original_method_ident = path_segment.ident;
1113                                        original_method_ident != ident
1114                                            && ident.as_str().starts_with(
1115                                                &original_method_ident.name.to_string(),
1116                                            )
1117                                    })
1118                                    .map(|ident| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}()", ident))
    })format!("{ident}()"))
1119                                    .peekable()
1120                            });
1121
1122                        if let Some(mut suggestions) = opt_suggestions
1123                            && suggestions.peek().is_some()
1124                        {
1125                            err.span_suggestions(
1126                                span,
1127                                "use mutable method",
1128                                suggestions,
1129                                Applicability::MaybeIncorrect,
1130                            );
1131                        }
1132                    }
1133                    AddrOf(BorrowKind::Ref, Mutability::Not, expr) => {
1134                        // We have `for _ in &i`, suggest `for _ in &mut i`.
1135                        err.span_suggestion_verbose(
1136                            expr.span.shrink_to_lo(),
1137                            "use a mutable iterator instead",
1138                            "mut ",
1139                            Applicability::MachineApplicable,
1140                        );
1141                    }
1142                    _ => {}
1143                }
1144            }
1145        }
1146    }
1147
1148    /// When modifying a binding from inside of an `Fn` closure, point at the binding definition.
1149    fn point_at_binding_outside_closure(
1150        &self,
1151        err: &mut Diag<'_>,
1152        local: Local,
1153        access_place: Place<'tcx>,
1154    ) {
1155        let place = access_place.as_ref();
1156        for (index, elem) in place.projection.into_iter().enumerate() {
1157            if let ProjectionElem::Deref = elem {
1158                if index == 0 {
1159                    if self.body.local_decls[local].is_ref_for_guard() {
1160                        continue;
1161                    }
1162                    if let LocalInfo::StaticRef { .. } = *self.body.local_decls[local].local_info()
1163                    {
1164                        continue;
1165                    }
1166                }
1167                if let Some(field) = self.is_upvar_field_projection(PlaceRef {
1168                    local,
1169                    projection: place.projection.split_at(index + 1).0,
1170                }) {
1171                    let var_index = field.index();
1172                    let upvar = self.upvars[var_index];
1173                    if let Some(hir_id) = upvar.info.capture_kind_expr_id {
1174                        let node = self.infcx.tcx.hir_node(hir_id);
1175                        if let hir::Node::Expr(expr) = node
1176                            && let hir::ExprKind::Path(hir::QPath::Resolved(None, path)) = expr.kind
1177                            && let hir::def::Res::Local(hir_id) = path.res
1178                            && let hir::Node::Pat(pat) = self.infcx.tcx.hir_node(hir_id)
1179                        {
1180                            let name = upvar.to_string(self.infcx.tcx);
1181                            err.span_label(
1182                                pat.span,
1183                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` declared here, outside the closure",
                name))
    })format!("`{name}` declared here, outside the closure"),
1184                            );
1185                            break;
1186                        }
1187                    }
1188                }
1189            }
1190        }
1191    }
1192    /// Targeted error when encountering an `FnMut` closure where an `Fn` closure was expected.
1193    fn expected_fn_found_fn_mut_call(&self, err: &mut Diag<'_>, sp: Span, act: &str) {
1194        err.span_label(sp, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot {0}", act))
    })format!("cannot {act}"));
1195
1196        let tcx = self.infcx.tcx;
1197        let closure_id = self.mir_hir_id();
1198        let closure_span = tcx.def_span(self.mir_def_id());
1199        let fn_call_id = tcx.parent_hir_id(closure_id);
1200        let node = tcx.hir_node(fn_call_id);
1201        let def_id = tcx.hir_enclosing_body_owner(fn_call_id);
1202        let mut look_at_return = true;
1203
1204        err.span_label(closure_span, "in this closure");
1205        let closure_arg_has_fn_trait_bound =
1206            |callee_def_id, input_index, generic_args: ty::GenericArgsRef<'tcx>| {
1207                let sig = tcx.fn_sig(callee_def_id).instantiate(tcx, generic_args).skip_binder();
1208                let Some(input_ty): Option<Ty<'tcx>> = sig.inputs().get(input_index).copied()
1209                else {
1210                    return false;
1211                };
1212
1213                tcx.predicates_of(callee_def_id)
1214                    .instantiate(tcx, generic_args)
1215                    .predicates
1216                    .iter()
1217                    .any(|predicate| {
1218                        predicate.as_trait_clause().is_some_and(|trait_pred| {
1219                            trait_pred.polarity() == ty::PredicatePolarity::Positive
1220                                && tcx.fn_trait_kind_from_def_id(trait_pred.def_id())
1221                                    == Some(ty::ClosureKind::Fn)
1222                                && trait_pred.self_ty().skip_binder().peel_refs()
1223                                    == input_ty.peel_refs()
1224                        })
1225                    })
1226            };
1227
1228        // If the HIR node is a function or method call, get the DefId
1229        // of the callee function or method, the span, and argument info for the call expr.
1230        let get_call_details =
1231            || -> Option<(DefId, Span, usize, usize, ty::GenericArgsRef<'tcx>)> {
1232                let hir::Node::Expr(hir::Expr { hir_id, kind, .. }) = node else {
1233                    return None;
1234                };
1235
1236                let typeck_results = tcx.typeck(def_id);
1237
1238                match kind {
1239                    hir::ExprKind::Call(expr, args) => {
1240                        if let Some(ty::FnDef(def_id, generic_args)) =
1241                            typeck_results.node_type_opt(expr.hir_id).as_ref().map(|ty| ty.kind())
1242                        {
1243                            let arg_pos = args.iter().position(|arg| arg.hir_id == closure_id)?;
1244                            Some((*def_id, expr.span, arg_pos, arg_pos, generic_args))
1245                        } else {
1246                            None
1247                        }
1248                    }
1249                    hir::ExprKind::MethodCall(_, _, args, span) => {
1250                        let arg_pos = args.iter().position(|arg| arg.hir_id == closure_id)?;
1251                        let def_id = typeck_results.type_dependent_def_id(*hir_id)?;
1252                        let generic_args = typeck_results.node_args_opt(*hir_id)?;
1253                        Some((def_id, *span, arg_pos, arg_pos + 1, generic_args))
1254                    }
1255                    _ => None,
1256                }
1257            };
1258
1259        // If we can detect the expression to be a function or method call where the closure was
1260        // an argument, we point at the function or method definition argument...
1261        if let Some((callee_def_id, call_span, arg_pos, input_index, generic_args)) =
1262            get_call_details()
1263        {
1264            let arg = match tcx.hir_get_if_local(callee_def_id) {
1265                Some(
1266                    hir::Node::Item(hir::Item {
1267                        kind: hir::ItemKind::Fn { ident, sig, .. }, ..
1268                    })
1269                    | hir::Node::TraitItem(hir::TraitItem {
1270                        ident,
1271                        kind: hir::TraitItemKind::Fn(sig, _),
1272                        ..
1273                    })
1274                    | hir::Node::ImplItem(hir::ImplItem {
1275                        ident,
1276                        kind: hir::ImplItemKind::Fn(sig, _),
1277                        ..
1278                    }),
1279                ) => Some(
1280                    sig.decl
1281                        .inputs
1282                        .get(
1283                            arg_pos
1284                                + if sig.decl.implicit_self().has_implicit_self() { 1 } else { 0 },
1285                        )
1286                        .map(|arg| arg.span)
1287                        .unwrap_or(ident.span),
1288                ),
1289                _ => None,
1290            };
1291            if let Some(span) = arg {
1292                err.span_label(span, "change this to accept `FnMut` instead of `Fn`");
1293                err.span_label(call_span, "expects `Fn` instead of `FnMut`");
1294                look_at_return = false;
1295            } else if closure_arg_has_fn_trait_bound(callee_def_id, input_index, generic_args) {
1296                // The callee is not local, so we cannot point at its argument declaration, but we
1297                // can still explain that this call site expects an `Fn` closure. Avoid falling
1298                // through to the enclosing function's return type, which is misleading in cases
1299                // like `flat_map(|_| external::map(|_| ...))`.
1300                err.span_label(call_span, "expects `Fn` instead of `FnMut`");
1301                look_at_return = false;
1302            }
1303        }
1304
1305        if look_at_return && tcx.hir_get_fn_id_for_return_block(closure_id).is_some() {
1306            // ...otherwise we are probably in the tail expression of the function, point at the
1307            // return type.
1308            match tcx.hir_node_by_def_id(tcx.hir_get_parent_item(fn_call_id).def_id) {
1309                hir::Node::Item(hir::Item {
1310                    kind: hir::ItemKind::Fn { ident, sig, .. }, ..
1311                })
1312                | hir::Node::TraitItem(hir::TraitItem {
1313                    ident,
1314                    kind: hir::TraitItemKind::Fn(sig, _),
1315                    ..
1316                })
1317                | hir::Node::ImplItem(hir::ImplItem {
1318                    ident,
1319                    kind: hir::ImplItemKind::Fn(sig, _),
1320                    ..
1321                }) => {
1322                    err.span_label(ident.span, "");
1323                    err.span_label(
1324                        sig.decl.output.span(),
1325                        "change this to return `FnMut` instead of `Fn`",
1326                    );
1327                }
1328                _ => {}
1329            }
1330        }
1331    }
1332
1333    fn suggest_using_iter_mut(&self, err: &mut Diag<'_>) {
1334        let source = self.body.source;
1335        if let InstanceKind::Item(def_id) = source.instance
1336            && let Some(Node::Expr(hir::Expr { hir_id, kind, .. })) =
1337                self.infcx.tcx.hir_get_if_local(def_id)
1338            && let ExprKind::Closure(hir::Closure { kind: hir::ClosureKind::Closure, .. }) = kind
1339            && let Node::Expr(expr) = self.infcx.tcx.parent_hir_node(*hir_id)
1340        {
1341            let mut cur_expr = expr;
1342            while let ExprKind::MethodCall(path_segment, recv, _, _) = cur_expr.kind {
1343                if path_segment.ident.name == sym::iter {
1344                    // Check that the type has an `iter_mut` method.
1345                    let res = self
1346                        .infcx
1347                        .tcx
1348                        .typeck(path_segment.hir_id.owner.def_id)
1349                        .type_dependent_def_id(cur_expr.hir_id)
1350                        .and_then(|def_id| self.infcx.tcx.impl_of_assoc(def_id))
1351                        .map(|def_id| self.infcx.tcx.associated_items(def_id))
1352                        .map(|assoc_items| {
1353                            assoc_items.filter_by_name_unhygienic(sym::iter_mut).peekable()
1354                        });
1355
1356                    if let Some(mut res) = res
1357                        && res.peek().is_some()
1358                    {
1359                        err.span_suggestion_verbose(
1360                            path_segment.ident.span,
1361                            "you may want to use `iter_mut` here",
1362                            "iter_mut",
1363                            Applicability::MaybeIncorrect,
1364                        );
1365                    }
1366                    break;
1367                } else {
1368                    cur_expr = recv;
1369                }
1370            }
1371        }
1372    }
1373
1374    fn suggest_make_local_mut(&self, err: &mut Diag<'_>, local: Local, name: Symbol) {
1375        let local_decl = &self.body.local_decls[local];
1376
1377        let (pointer_sigil, pointer_desc) =
1378            if local_decl.ty.is_ref() { ("&", "reference") } else { ("*const", "pointer") };
1379
1380        let (is_trait_sig, is_local, local_trait) = self.is_error_in_trait(local);
1381
1382        if is_trait_sig && !is_local {
1383            // Do not suggest changing the signature when the trait comes from another crate.
1384            err.span_label(
1385                local_decl.source_info.span,
1386                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this is an immutable {0}",
                pointer_desc))
    })format!("this is an immutable {pointer_desc}"),
1387            );
1388            return;
1389        }
1390
1391        // Do not suggest changing type if that is not under user control.
1392        if self.is_closure_arg_with_non_locally_decided_type(local) {
1393            return;
1394        }
1395
1396        let decl_span = local_decl.source_info.span;
1397
1398        let (amp_mut_sugg, local_var_ty_info) = match *local_decl.local_info() {
1399            LocalInfo::User(mir::BindingForm::ImplicitSelf(_)) => {
1400                let (span, suggestion) = suggest_ampmut_self(self.infcx.tcx, decl_span);
1401                let additional = local_trait.map(|span| suggest_ampmut_self(self.infcx.tcx, span));
1402                (AmpMutSugg::Type { span, suggestion, additional }, None)
1403            }
1404
1405            LocalInfo::User(mir::BindingForm::Var(mir::VarBindingForm {
1406                binding_mode: BindingMode(ByRef::No, _),
1407                opt_ty_info,
1408                ..
1409            })) => {
1410                // Check if the RHS is from desugaring.
1411                let first_assignment = find_assignments(&self.body, local).first().copied();
1412                let first_assignment_stmt = first_assignment
1413                    .and_then(|loc| self.body[loc.block].statements.get(loc.statement_index));
1414                {
    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/mutability_errors.rs:1414",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::TRACE,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(1414u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::tracing_core::field::FieldSet::new(&["first_assignment_stmt"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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(&debug(&first_assignment_stmt)
                                            as &dyn Value))])
            });
    } else { ; }
};trace!(?first_assignment_stmt);
1415                let opt_assignment_rhs_span =
1416                    first_assignment.map(|loc| self.body.source_info(loc).span);
1417                let mut source_span = opt_assignment_rhs_span;
1418                if let Some(mir::Statement {
1419                    source_info: _,
1420                    kind:
1421                        mir::StatementKind::Assign(box (
1422                            _,
1423                            mir::Rvalue::Use(mir::Operand::Copy(place), _),
1424                        )),
1425                    ..
1426                }) = first_assignment_stmt
1427                {
1428                    let local_span = self.body.local_decls[place.local].source_info.span;
1429                    // `&self` in async functions have a `desugaring_kind`, but the local we assign
1430                    // it with does not, so use the local_span for our checks later.
1431                    source_span = Some(local_span);
1432                    if let Some(DesugaringKind::ForLoop) = local_span.desugaring_kind() {
1433                        // On for loops, RHS points to the iterator part.
1434                        self.suggest_similar_mut_method_for_for_loop(err, local_span);
1435                        err.span_label(
1436                            local_span,
1437                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this iterator yields `{0}` {1}s",
                pointer_sigil, pointer_desc))
    })format!("this iterator yields `{pointer_sigil}` {pointer_desc}s",),
1438                        );
1439                        return;
1440                    }
1441                }
1442
1443                // Don't create labels for compiler-generated spans or spans not from users' code.
1444                if source_span.is_some_and(|s| {
1445                    s.desugaring_kind().is_some() || self.infcx.tcx.sess.source_map().is_imported(s)
1446                }) {
1447                    return;
1448                }
1449
1450                // This could be because we're in an `async fn`.
1451                if name == kw::SelfLower && opt_ty_info.is_none() {
1452                    let (span, suggestion) = suggest_ampmut_self(self.infcx.tcx, decl_span);
1453                    (AmpMutSugg::Type { span, suggestion, additional: None }, None)
1454                } else if let Some(sugg) =
1455                    suggest_ampmut(self.infcx, self.body(), first_assignment_stmt)
1456                {
1457                    (sugg, opt_ty_info)
1458                } else {
1459                    return;
1460                }
1461            }
1462
1463            LocalInfo::User(mir::BindingForm::Var(mir::VarBindingForm {
1464                binding_mode: BindingMode(ByRef::Yes(..), _),
1465                ..
1466            })) => {
1467                let pattern_span: Span = local_decl.source_info.span;
1468                let Some(span) = suggest_ref_mut(self.infcx.tcx, pattern_span) else {
1469                    return;
1470                };
1471                (AmpMutSugg::Type { span, suggestion: "mut ".to_owned(), additional: None }, None)
1472            }
1473
1474            _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
1475        };
1476
1477        let mut suggest = |suggs: Vec<_>, applicability, extra| {
1478            if suggs.iter().any(|(span, _)| self.infcx.tcx.sess.source_map().is_imported(*span)) {
1479                return;
1480            }
1481
1482            err.multipart_suggestion(
1483                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("consider changing this to be a mutable {1}{0}{2}",
                if is_trait_sig {
                    " in the `impl` method and the `trait` definition"
                } else { "" }, pointer_desc, extra))
    })format!(
1484                    "consider changing this to be a mutable {pointer_desc}{}{extra}",
1485                    if is_trait_sig {
1486                        " in the `impl` method and the `trait` definition"
1487                    } else {
1488                        ""
1489                    }
1490                ),
1491                suggs,
1492                applicability,
1493            );
1494        };
1495
1496        let (mut sugg, add_type_annotation_if_not_exists) = match amp_mut_sugg {
1497            AmpMutSugg::Type { span, suggestion, additional } => {
1498                let mut sugg = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(span, suggestion)]))vec![(span, suggestion)];
1499                sugg.extend(additional);
1500                suggest(sugg, Applicability::MachineApplicable, "");
1501                return;
1502            }
1503            AmpMutSugg::MapGetMut { span, suggestion } => {
1504                if self.infcx.tcx.sess.source_map().is_imported(span) {
1505                    return;
1506                }
1507                err.multipart_suggestion(
1508                    "consider using `get_mut`",
1509                    ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(span, suggestion)]))vec![(span, suggestion)],
1510                    Applicability::MaybeIncorrect,
1511                );
1512                return;
1513            }
1514            AmpMutSugg::Expr { span, suggestion } => {
1515                // `Expr` suggestions should change type annotations if they already exist (probably immut),
1516                // but do not add new type annotations.
1517                (::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(span, suggestion)]))vec![(span, suggestion)], false)
1518            }
1519            AmpMutSugg::ChangeBinding => (::alloc::vec::Vec::new()vec![], true),
1520        };
1521
1522        // Find a binding's type to make mutable.
1523        let (binding_exists, span) = match local_var_ty_info {
1524            // If this is a variable binding with an explicit type,
1525            // then we will suggest changing it to be mutable.
1526            // This is `Applicability::MachineApplicable`.
1527            Some(ty_span) => (true, ty_span),
1528
1529            // Otherwise, we'll suggest *adding* an annotated type, we'll suggest
1530            // the RHS's type for that.
1531            // This is `Applicability::HasPlaceholders`.
1532            None => (false, decl_span),
1533        };
1534
1535        if !binding_exists && !add_type_annotation_if_not_exists {
1536            suggest(sugg, Applicability::MachineApplicable, "");
1537            return;
1538        }
1539
1540        // If the binding already exists and is a reference with an explicit
1541        // lifetime, then we can suggest adding ` mut`. This is special-cased from
1542        // the path without an explicit lifetime.
1543        let (sugg_span, sugg_str, suggest_now) = if let Ok(src) = self.infcx.tcx.sess.source_map().span_to_snippet(span)
1544            && src.starts_with("&'")
1545            // Note that `&' a T` is invalid so this is correct.
1546            && let Some(ws_pos) = src.find(char::is_whitespace)
1547        {
1548            let span = span.with_lo(span.lo() + BytePos(ws_pos as u32)).shrink_to_lo();
1549            (span, " mut".to_owned(), true)
1550        // If there is already a binding, we modify it to be `mut`.
1551        } else if binding_exists {
1552            // Replace the sigil with the mutable version. We may be dealing
1553            // with parser recovery here and cannot assume the user actually
1554            // typed `&` or `*const`, so we compute the prefix from the snippet.
1555            let Ok(src) = self.infcx.tcx.sess.source_map().span_to_snippet(span) else {
1556                return;
1557            };
1558            let (prefix_len, replacement) = if local_decl.ty.is_ref() {
1559                (src.chars().next().map_or(0, char::len_utf8), "&mut ")
1560            } else {
1561                (src.find("const").map_or(1, |i| i + "const".len()), "*mut ")
1562            };
1563            let ws_len = src[prefix_len..].len() - src[prefix_len..].trim_start().len();
1564            let span = span.with_hi(span.lo() + BytePos((prefix_len + ws_len) as u32));
1565            (span, replacement.to_owned(), true)
1566        } else {
1567            // Otherwise, suggest that the user annotates the binding; We provide the
1568            // type of the local.
1569            let ty = local_decl.ty.builtin_deref(true).unwrap();
1570
1571            (span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}mut {1}",
                if local_decl.ty.is_ref() { "&" } else { "*" }, ty))
    })format!("{}mut {}", if local_decl.ty.is_ref() { "&" } else { "*" }, ty), false)
1572        };
1573
1574        if suggest_now {
1575            // Suggest changing `&x` to `&mut x` and changing `&T` to `&mut T` at the same time.
1576            let has_change = !sugg.is_empty();
1577            sugg.push((sugg_span, sugg_str));
1578            suggest(
1579                sugg,
1580                Applicability::MachineApplicable,
1581                // FIXME(fee1-dead) this somehow doesn't fire
1582                if has_change { " and changing the binding's type" } else { "" },
1583            );
1584            return;
1585        } else if !sugg.is_empty() {
1586            suggest(sugg, Applicability::MachineApplicable, "");
1587            return;
1588        }
1589
1590        let def_id = self.body.source.def_id();
1591        let hir_id = if let Some(local_def_id) = def_id.as_local()
1592            && let Some(body) = self.infcx.tcx.hir_maybe_body_owned_by(local_def_id)
1593        {
1594            BindingFinder { span: sugg_span }.visit_body(&body).break_value()
1595        } else {
1596            None
1597        };
1598        let node = hir_id.map(|hir_id| self.infcx.tcx.hir_node(hir_id));
1599
1600        let Some(hir::Node::LetStmt(local)) = node else {
1601            err.span_label(
1602                sugg_span,
1603                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("consider changing this binding\'s type to be: `{0}`",
                sugg_str))
    })format!("consider changing this binding's type to be: `{sugg_str}`"),
1604            );
1605            return;
1606        };
1607
1608        let tables = self.infcx.tcx.typeck(def_id.as_local().unwrap());
1609        if let Some(clone_trait) = self.infcx.tcx.lang_items().clone_trait()
1610            && let Some(expr) = local.init
1611            && let ty = tables.node_type_opt(expr.hir_id)
1612            && let Some(ty) = ty
1613            && let ty::Ref(..) = ty.kind()
1614        {
1615            match self
1616                .infcx
1617                .type_implements_trait_shallow(clone_trait, ty.peel_refs(), self.infcx.param_env)
1618                .as_deref()
1619            {
1620                Some([]) => {
1621                    // FIXME: This error message isn't useful, since we're just
1622                    // vaguely suggesting to clone a value that already
1623                    // implements `Clone`.
1624                    //
1625                    // A correct suggestion here would take into account the fact
1626                    // that inference may be affected by missing types on bindings,
1627                    // etc., to improve "tests/ui/borrowck/issue-91206.stderr", for
1628                    // example.
1629                }
1630                None => {
1631                    if let hir::ExprKind::MethodCall(segment, _rcvr, [], span) = expr.kind
1632                        && segment.ident.name == sym::clone
1633                    {
1634                        err.span_help(
1635                            span,
1636                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` doesn\'t implement `Clone`, so this call clones the reference `{1}`",
                ty.peel_refs(), ty))
    })format!(
1637                                "`{}` doesn't implement `Clone`, so this call clones \
1638                                             the reference `{ty}`",
1639                                ty.peel_refs(),
1640                            ),
1641                        );
1642                    }
1643                    // The type doesn't implement Clone.
1644                    let trait_ref = ty::Binder::dummy(ty::TraitRef::new(
1645                        self.infcx.tcx,
1646                        clone_trait,
1647                        [ty.peel_refs()],
1648                    ));
1649                    let obligation = traits::Obligation::new(
1650                        self.infcx.tcx,
1651                        traits::ObligationCause::dummy(),
1652                        self.infcx.param_env,
1653                        trait_ref,
1654                    );
1655                    self.infcx.err_ctxt().suggest_derive(
1656                        &obligation,
1657                        err,
1658                        trait_ref.upcast(self.infcx.tcx),
1659                    );
1660                }
1661                Some(errors) => {
1662                    if let hir::ExprKind::MethodCall(segment, _rcvr, [], span) = expr.kind
1663                        && segment.ident.name == sym::clone
1664                    {
1665                        err.span_help(
1666                            span,
1667                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` doesn\'t implement `Clone` because its implementations trait bounds could not be met, so this call clones the reference `{1}`",
                ty.peel_refs(), ty))
    })format!(
1668                                "`{}` doesn't implement `Clone` because its \
1669                                             implementations trait bounds could not be met, so \
1670                                             this call clones the reference `{ty}`",
1671                                ty.peel_refs(),
1672                            ),
1673                        );
1674                        err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the following trait bounds weren\'t met: {0}",
                errors.iter().map(|e|
                                e.obligation.predicate.to_string()).collect::<Vec<_>>().join("\n")))
    })format!(
1675                            "the following trait bounds weren't met: {}",
1676                            errors
1677                                .iter()
1678                                .map(|e| e.obligation.predicate.to_string())
1679                                .collect::<Vec<_>>()
1680                                .join("\n"),
1681                        ));
1682                    }
1683                    // The type doesn't implement Clone because of unmet obligations.
1684                    for error in errors {
1685                        if let traits::FulfillmentErrorCode::Select(
1686                            traits::SelectionError::Unimplemented,
1687                        ) = error.code
1688                            && let ty::PredicateKind::Clause(ty::ClauseKind::Trait(pred)) =
1689                                error.obligation.predicate.kind().skip_binder()
1690                        {
1691                            self.infcx.err_ctxt().suggest_derive(
1692                                &error.obligation,
1693                                err,
1694                                error.obligation.predicate.kind().rebind(pred),
1695                            );
1696                        }
1697                    }
1698                }
1699            }
1700        }
1701        let (changing, span, sugg) = match local.ty {
1702            Some(ty) => ("changing", ty.span, sugg_str),
1703            None => ("specifying", local.pat.span.shrink_to_hi(), ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(": {0}", sugg_str))
    })format!(": {sugg_str}")),
1704        };
1705        err.span_suggestion_verbose(
1706            span,
1707            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("consider {0} this binding\'s type",
                changing))
    })format!("consider {changing} this binding's type"),
1708            sugg,
1709            Applicability::HasPlaceholders,
1710        );
1711    }
1712
1713    /// Returns `true` if `local` is an argument in a closure passed to a
1714    /// function defined in another crate.
1715    ///
1716    /// For example, in the following code this function returns `true` for `x`
1717    /// since `Option::inspect()` is not defined in the current crate:
1718    ///
1719    /// ```text
1720    /// some_option.as_mut().inspect(|x| {
1721    /// ```
1722    fn is_closure_arg_with_non_locally_decided_type(&self, local: Local) -> bool {
1723        // We don't care about regular local variables, only args.
1724        if self.body.local_kind(local) != LocalKind::Arg {
1725            return false;
1726        }
1727
1728        // Make sure we are inside a closure.
1729        let InstanceKind::Item(body_def_id) = self.body.source.instance else {
1730            return false;
1731        };
1732        let Some(Node::Expr(hir::Expr { hir_id: body_hir_id, kind, .. })) =
1733            self.infcx.tcx.hir_get_if_local(body_def_id)
1734        else {
1735            return false;
1736        };
1737        let ExprKind::Closure(hir::Closure { kind: hir::ClosureKind::Closure, .. }) = kind else {
1738            return false;
1739        };
1740
1741        // Check if the method/function that our closure is passed to is defined
1742        // in another crate.
1743        let Node::Expr(closure_parent) = self.infcx.tcx.parent_hir_node(*body_hir_id) else {
1744            return false;
1745        };
1746        match closure_parent.kind {
1747            ExprKind::MethodCall(method, _, _, _) => self
1748                .infcx
1749                .tcx
1750                .typeck(method.hir_id.owner.def_id)
1751                .type_dependent_def_id(closure_parent.hir_id)
1752                .is_some_and(|def_id| !def_id.is_local()),
1753            ExprKind::Call(func, _) => self
1754                .infcx
1755                .tcx
1756                .typeck(func.hir_id.owner.def_id)
1757                .node_type_opt(func.hir_id)
1758                .and_then(|ty| match ty.kind() {
1759                    ty::FnDef(def_id, _) => Some(def_id),
1760                    _ => None,
1761                })
1762                .is_some_and(|def_id| !def_id.is_local()),
1763            _ => false,
1764        }
1765    }
1766}
1767
1768struct BindingFinder {
1769    span: Span,
1770}
1771
1772impl<'tcx> Visitor<'tcx> for BindingFinder {
1773    type Result = ControlFlow<hir::HirId>;
1774    fn visit_stmt(&mut self, s: &'tcx hir::Stmt<'tcx>) -> Self::Result {
1775        if let hir::StmtKind::Let(local) = s.kind
1776            && local.pat.span == self.span
1777        {
1778            ControlFlow::Break(local.hir_id)
1779        } else {
1780            hir::intravisit::walk_stmt(self, s)
1781        }
1782    }
1783
1784    fn visit_param(&mut self, param: &'tcx hir::Param<'tcx>) -> Self::Result {
1785        if let hir::Pat { kind: hir::PatKind::Ref(_, _, _), span, .. } = param.pat
1786            && *span == self.span
1787        {
1788            ControlFlow::Break(param.hir_id)
1789        } else {
1790            ControlFlow::Continue(())
1791        }
1792    }
1793}
1794
1795fn mut_borrow_of_mutable_ref(local_decl: &LocalDecl<'_>, local_name: Option<Symbol>) -> bool {
1796    {
    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/mutability_errors.rs:1796",
                        "rustc_borrowck::diagnostics::mutability_errors",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs"),
                        ::tracing_core::__macro_support::Option::Some(1796u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_borrowck::diagnostics::mutability_errors"),
                        ::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!("local_info: {0:?}, ty.kind(): {1:?}",
                                                    local_decl.local_info, local_decl.ty.kind()) as
                                            &dyn Value))])
            });
    } else { ; }
};debug!("local_info: {:?}, ty.kind(): {:?}", local_decl.local_info, local_decl.ty.kind());
1797
1798    match *local_decl.local_info() {
1799        // Check if mutably borrowing a mutable reference.
1800        LocalInfo::User(mir::BindingForm::Var(mir::VarBindingForm {
1801            binding_mode: BindingMode(ByRef::No, Mutability::Not),
1802            ..
1803        })) => #[allow(non_exhaustive_omitted_patterns)] match local_decl.ty.kind() {
    ty::Ref(_, _, hir::Mutability::Mut) => true,
    _ => false,
}matches!(local_decl.ty.kind(), ty::Ref(_, _, hir::Mutability::Mut)),
1804        LocalInfo::User(mir::BindingForm::ImplicitSelf(kind)) => {
1805            // Check if the user variable is a `&mut self` and we can therefore
1806            // suggest removing the `&mut`.
1807            //
1808            // Deliberately fall into this case for all implicit self types,
1809            // so that we don't fall into the next case with them.
1810            kind == hir::ImplicitSelfKind::RefMut
1811        }
1812        _ if Some(kw::SelfLower) == local_name => {
1813            // Otherwise, check if the name is the `self` keyword - in which case
1814            // we have an explicit self. Do the same thing in this case and check
1815            // for a `self: &mut Self` to suggest removing the `&mut`.
1816            #[allow(non_exhaustive_omitted_patterns)] match local_decl.ty.kind() {
    ty::Ref(_, _, hir::Mutability::Mut) => true,
    _ => false,
}matches!(local_decl.ty.kind(), ty::Ref(_, _, hir::Mutability::Mut))
1817        }
1818        _ => false,
1819    }
1820}
1821
1822fn suggest_ampmut_self(tcx: TyCtxt<'_>, span: Span) -> (Span, String) {
1823    match tcx.sess.source_map().span_to_snippet(span) {
1824        Ok(snippet) if snippet.ends_with("self") => {
1825            (span.with_hi(span.hi() - BytePos(4)).shrink_to_hi(), "mut ".to_string())
1826        }
1827        _ => (span, "&mut self".to_string()),
1828    }
1829}
1830
1831enum AmpMutSugg {
1832    /// Type suggestion. Changes `&self` to `&mut self`, `x: &T` to `x: &mut T`,
1833    /// `ref x` to `ref mut x`, etc.
1834    Type {
1835        span: Span,
1836        suggestion: String,
1837        additional: Option<(Span, String)>,
1838    },
1839    /// Suggestion for expressions, `&x` to `&mut x`, `&x[i]` to `&mut x[i]`, etc.
1840    Expr {
1841        span: Span,
1842        suggestion: String,
1843    },
1844    /// Suggests `.get_mut` in the case of `&map[&key]` for Hash/BTreeMap.
1845    MapGetMut {
1846        span: Span,
1847        suggestion: String,
1848    },
1849    ChangeBinding,
1850}
1851
1852// When we want to suggest a user change a local variable to be a `&mut`, there
1853// are three potential "obvious" things to highlight:
1854//
1855// let ident [: Type] [= RightHandSideExpression];
1856//     ^^^^^    ^^^^     ^^^^^^^^^^^^^^^^^^^^^^^
1857//     (1.)     (2.)              (3.)
1858//
1859// We can always fallback on highlighting the first. But chances are good that
1860// the user experience will be better if we highlight one of the others if possible;
1861// for example, if the RHS is present and the Type is not, then the type is going to
1862// be inferred *from* the RHS, which means we should highlight that (and suggest
1863// that they borrow the RHS mutably).
1864//
1865// This implementation attempts to emulate AST-borrowck prioritization
1866// by trying (3.), then (2.) and finally falling back on (1.).
1867fn suggest_ampmut<'tcx>(
1868    infcx: &crate::BorrowckInferCtxt<'tcx>,
1869    body: &Body<'tcx>,
1870    opt_assignment_rhs_stmt: Option<&Statement<'tcx>>,
1871) -> Option<AmpMutSugg> {
1872    let tcx = infcx.tcx;
1873    // If there is a RHS and it starts with a `&` from it, then check if it is
1874    // mutable, and if not, put suggest putting `mut ` to make it mutable.
1875    // We don't have to worry about lifetime annotations here because they are
1876    // not valid when taking a reference. For example, the following is not valid Rust:
1877    //
1878    // let x: &i32 = &'a 5;
1879    //                ^^ lifetime annotation not allowed
1880    //
1881    if let Some(rhs_stmt) = opt_assignment_rhs_stmt
1882        && let StatementKind::Assign(box (lhs, rvalue)) = &rhs_stmt.kind
1883        && let mut rhs_span = rhs_stmt.source_info.span
1884        && let Ok(mut rhs_str) = tcx.sess.source_map().span_to_snippet(rhs_span)
1885    {
1886        let mut rvalue = rvalue;
1887
1888        // Take some special care when handling `let _x = &*_y`:
1889        // We want to know if this is part of an overloaded index, so `let x = &a[0]`,
1890        // or whether this is a usertype ascription (`let _x: &T = y`).
1891        if let Rvalue::Ref(_, BorrowKind::Shared, place) = rvalue
1892            && place.projection.len() == 1
1893            && place.projection[0] == ProjectionElem::Deref
1894            && let Some(assign) = find_assignments(&body, place.local).first()
1895        {
1896            // If this is a usertype ascription (`let _x: &T = _y`) then pierce through it as either we want
1897            // to suggest `&mut` on the expression (handled here) or we return `None` and let the caller
1898            // suggest `&mut` on the type if the expression seems fine (e.g. `let _x: &T = &mut _y`).
1899            if let Some(user_ty_projs) = body.local_decls[lhs.local].user_ty.as_ref()
1900                && let [user_ty_proj] = user_ty_projs.contents.as_slice()
1901                && user_ty_proj.projs.is_empty()
1902                && let Either::Left(rhs_stmt_new) = body.stmt_at(*assign)
1903                && let StatementKind::Assign(box (_, rvalue_new)) = &rhs_stmt_new.kind
1904                && let rhs_span_new = rhs_stmt_new.source_info.span
1905                && let Ok(rhs_str_new) = tcx.sess.source_map().span_to_snippet(rhs_span_new)
1906            {
1907                (rvalue, rhs_span, rhs_str) = (rvalue_new, rhs_span_new, rhs_str_new);
1908            }
1909
1910            if let Either::Right(call) = body.stmt_at(*assign)
1911                && let TerminatorKind::Call {
1912                    func: Operand::Constant(box const_operand), args, ..
1913                } = &call.kind
1914                && let ty::FnDef(method_def_id, method_args) = *const_operand.ty().kind()
1915                && let Some(trait_) = tcx.trait_of_assoc(method_def_id)
1916                && tcx.is_lang_item(trait_, hir::LangItem::Index)
1917            {
1918                let trait_ref = ty::TraitRef::from_assoc(
1919                    tcx,
1920                    tcx.require_lang_item(hir::LangItem::IndexMut, rhs_span),
1921                    method_args,
1922                );
1923                // The type only implements `Index` but not `IndexMut`, we must not suggest `&mut`.
1924                if !infcx
1925                    .type_implements_trait(trait_ref.def_id, trait_ref.args, infcx.param_env)
1926                    .must_apply_considering_regions()
1927                {
1928                    // Suggest `get_mut` if type is a `BTreeMap` or `HashMap`.
1929                    if let ty::Adt(def, _) = trait_ref.self_ty().kind()
1930                        && [sym::BTreeMap, sym::HashMap]
1931                            .into_iter()
1932                            .any(|s| tcx.is_diagnostic_item(s, def.did()))
1933                        && let [map, key] = &**args
1934                        && let Ok(map) = tcx.sess.source_map().span_to_snippet(map.span)
1935                        && let Ok(key) = tcx.sess.source_map().span_to_snippet(key.span)
1936                    {
1937                        let span = rhs_span;
1938                        let suggestion = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}.get_mut({1}).unwrap()", map,
                key))
    })format!("{map}.get_mut({key}).unwrap()");
1939                        return Some(AmpMutSugg::MapGetMut { span, suggestion });
1940                    }
1941                    return None;
1942                }
1943            }
1944        }
1945
1946        let sugg = match rvalue {
1947            Rvalue::Ref(_, BorrowKind::Shared, _) if let Some(ref_idx) = rhs_str.find('&') => {
1948                // Shrink the span to just after the `&` in `&variable`.
1949                Some((
1950                    rhs_span.with_lo(rhs_span.lo() + BytePos(ref_idx as u32 + 1)).shrink_to_lo(),
1951                    "mut ".to_owned(),
1952                ))
1953            }
1954            Rvalue::RawPtr(RawPtrKind::Const, _) if let Some(const_idx) = rhs_str.find("const") => {
1955                // Suggest changing `&raw const` to `&raw mut` if applicable.
1956                let const_idx = const_idx as u32;
1957                Some((
1958                    rhs_span
1959                        .with_lo(rhs_span.lo() + BytePos(const_idx))
1960                        .with_hi(rhs_span.lo() + BytePos(const_idx + "const".len() as u32)),
1961                    "mut".to_owned(),
1962                ))
1963            }
1964            _ => None,
1965        };
1966
1967        if let Some((span, suggestion)) = sugg {
1968            return Some(AmpMutSugg::Expr { span, suggestion });
1969        }
1970    }
1971
1972    Some(AmpMutSugg::ChangeBinding)
1973}
1974
1975/// If the type is a `Coroutine`, `Closure`, or `CoroutineClosure`
1976fn is_closure_like(ty: Ty<'_>) -> bool {
1977    ty.is_closure() || ty.is_coroutine() || ty.is_coroutine_closure()
1978}
1979
1980/// Given a field that needs to be mutable, returns a span where the " mut " could go.
1981/// This function expects the local to be a reference to a struct in order to produce a span.
1982///
1983/// ```text
1984/// LL |     s: &'a   String
1985///    |           ^^^ returns a span taking up the space here
1986/// ```
1987fn get_mut_span_in_struct_field<'tcx>(
1988    tcx: TyCtxt<'tcx>,
1989    ty: Ty<'tcx>,
1990    field: FieldIdx,
1991) -> Option<Span> {
1992    // Expect our local to be a reference to a struct of some kind.
1993    if let ty::Ref(_, ty, _) = ty.kind()
1994        && let ty::Adt(def, _) = ty.kind()
1995        && let field = def.all_fields().nth(field.index())?
1996        // Now we're dealing with the actual struct that we're going to suggest a change to,
1997        // we can expect a field that is an immutable reference to a type.
1998        && let hir::Node::Field(field) = tcx.hir_node_by_def_id(field.did.as_local()?)
1999        && let hir::TyKind::Ref(lt, hir::MutTy { mutbl: hir::Mutability::Not, ty }) = field.ty.kind
2000    {
2001        return Some(lt.ident.span.between(ty.span));
2002    }
2003
2004    None
2005}
2006
2007/// If possible, suggest replacing `ref` with `ref mut`.
2008fn suggest_ref_mut(tcx: TyCtxt<'_>, span: Span) -> Option<Span> {
2009    let pattern_str = tcx.sess.source_map().span_to_snippet(span).ok()?;
2010    if let Some(rest) = pattern_str.strip_prefix("ref")
2011        && rest.starts_with(rustc_lexer::is_whitespace)
2012    {
2013        let span = span.with_lo(span.lo() + BytePos(4)).shrink_to_lo();
2014        Some(span)
2015    } else {
2016        None
2017    }
2018}