-
Notifications
You must be signed in to change notification settings - Fork 5
/
edit.ml
121 lines (104 loc) · 3.6 KB
/
edit.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
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
open Instr
open Types
module StringSet = Set.Make(String)
let fresh_one (existing : StringSet.t) (name : string) : string =
let rec get_basename pos =
if pos = 0
then name
else if String.get name pos = '_'
then String.sub name 0 pos
else get_basename (pos-1)
in
let name = get_basename ((String.length name) - 1) in
let cand i = name ^ "_" ^ (string_of_int i) in
let is_fresh cand_var = not (StringSet.mem cand_var existing) in
let rec find i =
let cand_var = cand i in
if is_fresh cand_var then cand_var else find (i+1) in
if is_fresh name
then name
else find 1
let freshener existing =
let existing = ref existing in
fun label ->
let next = fresh_one !existing label in
existing := StringSet.add next !existing;
next
let fresh_many f (names : string list) =
let replace acc old =
let fresh_var = f old in
(old, fresh_var) :: acc
in
List.fold_left replace [] names
let extract_vars instrs =
Array.fold_left
(fun acc instr -> VarSet.union (declared_vars instr) acc)
VarSet.empty instrs
let extract_labels instrs =
let add_label acc instr =
match[@warning "-4"] instr with
| Label (MergeLabel l | BranchLabel l) -> LabelSet.add l acc
| _ -> acc
in
Array.fold_left add_label LabelSet.empty instrs
let extract_bailout_labels instrs =
let add_label acc instr =
match[@warning "-4"] instr with
| Assume {label} -> LabelSet.add label acc
| _ -> acc
in
Array.fold_left add_label LabelSet.empty instrs
let fresh_version_label (func : afunction) label =
let existing = List.map (fun {label} -> label) func.body in
fresh_one (StringSet.of_list existing) label
let label_freshener instrs = freshener (extract_labels instrs)
let bailout_label_freshener instrs = freshener (extract_bailout_labels instrs)
let var_freshener instrs = freshener (extract_vars instrs)
type pc_map = int -> int
type result = Instr.instructions * pc_map
let subst_pc instrs start_pc len new_instrs pc =
if pc < start_pc then pc
else pc - len + Array.length new_instrs
let subst instrs pc len new_instrs =
Array.concat [
Array.sub instrs 0 pc;
new_instrs;
Array.sub instrs (pc + len) (Array.length instrs - pc - len);
], subst_pc instrs pc len new_instrs
let subst_many instrs substs =
let rec chunks cur_pc substs =
let chunk_upto pos = Array.sub instrs cur_pc (pos - cur_pc) in
match substs with
| (next_pc, len, insert_instrs) :: rest ->
if next_pc < cur_pc then invalid_arg "subst_many";
let rest, pc_map = chunks (next_pc + len) rest in
let pc_map pc =
subst_pc instrs next_pc len insert_instrs (pc_map pc)
in
let instrs = chunk_upto next_pc :: insert_instrs :: rest in
instrs, pc_map
| [] -> [chunk_upto (Array.length instrs)], fun pc -> pc
in
let compare (pc1, _, _) (pc2, _, _) =
if pc1 = pc2 then invalid_arg "subst_many";
Pervasives.compare (pc1 : int) pc2 in
let instrs, pc_map = chunks 0 (List.sort compare substs) in
Array.concat instrs, pc_map
let replace_uses old_name new_name = object (self)
inherit Instr.map as super
method! variable_use x =
if x = old_name then new_name else x
end
let replace_var var simple_exp = object (self)
inherit Instr.map as super
method! simple_expression e = match e with
| Constant _ -> e
| Var x -> if x = var then simple_exp else e
end
let replace_all_vars replacements = object (self)
inherit Instr.map as super
method replace x = try List.assoc x replacements with Not_found -> x
method! binder x = self#replace x
method! variable_assign x = self#replace x
method! variable_use x = self#replace x
end