-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoptimizer.py
108 lines (91 loc) · 3.23 KB
/
optimizer.py
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
from typing import List, Tuple
from bisect import bisect_right
from compiler import Instrument
def optimize(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
os = [
o_redundant_copy,
o_continuous_label,
o_immediate_jump,
o_unref_label,
o_dead_code,
remap_labels
]
optimized = False
for f in os:
r, hrm = f(hrm)
optimized = optimized or r
return optimized, hrm
def o_redundant_copy(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
opt = []
i = 0
while i < len(hrm):
opt.append(hrm[i])
if hrm[i].code.startswith("COPY"):
j = i + 1
while j < len(hrm) and hrm[j].code.startswith("COPY") and hrm[j].arg == hrm[i].arg:
j += 1
i = j
else:
i += 1
return len(opt) < len(hrm), opt
def o_unref_label(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
labs = set(i.arg for i in hrm if i.code == Instrument.LAB)
jmps = set(i.arg for i in hrm if i.code.startswith("JUMP"))
unrefs = labs.difference(jmps)
opt = [i for i in hrm if i.code != Instrument.LAB or i.arg not in unrefs]
return len(unrefs) > 0, opt
def o_continuous_label(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
glabs = []
i = 0
while i < len(hrm):
g = []
while hrm[i].code == Instrument.LAB:
g.append(hrm[i].arg)
i += 1
if len(g) > 1:
glabs.append(g)
i += 1
opt = hrm.copy()
for g in glabs:
if len(g) > 1:
t = g[0]
opt = [
(Instrument(ins.code, t) if (ins.code.startswith("JUMP") and ins.arg in g[1:]) else ins)
for ins in opt]
return len(glabs) > 0, opt
def o_dead_code(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
ijmp = [i for i in range(len(hrm)) if hrm[i].code == Instrument.JMP]
ilab = [i for i in range(len(hrm)) if hrm[i].code == Instrument.LAB]
if len(ijmp) == 0:
return False, hrm
ilab.append(len(hrm))
secs = [(j + 1, ilab[bisect_right(ilab, j)]) for j in ijmp]
secs = [(i, j) for i, j in secs if j - i > 0]
secs.reverse()
opt = hrm.copy()
for l, r in secs:
for _ in range(r - l):
opt.pop(l)
return len(secs) > 0, opt
def o_immediate_jump(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
ij = [(hrm[i].arg, hrm[i + 1].arg) for i in range(len(hrm) - 1)
if hrm[i].code == Instrument.LAB and hrm[i + 1].code == Instrument.JMP]
opt = hrm.copy()
for i, j in ij:
opt = [Instrument(ins.code, j) if (ins.code.startswith("JUMP") and ins.arg == i) else ins
for ins in hrm]
return len(ij) > 0, opt
def remap_labels(hrm: List[Instrument]) -> Tuple[bool, List[Instrument]]:
lab = [i.arg for i in hrm if i.code == Instrument.LAB]
lab = sorted(set(lab))
if lab == list(range(len(lab))):
return False, hrm
else:
opt = []
for i in hrm:
if i.code == Instrument.LAB or i.code.startswith("JUMP"):
idx = lab.index(i.arg)
opt.append(Instrument(i.code, idx))
else:
opt.append(i)
return True, opt