rustc_mir: don't build unused unwind cleanup blocks

The unused blocks are removed by SimplifyCfg, but they can cause a
significant performance slowdown before they are removed.
This commit is contained in:
Ariel Ben-Yehuda 2017-07-31 23:25:27 +03:00
parent 1447daa01d
commit 85c102757a
8 changed files with 181 additions and 153 deletions

View File

@ -200,6 +200,15 @@ pub struct BreakableScope<'tcx> {
pub break_destination: Lvalue<'tcx>, pub break_destination: Lvalue<'tcx>,
} }
impl DropKind {
fn may_panic(&self) -> bool {
match *self {
DropKind::Value { .. } => true,
DropKind::Storage => false
}
}
}
impl<'tcx> Scope<'tcx> { impl<'tcx> Scope<'tcx> {
/// Invalidate all the cached blocks in the scope. /// Invalidate all the cached blocks in the scope.
/// ///
@ -337,9 +346,13 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
mut block: BasicBlock) mut block: BasicBlock)
-> BlockAnd<()> { -> BlockAnd<()> {
debug!("pop_scope({:?}, {:?})", extent, block); debug!("pop_scope({:?}, {:?})", extent, block);
// We need to have `cached_block`s available for all the drops, so we call diverge_cleanup // If we are emitting a `drop` statement, we need to have the cached
// to make sure all the `cached_block`s are filled in. // diverge cleanup pads ready in case that drop panics.
self.diverge_cleanup(); let may_panic =
self.scopes.last().unwrap().drops.iter().any(|s| s.kind.may_panic());
if may_panic {
self.diverge_cleanup();
}
let scope = self.scopes.pop().unwrap(); let scope = self.scopes.pop().unwrap();
assert_eq!(scope.extent, extent.0); assert_eq!(scope.extent, extent.0);
unpack!(block = build_scope_drops(&mut self.cfg, unpack!(block = build_scope_drops(&mut self.cfg,
@ -370,6 +383,15 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
let len = self.scopes.len(); let len = self.scopes.len();
assert!(scope_count < len, "should not use `exit_scope` to pop ALL scopes"); assert!(scope_count < len, "should not use `exit_scope` to pop ALL scopes");
let tmp = self.get_unit_temp(); let tmp = self.get_unit_temp();
// If we are emitting a `drop` statement, we need to have the cached
// diverge cleanup pads ready in case that drop panics.
let may_panic =
self.scopes[(len - scope_count)..].iter().any(|s| s.drops.iter().any(|s| s.kind.may_panic()));
if may_panic {
self.diverge_cleanup();
}
{ {
let mut rest = &mut self.scopes[(len - scope_count)..]; let mut rest = &mut self.scopes[(len - scope_count)..];
while let Some((scope, rest_)) = {rest}.split_last_mut() { while let Some((scope, rest_)) = {rest}.split_last_mut() {
@ -736,45 +758,48 @@ fn build_scope_drops<'tcx>(cfg: &mut CFG<'tcx>,
mut block: BasicBlock, mut block: BasicBlock,
arg_count: usize) arg_count: usize)
-> BlockAnd<()> { -> BlockAnd<()> {
debug!("build_scope_drops({:?} -> {:?})", block, scope);
let mut iter = scope.drops.iter().rev().peekable(); let mut iter = scope.drops.iter().rev().peekable();
while let Some(drop_data) = iter.next() { while let Some(drop_data) = iter.next() {
let source_info = scope.source_info(drop_data.span); let source_info = scope.source_info(drop_data.span);
if let DropKind::Value { .. } = drop_data.kind {
// Try to find the next block with its cached block
// for us to diverge into in case the drop panics.
let on_diverge = iter.peek().iter().filter_map(|dd| {
match dd.kind {
DropKind::Value { cached_block } => cached_block,
DropKind::Storage => None
}
}).next();
// If theres no `cached_block`s within current scope,
// we must look for one in the enclosing scope.
let on_diverge = on_diverge.or_else(||{
earlier_scopes.iter().rev().flat_map(|s| s.cached_block()).next()
});
let next = cfg.start_new_block();
cfg.terminate(block, source_info, TerminatorKind::Drop {
location: drop_data.location.clone(),
target: next,
unwind: on_diverge
});
block = next;
}
match drop_data.kind { match drop_data.kind {
DropKind::Value { .. } | DropKind::Value { .. } => {
DropKind::Storage => { // Try to find the next block with its cached block
// Only temps and vars need their storage dead. // for us to diverge into in case the drop panics.
match drop_data.location { let on_diverge = iter.peek().iter().filter_map(|dd| {
Lvalue::Local(index) if index.index() > arg_count => {} match dd.kind {
_ => continue DropKind::Value { cached_block: None } =>
} span_bug!(drop_data.span, "cached block not present?"),
DropKind::Value { cached_block } => cached_block,
DropKind::Storage => None
}
}).next();
// If theres no `cached_block`s within current scope,
// we must look for one in the enclosing scope.
let on_diverge = on_diverge.or_else(|| {
earlier_scopes.iter().rev().flat_map(|s| s.cached_block()).next()
});
let next = cfg.start_new_block();
cfg.terminate(block, source_info, TerminatorKind::Drop {
location: drop_data.location.clone(),
target: next,
unwind: on_diverge
});
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 {
Lvalue::Local(index) if index.index() > arg_count => {
cfg.push(block, Statement { cfg.push(block, Statement {
source_info: source_info, source_info: source_info,
kind: StatementKind::StorageDead(drop_data.location.clone()) kind: StatementKind::StorageDead(drop_data.location.clone())
}); });
} }
_ => continue
} }
} }
block.unit() block.unit()

View File

@ -47,39 +47,39 @@ fn main() {
// StorageDead(_3); // StorageDead(_3);
// StorageLive(_4); // StorageLive(_4);
// _4 = std::option::Option<std::boxed::Box<u32>>::None; // _4 = std::option::Option<std::boxed::Box<u32>>::None;
// StorageLive(_5);
// StorageLive(_6); // StorageLive(_6);
// StorageLive(_7); // _6 = _4;
// _7 = _4; // replace(_5 <- _6) -> [return: bb1, unwind: bb7];
// replace(_6 <- _7) -> [return: bb6, unwind: bb7];
// } // }
// bb1: { // bb1: {
// resume; // drop(_6) -> [return: bb8, unwind: bb5];
// } // }
// bb2: { // bb2: {
// drop(_4) -> bb1; // resume;
// } // }
// bb3: { // bb3: {
// goto -> bb2; // drop(_4) -> bb2;
// } // }
// bb4: { // bb4: {
// drop(_6) -> bb3; // goto -> bb3;
// } // }
// bb5: { // bb5: {
// goto -> bb4; // drop(_5) -> bb4;
// } // }
// bb6: { // bb6: {
// drop(_7) -> [return: bb8, unwind: bb4]; // goto -> bb5;
// } // }
// bb7: { // bb7: {
// drop(_7) -> bb5; // drop(_6) -> bb6;
// } // }
// bb8: { // bb8: {
// StorageDead(_7); // StorageDead(_6);
// _0 = (); // _0 = ();
// drop(_6) -> [return: bb9, unwind: bb2]; // drop(_5) -> [return: bb9, unwind: bb3];
// } // }
// bb9: { // bb9: {
// StorageDead(_6); // StorageDead(_5);
// drop(_4) -> bb10; // drop(_4) -> bb10;
// } // }
// bb10: { // bb10: {

View File

@ -32,41 +32,41 @@ fn foo(i: i32) {
// START rustc.node4.SimplifyCfg-qualify-consts.after.mir // START rustc.node4.SimplifyCfg-qualify-consts.after.mir
// let mut _0: (); // let mut _0: ();
// let _1: D; // let _1: D;
// let _3: i32; // let _2: i32;
// let _4: &'6_2rce i32; // let _3: &'6_2rce i32;
// let _7: &'6_4rce i32; // let _7: &'6_4rce i32;
// let mut _5: (); // let mut _4: ();
// let mut _6: i32; // let mut _5: i32;
// // let mut _6: ();
// bb0: { // bb0: {
// StorageLive(_1); // StorageLive(_1);
// _1 = D::{{constructor}}(const 0i32,); // _1 = D::{{constructor}}(const 0i32,);
// StorageLive(_2);
// _2 = const 0i32;
// StorageLive(_3); // StorageLive(_3);
// _3 = const 0i32; // _3 = &'6_2rce _2;
// StorageLive(_4); // StorageLive(_5);
// _4 = &'6_2rce _3; // _5 = (*_3);
// StorageLive(_6); // _4 = const foo(_5) -> [return: bb1, unwind: bb3];
// _6 = (*_4);
// _5 = const foo(_6) -> [return: bb2, unwind: bb3];
// } // }
// bb1: { // bb1: {
// resume; // StorageDead(_5);
// }
// bb2: {
// StorageDead(_6);
// StorageLive(_7); // StorageLive(_7);
// _7 = &'6_4rce _3; // _7 = &'6_4rce _2;
// _0 = (); // _0 = ();
// StorageDead(_7); // StorageDead(_7);
// EndRegion('6_4rce); // EndRegion('6_4rce);
// StorageDead(_4);
// EndRegion('6_2rce);
// StorageDead(_3); // StorageDead(_3);
// EndRegion('6_2rce);
// StorageDead(_2);
// drop(_1) -> bb4; // drop(_1) -> bb4;
// } // }
// bb2: {
// resume;
// }
// bb3: { // bb3: {
// EndRegion('6_2rce); // EndRegion('6_2rce);
// drop(_1) -> bb1; // drop(_1) -> bb2;
// } // }
// bb4: { // bb4: {
// StorageDead(_1); // StorageDead(_1);

View File

@ -31,32 +31,31 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// let mut _0: (); // let mut _0: ();
// let _1: D; // let _1: D;
// let mut _2: (); // let mut _2: ();
// let mut _3: (); // let mut _3: [closure@NodeId(18) d:&'19mce D];
// let mut _4: [closure@NodeId(18) d: &'19mce D]; // let mut _4: &'19mce D;
// let mut _5: &'19mce D; // let mut _5: ();
//
// bb0: { // bb0: {
// StorageLive(_1); // StorageLive(_1);
// _1 = D::{{constructor}}(const 0i32,); // _1 = D::{{constructor}}(const 0i32,);
// StorageLive(_3);
// StorageLive(_4); // StorageLive(_4);
// StorageLive(_5); // _4 = &'19mce _1;
// _5 = &'19mce _1; // _3 = [closure@NodeId(18)] { d: _4 };
// _4 = [closure@NodeId(18)] { d: _5 }; // StorageDead(_4);
// StorageDead(_5); // _2 = const foo(_3) -> [return: bb1, unwind: bb3];
// _3 = const foo(_4) -> [return: bb2, unwind: bb3];
// } // }
// bb1: { // bb1: {
// resume; // StorageDead(_3);
// }
// bb2: {
// StorageDead(_4);
// EndRegion('19mce); // EndRegion('19mce);
// _0 = (); // _0 = ();
// drop(_1) -> bb4; // drop(_1) -> bb4;
// } // }
// bb2: {
// resume;
// }
// bb3: { // bb3: {
// EndRegion('19mce); // EndRegion('19mce);
// drop(_1) -> bb1; // drop(_1) -> bb2;
// } // }
// bb4: { // bb4: {
// StorageDead(_1); // StorageDead(_1);

View File

@ -27,35 +27,35 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// END RUST SOURCE // END RUST SOURCE
// START rustc.node4.SimplifyCfg-qualify-consts.after.mir // START rustc.node4.SimplifyCfg-qualify-consts.after.mir
// fn main() -> () {
// let mut _0: (); // let mut _0: ();
// let _1: D; // let _1: D;
// let mut _2: (); // let mut _2: ();
// let mut _3: (); // let mut _3: [closure@NodeId(22) d:&'23mce D];
// let mut _4: [closure@NodeId(22) d:&'23mce D]; // let mut _4: &'23mce D;
// let mut _5: &'23mce D; // let mut _5: ();
//
// bb0: { // bb0: {
// StorageLive(_1); // StorageLive(_1);
// _1 = D::{{constructor}}(const 0i32,); // _1 = D::{{constructor}}(const 0i32,);
// StorageLive(_3);
// StorageLive(_4); // StorageLive(_4);
// StorageLive(_5); // _4 = &'23mce _1;
// _5 = &'23mce _1; // _3 = [closure@NodeId(22)] { d: _4 };
// _4 = [closure@NodeId(22)] { d: _5 }; // StorageDead(_4);
// StorageDead(_5); // _2 = const foo(_3) -> [return: bb1, unwind: bb3];
// _3 = const foo(_4) -> [return: bb2, unwind: bb3];
// } // }
// bb1: { // bb1: {
// resume; // StorageDead(_3);
// }
// bb2: {
// StorageDead(_4);
// EndRegion('23mce); // EndRegion('23mce);
// _0 = (); // _0 = ();
// drop(_1) -> bb4; // drop(_1) -> bb4;
// } // }
// bb2: {
// resume;
// }
// bb3: { // bb3: {
// EndRegion('23mce); // EndRegion('23mce);
// drop(_1) -> bb1; // drop(_1) -> bb2;
// } // }
// bb4: { // bb4: {
// StorageDead(_1); // StorageDead(_1);

View File

@ -31,18 +31,18 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// let mut _0: (); // let mut _0: ();
// let _1: D; // let _1: D;
// let mut _2: (); // let mut _2: ();
// let mut _3: (); // let mut _3: [closure@NodeId(22) d:D];
// let mut _4: [closure@NodeId(22) d:D]; // let mut _4: D;
// let mut _5: D; // let mut _5: ();
// //
// bb0: { // bb0: {
// StorageLive(_1); // StorageLive(_1);
// _1 = D::{{constructor}}(const 0i32,); // _1 = D::{{constructor}}(const 0i32,);
// StorageLive(_3);
// StorageLive(_4); // StorageLive(_4);
// StorageLive(_5); // _4 = _1;
// _5 = _1; // _3 = [closure@NodeId(22)] { d: _4 };
// _4 = [closure@NodeId(22)] { d: _5 }; // drop(_4) -> [return: bb4, unwind: bb3];
// drop(_5) -> [return: bb4, unwind: bb3];
// } // }
// bb1: { // bb1: {
// resume; // resume;
@ -51,17 +51,17 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// drop(_1) -> bb1; // drop(_1) -> bb1;
// } // }
// bb3: { // bb3: {
// drop(_4) -> bb2; // drop(_3) -> bb2;
// } // }
// bb4: { // bb4: {
// StorageDead(_5); // StorageDead(_4);
// _3 = const foo(_4) -> [return: bb5, unwind: bb3]; // _2 = const foo(_3) -> [return: bb5, unwind: bb3];
// } // }
// bb5: { // bb5: {
// drop(_4) -> [return: bb6, unwind: bb2]; // drop(_3) -> [return: bb6, unwind: bb2];
// } // }
// bb6: { // bb6: {
// StorageDead(_4); // StorageDead(_3);
// _0 = (); // _0 = ();
// drop(_1) -> bb7; // drop(_1) -> bb7;
// } // }
@ -76,16 +76,16 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// fn main::{{closure}}(_1: [closure@NodeId(22) d:D]) -> i32 { // fn main::{{closure}}(_1: [closure@NodeId(22) d:D]) -> i32 {
// let mut _0: i32; // let mut _0: i32;
// let _2: &'14_0rce D; // let _2: &'14_0rce D;
// let mut _3: (); // let mut _3: i32;
// let mut _4: i32; // let mut _4: ();
// //
// bb0: { // bb0: {
// StorageLive(_2); // StorageLive(_2);
// _2 = &'14_0rce (_1.0: D); // _2 = &'14_0rce (_1.0: D);
// StorageLive(_4); // StorageLive(_3);
// _4 = ((*_2).0: i32); // _3 = ((*_2).0: i32);
// _0 = _4; // _0 = _3;
// StorageDead(_4); // StorageDead(_3);
// StorageDead(_2); // StorageDead(_2);
// EndRegion('14_0rce); // EndRegion('14_0rce);
// drop(_1) -> bb1; // drop(_1) -> bb1;

View File

@ -29,44 +29,43 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
// END RUST SOURCE // END RUST SOURCE
// START rustc.node4.SimplifyCfg-qualify-consts.after.mir // START rustc.node4.SimplifyCfg-qualify-consts.after.mir
// fn main() -> () { // fn main() -> () {
// let mut _0: (); // let mut _0: ();
// let _1: D; // let _1: D;
// let _3: &'6_1rce D; // let _2: &'6_1rce D;
// let mut _2: (); // let mut _3: ();
// let mut _4: (); // let mut _4: [closure@NodeId(22) r:&'6_1rce D];
// let mut _5: [closure@NodeId(22) r:&'6_1rce D]; // let mut _5: &'6_1rce D;
// let mut _6: &'6_1rce D; // let mut _6: ();
// // bb0: {
// bb0: { // StorageLive(_1);
// StorageLive(_1); // _1 = D::{{constructor}}(const 0i32,);
// _1 = D::{{constructor}}(const 0i32,); // StorageLive(_2);
// StorageLive(_3); // _2 = &'6_1rce _1;
// _3 = &'6_1rce _1; // StorageLive(_4);
// StorageLive(_5); // StorageLive(_5);
// StorageLive(_6); // _5 = _2;
// _6 = _3; // _4 = [closure@NodeId(22)] { r: _5 };
// _5 = [closure@NodeId(22)] { r: _6 }; // StorageDead(_5);
// StorageDead(_6); // _3 = const foo(_4) -> [return: bb1, unwind: bb3];
// _4 = const foo(_5) -> [return: bb2, unwind: bb3]; // }
// } // bb1: {
// bb1: { // StorageDead(_4);
// resume; // _0 = ();
// } // StorageDead(_2);
// bb2: { // EndRegion('6_1rce);
// StorageDead(_5); // drop(_1) -> bb4;
// _0 = (); // }
// StorageDead(_3); // bb2: {
// EndRegion('6_1rce); // resume;
// drop(_1) -> bb4; // }
// } // bb3: {
// bb3: { // EndRegion('6_1rce);
// EndRegion('6_1rce); // drop(_1) -> bb2;
// drop(_1) -> bb1; // }
// } // bb4: {
// bb4: { // StorageDead(_1);
// StorageDead(_1); // return;
// return; // }
// }
// } // }
// END rustc.node4.SimplifyCfg-qualify-consts.after.mir // END rustc.node4.SimplifyCfg-qualify-consts.after.mir

View File

@ -34,18 +34,23 @@ impl S {
// END RUST SOURCE // END RUST SOURCE
// START rustc.node4.ElaborateDrops.after.mir // START rustc.node4.ElaborateDrops.after.mir
// let mut _0: ();
// let _1: ();
// let mut _2: S; // let mut _2: S;
// let mut _3: (); // let mut _3: S;
// let mut _4: S; // let mut _4: S;
// let mut _5: S; // let mut _5: ();
// let mut _6: bool; // let mut _6: bool;
// //
// bb0: { // bb0: {
// END rustc.node4.ElaborateDrops.after.mir // END rustc.node4.ElaborateDrops.after.mir
// START rustc.node13.ElaborateDrops.after.mir // START rustc.node13.ElaborateDrops.after.mir
// let mut _2: (); // let mut _0: ();
// let mut _4: (); // let _1: S;
// let mut _5: S; // let mut _2: S;
// let mut _3: ();
// let mut _4: S;
// let mut _5: ();
// let mut _6: S; // let mut _6: S;
// let mut _7: bool; // let mut _7: bool;
// //