annotate python/c3/codegenerator.py @ 220:3f6c30a5d234

Major change in expression parsing to enable pointers and structs
author Windel Bouwman
date Sat, 06 Jul 2013 21:32:20 +0200
parents 8b2e5f3cd579
children 848c4b15fd0b
rev   line source
155
b28a11c01dbe Simplified IR classes
Windel Bouwman
parents: 151
diff changeset
1 import ir
b28a11c01dbe Simplified IR classes
Windel Bouwman
parents: 151
diff changeset
2 from . import astnodes
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
3 from .scope import boolType
220
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
4 from ppci import CompilerError
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
5
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
6 class CodeGenerator:
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
7 """ Generates intermediate code from a package """
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
8 def gencode(self, pkg):
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
9 assert type(pkg) is astnodes.Package
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
10 self.varMap = {} # Maps variables to storage locations.
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
11 self.funcMap = {}
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
12 self.builder = ir.Builder()
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
13 m = ir.Module(pkg.name)
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
14 self.builder.setModule(m)
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
15 self.genModule(pkg)
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
16 return m
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
17
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
18 # inner helpers:
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
19 def genModule(self, pkg):
175
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
20 # Take care of forward declarations:
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
21 for s in pkg.innerScope.Functions:
204
de3a68f677a5 Added long comment to c3 parser
Windel Bouwman
parents: 186
diff changeset
22 f = self.builder.newFunction(s.name)
175
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
23 self.funcMap[s] = f
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
24 for s in pkg.innerScope:
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
25 if type(s) is astnodes.Variable:
205
d77cb5962cc5 Added some handcoded arm code generation
Windel Bouwman
parents: 204
diff changeset
26 v = self.builder.newVariable(s.name)
186
46d62dadd61b Improved testsuite
Windel Bouwman
parents: 177
diff changeset
27 #self.builder.addIns(ir.Alloc(v))
46d62dadd61b Improved testsuite
Windel Bouwman
parents: 177
diff changeset
28 self.varMap[s] = v
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
29 elif type(s) is astnodes.Function:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
30 # TODO: handle arguments
175
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
31 f = self.funcMap[s]
204
de3a68f677a5 Added long comment to c3 parser
Windel Bouwman
parents: 186
diff changeset
32 self.builder.setFunction(f)
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
33 bb = self.builder.newBB()
172
5a7d37d615ee Added function to IR
Windel Bouwman
parents: 171
diff changeset
34 f.entry = bb
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
35 self.builder.setBB(bb)
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
36 # generate room for locals:
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
37
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
38 for sym in s.innerScope:
175
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
39 #print(sym, sym.isParameter)
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
40 # TODO: handle parameters different
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
41 v = self.builder.newTmp(sym.name)
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
42 self.builder.addIns(ir.Alloc(v))
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
43 self.varMap[sym] = v
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
44
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
45 self.genCode(s.body)
172
5a7d37d615ee Added function to IR
Windel Bouwman
parents: 171
diff changeset
46 # TODO handle return?
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
47 self.builder.addIns(ir.Return())
204
de3a68f677a5 Added long comment to c3 parser
Windel Bouwman
parents: 186
diff changeset
48 self.builder.setFunction(None)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
49 else:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
50 print(s)
158
9683a4cd848f Added some functions for code generation
Windel Bouwman
parents: 157
diff changeset
51
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
52 def genCode(self, code):
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
53 if type(code) is astnodes.CompoundStatement:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
54 for s in code.statements:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
55 self.genCode(s)
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
56 elif type(code) is astnodes.Assignment:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
57 re = self.genExprCode(code.rval)
220
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
58 # TODO: Handle pointers
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
59 loc = self.varMap[code.lval.target]
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
60 self.builder.addIns(ir.Store(loc, re))
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
61 elif type(code) is astnodes.IfStatement:
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
62 bbtrue = self.builder.newBB()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
63 bbfalse = self.builder.newBB()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
64 te = self.builder.newBB()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
65 self.genCondCode(code.condition, bbtrue, bbfalse)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
66 self.builder.setBB(bbtrue)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
67 self.genCode(code.truestatement)
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
68 self.builder.addIns(ir.Branch(te))
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
69 self.builder.setBB(bbfalse)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
70 self.genCode(code.falsestatement)
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
71 self.builder.addIns(ir.Branch(te))
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
72 self.builder.setBB(te)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
73 elif type(code) is astnodes.FunctionCall:
176
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
74 print('TODO')
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
75 pass
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
76 elif type(code) is astnodes.EmptyStatement:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
77 pass
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
78 elif type(code) is astnodes.ReturnStatement:
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
79 if code.expr:
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
80 re = self.genExprCode(code.expr)
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
81 self.builder.addIns(ir.Return(re))
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
82 else:
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
83 self.builder.addIns(ir.Return())
176
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
84 elif type(code) is astnodes.WhileStatement:
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
85 bbdo = self.builder.newBB()
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
86 bbtest = self.builder.newBB()
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
87 te = self.builder.newBB()
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
88 self.builder.addIns(ir.Branch(bbtest))
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
89 self.builder.setBB(bbtest)
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
90 self.genCondCode(code.condition, bbdo, te)
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
91 self.builder.setBB(bbdo)
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
92 self.genCode(code.statement)
176
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
93 self.builder.addIns(ir.Branch(bbtest))
5fd02aa38b42 Added while loop code generation
Windel Bouwman
parents: 175
diff changeset
94 self.builder.setBB(te)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
95 else:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
96 print('Unknown stmt:', code)
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
97 def genCondCode(self, expr, bbtrue, bbfalse):
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
98 # Implement sequential logical operators
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
99 assert expr.typ == boolType
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
100 if type(expr) is astnodes.Binop:
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
101 if expr.op == 'or':
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
102 l2 = self.builder.newBB()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
103 self.genCondCode(expr.a, bbtrue, l2)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
104 self.builder.setBB(l2)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
105 self.genCondCode(expr.b, bbtrue, bbfalse)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
106 elif expr.op == 'and':
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
107 l2 = self.builder.newBB()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
108 self.genCondCode(expr.a, l2, bbfalse)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
109 self.builder.setBB(l2)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
110 self.genCondCode(expr.b, bbtrue, bbfalse)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
111 elif expr.op in ['==', '>', '<']:
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
112 ta = self.genExprCode(expr.a)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
113 tb = self.genExprCode(expr.b)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
114 i = ir.ConditionalBranch(ta, expr.op, tb, bbtrue, bbfalse)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
115 self.builder.addIns(i)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
116 else:
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
117 raise NotImlementedError('Unknown condition {0}'.format(expr))
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
118 elif type(expr) is astnodes.Literal:
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
119 if expr.val:
205
d77cb5962cc5 Added some handcoded arm code generation
Windel Bouwman
parents: 204
diff changeset
120 self.builder.addIns(ir.Branch(bbtrue))
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
121 else:
205
d77cb5962cc5 Added some handcoded arm code generation
Windel Bouwman
parents: 204
diff changeset
122 self.builder.addIns(ir.Branch(bbfalse))
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
123 else:
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
124 print('Unknown cond', expr)
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
125 def genExprCode(self, expr):
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
126 if type(expr) is astnodes.Binop:
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
127 ra = self.genExprCode(expr.a)
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
128 rb = self.genExprCode(expr.b)
177
460db5669efa Added clean pass for IR
Windel Bouwman
parents: 176
diff changeset
129 ops = ['+', '-', '*', '/']
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
130 if expr.op in ops:
177
460db5669efa Added clean pass for IR
Windel Bouwman
parents: 176
diff changeset
131 tmpnames = {'+':'addtmp', '-':'subtmp', '*': 'multmp', '/':'divtmp'}
460db5669efa Added clean pass for IR
Windel Bouwman
parents: 176
diff changeset
132 tmp = self.builder.newTmp(tmpnames[expr.op])
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
133 op = expr.op
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
134 ins = ir.BinaryOperator(tmp, op, ra, rb)
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
135 self.builder.addIns(ins)
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
136 return tmp
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
137 else:
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
138 print('Unknown {0}'.format(expr))
177
460db5669efa Added clean pass for IR
Windel Bouwman
parents: 176
diff changeset
139 tmp = self.builder.newTmp()
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
140 # TODO
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
141 return tmp
220
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
142 elif type(expr) is astnodes.Unop:
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
143 if expr.op == '&':
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
144 # Address of operator
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
145 ra = self.genExprCode(expr.a)
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
146 tmp = self.builder.newTmp('addr')
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
147 return tmp
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
148 #self.builder.addIns(ins)
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
149 else:
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
150 print('Unknown {0}'.format(expr))
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
151 elif type(expr) is astnodes.Constant:
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
152 tmp = self.builder.newTmp()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
153 # TODO
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
154 return tmp
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
155 elif type(expr) is astnodes.VariableUse:
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
156 tmp = self.builder.newTmp()
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
157 loc = self.varMap[expr.target]
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
158 i = ir.Load(loc, tmp)
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
159 self.builder.addIns(i)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
160 return tmp
170
4348da5ca307 Cleanup of ir dir
Windel Bouwman
parents: 169
diff changeset
161 elif type(expr) is astnodes.Literal:
171
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
162 tmp = self.builder.newTmp()
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
163 ins = ir.ImmLoad(tmp, expr.val)
3eb9b9e2958d Improved IR code
Windel Bouwman
parents: 170
diff changeset
164 self.builder.addIns(ins)
158
9683a4cd848f Added some functions for code generation
Windel Bouwman
parents: 157
diff changeset
165 return tmp
175
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
166 elif type(expr) is astnodes.FunctionCall:
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
167 tmp = self.builder.newTmp("res")
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
168 args = []
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
169 for arg in expr.args:
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
170 ar = self.genExprCode(arg)
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
171 args.append(ar)
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
172 fn = self.funcMap[expr.proc]
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
173 ins = ir.Call(fn, args, tmp)
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
174 self.builder.addIns(ins)
a51b3c956386 Added function call in expressions
Windel Bouwman
parents: 174
diff changeset
175 return tmp
157
8f3924b6076e Added some code generator things
Windel Bouwman
parents: 156
diff changeset
176 else:
220
3f6c30a5d234 Major change in expression parsing to enable pointers and structs
Windel Bouwman
parents: 217
diff changeset
177 raise CompilerError('Unknown expr {}'.format(expr))
157
8f3924b6076e Added some code generator things
Windel Bouwman
parents: 156
diff changeset
178