package de.dhbwstuttgart.bytecode; import java.awt.List; import java.util.ArrayList; import java.util.HashMap; import java.util.Iterator; import org.objectweb.asm.ClassWriter; import org.objectweb.asm.FieldVisitor; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Opcodes; import org.objectweb.asm.Type; import de.dhbwstuttgart.parser.SyntaxTreeGenerator.AssignToLocal; import de.dhbwstuttgart.syntaxtree.*; import de.dhbwstuttgart.syntaxtree.statement.ArgumentList; import de.dhbwstuttgart.syntaxtree.statement.Assign; import de.dhbwstuttgart.syntaxtree.statement.AssignToField; import de.dhbwstuttgart.syntaxtree.statement.Binary; import de.dhbwstuttgart.syntaxtree.statement.Block; import de.dhbwstuttgart.syntaxtree.statement.CastExpr; import de.dhbwstuttgart.syntaxtree.statement.DoStmt; import de.dhbwstuttgart.syntaxtree.statement.EmptyStmt; import de.dhbwstuttgart.syntaxtree.statement.FieldVar; import de.dhbwstuttgart.syntaxtree.statement.ForStmt; import de.dhbwstuttgart.syntaxtree.statement.IfStmt; import de.dhbwstuttgart.syntaxtree.statement.InstanceOf; import de.dhbwstuttgart.syntaxtree.statement.LambdaExpression; import de.dhbwstuttgart.syntaxtree.statement.LocalVar; import de.dhbwstuttgart.syntaxtree.statement.LocalVarDecl; import de.dhbwstuttgart.syntaxtree.statement.MethodCall; import de.dhbwstuttgart.syntaxtree.statement.NewArray; import de.dhbwstuttgart.syntaxtree.statement.NewClass; import de.dhbwstuttgart.syntaxtree.statement.Receiver; import de.dhbwstuttgart.syntaxtree.statement.Return; import de.dhbwstuttgart.syntaxtree.statement.ReturnVoid; import de.dhbwstuttgart.syntaxtree.statement.StaticClassName; import de.dhbwstuttgart.syntaxtree.statement.Super; import de.dhbwstuttgart.syntaxtree.statement.SuperCall; import de.dhbwstuttgart.syntaxtree.statement.This; import de.dhbwstuttgart.syntaxtree.statement.UnaryPlus; import de.dhbwstuttgart.syntaxtree.statement.WhileStmt; import de.dhbwstuttgart.syntaxtree.statement.literal.Literal; import de.dhbwstuttgart.syntaxtree.statement.literal.Null; import de.dhbwstuttgart.syntaxtree.type.ExtendsWildcardType; import de.dhbwstuttgart.syntaxtree.type.GenericRefType; import de.dhbwstuttgart.syntaxtree.type.RefType; import de.dhbwstuttgart.syntaxtree.type.SuperWildcardType; import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder; public class BytecodeGen implements ASTVisitor { ClassWriter cw =new ClassWriter(ClassWriter.COMPUTE_FRAMES|ClassWriter.COMPUTE_MAXS); // String methDesc; String type; // stores parameter, local vars and the next index on the local variable table, which use for aload_i, astore_i,... HashMap paramsAndLocals = new HashMap<>(); byte[] bytecode; HashMap classFiles; public BytecodeGen(HashMap classFiles) { this.classFiles = classFiles; } @Override public void visit(SourceFile sourceFile) { for(ClassOrInterface cl : sourceFile.getClasses()) { BytecodeGen classGen = new BytecodeGen(classFiles); cl.accept(classGen); classGen.writeClass(cl.getClassName().toString()); } } private void writeClass(String name) { bytecode = cw.toByteArray(); classFiles.put(name, bytecode); } public HashMap getClassFiles() { return classFiles; } @Override public void visit(ClassOrInterface classOrInterface) { // access flages?? cw.visit(Opcodes.V1_8, classOrInterface.getModifiers(), classOrInterface.getClassName().toString() , null, classOrInterface.getSuperClass().toString(), null); // for each field in the class for(Field f : classOrInterface.getFieldDecl()) { f.accept(this); } for(Constructor c : classOrInterface.getConstructors()) { c.accept(this); } for(Method m : classOrInterface.getMethods()) { m.accept(this); } cw.visitSource(classOrInterface.getClassName().toString()+".jav", null); } @Override public void visit(Constructor field) { String methDesc = getConsDesc(field); MethodVisitor mv = cw.visitMethod(Opcodes.ACC_PUBLIC, "", methDesc, null, null); mv.visitCode(); BytecodeGenMethod gen = new BytecodeGenMethod(field, mv,paramsAndLocals,true); mv.visitInsn(Opcodes.RETURN); mv.visitEnd(); } private String getConsDesc(Constructor field) { String methDesc; methDesc = "("; field.getParameterList().accept(this); Iterator itr = field.getParameterList().iterator(); while(itr.hasNext()) { FormalParameter fp = itr.next(); methDesc = methDesc + "L"+fp.getType().toString() + ";"; } methDesc = methDesc + ")V"; return methDesc; } @Override public void visit(Method method) { // ParameterList pl = method.getParameterList(); String methDesc = getMethDesc(method); // methDesc = "("; // method.getParameterList().accept(this); // methDesc = methDesc + ")" + method.getReturnType().toString(); // create method descriptor (p1,p2,...)RT // String methDesc = "("; // Iterator itr = pl.iterator(); // while(itr.hasNext()) { // FormalParameter fp = itr.next(); //// fp.getType().accept(this); // methDesc = methDesc + "L"+fp.getType().toString() + ";"; // i++; // } // methDesc = methDesc + ")" + method.getReturnType().toString(); MethodVisitor mv = cw.visitMethod(Opcodes.ACC_PUBLIC, method.getName(), methDesc, null, null); mv.visitCode(); BytecodeGenMethod gen = new BytecodeGenMethod(method, mv,paramsAndLocals,false); } private String getMethDesc(Method method) { String methDesc; methDesc = "("; method.getParameterList().accept(this); Iterator itr = method.getParameterList().iterator(); while(itr.hasNext()) { FormalParameter fp = itr.next(); methDesc = methDesc + "L"+fp.getType().toString() + ";"; } methDesc = methDesc + ")" + method.getReturnType().toString(); return methDesc; } @Override public void visit(ParameterList formalParameters) { Iterator itr = formalParameters.iterator(); int i = 1; while(itr.hasNext()) { FormalParameter fp = itr.next(); paramsAndLocals.put(fp.getName(), i); fp.accept(this); i++; } } @Override public void visit(FormalParameter formalParameter) { formalParameter.getType().accept(this); } @Override public void visit(RefType refType) { type = "L"+refType.toString()+";"; } @Override public void visit(SuperWildcardType superWildcardType) { // TODO Auto-generated method stub } @Override public void visit(TypePlaceholder typePlaceholder) { // TODO Auto-generated method stub } @Override public void visit(ExtendsWildcardType extendsWildcardType) { // TODO Auto-generated method stub } @Override public void visit(GenericRefType genericRefType) { // TODO Auto-generated method stub } // ?? @Override public void visit(FieldVar fieldVar) { // cw.newField(fieldVar.receiver.toString(), fieldVar.fieldVarName.toString(), fieldVar.getType().toString()); } // access flages?? modifiers @Override public void visit(Field field) { FieldVisitor fv = cw.visitField(Opcodes.ACC_PRIVATE, field.getName(), "L"+field.getType()+";", null, null); fv.visitEnd(); } @Override public void visit(LambdaExpression lambdaExpression) { // TODO Auto-generated method stub } @Override public void visit(Assign assign) { // TODO Auto-generated method stub } @Override public void visit(Binary binary) { // TODO Auto-generated method stub } @Override public void visit(Block block) { // TODO Auto-generated method stub } @Override public void visit(CastExpr castExpr) { // TODO Auto-generated method stub } @Override public void visit(EmptyStmt emptyStmt) { // TODO Auto-generated method stub } @Override public void visit(ForStmt forStmt) { // TODO Auto-generated method stub } @Override public void visit(IfStmt ifStmt) { // TODO Auto-generated method stub } @Override public void visit(InstanceOf instanceOf) { // TODO Auto-generated method stub } @Override public void visit(LocalVar localVar) { // TODO Auto-generated method stub } @Override public void visit(LocalVarDecl localVarDecl) { // TODO Auto-generated method stub } @Override public void visit(MethodCall methodCall) { // TODO Auto-generated method stub } @Override public void visit(NewClass methodCall) { // TODO Auto-generated method stub } @Override public void visit(NewArray newArray) { // TODO Auto-generated method stub } @Override public void visit(Receiver receiver) { // TODO Auto-generated method stub } @Override public void visit(Return aReturn) { // TODO Auto-generated method stub } @Override public void visit(ReturnVoid aReturn) { // TODO Auto-generated method stub } @Override public void visit(StaticClassName staticClassName) { // TODO Auto-generated method stub } @Override public void visit(Super aSuper) { // TODO Auto-generated method stub } @Override public void visit(This aThis) { // TODO Auto-generated method stub } @Override public void visit(UnaryPlus unaryPlus) { // TODO Auto-generated method stub } @Override public void visit(WhileStmt whileStmt) { // TODO Auto-generated method stub } @Override public void visit(DoStmt whileStmt) { // TODO Auto-generated method stub } @Override public void visit(Null aNull) { // TODO Auto-generated method stub } // ??? @Override public void visit(Literal literal) { // TODO Auto-generated method stub } @Override public void visit(ArgumentList argumentList) { // TODO Auto-generated method stub } @Override public void visit(GenericTypeVar genericTypeVar) { // TODO Auto-generated method stub } @Override public void visit(GenericDeclarationList genericTypeVars) { // TODO Auto-generated method stub } @Override public void visit(AssignToField assignLeftSide) { // TODO Auto-generated method stub } @Override public void visit(AssignToLocal assignLeftSide) { // TODO Auto-generated method stub } @Override public void visit(SuperCall superCall) { // TODO Auto-generated method stub } }