Skip to content

Commit 0fc32d7

Browse files
authored
Rollup merge of rust-lang#83521 - sexxi-goose:quick-diagnostic-fix, r=nikomatsakis
2229: Fix diagnostic issue when using FakeReads in closures This PR fixes a diagnostic issue caused by rust-lang#82536. A temporary work around was used in this merged PR which involved feature gating the addition of FakeReads introduced as a result of pattern matching in closures. The fix involves adding an optional closure DefId to ForLet and ForMatchedPlace FakeReadCauses. This DefId will only be added if a closure pattern matches a Place starting with an Upvar. r? ``@nikomatsakis``
2 parents 81bffb1 + 42797e2 commit 0fc32d7

File tree

126 files changed

+1973
-1932
lines changed

Some content is hidden

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

126 files changed

+1973
-1932
lines changed

compiler/rustc_middle/src/mir/mod.rs

+16-4
Original file line numberDiff line numberDiff line change
@@ -1482,7 +1482,7 @@ pub enum StatementKind<'tcx> {
14821482
///
14831483
/// Note that this also is emitted for regular `let` bindings to ensure that locals that are
14841484
/// never accessed still get some sanity checks for, e.g., `let x: ! = ..;`
1485-
FakeRead(FakeReadCause, Box<Place<'tcx>>),
1485+
FakeRead(Box<(FakeReadCause, Place<'tcx>)>),
14861486

14871487
/// Write the discriminant for a variant to the enum Place.
14881488
SetDiscriminant { place: Box<Place<'tcx>>, variant_index: VariantIdx },
@@ -1575,7 +1575,12 @@ pub enum FakeReadCause {
15751575

15761576
/// `let x: !; match x {}` doesn't generate any read of x so we need to
15771577
/// generate a read of x to check that it is initialized and safe.
1578-
ForMatchedPlace,
1578+
///
1579+
/// If a closure pattern matches a Place starting with an Upvar, then we introduce a
1580+
/// FakeRead for that Place outside the closure, in such a case this option would be
1581+
/// Some(closure_def_id).
1582+
/// Otherwise, the value of the optional DefId will be None.
1583+
ForMatchedPlace(Option<DefId>),
15791584

15801585
/// A fake read of the RefWithinGuard version of a bind-by-value variable
15811586
/// in a match guard to ensure that it's value hasn't change by the time
@@ -1594,7 +1599,12 @@ pub enum FakeReadCause {
15941599
/// but in some cases it can affect the borrow checker, as in #53695.
15951600
/// Therefore, we insert a "fake read" here to ensure that we get
15961601
/// appropriate errors.
1597-
ForLet,
1602+
///
1603+
/// If a closure pattern matches a Place starting with an Upvar, then we introduce a
1604+
/// FakeRead for that Place outside the closure, in such a case this option would be
1605+
/// Some(closure_def_id).
1606+
/// Otherwise, the value of the optional DefId will be None.
1607+
ForLet(Option<DefId>),
15981608

15991609
/// If we have an index expression like
16001610
///
@@ -1618,7 +1628,9 @@ impl Debug for Statement<'_> {
16181628
use self::StatementKind::*;
16191629
match self.kind {
16201630
Assign(box (ref place, ref rv)) => write!(fmt, "{:?} = {:?}", place, rv),
1621-
FakeRead(ref cause, ref place) => write!(fmt, "FakeRead({:?}, {:?})", cause, place),
1631+
FakeRead(box (ref cause, ref place)) => {
1632+
write!(fmt, "FakeRead({:?}, {:?})", cause, place)
1633+
}
16221634
Retag(ref kind, ref place) => write!(
16231635
fmt,
16241636
"Retag({}{:?})",

compiler/rustc_middle/src/mir/visit.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -380,7 +380,7 @@ macro_rules! make_mir_visitor {
380380
) => {
381381
self.visit_assign(place, rvalue, location);
382382
}
383-
StatementKind::FakeRead(_, place) => {
383+
StatementKind::FakeRead(box (_, place)) => {
384384
self.visit_place(
385385
place,
386386
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -1728,7 +1728,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
17281728
impl<'tcx> Visitor<'tcx> for FakeReadCauseFinder<'tcx> {
17291729
fn visit_statement(&mut self, statement: &Statement<'tcx>, _: Location) {
17301730
match statement {
1731-
Statement { kind: StatementKind::FakeRead(cause, box place), .. }
1731+
Statement { kind: StatementKind::FakeRead(box (cause, place)), .. }
17321732
if *place == self.place =>
17331733
{
17341734
self.cause = Some(*cause);

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -515,7 +515,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
515515
let block = &self.body.basic_blocks()[location.block];
516516

517517
let kind = if let Some(&Statement {
518-
kind: StatementKind::FakeRead(FakeReadCause::ForLet, _),
518+
kind: StatementKind::FakeRead(box (FakeReadCause::ForLet(_), _)),
519519
..
520520
}) = block.statements.get(location.statement_index)
521521
{

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

+20-2
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,8 @@ use rustc_hir::def_id::DefId;
77
use rustc_hir::lang_items::LangItemGroup;
88
use rustc_hir::GeneratorKind;
99
use rustc_middle::mir::{
10-
AggregateKind, Constant, Field, Local, LocalInfo, LocalKind, Location, Operand, Place,
11-
PlaceRef, ProjectionElem, Rvalue, Statement, StatementKind, Terminator, TerminatorKind,
10+
AggregateKind, Constant, FakeReadCause, Field, Local, LocalInfo, LocalKind, Location, Operand,
11+
Place, PlaceRef, ProjectionElem, Rvalue, Statement, StatementKind, Terminator, TerminatorKind,
1212
};
1313
use rustc_middle::ty::print::Print;
1414
use rustc_middle::ty::{self, DefIdTree, Instance, Ty, TyCtxt};
@@ -795,6 +795,24 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
795795
}
796796
}
797797

798+
// StatementKind::FakeRead only contains a def_id if they are introduced as a result
799+
// of pattern matching within a closure.
800+
if let StatementKind::FakeRead(box (cause, ref place)) = stmt.kind {
801+
match cause {
802+
FakeReadCause::ForMatchedPlace(Some(closure_def_id))
803+
| FakeReadCause::ForLet(Some(closure_def_id)) => {
804+
debug!("move_spans: def_id={:?} place={:?}", closure_def_id, place);
805+
let places = &[Operand::Move(*place)];
806+
if let Some((args_span, generator_kind, var_span)) =
807+
self.closure_span(closure_def_id, moved_place, places)
808+
{
809+
return ClosureUse { generator_kind, args_span, var_span };
810+
}
811+
}
812+
_ => {}
813+
}
814+
}
815+
798816
let normal_ret =
799817
if moved_place.projection.iter().any(|p| matches!(p, ProjectionElem::Downcast(..))) {
800818
PatUse(stmt.source_info.span)

compiler/rustc_mir/src/borrow_check/invalidation.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ impl<'cx, 'tcx> Visitor<'tcx> for InvalidationGenerator<'cx, 'tcx> {
6363

6464
self.mutate_place(location, *lhs, Shallow(None), JustWrite);
6565
}
66-
StatementKind::FakeRead(_, _) => {
66+
StatementKind::FakeRead(box (_, _)) => {
6767
// Only relevant for initialized/liveness/safety checks.
6868
}
6969
StatementKind::SetDiscriminant { place, variant_index: _ } => {

compiler/rustc_mir/src/borrow_check/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -574,7 +574,7 @@ impl<'cx, 'tcx> dataflow::ResultsVisitor<'cx, 'tcx> for MirBorrowckCtxt<'cx, 'tc
574574

575575
self.mutate_place(location, (*lhs, span), Shallow(None), JustWrite, flow_state);
576576
}
577-
StatementKind::FakeRead(_, box ref place) => {
577+
StatementKind::FakeRead(box (_, ref place)) => {
578578
// Read for match doesn't access any memory and is used to
579579
// assert that a place is safe and live. So we don't have to
580580
// do any checks here.

compiler/rustc_mir/src/dataflow/move_paths/builder.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -293,8 +293,8 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
293293
}
294294
self.gather_rvalue(rval);
295295
}
296-
StatementKind::FakeRead(_, place) => {
297-
self.create_move_path(**place);
296+
StatementKind::FakeRead(box (_, place)) => {
297+
self.create_move_path(*place);
298298
}
299299
StatementKind::LlvmInlineAsm(ref asm) => {
300300
for (output, kind) in iter::zip(&*asm.outputs, &asm.asm.outputs) {

compiler/rustc_mir/src/transform/coverage/spans.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -683,10 +683,10 @@ pub(super) fn filtered_statement_span(
683683
// and `_1` is the `Place` for `somenum`.
684684
//
685685
// If and when the Issue is resolved, remove this special case match pattern:
686-
StatementKind::FakeRead(cause, _) if cause == FakeReadCause::ForGuardBinding => None,
686+
StatementKind::FakeRead(box (cause, _)) if cause == FakeReadCause::ForGuardBinding => None,
687687

688688
// Retain spans from all other statements
689-
StatementKind::FakeRead(_, _) // Not including `ForGuardBinding`
689+
StatementKind::FakeRead(box (_, _)) // Not including `ForGuardBinding`
690690
| StatementKind::CopyNonOverlapping(..)
691691
| StatementKind::Assign(_)
692692
| StatementKind::SetDiscriminant { .. }

compiler/rustc_mir_build/src/build/cfg.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@ impl<'tcx> CFG<'tcx> {
8080
cause: FakeReadCause,
8181
place: Place<'tcx>,
8282
) {
83-
let kind = StatementKind::FakeRead(cause, box place);
83+
let kind = StatementKind::FakeRead(box (cause, place));
8484
let stmt = Statement { source_info, kind };
8585
self.push(block, stmt);
8686
}

compiler/rustc_mir_build/src/build/expr/as_rvalue.rs

+14-18
Original file line numberDiff line numberDiff line change
@@ -179,24 +179,20 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
179179
// match x { _ => () } // fake read of `x`
180180
// };
181181
// ```
182-
// FIXME(RFC2229): Remove feature gate once diagnostics are improved
183-
if this.tcx.features().capture_disjoint_fields {
184-
for (thir_place, cause, hir_id) in fake_reads.into_iter() {
185-
let place_builder =
186-
unpack!(block = this.as_place_builder(block, thir_place));
187-
188-
if let Ok(place_builder_resolved) =
189-
place_builder.try_upvars_resolved(this.tcx, this.typeck_results)
190-
{
191-
let mir_place =
192-
place_builder_resolved.into_place(this.tcx, this.typeck_results);
193-
this.cfg.push_fake_read(
194-
block,
195-
this.source_info(this.tcx.hir().span(*hir_id)),
196-
*cause,
197-
mir_place,
198-
);
199-
}
182+
for (thir_place, cause, hir_id) in fake_reads.into_iter() {
183+
let place_builder = unpack!(block = this.as_place_builder(block, thir_place));
184+
185+
if let Ok(place_builder_resolved) =
186+
place_builder.try_upvars_resolved(this.tcx, this.typeck_results)
187+
{
188+
let mir_place =
189+
place_builder_resolved.into_place(this.tcx, this.typeck_results);
190+
this.cfg.push_fake_read(
191+
block,
192+
this.source_info(this.tcx.hir().span(*hir_id)),
193+
*cause,
194+
mir_place,
195+
);
200196
}
201197
}
202198

compiler/rustc_mir_build/src/build/matches/mod.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -139,7 +139,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
139139
// uninhabited value. If we get never patterns, those will check that
140140
// the place is initialized, and so this read would only be used to
141141
// check safety.
142-
let cause_matched_place = FakeReadCause::ForMatchedPlace;
142+
let cause_matched_place = FakeReadCause::ForMatchedPlace(None);
143143
let source_info = self.source_info(scrutinee_span);
144144

145145
if let Ok(scrutinee_builder) =
@@ -400,7 +400,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
400400

401401
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
402402
let source_info = self.source_info(irrefutable_pat.span);
403-
self.cfg.push_fake_read(block, source_info, FakeReadCause::ForLet, place);
403+
self.cfg.push_fake_read(block, source_info, FakeReadCause::ForLet(None), place);
404404

405405
self.schedule_drop_for_binding(var, irrefutable_pat.span, OutsideGuard);
406406
block.unit()
@@ -435,7 +435,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
435435

436436
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
437437
let pattern_source_info = self.source_info(irrefutable_pat.span);
438-
let cause_let = FakeReadCause::ForLet;
438+
let cause_let = FakeReadCause::ForLet(None);
439439
self.cfg.push_fake_read(block, pattern_source_info, cause_let, place);
440440

441441
let ty_source_info = self.source_info(user_ty_span);

compiler/rustc_typeck/src/expr_use_visitor.rs

+18-3
Original file line numberDiff line numberDiff line change
@@ -280,9 +280,14 @@ impl<'a, 'tcx> ExprUseVisitor<'a, 'tcx> {
280280
if needs_to_be_read {
281281
self.borrow_expr(&discr, ty::ImmBorrow);
282282
} else {
283+
let closure_def_id = match discr_place.place.base {
284+
PlaceBase::Upvar(upvar_id) => Some(upvar_id.closure_expr_id.to_def_id()),
285+
_ => None,
286+
};
287+
283288
self.delegate.fake_read(
284289
discr_place.place.clone(),
285-
FakeReadCause::ForMatchedPlace,
290+
FakeReadCause::ForMatchedPlace(closure_def_id),
286291
discr_place.hir_id,
287292
);
288293

@@ -578,9 +583,14 @@ impl<'a, 'tcx> ExprUseVisitor<'a, 'tcx> {
578583
}
579584

580585
fn walk_arm(&mut self, discr_place: &PlaceWithHirId<'tcx>, arm: &hir::Arm<'_>) {
586+
let closure_def_id = match discr_place.place.base {
587+
PlaceBase::Upvar(upvar_id) => Some(upvar_id.closure_expr_id.to_def_id()),
588+
_ => None,
589+
};
590+
581591
self.delegate.fake_read(
582592
discr_place.place.clone(),
583-
FakeReadCause::ForMatchedPlace,
593+
FakeReadCause::ForMatchedPlace(closure_def_id),
584594
discr_place.hir_id,
585595
);
586596
self.walk_pat(discr_place, &arm.pat);
@@ -595,9 +605,14 @@ impl<'a, 'tcx> ExprUseVisitor<'a, 'tcx> {
595605
/// Walks a pat that occurs in isolation (i.e., top-level of fn argument or
596606
/// let binding, and *not* a match arm or nested pat.)
597607
fn walk_irrefutable_pat(&mut self, discr_place: &PlaceWithHirId<'tcx>, pat: &hir::Pat<'_>) {
608+
let closure_def_id = match discr_place.place.base {
609+
PlaceBase::Upvar(upvar_id) => Some(upvar_id.closure_expr_id.to_def_id()),
610+
_ => None,
611+
};
612+
598613
self.delegate.fake_read(
599614
discr_place.place.clone(),
600-
FakeReadCause::ForLet,
615+
FakeReadCause::ForLet(closure_def_id),
601616
discr_place.hir_id,
602617
);
603618
self.walk_pat(discr_place, pat);

0 commit comments

Comments
 (0)