rust/src/analyze.rs

59 lines
1.5 KiB
Rust
Raw Normal View History

2018-08-09 03:46:56 -05:00
use crate::prelude::*;
use crate::rustc::mir::StatementKind::*;
2018-08-09 03:46:56 -05:00
bitflags! {
pub struct Flags: u8 {
const NOT_SSA = 0b00000001;
}
}
2018-08-14 13:31:16 -05:00
pub fn analyze<'a, 'tcx: 'a>(fx: &FunctionCx<'a, 'tcx, impl Backend>) -> HashMap<Local, Flags> {
2018-08-09 03:46:56 -05:00
let mut flag_map = HashMap::new();
for local in fx.mir.local_decls.indices() {
flag_map.insert(local, Flags::empty());
}
not_ssa(&mut flag_map, RETURN_PLACE);
for (local, local_decl) in fx.mir.local_decls.iter_enumerated() {
if fx.cton_type(local_decl.ty).is_none() {
not_ssa(&mut flag_map, local);
}
}
for bb in fx.mir.basic_blocks().iter() {
for stmt in bb.statements.iter() {
match &stmt.kind {
2018-10-10 12:07:13 -05:00
Assign(_, rval) => match &**rval {
Rvalue::Ref(_, _, place) => analyze_non_ssa_place(&mut flag_map, place),
_ => {}
},
2018-08-09 03:46:56 -05:00
_ => {}
}
}
match &bb.terminator().kind {
2018-08-09 04:25:14 -05:00
TerminatorKind::Call {
destination: Some((place, _)),
..
} => analyze_non_ssa_place(&mut flag_map, place),
2018-08-09 03:46:56 -05:00
_ => {}
}
}
flag_map
}
fn analyze_non_ssa_place(flag_map: &mut HashMap<Local, Flags>, place: &Place) {
match place {
Place::Local(local) => not_ssa(flag_map, local),
_ => {}
}
}
fn not_ssa<L: ::std::borrow::Borrow<Local>>(flag_map: &mut HashMap<Local, Flags>, local: L) {
*flag_map.get_mut(local.borrow()).unwrap() |= Flags::NOT_SSA;
2018-08-09 04:25:14 -05:00
}