Endabgabe #20

Merged
i22035 merged 137 commits from Endabgabe into main 2024-07-05 11:59:47 +00:00
10 changed files with 75 additions and 53 deletions
Showing only changes of commit 8ba58d492b - Show all commits

View File

@ -2,6 +2,7 @@ package ast.statementexpressions.methodcallstatementnexpressions;
import ast.expressions.IExpressionNode; import ast.expressions.IExpressionNode;
import ast.statements.IStatementNode; import ast.statements.IStatementNode;
import ast.type.type.ITypeNode;
import bytecode.visitor.MethodVisitor; import bytecode.visitor.MethodVisitor;
import semantic.SemanticVisitor; import semantic.SemanticVisitor;
import typechecker.TypeCheckResult; import typechecker.TypeCheckResult;
@ -11,6 +12,7 @@ import java.util.List;
public class MethodCallNode implements IStatementNode { public class MethodCallNode implements IStatementNode {
public TargetNode target; public TargetNode target;
public ITypeNode type;
public List<ChainedMethodNode> chainedMethods = new ArrayList<>(); public List<ChainedMethodNode> chainedMethods = new ArrayList<>();
public String identifier; public String identifier;
public List<IExpressionNode> parameters = new ArrayList<>(); public List<IExpressionNode> parameters = new ArrayList<>();

View File

@ -166,7 +166,7 @@ public class SemanticAnalyzer implements SemanticVisitor {
@Override @Override
public TypeCheckResult analyze(FieldNode toCheck) { public TypeCheckResult analyze(FieldNode toCheck) {
if (toCheck.type instanceof ReferenceType referenceType) { if (toCheck.type instanceof ReferenceType referenceType) {
if(!context.containsClass(referenceType.getIdentifier())){ if (!context.containsClass(referenceType.getIdentifier())) {
errors.add(new NotDeclaredException(referenceType.getIdentifier() + " not declared")); errors.add(new NotDeclaredException(referenceType.getIdentifier() + " not declared"));
} }
} }
@ -233,7 +233,7 @@ public class SemanticAnalyzer implements SemanticVisitor {
} }
for (IStatementNode statementNode : blockNode.statements) { for (IStatementNode statementNode : blockNode.statements) {
var result = statementNode.accept(this); var result = statementNode.accept(this);
if(!(statementNode instanceof IncrementNode) && !(statementNode instanceof DecrementNode)){ if (!(statementNode instanceof IncrementNode) && !(statementNode instanceof DecrementNode)) {
if (result.getType() != null) { if (result.getType() != null) {
if (blockReturnType == null) { if (blockReturnType == null) {
blockReturnType = result.getType(); blockReturnType = result.getType();
@ -258,6 +258,10 @@ public class SemanticAnalyzer implements SemanticVisitor {
} else { } else {
if (currentFields.get(toCheck.identifier) != null) { if (currentFields.get(toCheck.identifier) != null) {
var type = currentFields.get(toCheck.identifier); var type = currentFields.get(toCheck.identifier);
MemberAccessNode memberAccessNode = new MemberAccessNode(false);
memberAccessNode.identifiers.add(currentClass.identifier);
memberAccessNode.identifiers.add(toCheck.identifier);
toCheck.memberAccess = memberAccessNode;
toCheck.setTypeNode(type); toCheck.setTypeNode(type);
return new TypeCheckResult(true, type); return new TypeCheckResult(true, type);
} else if (currentScope.getLocalVar(toCheck.identifier) != null) { } else if (currentScope.getLocalVar(toCheck.identifier) != null) {
@ -320,44 +324,66 @@ public class SemanticAnalyzer implements SemanticVisitor {
@Override @Override
public TypeCheckResult analyze(MethodCallNode toCheck) { public TypeCheckResult analyze(MethodCallNode toCheck) {
if (toCheck.target.identifier != null) { if (toCheck.target != null) {
var targetType = currentScope.getLocalVar(toCheck.target.identifier); if (toCheck.target.identifier != null) {
if (targetType == null) { var targetType = currentScope.getLocalVar(toCheck.target.identifier);
targetType = currentFields.get(toCheck.target.identifier); if (targetType == null) {
} targetType = currentFields.get(toCheck.target.identifier);
if (targetType instanceof ReferenceType reference) { }
if (!toCheck.chainedMethods.isEmpty()) { if (targetType instanceof ReferenceType reference) {
for (ChainedMethodNode chainedMethod : toCheck.chainedMethods) { if (!toCheck.chainedMethods.isEmpty()) {
var type = getTypeFromMethod(chainedMethod, reference); for (ChainedMethodNode chainedMethod : toCheck.chainedMethods) {
if (type instanceof ReferenceType referenceType) var type = getTypeFromMethod(chainedMethod, reference);
reference = referenceType; if (type instanceof ReferenceType referenceType)
else reference = referenceType;
errors.add(new TypeMismatchException("Ein Basetyp hat keine funktionen")); else
errors.add(new TypeMismatchException("Ein Basetyp hat keine funktionen"));
}
}
var type = getTypeFromMethod(toCheck, reference);
if (type != null) {
return new TypeCheckResult(true, type);
} else {
return new TypeCheckResult(false, null);
}
}
} else {
if (toCheck.target.thisTar != null) {
if (toCheck.target.thisTar) {
var type = getTypeFromMethod(toCheck, new ReferenceType(currentClass.identifier));
if (type != null) {
return new TypeCheckResult(true, type);
}
}
} else {
var result = toCheck.target.accept(this);
if (result.getType() instanceof ReferenceType reference) {
return new TypeCheckResult(true, getTypeFromMethod(toCheck, reference));
} }
} }
var type = getTypeFromMethod(toCheck, reference);
if (type != null) {
return new TypeCheckResult(true, type);
} else {
return new TypeCheckResult(false, null);
}
} }
} else { } else {
if (toCheck.target.thisTar != null) {
if (toCheck.target.thisTar) { ReferenceType reference = new ReferenceType(currentClass.identifier);
var type = getTypeFromMethod(toCheck, new ReferenceType(currentClass.identifier)); if (!toCheck.chainedMethods.isEmpty()) {
if (type != null) { for (ChainedMethodNode chainedMethod : toCheck.chainedMethods) {
return new TypeCheckResult(true, type); var type = getTypeFromMethod(chainedMethod, reference);
if (type instanceof ReferenceType referenceType)
reference = referenceType;
else
errors.add(new TypeMismatchException("Ein Basetyp hat keine funktionen"));
} }
} }
var type = getTypeFromMethod(toCheck, reference);
if (type != null) {
return new TypeCheckResult(true, type);
} else { } else {
var result = toCheck.target.accept(this); return new TypeCheckResult(false, null);
if (result.getType() instanceof ReferenceType reference) {
return new TypeCheckResult(true, getTypeFromMethod(toCheck, reference));
}
} }
} }
return new TypeCheckResult(false, null); return new TypeCheckResult(false, null);
} }
@ -508,6 +534,10 @@ public class SemanticAnalyzer implements SemanticVisitor {
if (currentScope.contains(unary.identifier)) { if (currentScope.contains(unary.identifier)) {
return new TypeCheckResult(valid, currentScope.getLocalVar(unary.identifier)); return new TypeCheckResult(valid, currentScope.getLocalVar(unary.identifier));
} else if (currentFields.get(unary.identifier) != null) { } else if (currentFields.get(unary.identifier) != null) {
MemberAccessNode memberAccessNode = new MemberAccessNode(false);
memberAccessNode.identifiers.add(currentClass.identifier);
memberAccessNode.identifiers.add(unary.identifier);
unary.memberAccess = memberAccessNode;
return new TypeCheckResult(valid, currentFields.get(unary.identifier)); return new TypeCheckResult(valid, currentFields.get(unary.identifier));
} else if (unary.statement != null) { } else if (unary.statement != null) {
var result = unary.statement.accept(this); var result = unary.statement.accept(this);
@ -537,8 +567,16 @@ public class SemanticAnalyzer implements SemanticVisitor {
public TypeCheckResult analyze(MemberAccessNode memberAccessNode) { public TypeCheckResult analyze(MemberAccessNode memberAccessNode) {
ITypeNode currentType = null; ITypeNode currentType = null;
int start = 0;
if(!memberAccessNode.identifiers.isEmpty()){
if(currentFields.get(memberAccessNode.identifiers.get(0)) != null){
memberAccessNode.identifiers.add(0, currentClass.identifier);
start = 1;
}
}
for (int i = start; i < memberAccessNode.identifiers.size(); i++) {
for (String s : memberAccessNode.identifiers) { String s = memberAccessNode.identifiers.get(i);
if (currentType == null) { if (currentType == null) {
if (currentScope.getLocalVar(s) != null) { if (currentScope.getLocalVar(s) != null) {
currentType = currentScope.getLocalVar(s); currentType = currentScope.getLocalVar(s);

View File

@ -30,6 +30,7 @@ import semantic.exceptions.AlreadyDeclaredException;
import semantic.exceptions.MultipleReturnTypes; import semantic.exceptions.MultipleReturnTypes;
import semantic.exceptions.NotDeclaredException; import semantic.exceptions.NotDeclaredException;
import semantic.exceptions.TypeMismatchException; import semantic.exceptions.TypeMismatchException;
import semantic.exceptions.NotDeclaredException;
import static org.junit.jupiter.api.Assertions.*; import static org.junit.jupiter.api.Assertions.*;

View File

@ -1,19 +0,0 @@
// @expected: WrongOverloading
public class Test{
public void test(int x){
if(this.get()){
} else {
}
}
public boolean b;
public boolean get(int c){
return b;
}
}

View File

@ -6,7 +6,7 @@ public class Test{
public int test(){ public int test(){
return this.test(i); return test(i);
} }