From 3131427784b2c9f906a50b290f7d3cc215d0c0e8 Mon Sep 17 00:00:00 2001 From: Matthew Jasper Date: Sat, 15 Jun 2019 20:33:23 +0100 Subject: [PATCH] Use `Local`s instead of `Place`s in MIR drop generation --- src/librustc_mir/build/expr/as_rvalue.rs | 4 +- src/librustc_mir/build/expr/as_temp.rs | 4 +- src/librustc_mir/build/matches/mod.rs | 7 +- src/librustc_mir/build/mod.rs | 2 +- src/librustc_mir/build/scope.rs | 164 ++++++++++------------- 5 files changed, 81 insertions(+), 100 deletions(-) diff --git a/src/librustc_mir/build/expr/as_rvalue.rs b/src/librustc_mir/build/expr/as_rvalue.rs index 73ce2a5dc9b8c..17e7b1acc68f0 100644 --- a/src/librustc_mir/build/expr/as_rvalue.rs +++ b/src/librustc_mir/build/expr/as_rvalue.rs @@ -127,7 +127,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { this.schedule_drop_storage_and_value( expr_span, scope, - &Place::from(result), + result, value.ty, ); } @@ -559,7 +559,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { this.schedule_drop_storage_and_value( upvar_span, temp_lifetime, - &Place::from(temp), + temp, upvar_ty, ); } diff --git a/src/librustc_mir/build/expr/as_temp.rs b/src/librustc_mir/build/expr/as_temp.rs index 1b3ebac4a3d4d..1fe6be8bbc82e 100644 --- a/src/librustc_mir/build/expr/as_temp.rs +++ b/src/librustc_mir/build/expr/as_temp.rs @@ -88,7 +88,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { this.schedule_drop( expr_span, temp_lifetime, - temp_place, + temp, expr_ty, DropKind::Storage, ); @@ -101,7 +101,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { this.schedule_drop( expr_span, temp_lifetime, - temp_place, + temp, expr_ty, DropKind::Value, ); diff --git a/src/librustc_mir/build/matches/mod.rs b/src/librustc_mir/build/matches/mod.rs index cebfa681b5c47..f831f5105a468 100644 --- a/src/librustc_mir/build/matches/mod.rs +++ b/src/librustc_mir/build/matches/mod.rs @@ -531,11 +531,10 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { kind: StatementKind::StorageLive(local_id), }, ); - let place = Place::from(local_id); let var_ty = self.local_decls[local_id].ty; let region_scope = self.hir.region_scope_tree.var_scope(var.local_id); - self.schedule_drop(span, region_scope, &place, var_ty, DropKind::Storage); - place + self.schedule_drop(span, region_scope, local_id, var_ty, DropKind::Storage); + Place::Base(PlaceBase::Local(local_id)) } pub fn schedule_drop_for_binding(&mut self, var: HirId, span: Span, for_guard: ForGuard) { @@ -545,7 +544,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { self.schedule_drop( span, region_scope, - &Place::from(local_id), + local_id, var_ty, DropKind::Value, ); diff --git a/src/librustc_mir/build/mod.rs b/src/librustc_mir/build/mod.rs index ff4841cb57fbd..ad970de466cfd 100644 --- a/src/librustc_mir/build/mod.rs +++ b/src/librustc_mir/build/mod.rs @@ -809,7 +809,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { // Make sure we drop (parts of) the argument even when not matched on. self.schedule_drop( pattern.as_ref().map_or(ast_body.span, |pat| pat.span), - argument_scope, &place, ty, DropKind::Value, + argument_scope, local, ty, DropKind::Value, ); if let Some(pattern) = pattern { diff --git a/src/librustc_mir/build/scope.rs b/src/librustc_mir/build/scope.rs index ec2e970906c79..1b5fa1c9770f1 100644 --- a/src/librustc_mir/build/scope.rs +++ b/src/librustc_mir/build/scope.rs @@ -94,7 +94,7 @@ use std::collections::hash_map::Entry; use std::mem; #[derive(Debug)] -struct Scope<'tcx> { +struct Scope { /// The source scope this scope was created in. source_scope: SourceScope, @@ -121,7 +121,7 @@ struct Scope<'tcx> { /// out empty but grows as variables are declared during the /// building process. This is a stack, so we always drop from the /// end of the vector (top of the stack) first. - drops: Vec>, + drops: Vec, /// The cache for drop chain on “normal” exit into a particular BasicBlock. cached_exits: FxHashMap<(BasicBlock, region::Scope), BasicBlock>, @@ -135,18 +135,18 @@ struct Scope<'tcx> { #[derive(Debug, Default)] pub struct Scopes<'tcx> { - scopes: Vec>, + scopes: Vec, /// The current set of breakable scopes. See module comment for more details. breakable_scopes: Vec>, } #[derive(Debug)] -struct DropData<'tcx> { +struct DropData { /// span where drop obligation was incurred (typically where place was declared) span: Span, - /// place to drop - location: Place<'tcx>, + /// local to drop + local: Local, /// Whether this is a value Drop or a StorageDead. kind: DropKind, @@ -223,7 +223,7 @@ impl CachedBlock { } } -impl<'tcx> Scope<'tcx> { +impl Scope { /// Invalidates all the cached blocks in the scope. /// /// Should always be run for all inner scopes when a drop is pushed into some scope enclosing a @@ -285,7 +285,7 @@ impl<'tcx> Scopes<'tcx> { fn pop_scope( &mut self, region_scope: (region::Scope, SourceInfo), - ) -> (Scope<'tcx>, Option) { + ) -> (Scope, Option) { let scope = self.scopes.pop().unwrap(); assert_eq!(scope.region_scope, region_scope.0); let unwind_to = self.scopes.last() @@ -343,11 +343,11 @@ impl<'tcx> Scopes<'tcx> { scope_count } - fn iter_mut(&mut self) -> impl DoubleEndedIterator> + '_ { + fn iter_mut(&mut self) -> impl DoubleEndedIterator + '_ { self.scopes.iter_mut().rev() } - fn top_scopes(&mut self, count: usize) -> impl DoubleEndedIterator> + '_ { + fn top_scopes(&mut self, count: usize) -> impl DoubleEndedIterator + '_ { let len = self.len(); self.scopes[len - count..].iter_mut() } @@ -717,11 +717,11 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { &mut self, span: Span, region_scope: region::Scope, - place: &Place<'tcx>, + local: Local, place_ty: Ty<'tcx>, ) { - self.schedule_drop(span, region_scope, place, place_ty, DropKind::Storage); - self.schedule_drop(span, region_scope, place, place_ty, DropKind::Value); + self.schedule_drop(span, region_scope, local, place_ty, DropKind::Storage); + self.schedule_drop(span, region_scope, local, place_ty, DropKind::Value); } /// Indicates that `place` should be dropped on exit from @@ -733,7 +733,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { &mut self, span: Span, region_scope: region::Scope, - place: &Place<'tcx>, + local: Local, place_ty: Ty<'tcx>, drop_kind: DropKind, ) { @@ -741,17 +741,12 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { match drop_kind { DropKind::Value => if !needs_drop { return }, DropKind::Storage => { - match *place { - Place::Base(PlaceBase::Local(index)) => if index.index() <= self.arg_count { - span_bug!( - span, "`schedule_drop` called with index {} and arg_count {}", - index.index(), - self.arg_count, - ) - }, - _ => span_bug!( - span, "`schedule_drop` called with non-`Local` place {:?}", place - ), + if local.index() <= self.arg_count { + span_bug!( + span, "`schedule_drop` called with local {:?} and arg_count {}", + local, + self.arg_count, + ) } } } @@ -817,14 +812,14 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { scope.drops.push(DropData { span: scope_end, - location: place.clone(), + local, kind: drop_kind, cached_block: CachedBlock::default(), }); return; } } - span_bug!(span, "region scope {:?} not in scope to drop {:?}", region_scope, place); + span_bug!(span, "region scope {:?} not in scope to drop {:?}", region_scope, local); } // Other @@ -867,29 +862,23 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { bug!("Drop scheduled on top of condition variable") } DropKind::Storage => { - // Drop the storage for both value and storage drops. - // Only temps and vars need their storage dead. - match top_drop_data.location { - Place::Base(PlaceBase::Local(index)) => { - let source_info = top_scope.source_info(top_drop_data.span); - assert_eq!(index, cond_temp, "Drop scheduled on top of condition"); - self.cfg.push( - true_block, - Statement { - source_info, - kind: StatementKind::StorageDead(index) - }, - ); - self.cfg.push( - false_block, - Statement { - source_info, - kind: StatementKind::StorageDead(index) - }, - ); - } - _ => unreachable!(), - } + let source_info = top_scope.source_info(top_drop_data.span); + let local = top_drop_data.local; + assert_eq!(local, cond_temp, "Drop scheduled on top of condition"); + self.cfg.push( + true_block, + Statement { + source_info, + kind: StatementKind::StorageDead(local) + }, + ); + self.cfg.push( + false_block, + Statement { + source_info, + kind: StatementKind::StorageDead(local) + }, + ); } } @@ -1020,7 +1009,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { fn build_scope_drops<'tcx>( cfg: &mut CFG<'tcx>, is_generator: bool, - scope: &Scope<'tcx>, + scope: &Scope, mut block: BasicBlock, last_unwind_to: BasicBlock, arg_count: usize, @@ -1050,6 +1039,7 @@ fn build_scope_drops<'tcx>( for drop_idx in (0..scope.drops.len()).rev() { let drop_data = &scope.drops[drop_idx]; let source_info = scope.source_info(drop_data.span); + let local = drop_data.local; match drop_data.kind { DropKind::Value => { let unwind_to = get_unwind_to(scope, is_generator, drop_idx, generator_drop) @@ -1057,32 +1047,27 @@ fn build_scope_drops<'tcx>( let next = cfg.start_new_block(); cfg.terminate(block, source_info, TerminatorKind::Drop { - location: drop_data.location.clone(), + location: local.into(), target: next, unwind: Some(unwind_to) }); block = next; } DropKind::Storage => { - // Drop the storage for both value and storage drops. // Only temps and vars need their storage dead. - match drop_data.location { - Place::Base(PlaceBase::Local(index)) if index.index() > arg_count => { - cfg.push(block, Statement { - source_info, - kind: StatementKind::StorageDead(index) - }); - } - _ => unreachable!(), - } + assert!(local.index() > arg_count); + cfg.push(block, Statement { + source_info, + kind: StatementKind::StorageDead(local) + }); } } } block.unit() } -fn get_unwind_to<'tcx>( - scope: &Scope<'tcx>, +fn get_unwind_to( + scope: &Scope, is_generator: bool, unwind_from: usize, generator_drop: bool, @@ -1108,7 +1093,7 @@ fn get_unwind_to<'tcx>( fn build_diverge_scope<'tcx>(cfg: &mut CFG<'tcx>, span: Span, - scope: &mut Scope<'tcx>, + scope: &mut Scope, mut target: BasicBlock, generator_drop: bool, is_generator: bool) @@ -1152,26 +1137,20 @@ fn build_diverge_scope<'tcx>(cfg: &mut CFG<'tcx>, // this is not what clang does. match drop_data.kind { DropKind::Storage if is_generator => { - // Only temps and vars need their storage dead. - match drop_data.location { - Place::Base(PlaceBase::Local(index)) => { - storage_deads.push(Statement { - source_info: source_info(drop_data.span), - kind: StatementKind::StorageDead(index) - }); - if !target_built_by_us { - // We cannot add statements to an existing block, so we create a new - // block for our StorageDead statements. - let block = cfg.start_new_cleanup_block(); - let source_info = SourceInfo { span: DUMMY_SP, scope: source_scope }; - cfg.terminate(block, source_info, - TerminatorKind::Goto { target: target }); - target = block; - target_built_by_us = true; - } - } - _ => unreachable!(), - }; + storage_deads.push(Statement { + source_info: source_info(drop_data.span), + kind: StatementKind::StorageDead(drop_data.local) + }); + if !target_built_by_us { + // We cannot add statements to an existing block, so we create a new + // block for our StorageDead statements. + let block = cfg.start_new_cleanup_block(); + let source_info = SourceInfo { span: DUMMY_SP, scope: source_scope }; + cfg.terminate(block, source_info, + TerminatorKind::Goto { target: target }); + target = block; + target_built_by_us = true; + } *drop_data.cached_block.ref_mut(generator_drop) = Some(target); } DropKind::Storage => {} @@ -1184,12 +1163,15 @@ fn build_diverge_scope<'tcx>(cfg: &mut CFG<'tcx>, } else { push_storage_deads(cfg, target, &mut storage_deads); let block = cfg.start_new_cleanup_block(); - cfg.terminate(block, source_info(drop_data.span), - TerminatorKind::Drop { - location: drop_data.location.clone(), - target, - unwind: None - }); + cfg.terminate( + block, + source_info(drop_data.span), + TerminatorKind::Drop { + location: drop_data.local.into(), + target, + unwind: None + }, + ); *cached_block = Some(block); target_built_by_us = true; block