Compare commits
No commits in common. "8eba420d48631151a48faf9cac155d9e83c21ade" and "f6b34bf70b13995c64ef55ca5884cfec306a4bd4" have entirely different histories.
8eba420d48
...
f6b34bf70b
@ -1,7 +1,6 @@
|
|||||||
package ast.expressions.unaryexpressions;
|
package ast.expressions.unaryexpressions;
|
||||||
|
|
||||||
import ast.ASTNode;
|
import ast.ASTNode;
|
||||||
import ast.type.type.ITypeNode;
|
|
||||||
import bytecode.visitor.MethodVisitor;
|
import bytecode.visitor.MethodVisitor;
|
||||||
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
@ -14,7 +13,6 @@ import java.util.List;
|
|||||||
public class MemberAccessNode implements ASTNode, Visitable {
|
public class MemberAccessNode implements ASTNode, Visitable {
|
||||||
public Boolean thisExpr;
|
public Boolean thisExpr;
|
||||||
public List<String> identifiers = new ArrayList<>();
|
public List<String> identifiers = new ArrayList<>();
|
||||||
private ITypeNode typeNode;
|
|
||||||
|
|
||||||
public MemberAccessNode(Boolean thisExpr) {
|
public MemberAccessNode(Boolean thisExpr) {
|
||||||
this.thisExpr = thisExpr;
|
this.thisExpr = thisExpr;
|
||||||
@ -33,13 +31,5 @@ public class MemberAccessNode implements ASTNode, Visitable {
|
|||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ITypeNode getTypeNode() {
|
|
||||||
return typeNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setTypeNode(ITypeNode typeNode) {
|
|
||||||
this.typeNode = typeNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,6 @@ package ast.statementexpressions;
|
|||||||
|
|
||||||
import ast.expressions.unaryexpressions.MemberAccessNode;
|
import ast.expressions.unaryexpressions.MemberAccessNode;
|
||||||
import ast.type.type.ITypeNode;
|
import ast.type.type.ITypeNode;
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
|
|
||||||
|
@ -2,12 +2,10 @@ package ast.statementexpressions.crementexpressions;
|
|||||||
|
|
||||||
import ast.statementexpressions.AssignableNode;
|
import ast.statementexpressions.AssignableNode;
|
||||||
import ast.statementexpressions.IStatementExpressionNode;
|
import ast.statementexpressions.IStatementExpressionNode;
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
import visitor.Visitable;
|
|
||||||
|
|
||||||
public class DecrementNode implements IStatementExpressionNode, Visitable {
|
public class DecrementNode implements IStatementExpressionNode {
|
||||||
public CrementType crementType;
|
public CrementType crementType;
|
||||||
public AssignableNode assignableExpression;
|
public AssignableNode assignableExpression;
|
||||||
|
|
||||||
@ -20,9 +18,4 @@ public class DecrementNode implements IStatementExpressionNode, Visitable {
|
|||||||
public TypeCheckResult accept(SemanticVisitor visitor) {
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void accept(MethodVisitor methodVisitor) {
|
|
||||||
methodVisitor.visit(this);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -2,12 +2,10 @@ package ast.statementexpressions.crementexpressions;
|
|||||||
|
|
||||||
import ast.statementexpressions.AssignableNode;
|
import ast.statementexpressions.AssignableNode;
|
||||||
import ast.statementexpressions.IStatementExpressionNode;
|
import ast.statementexpressions.IStatementExpressionNode;
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import visitor.Visitable;
|
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
|
|
||||||
public class IncrementNode implements IStatementExpressionNode, Visitable {
|
public class IncrementNode implements IStatementExpressionNode {
|
||||||
public CrementType crementType;
|
public CrementType crementType;
|
||||||
public AssignableNode assignableExpression;
|
public AssignableNode assignableExpression;
|
||||||
|
|
||||||
@ -20,9 +18,4 @@ public class IncrementNode implements IStatementExpressionNode, Visitable {
|
|||||||
public TypeCheckResult accept(SemanticVisitor visitor) {
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void accept(MethodVisitor methodVisitor) {
|
|
||||||
methodVisitor.visit(this);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,11 @@ public class ChainedMethodNode implements ASTNode, Visitable {
|
|||||||
expressions.add(expression);
|
expressions.add(expression);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void accept(MethodVisitor methodVisitor) {
|
||||||
|
methodVisitor.visit(this);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult accept(SemanticVisitor visitor) {
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
package ast.statements;
|
package ast.statements;
|
||||||
|
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
import visitor.Visitable;
|
import visitor.Visitable;
|
||||||
@ -23,9 +22,4 @@ public class BlockNode implements IStatementNode, Visitable {
|
|||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void accept(MethodVisitor methodVisitor) {
|
|
||||||
methodVisitor.visit(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
package ast.statements;
|
package ast.statements;
|
||||||
|
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
package ast.statements;
|
package ast.statements;
|
||||||
|
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
|
|
||||||
@ -21,11 +20,6 @@ public class IfElseNode implements IStatementNode {
|
|||||||
elseIfStatements.add(elseIfStament);
|
elseIfStatements.add(elseIfStament);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void accept(MethodVisitor methodVisitor) {
|
|
||||||
methodVisitor.visit(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult accept(SemanticVisitor visitor) {
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
package ast.statements;
|
package ast.statements;
|
||||||
|
|
||||||
import ast.expressions.IExpressionNode;
|
import ast.expressions.IExpressionNode;
|
||||||
import bytecode.visitor.MethodVisitor;
|
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import semantic.TypeCheckResult;
|
import semantic.TypeCheckResult;
|
||||||
|
|
||||||
@ -18,11 +17,6 @@ public class WhileNode implements IStatementNode {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void accept(MethodVisitor methodVisitor) {
|
|
||||||
methodVisitor.visit(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult accept(SemanticVisitor visitor) {
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
|
@ -27,7 +27,7 @@ public class ClassCodeGen implements ClassVisitor {
|
|||||||
private final boolean generateClassFiles;
|
private final boolean generateClassFiles;
|
||||||
|
|
||||||
public ClassCodeGen(JarOutputStream jarOutputStream, String outputDirectory, boolean generateJar, boolean generateClassFiles) {
|
public ClassCodeGen(JarOutputStream jarOutputStream, String outputDirectory, boolean generateJar, boolean generateClassFiles) {
|
||||||
this.mapper = new Mapper();
|
mapper = new Mapper();
|
||||||
this.jarOutputStream = jarOutputStream;
|
this.jarOutputStream = jarOutputStream;
|
||||||
this.outputDirectory = outputDirectory;
|
this.outputDirectory = outputDirectory;
|
||||||
this.generateJar = generateJar;
|
this.generateJar = generateJar;
|
||||||
|
@ -3,7 +3,6 @@ package bytecode;
|
|||||||
import ast.parameters.ParameterNode;
|
import ast.parameters.ParameterNode;
|
||||||
import ast.type.*;
|
import ast.type.*;
|
||||||
import ast.type.type.BaseType;
|
import ast.type.type.BaseType;
|
||||||
import ast.type.type.ITypeNode;
|
|
||||||
import ast.type.type.ReferenceType;
|
import ast.type.type.ReferenceType;
|
||||||
import ast.type.type.TypeEnum;
|
import ast.type.type.TypeEnum;
|
||||||
import org.objectweb.asm.Opcodes;
|
import org.objectweb.asm.Opcodes;
|
||||||
@ -25,7 +24,7 @@ public class Mapper {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public String generateMethodDescriptor(ITypeNode type, List<ParameterNode> parameters) {
|
public String generateMethodDescriptor(BaseType type, List<ParameterNode> parameters) {
|
||||||
String descriptor = "(";
|
String descriptor = "(";
|
||||||
for (ParameterNode parameterNode : parameters) {
|
for (ParameterNode parameterNode : parameters) {
|
||||||
if(parameterNode.type instanceof BaseType) {
|
if(parameterNode.type instanceof BaseType) {
|
||||||
@ -36,11 +35,7 @@ public class Mapper {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
descriptor += ")";
|
descriptor += ")";
|
||||||
if(type instanceof BaseType) {
|
descriptor += getTypeChar(type);
|
||||||
descriptor += getTypeChar((BaseType) type);
|
|
||||||
} else if(type instanceof ReferenceType) {
|
|
||||||
descriptor += "L" + ((ReferenceType) type).getIdentifier() +";";
|
|
||||||
}
|
|
||||||
return descriptor;
|
return descriptor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,10 +14,9 @@ import ast.statementexpressions.NewDeclarationNode;
|
|||||||
import ast.statementexpressions.crementexpressions.CrementType;
|
import ast.statementexpressions.crementexpressions.CrementType;
|
||||||
import ast.statementexpressions.crementexpressions.DecrementNode;
|
import ast.statementexpressions.crementexpressions.DecrementNode;
|
||||||
import ast.statementexpressions.crementexpressions.IncrementNode;
|
import ast.statementexpressions.crementexpressions.IncrementNode;
|
||||||
|
import ast.statementexpressions.methodcallstatementnexpressions.ChainedMethodNode;
|
||||||
import ast.statementexpressions.methodcallstatementnexpressions.MethodCallNode;
|
import ast.statementexpressions.methodcallstatementnexpressions.MethodCallNode;
|
||||||
import ast.statements.*;
|
import ast.statements.*;
|
||||||
import ast.type.AccessModifierNode;
|
|
||||||
import ast.type.EnumAccessModifierNode;
|
|
||||||
import ast.type.ValueNode;
|
import ast.type.ValueNode;
|
||||||
import ast.type.type.BaseType;
|
import ast.type.type.BaseType;
|
||||||
import ast.type.type.ReferenceType;
|
import ast.type.type.ReferenceType;
|
||||||
@ -71,8 +70,10 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
|
|
||||||
// Visit all statements
|
// Visit all statements
|
||||||
for (IStatementNode statementNode : constructorNode.block.statements) {
|
for (IStatementNode statementNode : constructorNode.block.statements) {
|
||||||
|
if (statementNode != null) {
|
||||||
statementNode.accept(this);
|
statementNode.accept(this);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
methodVisitor.visitMaxs(0, 0);
|
methodVisitor.visitMaxs(0, 0);
|
||||||
methodVisitor.visitEnd();
|
methodVisitor.visitEnd();
|
||||||
@ -80,10 +81,8 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MainMethodNode mainMethodNode) {
|
public void visit(MainMethodNode mainMethodNode) {
|
||||||
AccessModifierNode accessModifierNode = new AccessModifierNode("");
|
methodVisitor = classWriter.visitMethod(mapper.mapAccessTypeToOpcode(mainMethodNode.accesModifier),
|
||||||
accessModifierNode.accessType = EnumAccessModifierNode.PUBLIC_STATIC;
|
mainMethodNode.getIdentifier(),
|
||||||
methodVisitor = classWriter.visitMethod(Opcodes.ACC_PUBLIC | Opcodes.ACC_STATIC,
|
|
||||||
"main",
|
|
||||||
"([Ljava/lang/String;)V",
|
"([Ljava/lang/String;)V",
|
||||||
null,
|
null,
|
||||||
null);
|
null);
|
||||||
@ -105,7 +104,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
public void visit(MethodNode methodNode) {
|
public void visit(MethodNode methodNode) {
|
||||||
methodVisitor = classWriter.visitMethod(mapper.mapAccessTypeToOpcode(methodNode.accesModifier),
|
methodVisitor = classWriter.visitMethod(mapper.mapAccessTypeToOpcode(methodNode.accesModifier),
|
||||||
methodNode.getIdentifier(),
|
methodNode.getIdentifier(),
|
||||||
mapper.generateMethodDescriptor(methodNode.getType(), methodNode.parameters),
|
mapper.generateMethodDescriptor((BaseType) methodNode.getType(), methodNode.parameters),
|
||||||
null,
|
null,
|
||||||
null);
|
null);
|
||||||
|
|
||||||
@ -241,7 +240,7 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
nonCalculationNode.unaryExpression.accept(this);
|
nonCalculationNode.unaryExpression.accept(this);
|
||||||
nonCalculationNode.expression.accept(this);
|
nonCalculationNode.expression.accept(this);
|
||||||
if (nonCalculationNode.unaryExpression.getType() instanceof BaseType && nonCalculationNode.expression.getType() instanceof BaseType) {
|
if (nonCalculationNode.unaryExpression.getType() instanceof BaseType && nonCalculationNode.expression.getType() instanceof BaseType) {
|
||||||
methodVisitor.visitJumpInsn(IF_ICMPEQ, labelFalse);
|
methodVisitor.visitJumpInsn(IF_ACMPEQ, labelFalse);
|
||||||
} else {
|
} else {
|
||||||
methodVisitor.visitJumpInsn(IF_ACMPEQ, labelFalse);
|
methodVisitor.visitJumpInsn(IF_ACMPEQ, labelFalse);
|
||||||
}
|
}
|
||||||
@ -266,17 +265,9 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
@Override
|
@Override
|
||||||
public void visit(MemberAccessNode memberAccessNode) {
|
public void visit(MemberAccessNode memberAccessNode) {
|
||||||
// Only used to get, not to put
|
// Only used to get, not to put
|
||||||
int localVarIndex = localVariables.indexOf("memberAccessNode.identifier"); // TODO
|
if (memberAccessNode.thisExpr) { // Field
|
||||||
if (localVarIndex >= 0) { // local var object
|
// methodVisitor.visitFieldInsn(GETFIELD, memberAccessNode.identifiers.get(0), memberAccessNode.identifiers.get(1), );
|
||||||
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
} else { // Object Attribut
|
||||||
} 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() + ";");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -322,17 +313,9 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
|
|
||||||
// Statements
|
// Statements
|
||||||
|
|
||||||
@Override
|
|
||||||
public void visit(BlockNode blockNode) {
|
|
||||||
for(IStatementNode statementNode : blockNode.statements) {
|
|
||||||
statementNode.accept(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(IfElseNode ifElseNode) {
|
public void visit(IfElseNode ifElseNode) {
|
||||||
Label elseLabel = new Label();
|
Label elseLabel = new Label();
|
||||||
Label endLabel = new Label();
|
|
||||||
|
|
||||||
Label[] elseIfLabels = new Label[ifElseNode.elseIfStatements.size()];
|
Label[] elseIfLabels = new Label[ifElseNode.elseIfStatements.size()];
|
||||||
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
||||||
@ -347,9 +330,9 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
// else if statements
|
// else if statements
|
||||||
methodVisitor.visitJumpInsn(IFEQ, elseIfLabels[0]);
|
methodVisitor.visitJumpInsn(IFEQ, elseIfLabels[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
ifElseNode.ifStatement.block.accept(this); // accept if block
|
ifElseNode.ifStatement.block.accept(this); // accept if block
|
||||||
|
|
||||||
|
Label endLabel = new Label();
|
||||||
methodVisitor.visitJumpInsn(GOTO, endLabel);
|
methodVisitor.visitJumpInsn(GOTO, endLabel);
|
||||||
|
|
||||||
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
for (int i = 0; i < ifElseNode.elseIfStatements.size(); i++) {
|
||||||
@ -368,8 +351,6 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
if (ifElseNode.elseStatement != null) {
|
if (ifElseNode.elseStatement != null) {
|
||||||
methodVisitor.visitLabel(elseLabel);
|
methodVisitor.visitLabel(elseLabel);
|
||||||
ifElseNode.elseStatement.block.accept(this);
|
ifElseNode.elseStatement.block.accept(this);
|
||||||
} else {
|
|
||||||
methodVisitor.visitLabel(elseLabel);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
methodVisitor.visitLabel(endLabel);
|
methodVisitor.visitLabel(endLabel);
|
||||||
@ -380,17 +361,22 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
if (localVariableDeclarationNode.expression != null) {
|
if (localVariableDeclarationNode.expression != null) {
|
||||||
// Process expression
|
// Process expression
|
||||||
localVariableDeclarationNode.expression.accept(this);
|
localVariableDeclarationNode.expression.accept(this);
|
||||||
// add local var to list if not in list
|
// Store result of expression in variable
|
||||||
if (!localVariables.contains(localVariableDeclarationNode.identifier)) {
|
if (localVariables.contains(localVariableDeclarationNode.identifier)) {
|
||||||
localVariables.add(localVariableDeclarationNode.identifier);
|
|
||||||
}
|
|
||||||
if (localVariableDeclarationNode.type instanceof BaseType) {
|
if (localVariableDeclarationNode.type instanceof BaseType) {
|
||||||
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
||||||
} else if (localVariableDeclarationNode.type instanceof ReferenceType) {
|
} else if (localVariableDeclarationNode.type instanceof ReferenceType) {
|
||||||
methodVisitor.visitVarInsn(ASTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
methodVisitor.visitVarInsn(ASTORE, localVariables.indexOf(localVariableDeclarationNode.identifier));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Local var declaration
|
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));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
if (!localVariables.contains(localVariableDeclarationNode.identifier)) {
|
if (!localVariables.contains(localVariableDeclarationNode.identifier)) {
|
||||||
localVariables.add(localVariableDeclarationNode.identifier);
|
localVariables.add(localVariableDeclarationNode.identifier);
|
||||||
}
|
}
|
||||||
@ -399,105 +385,38 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(AssignNode assignNode) {
|
public void visit(AssignNode assignNode) {
|
||||||
if (assignNode.assignable.memberAccess != null) { // this / object
|
// Process expression
|
||||||
if (assignNode.expression instanceof IncrementNode) {
|
if (assignNode.expression instanceof IncrementNode 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
|
if (incrementNode.crementType.equals(CrementType.PREFIX)) { // ++i
|
||||||
incrementNode.accept(this);
|
methodVisitor.visitIincInsn(localVariables.indexOf(incrementNode.assignableExpression.identifier), 1);
|
||||||
assignLocalVar(assignNode);
|
assign(assignNode);
|
||||||
} else { // i++
|
} else if (incrementNode.crementType.equals(CrementType.SUFFIX)) { // Suffix: i++
|
||||||
loadBeforeCrement(assignNode);
|
assign(assignNode);
|
||||||
assignLocalVar(assignNode);
|
methodVisitor.visitIincInsn(localVariables.indexOf(incrementNode.assignableExpression.identifier), 1);
|
||||||
incrementNode.accept(this);
|
|
||||||
}
|
}
|
||||||
} else if (assignNode.expression instanceof DecrementNode decrementNode) {
|
} else if (assignNode.expression instanceof DecrementNode decrementNode) {
|
||||||
if (decrementNode.crementType.equals(CrementType.PREFIX)) {
|
if (decrementNode.crementType.equals(CrementType.PREFIX)) {
|
||||||
decrementNode.accept(this);
|
methodVisitor.visitIincInsn(localVariables.indexOf(decrementNode.assignableExpression.identifier), -1);
|
||||||
assignLocalVar(assignNode);
|
assign(assignNode);
|
||||||
|
} else if (decrementNode.crementType.equals(CrementType.SUFFIX)) {
|
||||||
|
assign(assignNode);
|
||||||
|
methodVisitor.visitIincInsn(localVariables.indexOf(decrementNode.assignableExpression.identifier), 1);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
loadBeforeCrement(assignNode);
|
|
||||||
assignLocalVar(assignNode);
|
|
||||||
decrementNode.accept(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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 {
|
|
||||||
localVariables.add(assignNode.assignable.identifier);
|
|
||||||
methodVisitor.visitVarInsn(ALOAD, localVariables.indexOf(assignNode.assignable.identifier));
|
|
||||||
}
|
|
||||||
|
|
||||||
assignNode.expression.accept(this);
|
assignNode.expression.accept(this);
|
||||||
methodVisitor.visitInsn(DUP_X1);
|
assign(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() + ";");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void fieldOrObjectVarSuffixCrementAssign(AssignNode assignNode) {
|
private void assign(AssignNode assignNode) {
|
||||||
int localVarIndex = localVariables.indexOf(assignNode.assignable.identifier);
|
if (assignNode.assignable.memberAccess.thisExpr) {
|
||||||
if(localVarIndex >= 0) { // object
|
assignField(assignNode);
|
||||||
methodVisitor.visitVarInsn(ALOAD, localVarIndex);
|
|
||||||
} else if(assignNode.assignable.memberAccess.thisExpr) { // field
|
|
||||||
methodVisitor.visitVarInsn(ALOAD, 0);
|
|
||||||
} else {
|
} else {
|
||||||
localVariables.add(assignNode.assignable.identifier);
|
assignLocalVar(assignNode);
|
||||||
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) {
|
private void assignLocalVar(AssignNode assignNode) {
|
||||||
methodVisitor.visitInsn(DUP);
|
|
||||||
if (!localVariables.contains(assignNode.assignable.identifier)) {
|
|
||||||
localVariables.add(assignNode.assignable.identifier);
|
|
||||||
}
|
|
||||||
if (assignNode.expression instanceof BaseType) {
|
if (assignNode.expression instanceof BaseType) {
|
||||||
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(assignNode.assignable.identifier));
|
methodVisitor.visitVarInsn(ISTORE, localVariables.indexOf(assignNode.assignable.identifier));
|
||||||
} else if (assignNode.expression instanceof ReferenceType) {
|
} else if (assignNode.expression instanceof ReferenceType) {
|
||||||
@ -505,105 +424,12 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void assignFieldOrObjectVar(AssignNode assignNode) {
|
private void assignField(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) {
|
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()));
|
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) {
|
} else if (assignNode.expression instanceof ReferenceType) {
|
||||||
ReferenceType referenceType = (ReferenceType) assignNode.expression.getType();
|
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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -616,7 +442,8 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
expressionNode.accept(this);
|
expressionNode.accept(this);
|
||||||
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
||||||
}
|
}
|
||||||
methodVisitor.visitMethodInsn(INVOKESPECIAL, newDeclarationNode.identifier, "<init>", mapper.generateMethodDescriptor(new BaseType(TypeEnum.VOID), parameterNodes), false);
|
methodVisitor.visitMethodInsn(INVOKESPECIAL, newDeclarationNode.identifier, "<init>", mapper.generateMethodDescriptor(new BaseType(TypeEnum.VOID),parameterNodes), false);
|
||||||
|
localVariables.add(newDeclarationNode.identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -688,13 +515,19 @@ public class MethodCodeGen implements bytecode.visitor.MethodVisitor {
|
|||||||
methodVisitor.visitLabel(endOfLoopLabel);
|
methodVisitor.visitLabel(endOfLoopLabel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void visit(ChainedMethodNode chainedMethodNode) {
|
||||||
|
// TODO: Erstmal abwarten
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MethodCallNode methodCallNode) {
|
public void visit(MethodCallNode methodCallNode) {
|
||||||
List<ParameterNode> parameterNodes = new ArrayList<>();
|
List<ParameterNode> parameterNodes = new ArrayList<>();
|
||||||
for (IExpressionNode expressionNode : methodCallNode.parameters) {
|
for(IExpressionNode expressionNode : methodCallNode.parameters) {
|
||||||
expressionNode.accept(this);
|
expressionNode.accept(this);
|
||||||
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
parameterNodes.add(new ParameterNode(expressionNode.getType(), ""));
|
||||||
}
|
}
|
||||||
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, methodCallNode.target.memberAccess.identifiers.get(0), methodCallNode.identifier, mapper.generateMethodDescriptor(methodCallNode.type, parameterNodes), false);
|
// TODO: Klassenname und Returntype
|
||||||
|
//methodVisitor.visitMethodInsn(INVOKEVIRTUAL, classname, methodCallNode.identifier, mapper.generateMethodDescriptor(returntype, parameterNodes), false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,5 +5,6 @@ import ast.members.FieldNode;
|
|||||||
|
|
||||||
public interface ClassVisitor {
|
public interface ClassVisitor {
|
||||||
void visit(ClassNode classNode);
|
void visit(ClassNode classNode);
|
||||||
|
|
||||||
void visit(FieldNode fieldNode);
|
void visit(FieldNode fieldNode);
|
||||||
}
|
}
|
||||||
|
@ -9,8 +9,6 @@ import ast.members.MainMethodNode;
|
|||||||
import ast.members.MethodNode;
|
import ast.members.MethodNode;
|
||||||
import ast.statementexpressions.AssignNode;
|
import ast.statementexpressions.AssignNode;
|
||||||
import ast.statementexpressions.NewDeclarationNode;
|
import ast.statementexpressions.NewDeclarationNode;
|
||||||
import ast.statementexpressions.crementexpressions.DecrementNode;
|
|
||||||
import ast.statementexpressions.crementexpressions.IncrementNode;
|
|
||||||
import ast.statementexpressions.methodcallstatementnexpressions.ChainedMethodNode;
|
import ast.statementexpressions.methodcallstatementnexpressions.ChainedMethodNode;
|
||||||
import ast.statementexpressions.methodcallstatementnexpressions.MethodCallNode;
|
import ast.statementexpressions.methodcallstatementnexpressions.MethodCallNode;
|
||||||
import ast.statements.*;
|
import ast.statements.*;
|
||||||
@ -43,12 +41,8 @@ public interface MethodVisitor {
|
|||||||
void visit(UnaryNode unaryExpressionNode);
|
void visit(UnaryNode unaryExpressionNode);
|
||||||
|
|
||||||
// statements
|
// statements
|
||||||
void visit(BlockNode blockNode);
|
|
||||||
void visit(IfElseNode ifElseNode);
|
void visit(IfElseNode ifElseNode);
|
||||||
|
|
||||||
void visit(IncrementNode incrementNode);
|
|
||||||
void visit(DecrementNode decrementNode);
|
|
||||||
|
|
||||||
void visit(LocalVariableDeclarationNode localVariableDeclarationNode);
|
void visit(LocalVariableDeclarationNode localVariableDeclarationNode);
|
||||||
|
|
||||||
void visit(ReturnNode returnNode);
|
void visit(ReturnNode returnNode);
|
||||||
@ -56,6 +50,8 @@ public interface MethodVisitor {
|
|||||||
void visit(WhileNode whileNode);
|
void visit(WhileNode whileNode);
|
||||||
|
|
||||||
// statement expression
|
// statement expression
|
||||||
|
void visit(ChainedMethodNode chainedMethodNode);
|
||||||
|
|
||||||
void visit(MethodCallNode methodCallNode);
|
void visit(MethodCallNode methodCallNode);
|
||||||
|
|
||||||
void visit(AssignNode assignNode);
|
void visit(AssignNode assignNode);
|
||||||
|
Loading…
Reference in New Issue
Block a user