2015-11-10 14:38:36 -06:00
|
|
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2016-03-30 06:43:36 -05:00
|
|
|
use rustc::middle::const_val::ConstVal;
|
2016-03-22 10:30:57 -05:00
|
|
|
use rustc::ty::TyCtxt;
|
2015-11-19 09:37:34 -06:00
|
|
|
use rustc::mir::repr::*;
|
2016-05-02 16:26:41 -05:00
|
|
|
use rustc::mir::transform::{MirPass, MirSource, Pass};
|
2016-03-22 15:19:44 -05:00
|
|
|
use pretty;
|
2016-02-26 10:05:50 -06:00
|
|
|
|
|
|
|
use super::remove_dead_blocks::RemoveDeadBlocks;
|
2015-11-10 14:38:36 -06:00
|
|
|
|
|
|
|
pub struct SimplifyCfg;
|
|
|
|
|
|
|
|
impl SimplifyCfg {
|
|
|
|
pub fn new() -> SimplifyCfg {
|
|
|
|
SimplifyCfg
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove_goto_chains(&self, mir: &mut Mir) -> bool {
|
|
|
|
// Find the target at the end of the jump chain, return None if there is a loop
|
|
|
|
fn final_target(mir: &Mir, mut target: BasicBlock) -> Option<BasicBlock> {
|
|
|
|
// Keep track of already seen blocks to detect loops
|
|
|
|
let mut seen: Vec<BasicBlock> = Vec::with_capacity(8);
|
|
|
|
|
|
|
|
while mir.basic_block_data(target).statements.is_empty() {
|
2016-03-22 15:19:44 -05:00
|
|
|
// NB -- terminator may have been swapped with `None`
|
|
|
|
// below, in which case we have a cycle and just want
|
|
|
|
// to stop
|
|
|
|
if let Some(ref terminator) = mir.basic_block_data(target).terminator {
|
|
|
|
match terminator.kind {
|
|
|
|
TerminatorKind::Goto { target: next } => {
|
|
|
|
if seen.contains(&next) {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
seen.push(next);
|
|
|
|
target = next;
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
2016-03-22 15:19:44 -05:00
|
|
|
_ => break
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
2016-03-22 15:19:44 -05:00
|
|
|
} else {
|
|
|
|
break
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(target)
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut changed = false;
|
|
|
|
for bb in mir.all_basic_blocks() {
|
2015-12-18 16:44:32 -06:00
|
|
|
// Temporarily take ownership of the terminator we're modifying to keep borrowck happy
|
|
|
|
let mut terminator = mir.basic_block_data_mut(bb).terminator.take()
|
|
|
|
.expect("invalid terminator state");
|
2015-11-10 14:38:36 -06:00
|
|
|
|
2016-03-22 15:05:28 -05:00
|
|
|
debug!("remove_goto_chains: bb={:?} terminator={:?}", bb, terminator);
|
|
|
|
|
2015-11-10 14:38:36 -06:00
|
|
|
for target in terminator.successors_mut() {
|
|
|
|
let new_target = match final_target(mir, *target) {
|
|
|
|
Some(new_target) => new_target,
|
|
|
|
None if mir.basic_block_data(bb).statements.is_empty() => bb,
|
|
|
|
None => continue
|
|
|
|
};
|
|
|
|
changed |= *target != new_target;
|
|
|
|
*target = new_target;
|
|
|
|
}
|
2015-12-18 16:44:32 -06:00
|
|
|
mir.basic_block_data_mut(bb).terminator = Some(terminator);
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
|
|
|
changed
|
|
|
|
}
|
|
|
|
|
|
|
|
fn simplify_branches(&self, mir: &mut Mir) -> bool {
|
|
|
|
let mut changed = false;
|
|
|
|
|
|
|
|
for bb in mir.all_basic_blocks() {
|
2015-12-18 16:44:32 -06:00
|
|
|
let basic_block = mir.basic_block_data_mut(bb);
|
|
|
|
let mut terminator = basic_block.terminator_mut();
|
2016-03-10 08:55:15 -06:00
|
|
|
terminator.kind = match terminator.kind {
|
|
|
|
TerminatorKind::If { ref targets, .. } if targets.0 == targets.1 => {
|
2015-11-10 14:38:36 -06:00
|
|
|
changed = true;
|
2016-03-10 08:55:15 -06:00
|
|
|
TerminatorKind::Goto { target: targets.0 }
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
2016-01-31 11:17:15 -06:00
|
|
|
|
2016-03-10 08:55:15 -06:00
|
|
|
TerminatorKind::If { ref targets, cond: Operand::Constant(Constant {
|
2015-11-10 14:38:36 -06:00
|
|
|
literal: Literal::Value {
|
|
|
|
value: ConstVal::Bool(cond)
|
|
|
|
}, ..
|
|
|
|
}) } => {
|
|
|
|
changed = true;
|
2015-12-10 14:46:40 -06:00
|
|
|
if cond {
|
2016-03-10 08:55:15 -06:00
|
|
|
TerminatorKind::Goto { target: targets.0 }
|
2015-12-10 14:46:40 -06:00
|
|
|
} else {
|
2016-03-10 08:55:15 -06:00
|
|
|
TerminatorKind::Goto { target: targets.1 }
|
2015-12-10 14:46:40 -06:00
|
|
|
}
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
2016-01-31 11:17:15 -06:00
|
|
|
|
2016-03-10 08:55:15 -06:00
|
|
|
TerminatorKind::SwitchInt { ref targets, .. } if targets.len() == 1 => {
|
|
|
|
TerminatorKind::Goto { target: targets[0] }
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
2015-12-18 16:44:32 -06:00
|
|
|
_ => continue
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
changed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-26 10:05:50 -06:00
|
|
|
impl<'tcx> MirPass<'tcx> for SimplifyCfg {
|
2016-05-02 21:23:22 -05:00
|
|
|
fn run_pass<'a>(&mut self, tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|
|
|
src: MirSource, mir: &mut Mir<'tcx>) {
|
2016-03-22 15:19:44 -05:00
|
|
|
let mut counter = 0;
|
2015-11-10 14:38:36 -06:00
|
|
|
let mut changed = true;
|
|
|
|
while changed {
|
2016-05-02 16:26:41 -05:00
|
|
|
pretty::dump_mir(tcx, "simplify_cfg", &counter, src, mir, None);
|
2016-03-22 15:19:44 -05:00
|
|
|
counter += 1;
|
2015-11-10 14:38:36 -06:00
|
|
|
changed = self.simplify_branches(mir);
|
|
|
|
changed |= self.remove_goto_chains(mir);
|
2016-05-02 16:26:41 -05:00
|
|
|
RemoveDeadBlocks.run_pass(tcx, src, mir);
|
2015-11-10 14:38:36 -06:00
|
|
|
}
|
|
|
|
// FIXME: Should probably be moved into some kind of pass manager
|
|
|
|
mir.basic_blocks.shrink_to_fit();
|
|
|
|
}
|
|
|
|
}
|
2016-01-31 11:17:15 -06:00
|
|
|
|
2016-02-26 10:05:50 -06:00
|
|
|
impl Pass for SimplifyCfg {}
|