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