@@ -28,6 +28,7 @@ mod simplify;
28
28
mod test;
29
29
mod util;
30
30
31
+ use std:: assert_matches:: assert_matches;
31
32
use std:: borrow:: Borrow ;
32
33
use std:: mem;
33
34
@@ -74,6 +75,17 @@ pub(crate) enum EmitStorageLive {
74
75
No ,
75
76
}
76
77
78
+ /// Used by [`Builder::storage_live_binding`] and [`Builder::bind_matched_candidate_for_arm_body`]
79
+ /// to decide whether to schedule drops.
80
+ #[ derive( Clone , Copy , Debug ) ]
81
+ pub ( crate ) enum ScheduleDrops {
82
+ /// Yes, the relevant functions should also schedule drops as appropriate.
83
+ Yes ,
84
+ /// No, don't schedule drops. The caller has taken responsibility for any
85
+ /// appropriate drops.
86
+ No ,
87
+ }
88
+
77
89
impl < ' a , ' tcx > Builder < ' a , ' tcx > {
78
90
/// Lowers a condition in a way that ensures that variables bound in any let
79
91
/// expressions are definitely initialized in the if body.
@@ -535,7 +547,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
535
547
fake_borrow_temps,
536
548
scrutinee_span,
537
549
arm_match_scope,
538
- true ,
550
+ ScheduleDrops :: Yes ,
539
551
emit_storage_live,
540
552
)
541
553
} else {
@@ -554,7 +566,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
554
566
// To handle this we instead unschedule it's drop after each time
555
567
// we lower the guard.
556
568
let target_block = self . cfg . start_new_block ( ) ;
557
- let mut schedule_drops = true ;
569
+ let mut schedule_drops = ScheduleDrops :: Yes ;
558
570
let arm = arm_match_scope. unzip ( ) . 0 ;
559
571
// We keep a stack of all of the bindings and type ascriptions
560
572
// from the parent candidates that we visit, that also need to
@@ -576,7 +588,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
576
588
emit_storage_live,
577
589
) ;
578
590
if arm. is_none ( ) {
579
- schedule_drops = false ;
591
+ schedule_drops = ScheduleDrops :: No ;
580
592
}
581
593
self . cfg . goto ( binding_end, outer_source_info, target_block) ;
582
594
} ,
@@ -602,8 +614,13 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
602
614
match irrefutable_pat. kind {
603
615
// Optimize the case of `let x = ...` to write directly into `x`
604
616
PatKind :: Binding { mode : BindingMode ( ByRef :: No , _) , var, subpattern : None , .. } => {
605
- let place =
606
- self . storage_live_binding ( block, var, irrefutable_pat. span , OutsideGuard , true ) ;
617
+ let place = self . storage_live_binding (
618
+ block,
619
+ var,
620
+ irrefutable_pat. span ,
621
+ OutsideGuard ,
622
+ ScheduleDrops :: Yes ,
623
+ ) ;
607
624
unpack ! ( block = self . expr_into_dest( place, block, initializer_id) ) ;
608
625
609
626
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
@@ -636,8 +653,13 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
636
653
} ,
637
654
ascription : thir:: Ascription { ref annotation, variance : _ } ,
638
655
} => {
639
- let place =
640
- self . storage_live_binding ( block, var, irrefutable_pat. span , OutsideGuard , true ) ;
656
+ let place = self . storage_live_binding (
657
+ block,
658
+ var,
659
+ irrefutable_pat. span ,
660
+ OutsideGuard ,
661
+ ScheduleDrops :: Yes ,
662
+ ) ;
641
663
unpack ! ( block = self . expr_into_dest( place, block, initializer_id) ) ;
642
664
643
665
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
@@ -827,17 +849,21 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
827
849
var : LocalVarId ,
828
850
span : Span ,
829
851
for_guard : ForGuard ,
830
- schedule_drop : bool ,
852
+ schedule_drop : ScheduleDrops ,
831
853
) -> Place < ' tcx > {
832
854
let local_id = self . var_local_id ( var, for_guard) ;
833
855
let source_info = self . source_info ( span) ;
834
856
self . cfg . push ( block, Statement { source_info, kind : StatementKind :: StorageLive ( local_id) } ) ;
835
857
// Although there is almost always scope for given variable in corner cases
836
858
// like #92893 we might get variable with no scope.
837
- if let Some ( region_scope) = self . region_scope_tree . var_scope ( var. 0 . local_id )
838
- && schedule_drop
839
- {
840
- self . schedule_drop ( span, region_scope, local_id, DropKind :: Storage ) ;
859
+ if let Some ( region_scope) = self . region_scope_tree . var_scope ( var. 0 . local_id ) {
860
+ // Match exhaustively, so that it's easy to check how `ScheduleDrops` is used.
861
+ match schedule_drop {
862
+ ScheduleDrops :: Yes => {
863
+ self . schedule_drop ( span, region_scope, local_id, DropKind :: Storage ) ;
864
+ }
865
+ ScheduleDrops :: No => { }
866
+ }
841
867
}
842
868
Place :: from ( local_id)
843
869
}
@@ -2112,7 +2138,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
2112
2138
fake_borrows : & [ ( Place < ' tcx > , Local , FakeBorrowKind ) ] ,
2113
2139
scrutinee_span : Span ,
2114
2140
arm_match_scope : Option < ( & Arm < ' tcx > , region:: Scope ) > ,
2115
- schedule_drops : bool ,
2141
+ schedule_drops : ScheduleDrops ,
2116
2142
emit_storage_live : EmitStorageLive ,
2117
2143
) -> BasicBlock {
2118
2144
debug ! ( "bind_and_guard_matched_candidate(candidate={:?})" , candidate) ;
@@ -2323,10 +2349,14 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
2323
2349
let cause = FakeReadCause :: ForGuardBinding ;
2324
2350
self . cfg . push_fake_read ( post_guard_block, guard_end, cause, Place :: from ( local_id) ) ;
2325
2351
}
2326
- assert ! ( schedule_drops, "patterns with guards must schedule drops" ) ;
2352
+ assert_matches ! (
2353
+ schedule_drops,
2354
+ ScheduleDrops :: Yes ,
2355
+ "patterns with guards must schedule drops"
2356
+ ) ;
2327
2357
self . bind_matched_candidate_for_arm_body (
2328
2358
post_guard_block,
2329
- true ,
2359
+ ScheduleDrops :: Yes ,
2330
2360
by_value_bindings,
2331
2361
emit_storage_live,
2332
2362
) ;
@@ -2376,7 +2406,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
2376
2406
fn bind_matched_candidate_for_guard < ' b > (
2377
2407
& mut self ,
2378
2408
block : BasicBlock ,
2379
- schedule_drops : bool ,
2409
+ schedule_drops : ScheduleDrops ,
2380
2410
bindings : impl IntoIterator < Item = & ' b Binding < ' tcx > > ,
2381
2411
) where
2382
2412
' tcx : ' b ,
@@ -2429,7 +2459,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
2429
2459
fn bind_matched_candidate_for_arm_body < ' b > (
2430
2460
& mut self ,
2431
2461
block : BasicBlock ,
2432
- schedule_drops : bool ,
2462
+ schedule_drops : ScheduleDrops ,
2433
2463
bindings : impl IntoIterator < Item = & ' b Binding < ' tcx > > ,
2434
2464
emit_storage_live : EmitStorageLive ,
2435
2465
) where
@@ -2454,8 +2484,11 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
2454
2484
schedule_drops,
2455
2485
) ,
2456
2486
} ;
2457
- if schedule_drops {
2458
- self . schedule_drop_for_binding ( binding. var_id , binding. span , OutsideGuard ) ;
2487
+ match schedule_drops {
2488
+ ScheduleDrops :: Yes => {
2489
+ self . schedule_drop_for_binding ( binding. var_id , binding. span , OutsideGuard ) ;
2490
+ }
2491
+ ScheduleDrops :: No => { }
2459
2492
}
2460
2493
let rvalue = match binding. binding_mode . 0 {
2461
2494
ByRef :: No => Rvalue :: Use ( self . consume_by_copy_or_move ( binding. source ) ) ,
0 commit comments