-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathutil.py
171 lines (158 loc) · 5.73 KB
/
util.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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
#!/usr/bin/env python3
# Copyright 2022 David Guillen Fandos <david@davidgf.net>
# Simplifies C code and re-formats it (for doc generation purposes)
import tatsu, json
# Parses the AST, and performs simplifications like constant propagation
# and constant operations.
def simplify_code(code):
cgrammar = '''
start = lside:lsizeexpr op:'=' rside:expr $ ;
expr = ternexpr;
ternexpr = cond:ternexpr op:"?" rest:expr ":" resf:expr | boolorexpr;
boolorexpr = l:boolorexpr op:"||" r:boolandexpr | boolandexpr;
boolandexpr = l:boolandexpr op:"&&" r:bitworexpr | bitworexpr;
bitworexpr = l:bitworexpr op:"|" r:bitwxorexpr | bitwxorexpr;
bitwxorexpr = l:bitwxorexpr op:"^" r:bitwandexpr | bitwandexpr;
bitwandexpr = l:bitwandexpr op:"&" r:compequexpr | compequexpr;
compequexpr = l:compequexpr op:"==" r:cmpunequxpr | l:compequexpr op:"!=" cmpunequxpr | cmpunequxpr;
cmpunequxpr = l:cmpunequxpr op:"<" r:shiftexpr | l:cmpunequxpr op:">" r:shiftexpr |
l:cmpunequxpr op:"<=" r:shiftexpr | l:cmpunequxpr op:">=" r:shiftexpr | shiftexpr;
shiftexpr = l:shiftexpr op:">>" r:additionexpr | l:shiftexpr op:"<<" r:additionexpr | additionexpr;
additionexpr = l:additionexpr op:'+' r:multexpr | l:additionexpr op:'-' r:multexpr | multexpr;
multexpr = l:multexpr op:'*' r:unaryexpr | l:multexpr op:'/' r:unaryexpr |
l:multexpr op:'%' r:unaryexpr | unaryexpr;
unaryexpr = op:'-' e:simpleexpr | op:'+' e:simpleexpr | op:'~' e:simpleexpr | op:'!' e:simpleexpr | simpleexpr;
simpleexpr = '(' paren:expr ')' | fncall | unit;
fncall = funcname:fnname op:'(' args:fnargs ')' ;
fnargs = | ','.{ expr }+ | expr ;
fnname = /[A-Za-z][A-Za-z0-9]+/ ;
unit = consthex | constf32 | constint | vararray | variable ;
vararray = var:varname '[' idx:expr ']' ;
variable = var:varname ;
varname = /[A-Za-z][A-Za-z_0-9]+/ ;
constint = vali:/\d+/ ;
consthex = valh:/0x[0-9A-Fa-f]+/ ;
constf32 = valf:/\d+.\d+[f]/ ;
lsizeexpr = vararray | variable;
'''
unops = {
'+': lambda x: x,
'-': lambda x: -x,
'~': lambda x: ~x,
'!': lambda x: 0 if x else 1,
}
binops = {
'&': lambda x, y: x & y,
'|': lambda x, y: x | y,
'^': lambda x, y: x ^ y,
'+': lambda x, y: x + y,
'-': lambda x, y: x - y,
'*': lambda x, y: x * y,
'/': lambda x, y: x // y,
'%': lambda x, y: x % y,
'<': lambda x, y: 1 if x < y else 0,
'>': lambda x, y: 1 if x > y else 0,
'<=': lambda x, y: 1 if x <= y else 0,
'>=': lambda x, y: 1 if x >= y else 0,
'==': lambda x, y: 1 if x == y else 0,
'!=': lambda x, y: 1 if x != y else 0,
'>>': lambda x, y: x >> y,
'<<': lambda x, y: x << y,
'&&': lambda x, y: (x != 0) and (y != 0),
'||': lambda x, y: (x != 0) or (y != 0),
}
def _nonp(fn, *args):
if any(arg is None for arg in args):
return None
return fn(*args)
def _serialize(elem):
if "var" in elem:
if "idx" in elem:
return "%s[%s]" % (elem.var, _serialize(elem.idx))
return elem.var
if "paren" in elem:
return "(%s)" % _serialize(elem.paren)
if "cond" in elem:
return "%s ? %s : %s" % (_serialize(elem.cond), _serialize(elem.rest), _serialize(elem.resf))
if elem.op == "=":
return "%s = %s" % (_serialize(elem.lside), _serialize(elem.rside))
elif elem.op == "(":
return "%s(%s)" % (elem.funcname, ", ".join(_serialize(x) for x in elem.args))
elif "e" in elem and elem.op in unops:
return "%s%s" % (elem.op, _serialize(elem.e))
elif elem.op in binops:
return "%s %s %s" % (_serialize(elem.l), elem.op, _serialize(elem.r))
elif "vali" in elem:
return elem.vali
elif "valh" in elem:
return elem.valh
elif "valf" in elem:
return elem.valf
return str(elem)
def _eval(elem):
if "vali" in elem:
return int(elem.vali)
elif "valf" in elem:
return float(elem.valf.strip("f"))
elif "valh" in elem:
return int(elem.valh, 16)
elif "paren" in elem:
return _eval(elem.paren)
elif "cond" in elem:
condval = _eval(elem.cond)
if condval is None:
return None
if condval:
return _eval(elem.rest)
return _eval(elem.resf)
elif "e" in elem and elem.op in unops:
ev = _eval(elem.e)
return _nonp(unops[elem.op], ev)
elif elem.op in binops:
l, r = _eval(elem.l), _eval(elem.r)
return _nonp(binops[elem.op], l, r,)
return None
def _makecnst(v):
if isinstance(v, int):
return tatsu.ast.AST({"vali": str(v)})
if isinstance(v, float):
return tatsu.ast.AST({"valf": "%gf" % v})
return None
def _simplify(elem):
if "vali" in elem or "valf" in elem or "valh" in elem:
return elem
v = _eval(elem)
if v is not None:
return _makecnst(v)
if "var" in elem:
if "idx" in elem:
return tatsu.ast.AST({"var": elem.var, "idx": _simplify(elem.idx)})
return elem
if "paren" in elem:
return tatsu.ast.AST({"paren": _simplify(elem.paren)})
if "cond" in elem:
# Can simplify part of it, if subcondition is known
condval = _eval(elem.cond)
if condval is None:
return tatsu.ast.AST({"cond": _simplify(elem.cond), "rest": _simplify(elem.rest), "resf": _simplify(elem.resf)})
if condval:
return _simplify(elem.rest)
return _simplify(elem.resf)
if elem.op == "=":
return tatsu.ast.AST({"op": "=", "lside": _simplify(elem.lside), "rside": _simplify(elem.rside)})
elif elem.op == "(":
return tatsu.ast.AST({"op": "(", "funcname": elem.funcname, "args": [_simplify(x) for x in elem.args]})
elif "e" in elem and elem.op in unops:
return tatsu.ast.AST({"op": elem.op, "e": _simplify(elem.e)})
elif elem.op in binops:
l, r = _simplify(elem.l), _simplify(elem.r)
if elem.op in ["-", "+", ">>", "<<"] and _eval(r) == 0:
return l
return tatsu.ast.AST({"op": elem.op, "l": l, "r": r})
return elem
try:
ast = tatsu.parse(cgrammar, code)
except:
return None
ast = _simplify(ast)
return _serialize(ast)