]> git.lizzy.rs Git - rust.git/blob - src/analyze.rs
Rustup to rustc 1.37.0-nightly (2887008e0 2019-06-12)
[rust.git] / src / analyze.rs
1 use crate::prelude::*;
2
3 use rustc::mir::StatementKind::*;
4
5 bitflags::bitflags! {
6     pub struct Flags: u8 {
7         const NOT_SSA = 0b00000001;
8     }
9 }
10
11 pub fn analyze<'a, 'tcx: 'a>(fx: &FunctionCx<'a, 'tcx, impl Backend>) -> HashMap<Local, Flags> {
12     let mut flag_map = HashMap::new();
13
14     for local in fx.mir.local_decls.indices() {
15         flag_map.insert(local, Flags::empty());
16     }
17
18     not_ssa(&mut flag_map, RETURN_PLACE);
19
20     for (local, local_decl) in fx.mir.local_decls.iter_enumerated() {
21         if fx.clif_type(local_decl.ty).is_none() {
22             not_ssa(&mut flag_map, local);
23         }
24     }
25
26     for bb in fx.mir.basic_blocks().iter() {
27         for stmt in bb.statements.iter() {
28             match &stmt.kind {
29                 Assign(_, rval) => match &**rval {
30                     Rvalue::Ref(_, _, place) => analyze_non_ssa_place(&mut flag_map, place),
31                     _ => {}
32                 },
33                 _ => {}
34             }
35         }
36
37         match &bb.terminator().kind {
38             TerminatorKind::Call {
39                 destination: Some((place, _)),
40                 ..
41             } => analyze_non_ssa_place(&mut flag_map, place),
42             _ => {}
43         }
44     }
45
46     flag_map
47 }
48
49 fn analyze_non_ssa_place(flag_map: &mut HashMap<Local, Flags>, place: &Place) {
50     match place {
51         Place::Base(PlaceBase::Local(local)) => not_ssa(flag_map, local),
52         _ => {}
53     }
54 }
55
56 fn not_ssa<L: ::std::borrow::Borrow<Local>>(flag_map: &mut HashMap<Local, Flags>, local: L) {
57     *flag_map.get_mut(local.borrow()).unwrap() |= Flags::NOT_SSA;
58 }