diff --git a/AstBuilding.py b/AstBuilding.py index 177b623cc98260a8686862ff177e63ff86d1bd16..dbf919b220c100236f64e714c2d6dfc4035cb5b9 100644 --- a/AstBuilding.py +++ b/AstBuilding.py @@ -71,7 +71,7 @@ def arrayClassMemory(types): # Label for SIT pointer: "I_SIT_spot_typeName_array" # Label for method pointers: "V_typeName_methodName_param_array" code = "section .data\n" - code += "; Start of class memory layout\n" + code += "; Start of class memory layout\n" code += pLabel(name=t, type="array") + \ pLabel("SIT_spot_"+ t + "_array", "inter") + \ p(instruction="dd", arg1=64) @@ -104,28 +104,38 @@ def arrayClassMemory(types): # Preparation before code Gen: # 1. Calculating the size of the object from each class # (also simultaneosly creating fieldOffset because fieldOffset table is required to generate size) -# 2. SIT building +# 2. SIT building # 3. Creating class memory layout for every possible array types def codeGenPrep(ASTs): interM = [] types = ['boolean', 'byte', 'char', 'int', 'short'] # All possible types + j = 0 for t in ASTs: classInterNode = t[1].typeDcl - if classInterNode and classInterNode.__class__.__name__ == "ClassNode": + types.append(classInterNode) + + # assign each type a position in the subtype testing table + classInterNode.subTypeOffset = j * 4 + j += 1 + + if classInterNode.__class__.__name__ == "ClassNode": classInterNode.populateSizeAndFieldOffset() classInterNode.populateMethodOffset() types.append(classInterNode.name) else: # interfaceNode, get their methods to prep for SIT interM += classInterNode.methods - if interM: # prep SIT - for i in range(len(interM)): - interM[i].methodOffset = i * 4 - interM[i].isInterM = True - for t in ASTs: - classInterNode = t[1].typeDcl - if classInterNode and classInterNode.__class__.__name__ == "ClassNode": - classInterNode.SITsize = len(interM) + # prep SIT + for i in range(len(interM)): + interM[i].methodOffset = i * 4 + interM[i].isInterM = True + + # store SIT and subtype table size + for t in ASTs: + classInterNode = t[1].typeDcl + if classInterNode.__class__.__name__ == "ClassNode": + classInterNode.SITsize = len(interM) + classInterNode.subTypeSize = len(types) - 5 # no primitive types in subtype table return arrayClassMemory(types) diff --git a/CodeGenUtils.py b/CodeGenUtils.py index 4ea67d398fd23676abec5e7f6c63604a11585701..2602d677848c95e45ad71421eff47de512618fee 100644 --- a/CodeGenUtils.py +++ b/CodeGenUtils.py @@ -103,7 +103,7 @@ def iffalse(cond, label): result += cond.code result += p("jz", label) return result - + val = cond.getConstant() if val != None: if val == True: @@ -135,11 +135,8 @@ def genericHelperFunctions(): # Note: using jle instead of calling the function since we don't need to preserve eip upon calling the exception function (?) # saving and restoring ebx, a callee-save register code += pLabel(name="_Null_Check", type="helper", comment="helper function for null check") + \ - p(instruction="push", arg1="ebx", comment="saving ebx") + \ - p(instruction="mov", arg1="ebx", arg2="[G__Zero]") + \ - p(instruction="cmp", arg1="eax", arg2="ebx") + \ + p(instruction="cmp", arg1="eax", arg2="[G__Zero]") + \ p(instruction="jle", arg1="H__Throw_Exception") + \ - p(instruction="pop", arg1="ebx", comment="restoring ebx") + \ p(instruction="ret", arg1="") # Helper function to perform bounds check on array diff --git a/ExprPrimaryNodes.py b/ExprPrimaryNodes.py index 43399ec5a704914e041aa62bb4e33b728606bce0..875443891797d24426f33b1e1bfaed15477aae94 100644 --- a/ExprPrimaryNodes.py +++ b/ExprPrimaryNodes.py @@ -367,7 +367,6 @@ class CastNode(ASTNode): def checkType(self): self.left.checkType() - from pprint import pprint self.right.disambigName() self.right.checkType() if (self.left.myType.isNum() and self.right.myType.isNum()) \ @@ -377,6 +376,29 @@ class CastNode(ASTNode): return raise Exception("ERROR: Cannot cast type {} to type {}.".format(self.right.myType.name, self.left.myType.name)) + def codeGen(self): + if hasattr(self, "code") and self.code != "": + return + self.code = "" + + # get object at right + self.code += "; casting\n" + self.right.codeGen() + self.code += self.right.code + + # subtype test: + if not self.left.myType.isPrimitive: + # only test if right is not primitive (primitive types would be correctly static tested already) + offset = self.left.myType.typePointer.subTypeOffset + self.code += p("mov", "ebx", "[eax]", "start subtype test") # access class tag of left object + self.code += p("mov", "ebx", "[ebx + 4]") # access subtype testing column + self.code += p("mov", "ebx", "[ebx + " + str(offset) + "]") # ebx has isSubtype + + # exception if not subtype, else do nothing (object is already in eax) + self.code += p("cmp", "[G__Zero]", "ebx") + self.code += p("je", "H__Throw_Exception") + self.code += "; end of casting\n" + ################################################################################### # unqualCreate NEW name LPAREN args RPAREN class ClassCreateNode(ASTNode): @@ -549,6 +571,28 @@ class ExprNode(ASTNode): return self.code = "" + # instanceOf + if self.op == "instanceof": + + self.code += ("; evaluate instanceof\n") + self.left.codeGen() + self.code += self.left.code + + # only test if non-primitive + if not self.right.myType.isPrimitive: + offset = self.right.myType.typePointer.subTypeOffset + self.code += p("mov", "eax", "[eax]") # access class tag of left object + self.code += p("mov", "eax", "[eax + 4]") # access subtype testing column + self.code += p("mov", "eax", "[eax + " + str(offset) + "]") # subType column stores 0 or 1 already + else: # primitive type can be staticly evaluated + if self.right.myType.assignable(self.left.myType): + self.code += p("mov", "eax", "1") + else: + self.code += p("mov", "eax", "0") + + self.code += ("; end of instanceof\n") + return + # Unary operations: if not self.left: # get right output @@ -651,8 +695,6 @@ class ExprNode(ASTNode): self.code += p('pop', 'edx') # restore edx return - # if self.op == 'instanceof': TODO - # return # generate shorter code if self.op = comparison def getIfFalse(self, label): diff --git a/TypeNodes.py b/TypeNodes.py index 4a8d262de7247764b7a3daed3d668175d08a5e9e..71c91b7b13ed435a4c4a8a936410f05a9c31687e 100644 --- a/TypeNodes.py +++ b/TypeNodes.py @@ -17,6 +17,8 @@ class ClassInterNode(ASTNode): self.children = [] self.canonName = '' self.SITsize = 0 + self.subTypeSize = 0 + self.subTypeOffset = 0 # sets self.inherits = [] @@ -309,6 +311,8 @@ class ClassNode(ClassInterNode): def populateMethodOffset(self): if hasattr(self, "data"): return + + self.code = "" # For read-only section self.data = "" # For writeable data section if self.canonName == "java.lang.Object": self.data += p("global", "I_SIT_" + self.name) @@ -319,10 +323,12 @@ class ClassNode(ClassInterNode): self.data += self.label # SIT - self.data += pLabel("SIT_spot_"+ self.name, "inter") # type 'inter' for interface methods + self.data += pLabel("SIT_spot_"+ self.name, "inter") # type 'inter' for internal usage self.data += p("dd", "42") - # TODO: subtype testing tables + # subtype testing table + self.data += pLabel("subT_spot_"+ self.name, "inter") # type 'inter' for internal usage + self.data += p("dd", "42") ####### ADDING METHODS TO CLASS MEMORY LAYOUT AND FIELDS TO FIELD OFFSET TABLE ######### # Copying over the offsets of methods from superclass and DECLARING memory segment for the methods @@ -378,6 +384,12 @@ class ClassNode(ClassInterNode): self.data += pLabel("SIT_" + str(i), "inter") self.data += p("dd", "42") + # Layout subtype testing column + self.data += pLabel("subT_" + self.name, "inter") + for i in range(self.subTypeSize): + self.data += pLabel("subT_" + str(i), "inter") + self.data += p("dd", "0") # 0 for False, 1 for True + self.data += ";END OF CLASS MEMORY LAYOUT FOR CLASS " + self.name + "\n" @@ -429,19 +441,21 @@ class ClassNode(ClassInterNode): self.code += p("mov", "[eax]", "dword " + imLabel) self.code += "; End of fill in SIT.\n" - self.code += p(instruction="ret", arg1="") - self.code += "; End of function for filling in class memory layout\n" - + # fill in subtype testing column + self.code += "\n; Filling in subtype testing\n" + self.code += p("mov", "eax", "I_subT_spot_" + self.name) + self.code += p("mov", "[eax]", "dword I_subT_" + self.name) + for t in getSupers(self): # getSupers includes self + dlabel = "I_subT_" + str(t.subTypeOffset // 4) + self.code += p("mov", "eax", dlabel, "self is a subType of " + t.canonName) + self.code += p("mov", "[eax]", "dword 1") + self.code += "; End of fill in subtype testing.\n" + self.code += p(instruction="ret", arg1="") + self.code += "; End of function for filling in class memory layout\n" - - # print(self.name) - # print(self.fieldOffset) - - # print(self.code) - ########################################################### # Generating a function that allocates and initializes all static fields @@ -558,3 +572,12 @@ def safeReplace(cur, new, className): # quick fix for final method getClass from java.lang.Object if 'final' in cur.mods and cur.name != 'getClass': raise Exception("ERROR: In class {0}, {1} '{2}' in class '{3}' replaces final {1} in class/interface {4}".format(className, methodOrField, new.name, new.typeName, cur.typeName)) + + +# helper: get list of all super class/interface of a ClassInterNode +def getSupers(classType): + result = {classType} + for s in classType.super: + result.add(s) + result.union(getSupers(s)) + return result