Mercurial > lcfOS
view python/ks/irgenerator.py @ 158:9683a4cd848f
Added some functions for code generation
author | Windel Bouwman |
---|---|
date | Fri, 08 Mar 2013 16:52:44 +0100 |
parents | 91af0e40f868 |
children | 4fd075e8259c |
line wrap: on
line source
""" Generates ir code from ast tree. """ from .nodes import * from ...core.errors import Error from ... import core from .builtin import real, integer, boolean, char, void def coreType(typ): """ Return the correct core type given a type """ if type(typ) is BaseType: if typ is integer: return core.i32 if typ is void: return core.void elif type(typ) is ProcedureType: rType = coreType(typ.returntype) fpTypes = [coreType(p.typ) for p in typ.parameters] return core.FunctionType(rType, fpTypes) print(typ) raise NotImplementedError() class KsIrGenerator: def __init__(self): self.builder = core.IRBuilder() # Code generation functions: def genexprcode(self, node): """ Generate code for expressions! """ if isinstance(node, Binop): """ Handle a binary operation (two arguments) of some kind """ lhs = self.genexprcode(node.a) rhs = self.genexprcode(node.b) if node.op == '*': if node.typ.isType(integer): return self.builder.createMul(lhs, rhs) elif node.op == '+': if node.typ.isType(integer): return self.builder.createAdd(lhs, rhs) elif node.op == '-': if node.typ.isType(integer): return self.builder.createSub(lhs, rhs) Error('Unknown binop or type {0}'.format(node)) elif isinstance(node, Designator): # dereference, array index. Make sure that the result comes into a register if len(node.selectors) > 0: Error('Only integer types implemented') else: # No selectors, load variable directly print(node) #Error('Cannot load variable type {0}'.format(node.typ)) elif type(node) is Constant: return core.Constant(node.value, coreType(node.typ)) else: Error('Cannot generate expression code for: {0}'.format(node)) def gencode(self, node): """ Code generation function for AST nodes """ if isinstance(node, Module): # Create module: self.mod = core.Module(node.name) globs = node.symtable.getAllLocal(Variable) for g in globs: print('global:', g) # Loop over all functions: print(node.symtable) node.symtable.printTable() funcs = node.symtable.getAllLocal(Procedure) for f in funcs: self.gencode(f) # Create a function called init for this module: self.mod.dump() return self.mod elif type(node) is Procedure: ftype = coreType(node.typ) print('function', node, ftype) func = core.Function(ftype, node.name, self.mod) bb = core.BasicBlock() func.basicblocks.append(bb) self.builder.setInsertPoint(bb) self.gencode(node.block) self.builder.setInsertPoint(None) variables = node.symtable.getAllLocal(Variable) print(variables) elif isinstance(node, StatementSequence): for s in node.statements: self.gencode(s) elif type(node) is ProcedureCall: # Prepare parameters on the stack: print("TODO") elif type(node) is Assignment: if node.lval.typ.isType(integer): print('assign') rhs = self.genexprcode(node.rval) # Calculate the value that has to be stored. #self.gencode(node.lval) print("TODO: assigment") else: Error('Assignments of other types not implemented') elif type(node) is IfStatement: self.genexprcode(node.condition) print("TODO IF") if node.falsestatement: # If with else clause pass else: # If without else clause pass elif isinstance(node, WhileStatement): self.genexprcode(node.condition) self.gencode(node.dostatements) elif type(node) is ForStatement: # Initial load of iterator variable: self.genexprcode(node.begin) self.genexprcode(node.end) self.gencode(node.statements) Error('No implementation of FOR statement') elif isinstance(node, EmptyStatement): pass # That was easy :) elif type(node) is StringConstant: self.strings.append(node) elif type(node) is Designator: Error('Can only gencode for designator with selectors') else: print('not generating code for {0}'.format(node)) def generateIr(self, context, ast): """ ir generation front end """ # Create a new context for this code. self.context = context return self.gencode(ast)