-
Notifications
You must be signed in to change notification settings - Fork 5
/
transform_cleanup.ml
54 lines (51 loc) · 1.88 KB
/
transform_cleanup.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
open Instr
open Types
open Transform_utils
let remove_jmp : transform_instructions = fun ({instrs; _} as inp) ->
let pred = Analysis.predecessors instrs in
let succ = Analysis.successors instrs in
let transform pc =
if (pc+1) = Array.length instrs then Unchanged else
match[@warning "-4"] instrs.(pc), instrs.(pc+1) with
| Goto l1, Label (MergeLabel l2) when l1 = l2 && pred.(pc+1) = [pc] ->
Remove 2
| Label l, _ when
pred.(pc) = [pc-1] && succ.(pc-1) = [pc] ->
(* A label is unused if the previous instruction is the only predecessor
* unless the previous instruction jumps to it. The later can happen
* if its a goto (then we already remove it -- see above) or if its a branch (which
* is excluded by the second tests "succ (pc-1) = [pc]")
* TODO: we should implement some generic api for instructions like, "Instr.is_jmp" *)
Remove 1
| i, _ ->
Unchanged
in
change_instrs transform inp
let remove_unreachable_code : transform_instructions = fun ({instrs} as inp) ->
let reachable =
let merge _ _ _ = None in
let update _ _ = () in
Analysis.forward_analysis () instrs merge update
in
let transform pc =
match reachable pc with
| exception Analysis.UnreachableCode _ -> Remove 1
| () -> Unchanged
in
change_instrs transform inp
let remove_unused_decl : transform_instructions = fun ({instrs} as inp) ->
let open Analysis in
let required = Analysis.required inp in
let uses = Analysis.uses inp in
let aliased = Analysis.aliased inp in
let aliased var pc = VarSet.mem var (aliased pc) in
let transform pc =
match[@warning "-4"] instrs.(pc) with
| Decl_var (x, _) when PcSet.is_empty (required pc) ->
Remove 1
| Assign (x, _) when PcSet.is_empty (uses pc) && not (aliased x pc) ->
Remove 1
| _ ->
Unchanged
in
change_instrs transform inp