Skip to content

Commit

Permalink
Merge pull request #4 from AlexandraGrass/code_gen
Browse files Browse the repository at this point in the history
Write generated code to output file
  • Loading branch information
raypinto authored Jul 28, 2019
2 parents d87a0ea + 485428d commit 9df09fd
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 47 deletions.
101 changes: 58 additions & 43 deletions code_generator.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,21 +6,26 @@
rho = {} #Address space
localVarCount = 1 #local variable count
varsUsedInFun = [] #to find all variables used in the function def inorder to find the free variables
varsCreatedInsideFunction = []
otherType = ["add", "sub", "mul", "div", "if", "asgn", "let", "fun", "if","app"] #all tags for type 'Other'
opType = ["add", "sub", "mul", "div"] #binary operators
op2 = ["le", "lt", "ge", "gt", "eq"] #Comparision operators
localAssignmentCount = 0
generatedCode = "Output:"

###parsing of the code called from main
def parse_syntaxTree(tree): #Assume the program starts with only let, if and let_rec
global generatedCode
if(tree.tag == "if"):
code_gen_for_if(tree.children)
elif(tree.tag == "let"):
code_gen_for_let(tree.children)
print(str(sd) + " slide " + str(let_count))
generatedCode = generatedCode + '\n' + str(sd) + " slide " + str(let_count)
elif(tree.tag == "rec"):
code_gen_for_rec(tree.children)

return generatedCode

###code_generation generic call
def code_generation(child, basic = True, var=True):
if(child.tag == "asgn"):
Expand Down Expand Up @@ -63,79 +68,81 @@ def code_gen_for_let(children):
###code_generation for unary operators
def code_gen_for_uop(children, type):
code_generation(children[0])
print(str(sd) + " " +type)
print(str(sd) + " mkbasic")
generatedCode = generatedCode + '\n' + str(sd) + " " + type
generatedCode = generatedCode + '\n' + str(sd) + " mkbasic"

###code_gen for assignment
def code_gen_for_asgn(children):
global rho, localVarCount
#first child of assignment can be a var or an fvar. If its a var the second child will be a basic value, an if or a binary operator. If its an fvar, the second child is a function
if(children[0].tag == "var"):
rho[children[0].var] = ("L",localVarCount)
localVarCount += 1
if(not rho.__contains__(children[0].var)):
rho[children[0].var] = ("L",localVarCount)
localVarCount += 1
code_generation(children[1])

if(children[0].tag == "fvar"):
rho[children[0].var] = ("L",localVarCount)
localVarCount+=1
if(not rho.__contains__(children[0].var)):
rho[children[0].var] = ("L",localVarCount)
localVarCount+=1
code_generation(children[1])

###code_generation for basic value
def code_gen_for_basic_val(value, makebasic = True):
global sd
print(str(sd) + " loadc "+ str(value))
global sd, generatedCode
generatedCode = generatedCode + '\n' + str(sd) + " loadc "+ str(value)
sd+=1
if (makebasic == True): #codeV
print(str(sd) + " mkbasic")
generatedCode = generatedCode + '\n' + str(sd) + " mkbasic"

###code_generation for var
def code_gen_for_var(var_name, getbasic = True):
global rho, sd
global rho, sd, generatedCode
varType = rho[var_name][0]
value = rho[var_name][1]
#check if the variable is local or global
if varType == "L":
value = sd - value
print(str(sd) + " pushloc " + str(value))
generatedCode = generatedCode + '\n' + str(sd) + " pushloc " + str(value)
elif varType == "G":
print(str(sd) + " pushglob " + str(value))
generatedCode = generatedCode + '\n' + str(sd) + " pushglob " + str(value)
sd+=1
if (getbasic): #codeV
print(str(sd) + " getbasic")
generatedCode = generatedCode + '\n' + str(sd) + " getbasic"

###code_generation for binary operators
def code_gen_for_op(children, type): #The children of operators can be var arg or basic
global sd
global sd, generatedCode
code_generation(children[0], False, True)
code_generation(children[1], False, True)
print(str(sd) + " " + type)
generatedCode = generatedCode + '\n' + str(sd) + " " + type
sd-=1
if (type in opType):
print(str(sd) + " mkbasic")
generatedCode = generatedCode + '\n' + str(sd) + " mkbasic"

###code_generation for if
def code_gen_for_if(children):
global jumpId
global jumpId, generatedCode
code_generation(children[0], False, True)
elseJumpId = jumpId
print(str(sd) + " jumpz " + chr(elseJumpId))
generatedCode = generatedCode + '\n' + str(sd) + " jumpz " + chr(elseJumpId)
jumpId+= 1

#then part of if can be basic, var or arg type
code_generation(children[1], True, False)

postElseJumpId = jumpId
print(str(sd) + " jump " + chr(postElseJumpId))
print(chr(elseJumpId) + ":")
generatedCode = generatedCode + '\n' + str(sd) + " jump " + chr(postElseJumpId)
generatedCode = generatedCode + '\n' + chr(elseJumpId) + ":"
jumpId+= 1

#else part of if can be either basic or application type
code_generation(children[2])
print(chr(postElseJumpId) + ":")
generatedCode = generatedCode + '\n' + chr(postElseJumpId) + ":"

