Skip to content

Commit 35c6902

Browse files
authored
Rollup merge of rust-lang#84358 - sexxi-goose:print_captures_borrowck_rebased, r=nikomatsakis
Update closure capture error logging for disjoint captures for disjoint captures Improved error logging when `#![feature(capture_disjoint_fields)]` is used. Closes rust-lang/project-rfc-2229#8 Closes rust-lang/project-rfc-2229#36 Closes rust-lang/project-rfc-2229#39 Closes rust-lang#76005
2 parents 3a1cd0e + 404cc33 commit 35c6902

File tree

57 files changed

+534
-152
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

57 files changed

+534
-152
lines changed

compiler/rustc_middle/src/mir/mod.rs

+11
Original file line numberDiff line numberDiff line change
@@ -683,6 +683,15 @@ impl BorrowKind {
683683
BorrowKind::Mut { allow_two_phase_borrow } => allow_two_phase_borrow,
684684
}
685685
}
686+
687+
pub fn describe_mutability(&self) -> String {
688+
match *self {
689+
BorrowKind::Shared | BorrowKind::Shallow | BorrowKind::Unique => {
690+
"immutable".to_string()
691+
}
692+
BorrowKind::Mut { .. } => "mutable".to_string(),
693+
}
694+
}
686695
}
687696

688697
///////////////////////////////////////////////////////////////////////////
@@ -2369,6 +2378,7 @@ impl<'tcx> Debug for Rvalue<'tcx> {
23692378
};
23702379
let mut struct_fmt = fmt.debug_struct(&name);
23712380

2381+
// FIXME(project-rfc-2229#48): This should be a list of capture names/places
23722382
if let Some(upvars) = tcx.upvars_mentioned(def_id) {
23732383
for (&var_id, place) in iter::zip(upvars.keys(), places) {
23742384
let var_name = tcx.hir().name(var_id);
@@ -2388,6 +2398,7 @@ impl<'tcx> Debug for Rvalue<'tcx> {
23882398
let name = format!("[generator@{:?}]", tcx.hir().span(hir_id));
23892399
let mut struct_fmt = fmt.debug_struct(&name);
23902400

2401+
// FIXME(project-rfc-2229#48): This should be a list of capture names/places
23912402
if let Some(upvars) = tcx.upvars_mentioned(def_id) {
23922403
for (&var_id, place) in iter::zip(upvars.keys(), places) {
23932404
let var_name = tcx.hir().name(var_id);

compiler/rustc_middle/src/ty/closure.rs

+20
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,10 @@ pub struct CapturedPlace<'tcx> {
151151
}
152152

153153
impl CapturedPlace<'tcx> {
154+
pub fn to_string(&self, tcx: TyCtxt<'tcx>) -> String {
155+
place_to_string_for_capture(tcx, &self.place)
156+
}
157+
154158
/// Returns the hir-id of the root variable for the captured place.
155159
/// e.g., if `a.b.c` was captured, would return the hir-id for `a`.
156160
pub fn get_root_variable(&self) -> hir::HirId {
@@ -168,6 +172,22 @@ impl CapturedPlace<'tcx> {
168172
}
169173
}
170174

175+
/// Return span pointing to use that resulted in selecting the captured path
176+
pub fn get_path_span(&self, tcx: TyCtxt<'tcx>) -> Span {
177+
if let Some(path_expr_id) = self.info.path_expr_id {
178+
tcx.hir().span(path_expr_id)
179+
} else if let Some(capture_kind_expr_id) = self.info.capture_kind_expr_id {
180+
tcx.hir().span(capture_kind_expr_id)
181+
} else {
182+
// Fallback on upvars mentioned if neither path or capture expr id is captured
183+
184+
// Safe to unwrap since we know this place is captured by the closure, therefore the closure must have upvars.
185+
tcx.upvars_mentioned(self.get_closure_local_def_id()).unwrap()
186+
[&self.get_root_variable()]
187+
.span
188+
}
189+
}
190+
171191
/// Return span pointing to use that resulted in selecting the current capture kind
172192
pub fn get_capture_kind_span(&self, tcx: TyCtxt<'tcx>) -> Span {
173193
if let Some(capture_kind_expr_id) = self.info.capture_kind_expr_id {

compiler/rustc_mir/src/borrow_check/diagnostics/conflict_errors.rs

+33-15
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
9999
);
100100
err.span_label(span, format!("use of possibly-uninitialized {}", item_msg));
101101

102-
use_spans.var_span_label(
102+
use_spans.var_span_label_path_only(
103103
&mut err,
104104
format!("{} occurs due to use{}", desired_action.as_noun(), use_spans.describe()),
105105
);
@@ -255,6 +255,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
255255
partially_str,
256256
move_spans.describe()
257257
),
258+
"moved",
258259
);
259260
}
260261
}
@@ -304,7 +305,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
304305
}
305306
}
306307

