annotate python/ppci/c3/codegenerator.py @ 395:3b0c495e3008

Speed improvements
author Windel Bouwman
date Fri, 23 May 2014 14:28:03 +0200
parents 988f3fb861e4
children
rev   line source
255
7416c923a02a Added more logging
Windel Bouwman
parents: 252
diff changeset
1 import logging
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
2 import struct
301
6753763d3bec merge codegen into ppci package
Windel Bouwman
parents: 300
diff changeset
3 from .. import ir
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
4 from .. import irutils
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
5 from . import astnodes as ast
230
88a1e0baef65 Added some tests for IR-code
Windel Bouwman
parents: 228
diff changeset
6
228
7f18ed9b6b7e Removal of emptystatement class
Windel Bouwman
parents: 227
diff changeset
7
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
8 class SemanticError(Exception):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
9 """ Error thrown when a semantic issue is observed """
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
10 def __init__(self, msg, loc):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
11 super().__init__()
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
12 self.msg = msg
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
13 self.loc = loc
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
14
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
15
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
16 class CodeGenerator:
288
a747a45dcd78 Various styling work
Windel Bouwman
parents: 287
diff changeset
17 """
274
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
18 Generates intermediate (IR) code from a package. The entry function is
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
19 'genModule'. The main task of this part is to rewrite complex control
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
20 structures, such as while and for loops into simple conditional
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
21 jump statements. Also complex conditional statements are simplified.
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
22 Such as 'and' and 'or' statements are rewritten in conditional jumps.
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
23 And structured datatypes are rewritten.
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
24
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
25 Type checking is done in one run with code generation.
274
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
26 """
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
27 def __init__(self, diag):
261
444b9df2ed99 try to split up code generation
Windel Bouwman
parents: 259
diff changeset
28 self.logger = logging.getLogger('c3cgen')
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
29 self.builder = irutils.Builder()
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
30 self.diag = diag
261
444b9df2ed99 try to split up code generation
Windel Bouwman
parents: 259
diff changeset
31
217
8b2e5f3cd579 Removed some stale python source files
Windel Bouwman
parents: 205
diff changeset
32 def gencode(self, pkg):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
33 """ Generate code for a single module """
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
34 self.builder.prepare()
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
35 assert type(pkg) is ast.Package
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
36 self.pkg = pkg
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
37 self.intType = pkg.scope['int']
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
38 self.boolType = pkg.scope['bool']
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
39 self.pointerSize = 4
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
40 self.logger.debug('Generating ir-code for {}'.format(pkg.name),
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
41 extra={'c3_ast': pkg})
288
a747a45dcd78 Various styling work
Windel Bouwman
parents: 287
diff changeset
42 self.varMap = {} # Maps variables to storage locations.
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
43 self.builder.m = ir.Module(pkg.name)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
44 try:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
45 for typ in pkg.Types:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
46 self.check_type(typ)
362
c05ab629976a Added CPUID for arm
Windel Bouwman
parents: 360
diff changeset
47 # Only generate function if function contains a body:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
48 real_functions = list(filter(
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
49 lambda f: f.body, pkg.Functions))
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
50 for v in pkg.innerScope.Variables:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
51 v2 = ir.GlobalVariable(v.name, ir.i32)
364
c49459768aaa Work on globals
Windel Bouwman
parents: 363
diff changeset
52 self.varMap[v] = v2
363
396e5cefba13 Removed debuginfo instruction
Windel Bouwman
parents: 362
diff changeset
53 if not v.isLocal:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
54 self.builder.m.add_variable(v2)
362
c05ab629976a Added CPUID for arm
Windel Bouwman
parents: 360
diff changeset
55 for s in real_functions:
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
56 self.gen_function(s)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
57 except SemanticError as e:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
58 self.error(e.msg, e.loc)
313
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
59 if self.pkg.ok:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
60 return self.builder.m
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
61
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
62 def error(self, msg, loc=None):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
63 self.pkg.ok = False
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
64 self.diag.error(msg, loc)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
65
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
66 def gen_function(self, fn):
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
67 # TODO: handle arguments
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
68 f = self.builder.new_function(fn.name)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
69 f.return_value = self.builder.newTemp()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
70 self.builder.setFunction(f)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
71 l2 = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
72 self.builder.emit(ir.Jump(l2))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
73 self.builder.setBlock(l2)
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
74 # generate room for locals:
174
3eb06f5fb987 Added memory alloc for locals
Windel Bouwman
parents: 172
diff changeset
75
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
76 for sym in fn.innerScope:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
77 self.check_type(sym.typ)
272
e64bae57cda8 refactor ir
Windel Bouwman
parents: 269
diff changeset
78 if sym.isParameter:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
79 p = ir.Parameter(sym.name, ir.i32)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
80 variable = ir.LocalVariable(sym.name + '_copy', ir.i32)
316
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
81 f.addParameter(p)
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
82 f.addLocal(variable)
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
83 # Move parameter into local copy:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
84 self.builder.emit(ir.Move(ir.Mem(variable), p))
274
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
85 elif sym.isLocal:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
86 variable = ir.LocalVariable(sym.name, ir.i32)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
87 f.addLocal(variable)
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
88 elif isinstance(sym, ast.Variable):
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
89 variable = ir.LocalVariable(sym.name, ir.i32)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
90 f.addLocal(variable)
274
ea93e0a7a31e Move docs
Windel Bouwman
parents: 272
diff changeset
91 else:
275
6f2423df0675 Fixed serve arm-as
Windel Bouwman
parents: 274
diff changeset
92 raise NotImplementedError('{}'.format(sym))
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
93 self.varMap[sym] = variable
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
94
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
95 self.gen_stmt(fn.body)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
96 self.builder.emit(ir.Move(f.return_value, ir.Const(0)))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
97 self.builder.emit(ir.Jump(f.epiloog))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
98 self.builder.setFunction(None)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
99
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
100 def gen_stmt(self, code):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
101 """ Generate code for a statement """
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
102 try:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
103 assert isinstance(code, ast.Statement)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
104 self.builder.setLoc(code.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
105 if type(code) is ast.Compound:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
106 for s in code.statements:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
107 self.gen_stmt(s)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
108 elif type(code) is ast.Empty:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
109 pass
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
110 elif type(code) is ast.Assignment:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
111 self.gen_assignment_stmt(code)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
112 elif type(code) is ast.ExpressionStatement:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
113 self.builder.emit(ir.Exp(self.gen_expr_code(code.ex)))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
114 elif type(code) is ast.If:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
115 self.gen_if_stmt(code)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
116 elif type(code) is ast.Return:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
117 re = self.gen_expr_code(code.expr)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
118 self.builder.emit(ir.Move(self.builder.fn.return_value, re))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
119 self.builder.emit(ir.Jump(self.builder.fn.epiloog))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
120 b = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
121 self.builder.setBlock(b)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
122 elif type(code) is ast.While:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
123 self.gen_while(code)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
124 elif type(code) is ast.For:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
125 self.gen_for_stmt(code)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
126 elif type(code) is ast.Switch:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
127 raise NotImplementedError('Unknown stmt {}'.format(code))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
128 else:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
129 raise NotImplementedError('Unknown stmt {}'.format(code))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
130 except SemanticError as e:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
131 self.error(e.msg, e.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
132
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
133 def gen_assignment_stmt(self, code):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
134 """ Generate code for assignment statement """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
135 lval = self.gen_expr_code(code.lval)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
136 rval = self.gen_expr_code(code.rval)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
137 if not self.equal_types(code.lval.typ, code.rval.typ):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
138 raise SemanticError('Cannot assign {} to {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
139 .format(code.rval.typ, code.lval.typ),
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
140 code.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
141 if not code.lval.lvalue:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
142 raise SemanticError('No valid lvalue {}'.format(code.lval),
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
143 code.lval.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
144 self.builder.emit(ir.Move(lval, rval))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
145
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
146 def gen_if_stmt(self, code):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
147 """ Generate code for if statement """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
148 true_block = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
149 bbfalse = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
150 te = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
151 self.gen_cond_code(code.condition, true_block, bbfalse)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
152 self.builder.setBlock(true_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
153 self.gen_stmt(code.truestatement)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
154 self.builder.emit(ir.Jump(te))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
155 self.builder.setBlock(bbfalse)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
156 self.gen_stmt(code.falsestatement)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
157 self.builder.emit(ir.Jump(te))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
158 self.builder.setBlock(te)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
159
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
160 def gen_while(self, code):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
161 """ Generate code for while statement """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
162 bbdo = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
163 test_block = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
164 final_block = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
165 self.builder.emit(ir.Jump(test_block))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
166 self.builder.setBlock(test_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
167 self.gen_cond_code(code.condition, bbdo, final_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
168 self.builder.setBlock(bbdo)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
169 self.gen_stmt(code.statement)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
170 self.builder.emit(ir.Jump(test_block))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
171 self.builder.setBlock(final_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
172
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
173 def gen_for_stmt(self, code):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
174 """ Generate for statement code """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
175 bbdo = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
176 test_block = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
177 final_block = self.builder.newBlock()
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
178 self.gen_stmt(code.init)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
179 self.builder.emit(ir.Jump(test_block))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
180 self.builder.setBlock(test_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
181 self.gen_cond_code(code.condition, bbdo, final_block)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
182 self.builder.setBlock(bbdo)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
183 self.gen_stmt(code.statement)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
184 self.gen_stmt(code.final)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
185 self.builder.emit(ir.Jump(test_block))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
186 self.builder.setBlock(final_block)
230
88a1e0baef65 Added some tests for IR-code
Windel Bouwman
parents: 228
diff changeset
187
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
188 def gen_cond_code(self, expr, bbtrue, bbfalse):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
189 """ Generate conditional logic.
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
190 Implement sequential logical operators. """
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
191 if type(expr) is ast.Binop:
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
192 if expr.op == 'or':
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
193 l2 = self.builder.newBlock()
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
194 self.gen_cond_code(expr.a, bbtrue, l2)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
195 if not self.equal_types(expr.a.typ, self.boolType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
196 raise SemanticError('Must be boolean', expr.a.loc)
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
197 self.builder.setBlock(l2)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
198 self.gen_cond_code(expr.b, bbtrue, bbfalse)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
199 if not self.equal_types(expr.b.typ, self.boolType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
200 raise SemanticError('Must be boolean', expr.b.loc)
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
201 elif expr.op == 'and':
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
202 l2 = self.builder.newBlock()
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
203 self.gen_cond_code(expr.a, l2, bbfalse)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
204 if not self.equal_types(expr.a.typ, self.boolType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
205 self.error('Must be boolean', expr.a.loc)
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
206 self.builder.setBlock(l2)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
207 self.gen_cond_code(expr.b, bbtrue, bbfalse)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
208 if not self.equal_types(expr.b.typ, self.boolType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
209 raise SemanticError('Must be boolean', expr.b.loc)
305
0615b5308710 Updated docs
Windel Bouwman
parents: 303
diff changeset
210 elif expr.op in ['==', '>', '<', '!=', '<=', '>=']:
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
211 ta = self.gen_expr_code(expr.a)
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
212 tb = self.gen_expr_code(expr.b)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
213 if not self.equal_types(expr.a.typ, expr.b.typ):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
214 raise SemanticError('Types unequal {} != {}'
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
215 .format(expr.a.typ, expr.b.typ),
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
216 expr.loc)
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
217 self.builder.emit(ir.CJump(ta, expr.op, tb, bbtrue, bbfalse))
268
5ec7580976d9 Op naar tree-IR
Windel Bouwman
parents: 261
diff changeset
218 else:
311
ff665880a6b0 Added testcase for kernel and userspace
Windel Bouwman
parents: 308
diff changeset
219 raise SemanticError('non-bool: {}'.format(expr.op), expr.loc)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
220 expr.typ = self.boolType
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
221 elif type(expr) is ast.Literal:
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
222 self.gen_expr_code(expr)
288
a747a45dcd78 Various styling work
Windel Bouwman
parents: 287
diff changeset
223 if expr.val:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
224 self.builder.emit(ir.Jump(bbtrue))
288
a747a45dcd78 Various styling work
Windel Bouwman
parents: 287
diff changeset
225 else:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
226 self.builder.emit(ir.Jump(bbfalse))
228
7f18ed9b6b7e Removal of emptystatement class
Windel Bouwman
parents: 227
diff changeset
227 else:
288
a747a45dcd78 Various styling work
Windel Bouwman
parents: 287
diff changeset
228 raise NotImplementedError('Unknown cond {}'.format(expr))
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
229
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
230 # Check that the condition is a boolean value:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
231 if not self.equal_types(expr.typ, self.boolType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
232 self.error('Condition must be boolean', expr.loc)
230
88a1e0baef65 Added some tests for IR-code
Windel Bouwman
parents: 228
diff changeset
233
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
234 def gen_expr_code(self, expr):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
235 """ Generate code for an expression. Return the generated ir-value """
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
236 assert isinstance(expr, ast.Expression)
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
237 if type(expr) is ast.Binop:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
238 expr.lvalue = False
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
239 if expr.op in ['+', '-', '*', '/', '<<', '>>', '|', '&']:
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
240 ra = self.gen_expr_code(expr.a)
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
241 rb = self.gen_expr_code(expr.b)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
242 if self.equal_types(expr.a.typ, self.intType) and \
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
243 self.equal_types(expr.b.typ, self.intType):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
244 expr.typ = expr.a.typ
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
245 elif self.equal_types(expr.b.typ, self.intType) and \
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
246 type(expr.a.typ) is ast.PointerType:
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
247 # Special case for pointer arithmatic TODO: coerce!
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
248 expr.typ = expr.a.typ
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
249 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
250 raise SemanticError('Can only add integers', expr.loc)
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
251 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
252 raise NotImplementedError("Cannot use equality as expressions")
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
253 return ir.Binop(ra, expr.op, rb, "op", ir.i32)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
254 elif type(expr) is ast.Unop:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
255 if expr.op == '&':
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
256 ra = self.gen_expr_code(expr.a)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
257 expr.typ = ast.PointerType(expr.a.typ)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
258 if not expr.a.lvalue:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
259 raise SemanticError('No valid lvalue', expr.a.loc)
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
260 expr.lvalue = False
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
261 assert type(ra) is ir.Mem
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
262 return ra.e
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
263 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
264 raise NotImplementedError('Unknown unop {0}'.format(expr.op))
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
265 elif type(expr) is ast.Identifier:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
266 # Generate code for this identifier.
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
267 tg = self.resolveSymbol(expr)
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
268 expr.kind = type(tg)
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
269 expr.typ = tg.typ
279
2ccd57b1d78c Fix register allocator to do burn2 OK
Windel Bouwman
parents: 275
diff changeset
270 # This returns the dereferenced variable.
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
271 if isinstance(tg, ast.Variable):
313
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
272 expr.lvalue = True
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
273 return ir.Mem(self.varMap[tg])
313
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
274 elif isinstance(tg, ast.Constant):
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
275 c_val = self.gen_expr_code(tg.value)
313
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
276 return self.evalConst(c_val)
280
02385f62f250 Rework from str interface to Instruction interface
Windel Bouwman
parents: 279
diff changeset
277 else:
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
278 raise NotImplementedError(str(tg))
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
279 elif type(expr) is ast.Deref:
222
c3f1ce8b638f Fixup of parser
Windel Bouwman
parents: 221
diff changeset
280 # dereference pointer type:
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
281 addr = self.gen_expr_code(expr.ptr)
316
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
282 ptr_typ = self.the_type(expr.ptr.typ)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
283 expr.lvalue = True
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
284 if type(ptr_typ) is ast.PointerType:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
285 expr.typ = ptr_typ.ptype
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
286 return ir.Mem(addr)
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
287 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
288 raise SemanticError('Cannot deref non-pointer', expr.loc)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
289 elif type(expr) is ast.Member:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
290 return self.gen_member_expr(expr)
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
291 elif type(expr) is ast.Index:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
292 return self.gen_index_expr(expr)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
293 elif type(expr) is ast.Literal:
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
294 return self.gen_literal_expr(expr)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
295 elif type(expr) is ast.TypeCast:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
296 return self.gen_type_cast(expr)
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
297 elif type(expr) is ast.Sizeof:
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
298 # The type of this expression is int:
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
299 expr.typ = self.intType
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
300 self.check_type(expr.query_typ)
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
301 type_size = self.size_of(expr.query_typ)
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
302 return ir.Const(type_size)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
303 elif type(expr) is ast.FunctionCall:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
304 return self.gen_function_call(expr)
225
1c7364bd74c7 Fixed pointer deref
Windel Bouwman
parents: 222
diff changeset
305 else:
259
ac603eb66b63 Added function call
Windel Bouwman
parents: 255
diff changeset
306 raise NotImplementedError('Unknown expr {}'.format(expr))
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
307
394
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
308 def gen_member_expr(self, expr):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
309 base = self.gen_expr_code(expr.base)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
310 expr.lvalue = expr.base.lvalue
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
311 basetype = self.the_type(expr.base.typ)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
312 if type(basetype) is ast.StructureType:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
313 if basetype.hasField(expr.field):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
314 expr.typ = basetype.fieldType(expr.field)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
315 else:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
316 raise SemanticError('{} does not contain field {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
317 .format(basetype, expr.field),
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
318 expr.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
319 else:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
320 raise SemanticError('Cannot select {} of non-structure type {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
321 .format(expr.field, basetype), expr.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
322
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
323 assert type(base) is ir.Mem, type(base)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
324 bt = self.the_type(expr.base.typ)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
325 offset = ir.Const(bt.fieldOffset(expr.field))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
326 addr = ir.Add(base.e, offset, "mem_addr", ir.i32)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
327 return ir.Mem(addr)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
328
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
329 def gen_index_expr(self, expr):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
330 """ Array indexing """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
331 base = self.gen_expr_code(expr.base)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
332 idx = self.gen_expr_code(expr.i)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
333 base_typ = self.the_type(expr.base.typ)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
334 if not isinstance(base_typ, ast.ArrayType):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
335 raise SemanticError('Cannot index non-array type {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
336 .format(base_typ),
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
337 expr.base.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
338 idx_type = self.the_type(expr.i.typ)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
339 if not self.equal_types(idx_type, self.intType):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
340 raise SemanticError('Index must be int not {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
341 .format(idx_type), expr.i.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
342 assert type(base) is ir.Mem
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
343 element_type = self.the_type(base_typ.element_type)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
344 element_size = self.size_of(element_type)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
345 expr.typ = base_typ.element_type
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
346 expr.lvalue = True
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
347
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
348 offset = ir.Mul(idx, ir.Const(element_size), "element_offset", ir.i32)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
349 addr = ir.Add(base.e, offset, "element_address", ir.i32)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
350 return ir.Mem(addr)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
351
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
352 def gen_literal_expr(self, expr):
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
353 """ Generate code for literal """
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
354 expr.lvalue = False
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
355 typemap = {int: 'int',
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
356 float: 'double',
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
357 bool: 'bool',
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
358 str: 'string'}
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
359 if type(expr.val) in typemap:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
360 expr.typ = self.pkg.scope[typemap[type(expr.val)]]
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
361 else:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
362 raise SemanticError('Unknown literal type {}'
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
363 .format(expr.val), expr.loc)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
364 # Construct correct const value:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
365 if type(expr.val) is str:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
366 cval = self.pack_string(expr.val)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
367 return ir.Addr(ir.Const(cval))
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
368 else:
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
369 return ir.Const(expr.val)
988f3fb861e4 c3 code generator rewrite
Windel Bouwman
parents: 393
diff changeset
370
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
371 def pack_string(self, txt):
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
372 """ Pack a string using 4 bytes length followed by text data """
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
373 length = struct.pack('<I', len(txt))
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
374 data = txt.encode('ascii')
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
375 return length + data
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
376
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
377 def gen_type_cast(self, expr):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
378 """ Generate code for type casting """
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
379 ar = self.gen_expr_code(expr.a)
316
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
380 from_type = self.the_type(expr.a.typ)
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
381 to_type = self.the_type(expr.to_type)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
382 if isinstance(from_type, ast.PointerType) and \
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
383 isinstance(to_type, ast.PointerType):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
384 expr.typ = expr.to_type
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
385 return ar
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
386 elif self.equal_types(self.intType, from_type) and \
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
387 isinstance(to_type, ast.PointerType):
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
388 expr.typ = expr.to_type
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
389 return ar
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
390 elif self.equal_types(self.intType, to_type) \
364
c49459768aaa Work on globals
Windel Bouwman
parents: 363
diff changeset
391 and isinstance(from_type, ast.PointerType):
c49459768aaa Work on globals
Windel Bouwman
parents: 363
diff changeset
392 expr.typ = expr.to_type
c49459768aaa Work on globals
Windel Bouwman
parents: 363
diff changeset
393 return ar
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
394 elif type(from_type) is ast.BaseType and from_type.name == 'byte' and \
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
395 type(to_type) is ast.BaseType and to_type.name == 'int':
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
396 expr.typ = expr.to_type
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
397 return ar
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
398 else:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
399 raise SemanticError('Cannot cast {} to {}'
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
400 .format(from_type, to_type), expr.loc)
353
b8ad45b3a573 Started with strings
Windel Bouwman
parents: 336
diff changeset
401
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
402 def gen_function_call(self, expr):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
403 """ Generate code for a function call """
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
404 # Evaluate the arguments:
393
6ae782a085e0 Added init program
Windel Bouwman
parents: 389
diff changeset
405 args = [self.gen_expr_code(e) for e in expr.args]
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
406 # Check arguments:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
407 tg = self.resolveSymbol(expr.proc)
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
408 if type(tg) is not ast.Function:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
409 raise SemanticError('cannot call {}'.format(tg))
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
410 ftyp = tg.typ
336
d1ecc493384e Added spiffy armtoken class for bit fiddeling. Added cool test that checks for build repeatability
Windel Bouwman
parents: 334
diff changeset
411 fname = tg.package.name + '_' + tg.name
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
412 ptypes = ftyp.parametertypes
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
413 if len(expr.args) != len(ptypes):
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
414 raise SemanticError('{} requires {} arguments, {} given'
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
415 .format(fname, len(ptypes), len(expr.args)),
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
416 expr.loc)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
417 for arg, at in zip(expr.args, ptypes):
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
418 if not self.equal_types(arg.typ, at):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
419 raise SemanticError('Got {}, expected {}'
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
420 .format(arg.typ, at), arg.loc)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
421 # determine return type:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
422 expr.typ = ftyp.returntype
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
423 return ir.Call(fname, args)
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
424
313
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
425 def evalConst(self, c):
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
426 if isinstance(c, ir.Const):
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
427 return c
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
428 else:
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
429 raise SemanticError('Cannot evaluate constant {}'.format(c))
04cf4d26a3bc Added constant function
Windel Bouwman
parents: 312
diff changeset
430
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
431 def resolveSymbol(self, sym):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
432 if type(sym) is ast.Member:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
433 base = self.resolveSymbol(sym.base)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
434 if type(base) is not ast.Package:
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
435 raise SemanticError('Base is not a package', sym.loc)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
436 scope = base.innerScope
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
437 name = sym.field
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
438 elif type(sym) is ast.Identifier:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
439 scope = sym.scope
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
440 name = sym.target
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
441 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
442 raise NotImplementedError(str(sym))
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
443 if name in scope:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
444 s = scope[name]
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
445 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
446 raise SemanticError('{} undefined'.format(name), sym.loc)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
447 assert isinstance(s, ast.Symbol)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
448 return s
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
449
316
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
450 def size_of(self, t):
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
451 """ Determine the byte size of a type """
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
452 t = self.the_type(t)
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
453 if type(t) is ast.BaseType:
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
454 return t.bytesize
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
455 elif type(t) is ast.StructureType:
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
456 return sum(self.size_of(mem.typ) for mem in t.mems)
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
457 elif type(t) is ast.ArrayType:
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
458 return t.size * self.size_of(t.element_type)
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
459 elif type(t) is ast.PointerType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
460 return self.pointerSize
316
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
461 else:
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
462 raise NotImplementedError(str(t))
56e6ff84f646 Fixed burn led demo
Windel Bouwman
parents: 315
diff changeset
463
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
464 def the_type(self, t, reveil_defined=True):
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
465 """ Recurse until a 'real' type is found
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
466 When reveil_defined is True, defined types are resolved to
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
467 their backing types.
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
468 """
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
469 if type(t) is ast.DefinedType:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
470 if reveil_defined:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
471 t = self.the_type(t.typ)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
472 elif type(t) in [ast.Identifier, ast.Member]:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
473 t = self.the_type(self.resolveSymbol(t), reveil_defined)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
474 elif isinstance(t, ast.Type):
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
475 pass
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
476 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
477 raise NotImplementedError(str(t))
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
478 assert isinstance(t, ast.Type)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
479 return t
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
480
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
481 def equal_types(self, a, b, byname=False):
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
482 """ Compare types a and b for structural equavalence.
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
483 if byname is True stop on defined types.
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
484 """
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
485 # Recurse into named types:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
486 a = self.the_type(a, not byname)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
487 b = self.the_type(b, not byname)
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
488
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
489 # Check types for sanity:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
490 self.check_type(a)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
491 self.check_type(b)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
492
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
493 # Do structural equivalence check:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
494 if type(a) is type(b):
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
495 if type(a) is ast.BaseType:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
496 return a.name == b.name
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
497 elif type(a) is ast.PointerType:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
498 # If a pointed type is detected, stop structural
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
499 # equivalence:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
500 return self.equal_types(a.ptype, b.ptype, byname=True)
308
2e7f55319858 Merged analyse into codegenerator
Windel Bouwman
parents: 307
diff changeset
501 elif type(a) is ast.StructureType:
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
502 if len(a.mems) != len(b.mems):
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
503 return False
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
504 return all(self.equal_types(am.typ, bm.typ) for am, bm in
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
505 zip(a.mems, b.mems))
354
5477e499b039 Added some sort of string functionality
Windel Bouwman
parents: 353
diff changeset
506 elif type(a) is ast.ArrayType:
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
507 return self.equal_types(a.element_type, b.element_type)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
508 elif type(a) is ast.DefinedType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
509 # Try by name in case of defined types:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
510 return a.name == b.name
307
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
511 else:
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
512 raise NotImplementedError('{} not implemented'.format(type(a)))
e609d5296ee9 Massive rewrite of codegenerator
Windel Bouwman
parents: 306
diff changeset
513 return False
389
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
514
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
515 def check_type(self, t, first=True, byname=False):
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
516 """ Determine struct offsets and check for recursiveness by using
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
517 mark and sweep algorithm.
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
518 The calling function could call this function with first set
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
519 to clear the marks.
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
520 """
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
521
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
522 # Reset the mark and sweep:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
523 if first:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
524 self.got_types = set()
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
525
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
526 # Resolve the type:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
527 t = self.the_type(t, not byname)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
528
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
529 # Check for recursion:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
530 if t in self.got_types:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
531 raise SemanticError('Recursive data type {}'.format(t), None)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
532
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
533 if type(t) is ast.BaseType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
534 pass
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
535 elif type(t) is ast.PointerType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
536 # If a pointed type is detected, stop structural
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
537 # equivalence:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
538 self.check_type(t.ptype, first=False, byname=True)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
539 elif type(t) is ast.StructureType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
540 self.got_types.add(t)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
541 # Setup offsets of fields. Is this the right place?:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
542 offset = 0
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
543 for struct_member in t.mems:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
544 self.check_type(struct_member.typ, first=False)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
545 struct_member.offset = offset
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
546 offset = offset + self.size_of(struct_member.typ)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
547 elif type(t) is ast.ArrayType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
548 self.check_type(t.element_type, first=False)
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
549 elif type(t) is ast.DefinedType:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
550 pass
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
551 else:
2ec730e45ea1 Added check for recursive struct
Windel Bouwman
parents: 364
diff changeset
552 raise NotImplementedError('{} not implemented'.format(type(t)))