@@ -725,30 +725,79 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
725
725
///
726
726
/// Unlike other kinds of early exits, tail calls do not go through the drop tree.
727
727
/// Instead, all scheduled drops are immediately added to the CFG.
728
- pub ( crate ) fn break_for_tail_call ( & mut self , mut block : BasicBlock ) -> BlockAnd < ( ) > {
728
+ pub ( crate ) fn break_for_tail_call (
729
+ & mut self ,
730
+ mut block : BasicBlock ,
731
+ args : & [ Operand < ' tcx > ] ,
732
+ source_info : SourceInfo ,
733
+ ) -> BlockAnd < ( ) > {
734
+ let arg_drops: Vec < _ > = args
735
+ . iter ( )
736
+ . rev ( )
737
+ . filter_map ( |arg| match arg {
738
+ Operand :: Copy ( _) => bug ! ( "copy op in tail call args" ) ,
739
+ Operand :: Move ( place) => {
740
+ let local =
741
+ place. as_local ( ) . unwrap_or_else ( || bug ! ( "projection in tail call args" ) ) ;
742
+
743
+ Some ( DropData { source_info, local, kind : DropKind :: Value } )
744
+ }
745
+ Operand :: Constant ( _) => None ,
746
+ } )
747
+ . collect ( ) ;
748
+
749
+ let mut unwind_to = self . diverge_cleanup_target (
750
+ self . scopes . scopes . iter ( ) . rev ( ) . nth ( 1 ) . unwrap ( ) . region_scope ,
751
+ DUMMY_SP ,
752
+ ) ;
753
+ let unwind_drops = & mut self . scopes . unwind_drops ;
754
+
729
755
// the innermost scope contains only the destructors for the tail call arguments
730
756
// we only want to drop these in case of a panic, so we skip it
731
757
for scope in self . scopes . scopes [ 1 ..] . iter ( ) . rev ( ) . skip ( 1 ) {
732
- for drop in scope. drops . iter ( ) . rev ( ) {
733
- match drop. kind {
758
+ // FIXME(explicit_tail_calls) code duplication with `build_scope_drops`
759
+ for drop_data in scope. drops . iter ( ) . rev ( ) {
760
+ let source_info = drop_data. source_info ;
761
+ let local = drop_data. local ;
762
+
763
+ match drop_data. kind {
734
764
DropKind :: Value => {
735
- let target = self . cfg . start_new_block ( ) ;
736
- let terminator = TerminatorKind :: Drop {
737
- target,
738
- // The caller will handle this if needed.
739
- unwind : UnwindAction :: Terminate ,
740
- place : drop. local . into ( ) ,
741
- replace : false ,
742
- } ;
743
- self . cfg . terminate ( block, drop. source_info , terminator) ;
744
- block = target;
765
+ // `unwind_to` should drop the value that we're about to
766
+ // schedule. If dropping this value panics, then we continue
767
+ // with the *next* value on the unwind path.
768
+ debug_assert_eq ! ( unwind_drops. drops[ unwind_to] . 0 . local, drop_data. local) ;
769
+ debug_assert_eq ! ( unwind_drops. drops[ unwind_to] . 0 . kind, drop_data. kind) ;
770
+ unwind_to = unwind_drops. drops [ unwind_to] . 1 ;
771
+
772
+ let mut unwind_entry_point = unwind_to;
773
+
774
+ // the tail call arguments must be dropped if any of these drops panic
775
+ for drop in arg_drops. iter ( ) . copied ( ) {
776
+ unwind_entry_point = unwind_drops. add_drop ( drop, unwind_entry_point) ;
777
+ }
778
+
779
+ unwind_drops. add_entry ( block, unwind_entry_point) ;
780
+
781
+ let next = self . cfg . start_new_block ( ) ;
782
+ self . cfg . terminate (
783
+ block,
784
+ source_info,
785
+ TerminatorKind :: Drop {
786
+ place : local. into ( ) ,
787
+ target : next,
788
+ unwind : UnwindAction :: Continue ,
789
+ replace : false ,
790
+ } ,
791
+ ) ;
792
+ block = next;
745
793
}
746
794
DropKind :: Storage => {
747
- let stmt = Statement {
748
- source_info : drop. source_info ,
749
- kind : StatementKind :: StorageDead ( drop. local ) ,
750
- } ;
751
- self . cfg . push ( block, stmt) ;
795
+ // Only temps and vars need their storage dead.
796
+ assert ! ( local. index( ) > self . arg_count) ;
797
+ self . cfg . push (
798
+ block,
799
+ Statement { source_info, kind : StatementKind :: StorageDead ( local) } ,
800
+ ) ;
752
801
}
753
802
}
754
803
}
0 commit comments