307-
use_spans.var_span_label(
308+
use_spans.var_span_label_path_only(
308309
&mut err,
309310
format!("{} occurs due to use{}", desired_action.as_noun(), use_spans.describe()),
310311
);
@@ -434,13 +435,16 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
434435
err.span_label(borrow_span, format!("borrow of {} occurs here", borrow_msg));
435436
err.span_label(span, format!("move out of {} occurs here", value_msg));
436437

437-
borrow_spans.var_span_label(
438+
borrow_spans.var_span_label_path_only(
438439
&mut err,
439440
format!("borrow occurs due to use{}", borrow_spans.describe()),
440441
);
441442

442-
move_spans
443-
.var_span_label(&mut err, format!("move occurs due to use{}", move_spans.describe()));
443+
move_spans.var_span_label(
444+
&mut err,
445+
format!("move occurs due to use{}", move_spans.describe()),
446+
"moved",
447+
);
444448

445449
self.explain_why_borrow_contains_point(location, borrow, None)
446450
.add_explanation_to_diagnostic(
@@ -468,18 +472,24 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
468472
let use_spans = self.move_spans(place.as_ref(), location);
469473
let span = use_spans.var_or_use();
470474

475+
// If the attempted use is in a closure then we do not care about the path span of the place we are currently trying to use
476+
// we call `var_span_label` on `borrow_spans` to annotate if the existing borrow was in a closure
471477
let mut err = self.cannot_use_when_mutably_borrowed(
472478
span,
473479
&self.describe_any_place(place.as_ref()),
474480
borrow_span,
475481
&self.describe_any_place(borrow.borrowed_place.as_ref()),
476482
);
477483

478-
borrow_spans.var_span_label(&mut err, {
479-
let place = &borrow.borrowed_place;
480-
let desc_place = self.describe_any_place(place.as_ref());
481-
format!("borrow occurs due to use of {}{}", desc_place, borrow_spans.describe())
482-
});
484+
borrow_spans.var_span_label(
485+
&mut err,
486+
{
487+
let place = &borrow.borrowed_place;
488+
let desc_place = self.describe_any_place(place.as_ref());
489+
format!("borrow occurs due to use of {}{}", desc_place, borrow_spans.describe())
490+
},
491+
"mutable",
492+
);
483493

484494
self.explain_why_borrow_contains_point(location, borrow, None)
485495
.add_explanation_to_diagnostic(
@@ -591,6 +601,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
591601
desc_place,
592602
borrow_spans.describe(),
593603
),
604+
"immutable",
594605
);
595606

596607
return err;
@@ -667,7 +678,8 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
667678
if issued_spans == borrow_spans {
668679
borrow_spans.var_span_label(
669680
&mut err,
670-
format!("borrows occur due to use of {}{}", desc_place, borrow_spans.describe()),
681+
format!("borrows occur due to use of {}{}", desc_place, borrow_spans.describe(),),
682+
gen_borrow_kind.describe_mutability(),
671683
);
672684
} else {
673685
let borrow_place = &issued_borrow.borrowed_place;
@@ -679,6 +691,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
679691
borrow_place_desc,
680692
issued_spans.describe(),
681693
),
694+
issued_borrow.kind.describe_mutability(),
682695
);
683696

684697
borrow_spans.var_span_label(
@@ -688,6 +701,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
688701
desc_place,
689702
borrow_spans.describe(),
690703
),
704+
gen_borrow_kind.describe_mutability(),
691705
);
692706
}
693707

@@ -847,7 +861,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
847861
self.prefixes(borrow.borrowed_place.as_ref(), PrefixSet::All).last().unwrap();
848862