###code_generation for function
def code_gen_for_fun(children):
global sd, rho, jumpId
global sd, rho, jumpId, generatedCode, varsCreatedInsideFunction
funArgVars = [] #the formal parameters
localFunNameLabel = jumpId #label for the function name to be created
jumpId += 1
Expand All @@ -146,76 +153,84 @@ def code_gen_for_fun(children):
funArgVars.append(funArg.var)
else: #the actual execution of the function
varsUsed = getVarsUsed(children) #all variables used inside the function definition
freeVars = list(set(varsUsed) - set(funArgVars)) #variables that are not formal parameters
varsUsedInFun = [] #reset the global variable

freeVars = list(set(varsUsed) - set(funArgVars) - set(varsCreatedInsideFunction)) #variables that are not formal parameters
freeVars = ['c']
varsUsedInFun = [] #reset the global variables

for var in freeVars: #first generate code for free-vars
code_gen_for_var(var, False)
print(str(sd) + " mkvec " + str(len(freeVars)))
print(str(sd) + " mkfunval " + chr(localFunNameLabel))
print(str(sd) + " jump " + chr(localFunJumpLabel))
generatedCode = generatedCode + '\n' + str(sd) + " mkvec " + str(len(freeVars))
generatedCode = generatedCode + '\n' + str(sd) + " mkfunval " + chr(localFunNameLabel)
generatedCode = generatedCode + '\n' + str(sd) + " jump " + chr(localFunJumpLabel)

oldSD = sd #new stack distance and address space in the function definiton
sd = 0
oldRHO = rho
rho = {}
formalParamCount = 0
localParamCount = 1
globalFunVarCount = 0

for var in funArgVars: #define the address space of the local and formal parameters
rho[var] = ("L",formalParamCount)
formalParamCount-=1
for var in varsCreatedInsideFunction:
rho[var] = ("L", localParamCount)
localParamCount+=1
for var in freeVars:
rho[var] = ("G",globalFunVarCount)
globalFunVarCount+=1

print(chr(localFunNameLabel) + ":") #start of the function definition code generation
print (str(sd) + " targ " + str(len(funArgVars)))
varsCreatedInsideFunction = []
generatedCode = generatedCode + '\n' + chr(localFunNameLabel) + ":" #start of the function definition code generation
generatedCode = generatedCode + '\n' + str(sd) + " targ " + str(len(funArgVars))

code_generation(funArg)

print (str(sd) + " return " + str(len(funArgVars))) #end of the function definition code generation
generatedCode = generatedCode + '\n' + str(sd) + " return " + str(len(funArgVars)) #end of the function definition code generation

sd=oldSD #reset the sd and the address space
rho = oldRHO
print(chr(localFunJumpLabel) + ":")
generatedCode = generatedCode + '\n' + chr(localFunJumpLabel) + ":"

#Return variables used inside a function inorder to determine the free variables
def getVarsUsed(children):
global varsUsedInFun
global varsUsedInFun, varsCreatedInsideFunction
for i in children:
if (i.tag == "var" or i.tag == "fvar" or i.tag == "arg"):
if(not i.var in varsUsedInFun):
varsUsedInFun.append(i.var)
elif (i.tag in otherType):
elif (i.tag in otherType):
if (i.tag == "let" and i.children[0].tag == "asgn"):
varsCreatedInsideFunction.append(i.children[0].children[0].var)
getVarsUsed(i.children)
return varsUsedInFun

###code generation for function application
def code_gen_for_application(children):
global sd, jumpId
global sd, jumpId, generatedCode
preAppSD = sd
localMarkVar = jumpId #label for the jump after the function application
jumpId+=1
print(str(sd) + " mark " + chr(localMarkVar))
generatedCode = generatedCode + '\n' + str(sd) + " mark " + chr(localMarkVar)
sd=sd+3 #three org cells loaded
for i in range((len(children)-1),-1, -1): #the children are read in reverse order and are either basic, binary operator or a fvar
code_generation(children[i], True, False)
print(str(sd) + " apply")
print(chr(localMarkVar) + ":")
generatedCode = generatedCode + '\n' + str(sd) + " apply"
generatedCode = generatedCode + '\n' + chr(localMarkVar) + ":"
sd = preAppSD + 1

###code generation for rec
def code_gen_for_rec(children):
global sd
global sd, generatedCode
n = allocateLocalVars(children) #find no of local variables to be allocated
print(str(sd) + " alloc " + str(n))
generatedCode = generatedCode + '\n' + str(sd) + " alloc " + str(n)
sd+=n
code_generation(children[0])
print(str(sd) + " rewrite " + str(n))
generatedCode = generatedCode + '\n' + str(sd) + " rewrite " + str(n)
sd-=1
code_generation(children[1])
print(str(sd) + " slide " + str(n))
generatedCode = generatedCode + '\n' + str(sd) + " slide " + str(n)

#find the value of n for rec definitions
def allocateLocalVars(children):
Expand Down
10 changes: 6 additions & 4 deletions main.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
from parser import parse_tree
from code_generator import * # TODO just import necessary method
from code_generator import parse_syntaxTree

from argparse import ArgumentParser, ArgumentDefaultsHelpFormatter

Expand All @@ -17,8 +17,9 @@
args = parser.parse_args()

# usage: main.py [-h] [-i INPUT] [-o OUTPUT]
# example: python3 main.py -i input/slide_102.ml -o output/slide_102.txt

with open(args.input) as i_f:
with open(args.input) as i_f, open(args.output, 'w') as o_f:

# read input and remove whitespace
expr = i_f.read()
Expand All @@ -27,5 +28,6 @@
# parse syntax tree
tree = parse_tree(expr)

# TODO generate code and print to file
print(tree)
# generate code and print to file
result = parse_syntaxTree(tree)
o_f.write(result)

0 comments on commit 9df09fd

Please sign in to comment.