|
|
|
@ -14,9 +14,10 @@ import ast.statementexpressions.NewDeclarationNode;
|
|
|
|
|
import ast.statementexpressions.crementexpressions.CrementType;
|
|
|
|
|
import ast.statementexpressions.crementexpressions.DecrementNode;
|
|
|
|
|
import ast.statementexpressions.crementexpressions.IncrementNode;
|
|
|
|
|
import ast.statementexpressions.methodcallstatementnexpressions.ChainedMethodNode;
|
|
|
|
|
import ast.statementexpressions.methodcallstatementnexpressions.MethodCallNode;
|
|
|
|
|
import ast.statements.*;
|
|
|
|
|
import ast.type.AccessModifierNode;
|
|
|
|
|
import ast.type.EnumAccessModifierNode;
|
|
|
|
|
import ast.type.ValueNode;
|
|
|
|
|
import ast.type.type.BaseType;
|
|
|
|
|
import ast.type.type.ReferenceType;
|
|
|
|
@ -70,9 +71,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
|
|
|
|
|
// Visit all statements
|
|
|
|
|
for (IStatementNode statementNode : constructorNode.block.statements) {
|
|
|
|
|
if (statementNode != null) {
|
|
|
|
|
statementNode.accept(this);
|
|
|
|
|
}
|
|
|
|
|
statementNode.accept(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
methodVisitor.visitMaxs(0, 0);
|
|
|
|
@ -81,8 +80,10 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(MainMethodNode mainMethodNode) {
|
|
|
|
|
methodVisitor = classWriter.visitMethod(mapper.mapAccessTypeToOpcode(mainMethodNode.accesModifier),
|
|
|
|
|
mainMethodNode.getIdentifier(),
|
|
|
|
|
AccessModifierNode accessModifierNode = new AccessModifierNode("");
|
|
|
|
|
accessModifierNode.accessType = EnumAccessModifierNode.PUBLIC_STATIC;
|
|
|
|
|
methodVisitor = classWriter.visitMethod(Opcodes.ACC_PUBLIC | Opcodes.ACC_STATIC,
|
|
|
|
|
"main",
|
|
|
|
|
"([Ljava/lang/String;)V",
|
|
|
|
|
null,
|
|
|
|
|
null);
|
|
|
|
@ -104,7 +105,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
public void visit(MethodNode methodNode) {
|
|
|
|
|
methodVisitor = classWriter.visitMethod(mapper.mapAccessTypeToOpcode(methodNode.accesModifier),
|
|
|
|
|
methodNode.getIdentifier(),
|
|
|
|
|
mapper.generateMethodDescriptor((BaseType) methodNode.getType(), methodNode.parameters),
|
|
|
|
|
mapper.generateMethodDescriptor(methodNode.getType(), methodNode.parameters),
|
|
|
|
|
null,
|
|
|
|
|
null);
|
|
|
|
|
|
|
|
|
@ -240,7 +241,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
nonCalculationNode.unaryExpression.accept(this);
|
|
|
|
|
nonCalculationNode.expression.accept(this);
|
|
|
|
|
if (nonCalculationNode.unaryExpression.getType() instanceof BaseType && nonCalculationNode.expression.getType() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitJumpInsn(IF_ACMPEQ, labelFalse);
|
|
|
|
|
methodVisitor.visitJumpInsn(IF_ICMPEQ, labelFalse);
|
|
|
|
|
} else {
|
|
|
|
|
methodVisitor.visitJumpInsn(IF_ACMPEQ, labelFalse);
|
|
|
|
|
}
|
|
|
|
@ -265,9 +266,17 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(MemberAccessNode memberAccessNode) {
|
|
|
|
|
// Only used to get, not to put
|
|
|
|
|
if (memberAccessNode.thisExpr) { // Field
|
|
|
|
|
// methodVisitor.visitFieldInsn(GETFIELD, memberAccessNode.identifiers.get(0), memberAccessNode.identifiers.get(1), );
|
|
|
|
|
} else { // Object Attribut
|
|
|
|
|
int localVarIndex = localVariables.indexOf("memberAccessNode.identifier"); // TODO
|
|
|
|
|
if (localVarIndex >= 0) { // local var object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else { // this field
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (memberAccessNode.getTypeNode() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, memberAccessNode.identifiers.get(0), memberAccessNode.identifiers.get(1), mapper.getTypeChar((BaseType) memberAccessNode.getTypeNode()));
|
|
|
|
|
} else if (memberAccessNode.getTypeNode() instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, memberAccessNode.identifiers.get(0), memberAccessNode.identifiers.get(1), "L" + ((ReferenceType) memberAccessNode.getTypeNode()).getIdentifier() + ";");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -313,9 +322,17 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
|
|
|
|
|
// Statements
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(BlockNode blockNode) {
|
|
|
|
|
for(IStatementNode statementNode : blockNode.statements) {
|
|
|
|
|
statementNode.accept(this);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(IfElseNode ifElseNode) {
|
|
|
|
|
Label elseLabel = new Label();
|
|
|
|
|
Label endLabel = new Label();
|
|
|
|
|
|
|
|
|
|
Label[] elseIfLabels = new Label[ifElseNode.elseIfStatements.size()];
|
|
|
|
|
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
|
|
|
@ -330,9 +347,9 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
// else if statements
|
|
|
|
|
methodVisitor.visitJumpInsn(IFEQ, elseIfLabels[0]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ifElseNode.ifStatement.block.accept(this); // accept if block
|
|
|
|
|
|
|
|
|
|
Label endLabel = new Label();
|
|
|
|
|
methodVisitor.visitJumpInsn(GOTO, endLabel);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
|
|
|
@ -351,6 +368,8 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
if (ifElseNode.elseStatement != null) {
|
|
|
|
|
methodVisitor.visitLabel(elseLabel);
|
|
|
|
|
ifElseNode.elseStatement.block.accept(this);
|
|
|
|
|
} else {
|
|
|
|
|
methodVisitor.visitLabel(elseLabel);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
methodVisitor.visitLabel(endLabel);
|
|
|
|
@ -361,22 +380,17 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
if (localVariableDeclarationNode.expression != null) {
|
|
|
|
|
// Process expression
|
|
|
|
|
localVariableDeclarationNode.expression.accept(this);
|
|
|
|
|
// Store result of expression in variable
|
|
|
|
|
if (localVariables.contains(localVariableDeclarationNode.identifier)) {
|
|
|
|
|
if (localVariableDeclarationNode.type instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
} else if (localVariableDeclarationNode.type instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ASTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// add local var to list if not in list
|
|
|
|
|
if (!localVariables.contains(localVariableDeclarationNode.identifier)) {
|
|
|
|
|
localVariables.add(localVariableDeclarationNode.identifier);
|
|
|
|
|
if (localVariableDeclarationNode.type instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
} else if (localVariableDeclarationNode.type instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ASTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (localVariableDeclarationNode.type instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
} else if (localVariableDeclarationNode.type instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ASTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Local var declaration
|
|
|
|
|
if (!localVariables.contains(localVariableDeclarationNode.identifier)) {
|
|
|
|
|
localVariables.add(localVariableDeclarationNode.identifier);
|
|
|
|
|
}
|
|
|
|
@ -385,38 +399,105 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(AssignNode assignNode) {
|
|
|
|
|
// Process expression
|
|
|
|
|
if (assignNode.expression instanceof IncrementNode incrementNode) {
|
|
|
|
|
if (assignNode.assignable.memberAccess != null) { // this / object
|
|
|
|
|
if (assignNode.expression instanceof IncrementNode) {
|
|
|
|
|
if (((IncrementNode) assignNode.expression).crementType.equals(CrementType.PREFIX)) { // ++i
|
|
|
|
|
fieldOrObjectVarPrefixCrementAssign(assignNode);
|
|
|
|
|
} else { // i++
|
|
|
|
|
fieldOrObjectVarSuffixCrementAssign(assignNode);
|
|
|
|
|
}
|
|
|
|
|
} else if (assignNode.expression instanceof DecrementNode) {
|
|
|
|
|
if (((DecrementNode) assignNode.expression).crementType.equals(CrementType.PREFIX)) {
|
|
|
|
|
fieldOrObjectVarPrefixCrementAssign(assignNode);
|
|
|
|
|
} else {
|
|
|
|
|
fieldOrObjectVarSuffixCrementAssign(assignNode);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
assignFieldOrObjectVar(assignNode);
|
|
|
|
|
}
|
|
|
|
|
} else { // local var
|
|
|
|
|
if (assignNode.expression instanceof IncrementNode || assignNode.expression instanceof DecrementNode) {
|
|
|
|
|
localVarCrementAssign(assignNode);
|
|
|
|
|
} else {
|
|
|
|
|
assignNode.expression.accept(this);
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void localVarCrementAssign(AssignNode assignNode) {
|
|
|
|
|
if (assignNode.expression instanceof IncrementNode) {
|
|
|
|
|
IncrementNode incrementNode = (IncrementNode) assignNode.expression;
|
|
|
|
|
if (incrementNode.crementType.equals(CrementType.PREFIX)) { // ++i
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(incrementNode.assignableExpression.identifier), 1);
|
|
|
|
|
assign(assignNode);
|
|
|
|
|
} else if (incrementNode.crementType.equals(CrementType.SUFFIX)) { // Suffix: i++
|
|
|
|
|
assign(assignNode);
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(incrementNode.assignableExpression.identifier), 1);
|
|
|
|
|
incrementNode.accept(this);
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
} else { // i++
|
|
|
|
|
loadBeforeCrement(assignNode);
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
incrementNode.accept(this);
|
|
|
|
|
}
|
|
|
|
|
} else if (assignNode.expression instanceof DecrementNode decrementNode) {
|
|
|
|
|
if (decrementNode.crementType.equals(CrementType.PREFIX)) {
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(decrementNode.assignableExpression.identifier), -1);
|
|
|
|
|
assign(assignNode);
|
|
|
|
|
} else if (decrementNode.crementType.equals(CrementType.SUFFIX)) {
|
|
|
|
|
assign(assignNode);
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(decrementNode.assignableExpression.identifier), 1);
|
|
|
|
|
decrementNode.accept(this);
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
} else {
|
|
|
|
|
loadBeforeCrement(assignNode);
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
decrementNode.accept(this);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
assignNode.expression.accept(this);
|
|
|
|
|
assign(assignNode);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void assign(AssignNode assignNode) {
|
|
|
|
|
if (assignNode.assignable.memberAccess.thisExpr) {
|
|
|
|
|
assignField(assignNode);
|
|
|
|
|
private void fieldOrObjectVarPrefixCrementAssign(AssignNode assignNode) {
|
|
|
|
|
int localVarIndex = localVariables.indexOf(assignNode.assignable.identifier);
|
|
|
|
|
if(localVarIndex >= 0) { // object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else if(assignNode.assignable.memberAccess.thisExpr) { // field
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
} else {
|
|
|
|
|
assignLocalVar(assignNode);
|
|
|
|
|
localVariables.add(assignNode.assignable.identifier);
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assignNode.expression.accept(this);
|
|
|
|
|
methodVisitor.visitInsn(DUP_X1);
|
|
|
|
|
|
|
|
|
|
if (assignNode.expression instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) assignNode.expression.getType()));
|
|
|
|
|
} else if (assignNode.expression instanceof ReferenceType) {
|
|
|
|
|
ReferenceType referenceType = (ReferenceType) assignNode.expression.getType();
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), "L" + referenceType.getIdentifier() + ";");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void fieldOrObjectVarSuffixCrementAssign(AssignNode assignNode) {
|
|
|
|
|
int localVarIndex = localVariables.indexOf(assignNode.assignable.identifier);
|
|
|
|
|
if(localVarIndex >= 0) { // object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else if(assignNode.assignable.memberAccess.thisExpr) { // field
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
} else {
|
|
|
|
|
localVariables.add(assignNode.assignable.identifier);
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
loadBeforeCrement(assignNode);
|
|
|
|
|
|
|
|
|
|
if (assignNode.expression instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) assignNode.expression.getType()));
|
|
|
|
|
} else if (assignNode.expression instanceof ReferenceType) {
|
|
|
|
|
ReferenceType referenceType = (ReferenceType) assignNode.expression.getType();
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), "L" + referenceType.getIdentifier() + ";");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assignNode.expression.accept(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void assignLocalVar(AssignNode assignNode) {
|
|
|
|
|
methodVisitor.visitInsn(DUP);
|
|
|
|
|
if (!localVariables.contains(assignNode.assignable.identifier)) {
|
|
|
|
|
localVariables.add(assignNode.assignable.identifier);
|
|
|
|
|
}
|
|
|
|
|
if (assignNode.expression instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
} else if (assignNode.expression instanceof ReferenceType) {
|
|
|
|
@ -424,12 +505,105 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void assignField(AssignNode assignNode) {
|
|
|
|
|
private void assignFieldOrObjectVar(AssignNode assignNode) {
|
|
|
|
|
int localVarIndex = localVariables.indexOf(assignNode.assignable.identifier);
|
|
|
|
|
if (localVarIndex >= 0) { // object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else if (assignNode.assignable.memberAccess.thisExpr) { // this
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
} else {
|
|
|
|
|
localVariables.add(assignNode.assignable.identifier);
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assignNode.expression.accept(this);
|
|
|
|
|
methodVisitor.visitInsn(DUP_X1);
|
|
|
|
|
|
|
|
|
|
if (assignNode.expression instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) assignNode.expression.getType()));
|
|
|
|
|
} else if (assignNode.expression instanceof ReferenceType) {
|
|
|
|
|
ReferenceType referenceType = (ReferenceType) assignNode.expression.getType();
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), "L"+referenceType.getIdentifier()+";");
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, assignNode.assignable.memberAccess.identifiers.get(0), assignNode.assignable.memberAccess.identifiers.get(1), "L" + referenceType.getIdentifier() + ";");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void loadBeforeCrement(AssignNode assignNode) {
|
|
|
|
|
if(assignNode.expression instanceof IncrementNode) {
|
|
|
|
|
IncrementNode incrementNode = (IncrementNode) assignNode.expression;
|
|
|
|
|
if(incrementNode.assignableExpression.memberAccess != null) {
|
|
|
|
|
incrementNode.assignableExpression.memberAccess.accept(this);
|
|
|
|
|
} else {
|
|
|
|
|
if(assignNode.assignable.typeNode instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ILOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
} else if(assignNode.assignable.typeNode instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if(assignNode.expression instanceof DecrementNode) {
|
|
|
|
|
DecrementNode decrementNode = (DecrementNode) assignNode.expression;
|
|
|
|
|
if(decrementNode.assignableExpression.memberAccess != null) {
|
|
|
|
|
decrementNode.assignableExpression.memberAccess.accept(this);
|
|
|
|
|
} else {
|
|
|
|
|
if(assignNode.assignable.typeNode instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ILOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
} else if(assignNode.assignable.typeNode instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(IncrementNode incrementNode) {
|
|
|
|
|
if (incrementNode.assignableExpression.memberAccess != null) { // Object var / field
|
|
|
|
|
int localVarIndex = localVariables.indexOf(incrementNode.assignableExpression.identifier);
|
|
|
|
|
if (localVarIndex >= 0) { // object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else { // this
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
}
|
|
|
|
|
if (incrementNode.assignableExpression.memberAccess.getTypeNode() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, incrementNode.assignableExpression.memberAccess.identifiers.get(0), incrementNode.assignableExpression.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) incrementNode.assignableExpression.memberAccess.getTypeNode()));
|
|
|
|
|
} else if (incrementNode.assignableExpression.memberAccess.getTypeNode() instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, incrementNode.assignableExpression.memberAccess.identifiers.get(0), incrementNode.assignableExpression.memberAccess.identifiers.get(1), "L" + (((ReferenceType) incrementNode.assignableExpression.memberAccess.getTypeNode()).getIdentifier() + ";"));
|
|
|
|
|
}
|
|
|
|
|
methodVisitor.visitInsn(DUP);
|
|
|
|
|
methodVisitor.visitInsn(ICONST_1);
|
|
|
|
|
methodVisitor.visitInsn(IADD);
|
|
|
|
|
if (incrementNode.assignableExpression.memberAccess.getTypeNode() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, incrementNode.assignableExpression.memberAccess.identifiers.get(0), incrementNode.assignableExpression.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) incrementNode.assignableExpression.memberAccess.getTypeNode()));
|
|
|
|
|
} else if (incrementNode.assignableExpression.memberAccess.getTypeNode() instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, incrementNode.assignableExpression.memberAccess.identifiers.get(0), incrementNode.assignableExpression.memberAccess.identifiers.get(1), "L" + (((ReferenceType) incrementNode.assignableExpression.memberAccess.getTypeNode()).getIdentifier() + ";"));
|
|
|
|
|
}
|
|
|
|
|
} else { // local var
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(incrementNode.assignableExpression.identifier), 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(DecrementNode decrementNode) {
|
|
|
|
|
if (decrementNode.assignableExpression.memberAccess != null) { // Object var / field
|
|
|
|
|
int localVarIndex = localVariables.indexOf(decrementNode.assignableExpression.identifier);
|
|
|
|
|
if (localVarIndex >= 0) { // object
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
|
|
|
} else { // this
|
|
|
|
|
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
|
|
|
}
|
|
|
|
|
methodVisitor.visitInsn(DUP);
|
|
|
|
|
if (decrementNode.assignableExpression.memberAccess.getTypeNode() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, decrementNode.assignableExpression.memberAccess.identifiers.get(0), decrementNode.assignableExpression.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) decrementNode.assignableExpression.memberAccess.getTypeNode()));
|
|
|
|
|
} else if (decrementNode.assignableExpression.memberAccess.getTypeNode() instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(GETFIELD, decrementNode.assignableExpression.memberAccess.identifiers.get(0), decrementNode.assignableExpression.memberAccess.identifiers.get(1), "L" + (((ReferenceType) decrementNode.assignableExpression.memberAccess.getTypeNode()).getIdentifier() + ";"));
|
|
|
|
|
}
|
|
|
|
|
methodVisitor.visitInsn(ICONST_1);
|
|
|
|
|
methodVisitor.visitInsn(ISUB);
|
|
|
|
|
if (decrementNode.assignableExpression.memberAccess.getTypeNode() instanceof BaseType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, decrementNode.assignableExpression.memberAccess.identifiers.get(0), decrementNode.assignableExpression.memberAccess.identifiers.get(1), mapper.getTypeChar((BaseType) decrementNode.assignableExpression.memberAccess.getTypeNode()));
|
|
|
|
|
} else if (decrementNode.assignableExpression.memberAccess.getTypeNode() instanceof ReferenceType) {
|
|
|
|
|
methodVisitor.visitFieldInsn(PUTFIELD, decrementNode.assignableExpression.memberAccess.identifiers.get(0), decrementNode.assignableExpression.memberAccess.identifiers.get(1), "L" + (((ReferenceType) decrementNode.assignableExpression.memberAccess.getTypeNode()).getIdentifier() + ";"));
|
|
|
|
|
}
|
|
|
|
|
} else { // local var
|
|
|
|
|
methodVisitor.visitIincInsn(localVariables.indexOf(decrementNode.assignableExpression.identifier), -1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -442,8 +616,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
expressionNode.accept(this);
|
|
|
|
|
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
|
|
|
|
}
|
|
|
|
|
methodVisitor.visitMethodInsn(INVOKESPECIAL, newDeclarationNode.identifier, "<init>", mapper.generateMethodDescriptor(new BaseType(TypeEnum.VOID),parameterNodes), false);
|
|
|
|
|
localVariables.add(newDeclarationNode.identifier);
|
|
|
|
|
methodVisitor.visitMethodInsn(INVOKESPECIAL, newDeclarationNode.identifier, "<init>", mapper.generateMethodDescriptor(new BaseType(TypeEnum.VOID), parameterNodes), false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
@ -515,19 +688,13 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|
|
|
|
methodVisitor.visitLabel(endOfLoopLabel);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(ChainedMethodNode chainedMethodNode) {
|
|
|
|
|
// TODO: Erstmal abwarten
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void visit(MethodCallNode methodCallNode) {
|
|
|
|
|
List<ParameterNode> parameterNodes = new ArrayList<>();
|
|
|
|
|
for(IExpressionNode expressionNode : methodCallNode.parameters) {
|
|
|
|
|
for (IExpressionNode expressionNode : methodCallNode.parameters) {
|
|
|
|
|
expressionNode.accept(this);
|
|
|
|
|
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
|
|
|
|
}
|
|
|
|
|
// TODO: Klassenname und Returntype
|
|
|
|
|
//methodVisitor.visitMethodInsn(INVOKEVIRTUAL, classname, methodCallNode.identifier, mapper.generateMethodDescriptor(returntype, parameterNodes), false);
|
|
|
|
|
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, methodCallNode.target.memberAccess.identifiers.get(0), methodCallNode.identifier, mapper.generateMethodDescriptor(methodCallNode.type, parameterNodes), false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|