849863
let borrow_spans = self.retrieve_borrow_spans(borrow);
850-
let borrow_span = borrow_spans.var_or_use();
864+
let borrow_span = borrow_spans.var_or_use_path_span();
851865

852866
assert!(root_place.projection.is_empty());
853867
let proper_span = self.body.local_decls[root_place.local].source_info.span;
@@ -987,7 +1001,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
9871001
location, name, borrow, drop_span, borrow_spans
9881002
);
9891003

990-
let borrow_span = borrow_spans.var_or_use();
1004+
let borrow_span = borrow_spans.var_or_use_path_span();
9911005
if let BorrowExplanation::MustBeValidFor {
9921006
category,
9931007
span,
@@ -1575,6 +1589,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
15751589
loan_spans.var_span_label(
15761590
&mut err,
15771591
format!("borrow occurs due to use{}", loan_spans.describe()),
1592+
loan.kind.describe_mutability(),
15781593
);
15791594

15801595
err.buffer(&mut self.errors_buffer);
@@ -1585,8 +1600,11 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
15851600

15861601
let mut err = self.cannot_assign_to_borrowed(span, loan_span, &descr_place);
15871602

1588-
loan_spans
1589-
.var_span_label(&mut err, format!("borrow occurs due to use{}", loan_spans.describe()));
1603+
loan_spans.var_span_label(
1604+
&mut err,
1605+
format!("borrow occurs due to use{}", loan_spans.describe()),
1606+
loan.kind.describe_mutability(),
1607+
);
15901608

15911609
self.explain_why_borrow_contains_point(location, loan, None).add_explanation_to_diagnostic(
15921610
self.infcx.tcx,

compiler/rustc_mir/src/borrow_check/diagnostics/explain_borrow.rs

+54-17
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,8 @@ use super::{find_use, RegionName, UseSpans};
2424

2525
#[derive(Debug)]
2626
pub(in crate::borrow_check) enum BorrowExplanation {
27-
UsedLater(LaterUseKind, Span),
28-
UsedLaterInLoop(LaterUseKind, Span),
27+
UsedLater(LaterUseKind, Span, Option<Span>),
28+
UsedLaterInLoop(LaterUseKind, Span, Option<Span>),
2929
UsedLaterWhenDropped {
3030
drop_loc: Location,
3131
dropped_local: Local,
@@ -67,22 +67,39 @@ impl BorrowExplanation {
6767
borrow_span: Option<Span>,
6868
) {
6969
match *self {
70-
BorrowExplanation::UsedLater(later_use_kind, var_or_use_span) => {
70+
BorrowExplanation::UsedLater(later_use_kind, var_or_use_span, path_span) => {
7171
let message = match later_use_kind {
7272
LaterUseKind::TraitCapture => "captured here by trait object",
7373
LaterUseKind::ClosureCapture => "captured here by closure",
7474
LaterUseKind::Call => "used by call",
7575
LaterUseKind::FakeLetRead => "stored here",
7676
LaterUseKind::Other => "used here",
7777
};
78-
if !borrow_span.map_or(false, |sp| sp.overlaps(var_or_use_span)) {
79-
err.span_label(
80-
var_or_use_span,
81-
format!("{}borrow later {}", borrow_desc, message),
82-
);
78+
// We can use `var_or_use_span` if either `path_span` is not present, or both spans are the same
79+
if path_span.map(|path_span| path_span == var_or_use_span).unwrap_or(true) {
80+
if borrow_span.map(|sp| !sp.overlaps(var_or_use_span)).unwrap_or(true) {
81+
err.span_label(
82+
var_or_use_span,
83+
format!("{}borrow later {}", borrow_desc, message),
84+
);
85+
}
86+
} else {
87+
// path_span must be `Some` as otherwise the if condition is true
88+
let path_span = path_span.unwrap();
89+
// path_span is only present in the case of closure capture
90+
assert!(matches!(later_use_kind, LaterUseKind::ClosureCapture));
91+
if !borrow_span.map_or(false, |sp| sp.overlaps(var_or_use_span)) {
92+
let path_label = "used here by closure";
93+
let capture_kind_label = message;
94+
err.span_label(
95+
var_or_use_span,
96+
format!("{}borrow later {}", borrow_desc, capture_kind_label),
97+
);
98+
err.span_label(path_span, path_label);
99+
}
83100
}
84101
}
85-
BorrowExplanation::UsedLaterInLoop(later_use_kind, var_or_use_span) => {
102+
BorrowExplanation::UsedLaterInLoop(later_use_kind, var_or_use_span, path_span) => {
86103
let message = match later_use_kind {
87104
LaterUseKind::TraitCapture => {
88105
"borrow captured here by trait object, in later iteration of loop"
@@ -94,7 +111,24 @@ impl BorrowExplanation {
94111
LaterUseKind::FakeLetRead => "borrow later stored here",
95112
LaterUseKind::Other => "borrow used here, in later iteration of loop",
96113
};
97-
err.span_label(var_or_use_span, format!("{}{}", borrow_desc, message));
114+
// We can use `var_or_use_span` if either `path_span` is not present, or both spans are the same
115+
if path_span.map(|path_span| path_span == var_or_use_span).unwrap_or(true) {
116+
err.span_label(var_or_use_span, format!("{}{}", borrow_desc, message));
117+
} else {
118+
// path_span must be `Some` as otherwise the if condition is true
119+
let path_span = path_span.unwrap();
120+
// path_span is only present in the case of closure capture
121+
assert!(matches!(later_use_kind, LaterUseKind::ClosureCapture));
122+
if borrow_span.map(|sp| !sp.overlaps(var_or_use_span)).unwrap_or(true) {
123+
let path_label = "used here by closure";
124+
let capture_kind_label = message;
125+
err.span_label(
126+
var_or_use_span,
127+
format!("{}borrow later {}", borrow_desc, capture_kind_label),
128+
);
129+
err.span_label(path_span, path_label);
130+
}
131+
}
98132
}
99133
BorrowExplanation::UsedLaterWhenDropped {
100134
drop_loc,
@@ -311,13 +345,13 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
311345
let borrow_location = location;
312346
if self.is_use_in_later_iteration_of_loop(borrow_location, location) {
313347
let later_use = self.later_use_kind(borrow, spans, location);
314-
BorrowExplanation::UsedLaterInLoop(later_use.0, later_use.1)
348+
BorrowExplanation::UsedLaterInLoop(later_use.0, later_use.1, later_use.2)
315349
} else {
316350
// Check if the location represents a `FakeRead`, and adapt the error
317351
// message to the `FakeReadCause` it is from: in particular,
318352
// the ones inserted in optimized `let var = <expr>` patterns.
319353
let later_use = self.later_use_kind(borrow, spans, location);
320-
BorrowExplanation::UsedLater(later_use.0, later_use.1)
354+
BorrowExplanation::UsedLater(later_use.0, later_use.1, later_use.2)
321355
}
322356
}
323357

@@ -498,16 +532,19 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
498532
}
499533

500534
/// Determine how the borrow was later used.
535+
/// First span returned points to the location of the conflicting use
536+
/// Second span if `Some` is returned in the case of closures and points
537+
/// to the use of the path
501538
fn later_use_kind(
502539
&self,
503540
borrow: &BorrowData<'tcx>,
504541
use_spans: UseSpans<'tcx>,
505542
location: Location,
506-
) -> (LaterUseKind, Span) {
543+
) -> (LaterUseKind, Span, Option<Span>) {
507544
match use_spans {
508-
UseSpans::ClosureUse { var_span, .. } => {
545+
UseSpans::ClosureUse { capture_kind_span, path_span, .. } => {
509546
// Used in a closure.
510-
(LaterUseKind::ClosureCapture, var_span)
547+
(LaterUseKind::ClosureCapture, capture_kind_span, Some(path_span))
511548
}
512549
UseSpans::PatUse(span)
513550
| UseSpans::OtherUse(span)
@@ -542,15 +579,15 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
542579
}
543580
}
544581
};
545-
return (LaterUseKind::Call, function_span);
582+
return (LaterUseKind::Call, function_span, None);
546583
} else {
547584
LaterUseKind::Other
548585
}
549586
} else {
550587
LaterUseKind::Other
551588
};
552589

553-
(kind, span)
590+
(kind, span, None)
554591
}
555592
}
556593
}

0 commit comments

Comments
 (0)