From 4ecf526b14f4903b00300ae3eadf3c34c86c8517 Mon Sep 17 00:00:00 2001 From: JanUlrich Date: Wed, 29 Mar 2017 17:28:29 +0200 Subject: [PATCH] =?UTF-8?q?Generics=20anf=C3=BCgen?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/de/dhbwstuttgart/core/JavaTXCompiler.java | 2 + .../SyntaxTreeGenerator/GenericContext.java | 14 +++++ .../SyntaxTreeGenerator/GenericsRegistry.java | 6 +-- .../StatementGenerator.java | 5 +- .../SyntaxTreeGenerator/SyntacticSugar.java | 16 ++++++ .../SyntaxTreeGenerator.java | 9 ++-- .../SyntaxTreeGenerator/TypeGenerator.java | 14 ++--- .../syntaxtree/ClassOrInterface.java | 4 ++ .../syntaxtree/GenericDeclarationList.java | 2 +- .../syntaxtree/GenericTypeVar.java | 8 ++- src/de/dhbwstuttgart/syntaxtree/Method.java | 16 +++++- .../syntaxtree/factory/ASTFactory.java | 54 ++++++++++--------- .../syntaxtree/statement/Assign.java | 7 +-- .../syntaxtree/statement/Block.java | 2 - .../syntaxtree/statement/FieldVar.java | 10 ++-- .../syntaxtree/statement/MethodCall.java | 11 ++-- .../syntaxtree/statement/Return.java | 6 ++- .../syntaxtree/statement/This.java | 3 +- .../syntaxtree/type/GenericRefType.java | 5 ++ .../syntaxtree/type/RefType.java | 5 ++ .../type/RefTypeOrTPHOrWildcardOrGeneric.java | 8 +-- .../typecheck/GenericTypeName.java | 17 +++++- .../TypeInferenceBlockInformation.java | 6 ++- .../constraints/ConstraintsFactory.java | 47 ++++++++++++++++ .../typeinference/constraints/Pair.java | 5 +- test/typeinference/JavaTXCompilerTest.java | 2 +- 26 files changed, 209 insertions(+), 75 deletions(-) create mode 100644 src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericContext.java create mode 100644 src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntacticSugar.java mode change 100755 => 100644 src/de/dhbwstuttgart/syntaxtree/type/RefTypeOrTPHOrWildcardOrGeneric.java create mode 100644 src/de/dhbwstuttgart/typeinference/constraints/ConstraintsFactory.java mode change 100755 => 100644 src/de/dhbwstuttgart/typeinference/constraints/Pair.java diff --git a/src/de/dhbwstuttgart/core/JavaTXCompiler.java b/src/de/dhbwstuttgart/core/JavaTXCompiler.java index 6b114e741..7de0e9384 100644 --- a/src/de/dhbwstuttgart/core/JavaTXCompiler.java +++ b/src/de/dhbwstuttgart/core/JavaTXCompiler.java @@ -39,6 +39,8 @@ public class JavaTXCompiler { for(Constraint constraint : xCons){ xConsSet.addAll(constraint); } + + System.out.println(xConsSet); Set> result = unify.unify(xConsSet, finiteClosure); System.out.println(result); } diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericContext.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericContext.java new file mode 100644 index 000000000..88f94ee81 --- /dev/null +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericContext.java @@ -0,0 +1,14 @@ +package de.dhbwstuttgart.parser.SyntaxTreeGenerator; + +import de.dhbwstuttgart.typecheck.JavaClassName; + +public class GenericContext { + public final String parentMethod; + public final JavaClassName parentClass; + + public GenericContext(JavaClassName parentClass, String parentMethod) { + if(parentMethod == null)parentMethod = ""; + this.parentClass = parentClass; + this.parentMethod = parentMethod; + } +} diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericsRegistry.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericsRegistry.java index 8684ae1c7..08212fc94 100644 --- a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericsRegistry.java +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/GenericsRegistry.java @@ -1,7 +1,7 @@ package de.dhbwstuttgart.parser.SyntaxTreeGenerator; -import java.util.HashSet; +import java.util.HashMap; -public class GenericsRegistry extends HashSet { +public class GenericsRegistry extends HashMap { -} +} \ No newline at end of file diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/StatementGenerator.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/StatementGenerator.java index 90b21d0b5..51542c344 100644 --- a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/StatementGenerator.java +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/StatementGenerator.java @@ -10,6 +10,7 @@ import de.dhbwstuttgart.syntaxtree.statement.literal.*; import de.dhbwstuttgart.syntaxtree.type.RefType; import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder; +import de.dhbwstuttgart.typecheck.JavaClassName; import de.dhbwstuttgart.typecheck.JavaClassRegistry; import org.antlr.v4.runtime.Token; import org.antlr.v4.runtime.tree.TerminalNode; @@ -33,12 +34,12 @@ public class StatementGenerator { } - public Method convert(Java8Parser.MethodDeclarationContext methodDeclarationContext) { + public Method convert(Java8Parser.MethodDeclarationContext methodDeclarationContext, JavaClassName parentClass) { Java8Parser.MethodHeaderContext header = methodDeclarationContext.methodHeader(); String name = header.methodDeclarator().Identifier().getText(); GenericDeclarationList gtvDeclarations = new GenericDeclarationList(new ArrayList<>(), methodDeclarationContext.methodHeader().getStart()); if(methodDeclarationContext.methodHeader().typeParameters() != null){ - gtvDeclarations = TypeGenerator.convert(methodDeclarationContext.methodHeader().typeParameters(), reg, generics); + gtvDeclarations = TypeGenerator.convert(methodDeclarationContext.methodHeader().typeParameters(), parentClass, name,reg, generics); } RefTypeOrTPHOrWildcardOrGeneric retType; if(header.result() != null){ diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntacticSugar.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntacticSugar.java new file mode 100644 index 000000000..2602f58a4 --- /dev/null +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntacticSugar.java @@ -0,0 +1,16 @@ +package de.dhbwstuttgart.parser.SyntaxTreeGenerator; + +import de.dhbwstuttgart.syntaxtree.statement.Return; +import de.dhbwstuttgart.syntaxtree.statement.Statement; + +import java.util.List; + +public class SyntacticSugar { + public List addTrailingReturn(List statements){ + Statement lastStmt = statements.get(statements.size()-1); + if(lastStmt instanceof Return)return statements; + + return statements; + } + +} diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntaxTreeGenerator.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntaxTreeGenerator.java index 725c8e7ee..68b67fe5f 100644 --- a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntaxTreeGenerator.java +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/SyntaxTreeGenerator.java @@ -200,10 +200,9 @@ public class SyntaxTreeGenerator{ } } JavaClassName name = reg.getName(ctx.Identifier().getText()); - GenericDeclarationList genericClassParameters = TypeGenerator.convert(ctx.typeParameters(), reg, generics); - Block class_block = null; + GenericDeclarationList genericClassParameters = TypeGenerator.convert(ctx.typeParameters(), name, "",reg, generics); List fielddecl = convertFields(ctx.classBody()); - List methods = convertMethods(ctx.classBody()); + List methods = convertMethods(ctx.classBody(), name); Token offset = ctx.getStart(); RefType superClass ; @@ -221,7 +220,7 @@ public class SyntaxTreeGenerator{ throw new NotImplementedException(); } - private List convertMethods(Java8Parser.ClassBodyContext classBodyContext) { + private List convertMethods(Java8Parser.ClassBodyContext classBodyContext, JavaClassName parentClass) { List ret = new ArrayList<>(); for(Java8Parser.ClassBodyDeclarationContext classMember : classBodyContext.classBodyDeclaration()){ if(classMember.classMemberDeclaration() != null){ @@ -230,7 +229,7 @@ public class SyntaxTreeGenerator{ //Do nothing! }else if(classMemberDeclarationContext.methodDeclaration()!= null){ StatementGenerator stmtGen = new StatementGenerator(reg, generics); - ret.add(stmtGen.convert(classMemberDeclarationContext.methodDeclaration())); + ret.add(stmtGen.convert(classMemberDeclarationContext.methodDeclaration(), parentClass)); } } } diff --git a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/TypeGenerator.java b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/TypeGenerator.java index 8c2ad2695..fa37c0319 100644 --- a/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/TypeGenerator.java +++ b/src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/TypeGenerator.java @@ -25,8 +25,8 @@ public class TypeGenerator { public static RefTypeOrTPHOrWildcardOrGeneric convert(Java8Parser.UnannClassOrInterfaceTypeContext unannClassOrInterfaceTypeContext, JavaClassRegistry reg, GenericsRegistry generics) { String name = unannClassOrInterfaceTypeContext.getText(); if(!reg.contains(name)){ //Dann könnte es ein Generische Type sein: - if(generics.contains(name)){ - return new GenericRefType(new GenericTypeName(name), unannClassOrInterfaceTypeContext.getStart()); + if(generics.keySet().contains(name)){ + return new GenericRefType(new GenericTypeName(generics.get(name),name), unannClassOrInterfaceTypeContext.getStart()); }else{ throw new TypeinferenceException("Der Typ "+ name + " ist nicht vorhanden",unannClassOrInterfaceTypeContext.getStart()); } @@ -48,25 +48,25 @@ public class TypeGenerator { return TypeGenerator.convert(unannTypeContext.unannReferenceType().unannClassOrInterfaceType(), reg, genericsRegistry); } - public static GenericDeclarationList convert(Java8Parser.TypeParametersContext typeParametersContext, JavaClassRegistry reg, GenericsRegistry generics) { + public static GenericDeclarationList convert(Java8Parser.TypeParametersContext typeParametersContext, JavaClassName parentClass, String parentMethod, JavaClassRegistry reg, GenericsRegistry generics) { if(typeParametersContext == null){ return new GenericDeclarationList(new ArrayList<>(), new NullToken()); } Token endOffset = typeParametersContext.getStop(); List typeVars = new ArrayList<>(); for(Java8Parser.TypeParameterContext typeParameter : typeParametersContext.typeParameterList().typeParameter()){ - typeVars.add(convert(typeParameter, reg, generics)); + typeVars.add(convert(typeParameter, parentClass, parentMethod, reg, generics)); endOffset = typeParameter.getStop(); } return new GenericDeclarationList(typeVars, endOffset); } - public static GenericTypeVar convert(Java8Parser.TypeParameterContext typeParameter, JavaClassRegistry reg, GenericsRegistry generics) { + public static GenericTypeVar convert(Java8Parser.TypeParameterContext typeParameter, JavaClassName parentClass, String parentMethod, JavaClassRegistry reg, GenericsRegistry generics) { String name = typeParameter.Identifier().getText(); List bounds = TypeGenerator.convert(typeParameter.typeBound(),reg, generics); - GenericTypeVar ret = new GenericTypeVar(name, bounds, typeParameter.getStart(), typeParameter.getStop()); - generics.add(name); + GenericTypeVar ret = new GenericTypeVar(new GenericTypeName(new GenericContext(parentClass, parentMethod), name), bounds, typeParameter.getStart(), typeParameter.getStop()); + generics.put(name, new GenericContext(parentClass, parentMethod)); return ret; } diff --git a/src/de/dhbwstuttgart/syntaxtree/ClassOrInterface.java b/src/de/dhbwstuttgart/syntaxtree/ClassOrInterface.java index a69a55e37..ca654563c 100755 --- a/src/de/dhbwstuttgart/syntaxtree/ClassOrInterface.java +++ b/src/de/dhbwstuttgart/syntaxtree/ClassOrInterface.java @@ -81,4 +81,8 @@ public class ClassOrInterface extends GTVDeclarationContext implements IItemWith public RefTypeOrTPHOrWildcardOrGeneric getSuperClass() { return superClass; } + + public Iterable getGenerics() { + return this.genericClassParameters; + } } diff --git a/src/de/dhbwstuttgart/syntaxtree/GenericDeclarationList.java b/src/de/dhbwstuttgart/syntaxtree/GenericDeclarationList.java index 464f5aaa9..ceea42b1a 100644 --- a/src/de/dhbwstuttgart/syntaxtree/GenericDeclarationList.java +++ b/src/de/dhbwstuttgart/syntaxtree/GenericDeclarationList.java @@ -24,6 +24,6 @@ public class GenericDeclarationList extends SyntaxTreeNode implements Iterable iterator() { - return null; + return gtvs.iterator(); } } diff --git a/src/de/dhbwstuttgart/syntaxtree/GenericTypeVar.java b/src/de/dhbwstuttgart/syntaxtree/GenericTypeVar.java index 4c30e6d8c..5d151046e 100755 --- a/src/de/dhbwstuttgart/syntaxtree/GenericTypeVar.java +++ b/src/de/dhbwstuttgart/syntaxtree/GenericTypeVar.java @@ -1,6 +1,7 @@ package de.dhbwstuttgart.syntaxtree; import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; +import de.dhbwstuttgart.typecheck.GenericTypeName; import org.antlr.v4.runtime.Token; import java.util.ArrayList; @@ -23,9 +24,9 @@ public class GenericTypeVar extends SyntaxTreeNode */ List bounds=new ArrayList(); private Token endOffset; - private String name; + private GenericTypeName name; - public GenericTypeVar(String s, List bounds, Token offset, Token endOffset) + public GenericTypeVar(GenericTypeName s, List bounds, Token offset, Token endOffset) { super(offset); name = s; @@ -47,4 +48,7 @@ public class GenericTypeVar extends SyntaxTreeNode return "BoGTV " + this.name; } + public GenericTypeName getName() { + return name; + } } diff --git a/src/de/dhbwstuttgart/syntaxtree/Method.java b/src/de/dhbwstuttgart/syntaxtree/Method.java index e117dba54..af33f4534 100755 --- a/src/de/dhbwstuttgart/syntaxtree/Method.java +++ b/src/de/dhbwstuttgart/syntaxtree/Method.java @@ -5,8 +5,11 @@ import java.util.ArrayList; import de.dhbwstuttgart.parser.NullToken; import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceBlockInformation; +import de.dhbwstuttgart.typeinference.constraints.Constraint; import de.dhbwstuttgart.typeinference.constraints.ConstraintSet; import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceInformation; +import de.dhbwstuttgart.typeinference.constraints.ConstraintsFactory; +import de.dhbwstuttgart.typeinference.constraints.Pair; import org.antlr.v4.runtime.Token; import de.dhbwstuttgart.core.IItemWithOffset; @@ -25,21 +28,32 @@ public class Method extends Field implements IItemWithOffset private Block block; private ParameterList parameterlist = new ParameterList(new ArrayList<>(), new NullToken()); private ExceptionList exceptionlist; + private GenericDeclarationList generics; public Method(String name, RefTypeOrTPHOrWildcardOrGeneric returnType, int modifiers, ParameterList parameterList, Block block, GenericDeclarationList gtvDeclarations, Token offset) { super(name, returnType, modifiers, offset); this.parameterlist = parameterList; this.block = block; + this.generics = gtvDeclarations; } public ConstraintSet getConstraints(TypeInferenceInformation info, ClassOrInterface currentClass) { ConstraintSet ret = new ConstraintSet(); - ret.addAll(block.getConstraints(new TypeInferenceBlockInformation(info.getAvailableClasses(), currentClass, this))); + TypeInferenceBlockInformation blockInfo = new TypeInferenceBlockInformation(info.getAvailableClasses(), currentClass, this); + ret.addAll(block.getConstraints(blockInfo)); + /* + ret.addUndConstraint( + ConstraintsFactory.createPair(block.getType(), this.getType(), blockInfo)); + */ return ret; } public ParameterList getParameterList() { return parameterlist; } + + public Iterable getGenerics() { + return generics; + } } diff --git a/src/de/dhbwstuttgart/syntaxtree/factory/ASTFactory.java b/src/de/dhbwstuttgart/syntaxtree/factory/ASTFactory.java index 1345edb6d..c12dbd46e 100644 --- a/src/de/dhbwstuttgart/syntaxtree/factory/ASTFactory.java +++ b/src/de/dhbwstuttgart/syntaxtree/factory/ASTFactory.java @@ -6,10 +6,10 @@ import java.util.ArrayList; import java.util.List; import de.dhbwstuttgart.parser.NullToken; +import de.dhbwstuttgart.parser.SyntaxTreeGenerator.GenericContext; import de.dhbwstuttgart.syntaxtree.Field; import de.dhbwstuttgart.syntaxtree.Method; -import de.dhbwstuttgart.syntaxtree.type.GenericRefType; -import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; +import de.dhbwstuttgart.syntaxtree.type.*; import de.dhbwstuttgart.syntaxtree.type.Void; import de.dhbwstuttgart.typecheck.GenericTypeName; import de.dhbwstuttgart.typecheck.JavaClassName; @@ -17,7 +17,6 @@ import de.dhbwstuttgart.typecheck.JavaClassRegistry; import de.dhbwstuttgart.syntaxtree.*; import de.dhbwstuttgart.syntaxtree.statement.Block; import de.dhbwstuttgart.syntaxtree.statement.Statement; -import de.dhbwstuttgart.syntaxtree.type.RefType; import org.antlr.v4.runtime.Token; /** @@ -48,13 +47,13 @@ public class ASTFactory { java.lang.Class superjreClass = jreClass.getSuperclass(); RefType superClass; if(superjreClass != null){ - superClass = (RefType) createType(superjreClass); + superClass = (RefType) createType(superjreClass, name, ""); }else{//Jede Klasse und jedes Interface erbt von Object: (auch Object selbst!) - superClass = (RefType) createType(java.lang.Object.class); + superClass = (RefType) createType(java.lang.Object.class, name, ""); } List implementedInterfaces = new ArrayList<>(); for(java.lang.Class jreInterface : jreClass.getInterfaces()){ - implementedInterfaces.add((RefType) createType(jreInterface)); + implementedInterfaces.add((RefType) createType(jreInterface, name, "")); } GenericDeclarationList genericDeclarationList = createGenerics(jreClass.getTypeParameters(), jreClass, null); @@ -64,11 +63,11 @@ public class ASTFactory { private de.dhbwstuttgart.syntaxtree.Constructor createConstructor(Constructor constructor, Class inClass) { String name = constructor.getName(); - RefTypeOrTPHOrWildcardOrGeneric returnType = createType(inClass); + RefTypeOrTPHOrWildcardOrGeneric returnType = createType(inClass, names.getName(inClass.getName()), name); Parameter[] jreParams = constructor.getParameters(); List params = new ArrayList<>(); for(Parameter jreParam : jreParams){ - RefTypeOrTPHOrWildcardOrGeneric paramType = createType(jreParam.getType()); + RefTypeOrTPHOrWildcardOrGeneric paramType = createType(jreParam.getType(),names.getName(inClass.getName()), name); params.add(new FormalParameter(jreParam.getName(),paramType, new NullToken())); } ParameterList parameterList = new ParameterList(params, new NullToken()); @@ -83,11 +82,11 @@ public class ASTFactory { public Method createMethod(java.lang.reflect.Method jreMethod, java.lang.Class inClass){ String name = jreMethod.getName(); RefTypeOrTPHOrWildcardOrGeneric returnType; - returnType = createType(jreMethod.getReturnType()); + returnType = createType(jreMethod.getReturnType(),names.getName(inClass.getName()), name); Parameter[] jreParams = jreMethod.getParameters(); List params = new ArrayList<>(); for(Parameter jreParam : jreParams){ - RefTypeOrTPHOrWildcardOrGeneric paramType = createType(jreParam.getType()); + RefTypeOrTPHOrWildcardOrGeneric paramType = createType(jreParam.getType(),names.getName(inClass.getName()), name); params.add(new FormalParameter(jreParam.getName(),paramType, new NullToken())); } ParameterList parameterList = new ParameterList(params, new NullToken()); @@ -99,14 +98,14 @@ public class ASTFactory { return new Method(name,returnType, modifier, parameterList, block, gtvDeclarations, offset); } - public GenericDeclarationList createGenerics(TypeVariable[] typeParameters, Class context, String methodName){ - List gtvs = new ArrayList<>(); - for(TypeVariable jreTV : typeParameters){ - de.dhbwstuttgart.syntaxtree.GenericTypeVar gtv = createGeneric(jreTV, jreTV.getName()); - gtvs.add(gtv); - } - return new GenericDeclarationList(gtvs,new NullToken()); - } + public GenericDeclarationList createGenerics(TypeVariable[] typeParameters, Class context, String methodName){ + List gtvs = new ArrayList<>(); + for(TypeVariable jreTV : typeParameters){ + de.dhbwstuttgart.syntaxtree.GenericTypeVar gtv = createGeneric(jreTV, jreTV.getName(), context, methodName); + gtvs.add(gtv); + } + return new GenericDeclarationList(gtvs,new NullToken()); + } /* public RefType createType(java.lang.Class jreClass){ @@ -119,7 +118,7 @@ public class ASTFactory { } */ - public RefTypeOrTPHOrWildcardOrGeneric createType(java.lang.reflect.Type type){ + public RefTypeOrTPHOrWildcardOrGeneric createType(java.lang.reflect.Type type, JavaClassName parentClass, String parentMethod){ if(type.getTypeName().equals("void")){ return new Void(new NullToken()); }else if(type.getTypeName().equals("int")){ @@ -139,12 +138,14 @@ public class ASTFactory { }else{ if(type instanceof TypeVariable){ //GTVDeclarationContext via "(TypeVariable) type).getGenericDeclaration()" - return new GenericRefType(new GenericTypeName(type.getTypeName()), new NullToken()); + return new GenericRefType( + new GenericTypeName(new GenericContext(parentClass, parentMethod),type.getTypeName()), + new NullToken()); } List params = new ArrayList<>(); if(type instanceof ParameterizedType){ for(Type t : ((ParameterizedType)type).getActualTypeArguments()){ - params.add(createType(t)); + params.add(createType(t, parentClass, parentMethod)); } } RefType ret = new RefType(this.names.getName(type.getTypeName()), params, new NullToken()); @@ -152,15 +153,18 @@ public class ASTFactory { } } - public de.dhbwstuttgart.syntaxtree.GenericTypeVar createGeneric(TypeVariable jreTypeVar, String name){ + public de.dhbwstuttgart.syntaxtree.GenericTypeVar createGeneric(TypeVariable jreTypeVar, String jreTVName, Class context, String parentMethod){ + JavaClassName parentClass = names.getName(context.getName()); List genericBounds = new ArrayList<>(); java.lang.reflect.Type[] bounds = jreTypeVar.getBounds(); if(bounds.length > 0){ for(java.lang.reflect.Type bound : bounds){ - genericBounds.add((RefType) createType(bound)); + genericBounds.add((RefType) createType(bound, parentClass, parentMethod)); } } - return new de.dhbwstuttgart.syntaxtree.GenericTypeVar(name, genericBounds, new NullToken(), new NullToken()); + return new de.dhbwstuttgart.syntaxtree.GenericTypeVar( + new GenericTypeName(new GenericContext(parentClass, parentMethod), jreTVName) + , genericBounds, new NullToken(), new NullToken()); } public ClassOrInterface createObjectClass() { @@ -212,4 +216,4 @@ public class ASTFactory { return createObjectClass().getType(); } */ -} +} \ No newline at end of file diff --git a/src/de/dhbwstuttgart/syntaxtree/statement/Assign.java b/src/de/dhbwstuttgart/syntaxtree/statement/Assign.java index a0b65973b..ae536e812 100755 --- a/src/de/dhbwstuttgart/syntaxtree/statement/Assign.java +++ b/src/de/dhbwstuttgart/syntaxtree/statement/Assign.java @@ -3,8 +3,8 @@ package de.dhbwstuttgart.syntaxtree.statement; import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceBlockInformation; import de.dhbwstuttgart.typeinference.constraints.ConstraintSet; -import de.dhbwstuttgart.typeinference.constraints.Pair; -import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceInformation; +import de.dhbwstuttgart.typeinference.constraints.ConstraintsFactory; +import de.dhbwstuttgart.typeinference.unify.model.PairOperator; import org.antlr.v4.runtime.Token; @@ -23,7 +23,8 @@ public class Assign extends Statement public ConstraintSet getConstraints(TypeInferenceBlockInformation info) { ConstraintSet ret = lefSide.getConstraints(info); ret.addAll(rightSide.getConstraints(info)); - ret.addUndConstraint(new Pair(rightSide.getType(), lefSide.getType())); + ret.addUndConstraint(ConstraintsFactory.createPair( + rightSide.getType(), lefSide.getType(), PairOperator.SMALLERDOT, info)); return ret; } } diff --git a/src/de/dhbwstuttgart/syntaxtree/statement/Block.java b/src/de/dhbwstuttgart/syntaxtree/statement/Block.java index 56b9a1f73..04e4a1066 100755 --- a/src/de/dhbwstuttgart/syntaxtree/statement/Block.java +++ b/src/de/dhbwstuttgart/syntaxtree/statement/Block.java @@ -31,8 +31,6 @@ public class Block extends Statement } return ret; } - - } diff --git a/src/de/dhbwstuttgart/syntaxtree/statement/FieldVar.java b/src/de/dhbwstuttgart/syntaxtree/statement/FieldVar.java index 0eac0866c..13274065d 100644 --- a/src/de/dhbwstuttgart/syntaxtree/statement/FieldVar.java +++ b/src/de/dhbwstuttgart/syntaxtree/statement/FieldVar.java @@ -6,9 +6,9 @@ import de.dhbwstuttgart.typeinference.assumptions.FieldAssumption; import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceBlockInformation; import de.dhbwstuttgart.typeinference.constraints.Constraint; import de.dhbwstuttgart.typeinference.constraints.ConstraintSet; -import de.dhbwstuttgart.typeinference.constraints.Pair; +import de.dhbwstuttgart.typeinference.constraints.ConstraintsFactory; +import de.dhbwstuttgart.typeinference.unify.model.PairOperator; import org.antlr.v4.runtime.Token; -import sun.reflect.generics.reflectiveObjects.NotImplementedException; import java.util.HashSet; import java.util.Set; @@ -31,8 +31,10 @@ public class FieldVar extends Expression { ret.addAll(receiver.getConstraints(info)); for(FieldAssumption fieldAssumption : info.getFields(fieldVarName)){ Constraint constraint = new Constraint(); - constraint.add(new Pair(receiver.getType(),fieldAssumption.getReceiverType())); - constraint.add(new Pair(this.getType(),fieldAssumption.getType())); + constraint.add(ConstraintsFactory.createPair( + receiver.getType(),fieldAssumption.getReceiverType(), info)); + constraint.add(ConstraintsFactory.createPair( + this.getType(),fieldAssumption.getType(), info)); oderConstraints.add(constraint); } if(oderConstraints.size() == 0) diff --git a/src/de/dhbwstuttgart/syntaxtree/statement/MethodCall.java b/src/de/dhbwstuttgart/syntaxtree/statement/MethodCall.java index a84ea30dc..0c2accd5b 100755 --- a/src/de/dhbwstuttgart/syntaxtree/statement/MethodCall.java +++ b/src/de/dhbwstuttgart/syntaxtree/statement/MethodCall.java @@ -4,8 +4,9 @@ import de.dhbwstuttgart.exceptions.TypeinferenceException; import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceBlockInformation; import de.dhbwstuttgart.typeinference.constraints.Constraint; import de.dhbwstuttgart.typeinference.constraints.ConstraintSet; -import de.dhbwstuttgart.typeinference.constraints.Pair; +import de.dhbwstuttgart.typeinference.constraints.ConstraintsFactory; import de.dhbwstuttgart.typeinference.assumptions.MethodAssumption; +import de.dhbwstuttgart.typeinference.unify.model.PairOperator; import org.antlr.v4.runtime.Token; import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder; @@ -38,11 +39,11 @@ public class MethodCall extends Statement Set methodConstraints = new HashSet<>(); for(MethodAssumption m : info.getMethods(name, arglist)){ Constraint methodConstraint = new Constraint(); - methodConstraint.add(new Pair(receiver.getType(), m.getReceiverType())); - methodConstraint.add(new Pair(m.getReturnType(), this.getType())); + methodConstraint.add(ConstraintsFactory.createPair(receiver.getType(), m.getReceiverType(), PairOperator.SMALLERDOT, info)); + methodConstraint.add(ConstraintsFactory.createPair(m.getReturnType(), this.getType(), PairOperator.SMALLERDOT, info)); for(int i = 0;i availableClasses, ClassOrInterface currentClass, Method methodContext) { + public TypeInferenceBlockInformation(Set availableClasses, @NotNull ClassOrInterface currentClass, Method methodContext) { super(availableClasses); this.methodContext = methodContext; this.currentClass = currentClass; @@ -19,4 +20,7 @@ public class TypeInferenceBlockInformation extends TypeInferenceInformation { public ClassOrInterface getCurrentClass() { return currentClass; } + public Method getCurrentMethod() { + return methodContext; + } } diff --git a/src/de/dhbwstuttgart/typeinference/constraints/ConstraintsFactory.java b/src/de/dhbwstuttgart/typeinference/constraints/ConstraintsFactory.java new file mode 100644 index 000000000..6a13da400 --- /dev/null +++ b/src/de/dhbwstuttgart/typeinference/constraints/ConstraintsFactory.java @@ -0,0 +1,47 @@ +package de.dhbwstuttgart.typeinference.constraints; + +import de.dhbwstuttgart.syntaxtree.GenericTypeVar; +import de.dhbwstuttgart.syntaxtree.type.GenericRefType; +import de.dhbwstuttgart.syntaxtree.type.RefType; +import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; +import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder; +import de.dhbwstuttgart.typeinference.assumptions.TypeInferenceBlockInformation; +import de.dhbwstuttgart.typeinference.unify.model.PairOperator; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +public class ConstraintsFactory { + public static Pair createPair(RefTypeOrTPHOrWildcardOrGeneric t1, RefTypeOrTPHOrWildcardOrGeneric t2, PairOperator equalsdot, TypeInferenceBlockInformation info){ + //Check whether Generics are in the same class: + return new Pair(checkGeneric(t1, info), checkGeneric(t2, info)); + } + public static Pair createPair(RefTypeOrTPHOrWildcardOrGeneric t1, + RefTypeOrTPHOrWildcardOrGeneric t2, TypeInferenceBlockInformation info){ + return createPair(t1,t2,PairOperator.SMALLERDOT, info); + } + + private static RefTypeOrTPHOrWildcardOrGeneric checkGeneric(RefTypeOrTPHOrWildcardOrGeneric type, TypeInferenceBlockInformation info){ + if(type instanceof GenericRefType){ + List genericsInContext = new ArrayList<>(); + for(GenericTypeVar genericTypeVar : info.getCurrentClass().getGenerics()){ + genericsInContext.add(genericTypeVar); + } + if(info.getCurrentMethod() != null)for(GenericTypeVar genericTypeVar : info.getCurrentMethod().getGenerics()){ + genericsInContext.add(genericTypeVar); + } + + for(GenericTypeVar genericTypeVar : genericsInContext){ + if(genericTypeVar.getName().equals(((GenericRefType)type).getName())){ + return new RefType(((GenericRefType)type).getName(),type.getOffset()); + } + } + //Nicht in den Generics in diesem Kontext enthalten: + return TypePlaceholder.fresh(type.getOffset()); + }else{ + return type; + } + } + +} diff --git a/src/de/dhbwstuttgart/typeinference/constraints/Pair.java b/src/de/dhbwstuttgart/typeinference/constraints/Pair.java old mode 100755 new mode 100644 index 08611bb3d..72f983265 --- a/src/de/dhbwstuttgart/typeinference/constraints/Pair.java +++ b/src/de/dhbwstuttgart/typeinference/constraints/Pair.java @@ -4,6 +4,7 @@ import java.io.Serializable; import com.sun.istack.internal.NotNull; import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric; import de.dhbwstuttgart.typeinference.unify.model.PairOperator; +import de.dhbwstuttgart.typeinference.unify.model.UnifyPair; public class Pair implements Serializable @@ -13,7 +14,7 @@ public class Pair implements Serializable private PairOperator eOperator = PairOperator.SMALLER; - public Pair(@NotNull RefTypeOrTPHOrWildcardOrGeneric TA1, @NotNull RefTypeOrTPHOrWildcardOrGeneric TA2 ) + Pair(@NotNull RefTypeOrTPHOrWildcardOrGeneric TA1, @NotNull RefTypeOrTPHOrWildcardOrGeneric TA2 ) { this.TA1 = TA1; this.TA2 = TA2; @@ -22,7 +23,7 @@ public class Pair implements Serializable eOperator = PairOperator.SMALLER; } - public Pair(RefTypeOrTPHOrWildcardOrGeneric TA1, RefTypeOrTPHOrWildcardOrGeneric TA2, PairOperator eOp) + Pair(@NotNull RefTypeOrTPHOrWildcardOrGeneric TA1, @NotNull RefTypeOrTPHOrWildcardOrGeneric TA2, PairOperator eOp) { // Konstruktor this(TA1,TA2); diff --git a/test/typeinference/JavaTXCompilerTest.java b/test/typeinference/JavaTXCompilerTest.java index 1de2cc878..8760e3217 100644 --- a/test/typeinference/JavaTXCompilerTest.java +++ b/test/typeinference/JavaTXCompilerTest.java @@ -16,7 +16,7 @@ public class JavaTXCompilerTest { @Test public void test() throws IOException, ClassNotFoundException { JavaTXCompiler compiler = new JavaTXCompiler(); - compiler.parse(new File(rootDirectory+"Methods.jav")); + //compiler.parse(new File(rootDirectory+"Methods.jav")); compiler.parse(new File(rootDirectory+"Generics.jav")); compiler.typeInference(); }