FC Generator
This commit is contained in:
parent
8d12821c68
commit
0610ea8e63
@ -89,6 +89,7 @@ public class JavaTXCompiler {
|
|||||||
final ConstraintSet<Pair> cons = getConstraints();
|
final ConstraintSet<Pair> cons = getConstraints();
|
||||||
|
|
||||||
FiniteClosure finiteClosure = UnifyTypeFactory.generateFC(allClasses);
|
FiniteClosure finiteClosure = UnifyTypeFactory.generateFC(allClasses);
|
||||||
|
System.out.println(finiteClosure);
|
||||||
ConstraintSet<UnifyPair> unifyCons = UnifyTypeFactory.convert(cons);
|
ConstraintSet<UnifyPair> unifyCons = UnifyTypeFactory.convert(cons);
|
||||||
|
|
||||||
TypeUnify unify = new TypeUnify();
|
TypeUnify unify = new TypeUnify();
|
||||||
|
148
src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/FCGenerator.java
Normal file
148
src/de/dhbwstuttgart/parser/SyntaxTreeGenerator/FCGenerator.java
Normal file
@ -0,0 +1,148 @@
|
|||||||
|
package de.dhbwstuttgart.parser.SyntaxTreeGenerator;
|
||||||
|
|
||||||
|
import de.dhbwstuttgart.exceptions.DebugException;
|
||||||
|
import de.dhbwstuttgart.exceptions.NotImplementedException;
|
||||||
|
import de.dhbwstuttgart.syntaxtree.ClassOrInterface;
|
||||||
|
import de.dhbwstuttgart.syntaxtree.GenericTypeVar;
|
||||||
|
import de.dhbwstuttgart.syntaxtree.factory.ASTFactory;
|
||||||
|
import de.dhbwstuttgart.syntaxtree.factory.UnifyTypeFactory;
|
||||||
|
import de.dhbwstuttgart.syntaxtree.type.*;
|
||||||
|
import de.dhbwstuttgart.typeinference.constraints.Pair;
|
||||||
|
import de.dhbwstuttgart.typeinference.unify.model.*;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
public class FCGenerator {
|
||||||
|
/**
|
||||||
|
* Baut die FiniteClosure aus availableClasses.
|
||||||
|
* Klassen welche nicht in availableClasses vorkommen werden im Java Classpath nachgeschlagen.
|
||||||
|
*
|
||||||
|
* @param availableClasses - Alle geparsten Klassen
|
||||||
|
*/
|
||||||
|
public static Set<UnifyPair> toFC(Collection<ClassOrInterface> availableClasses) throws ClassNotFoundException {
|
||||||
|
HashSet<UnifyPair> pairs = new HashSet<>();
|
||||||
|
for(ClassOrInterface cly : availableClasses){
|
||||||
|
pairs.addAll(getSuperTypes(cly, availableClasses));
|
||||||
|
}
|
||||||
|
System.out.println(pairs);
|
||||||
|
return pairs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bildet eine Kette vom übergebenen Typ bis hin zum höchsten bekannten Typ
|
||||||
|
* Als Generics werden TPHs benutzt, welche der Unifikationsalgorithmus korrekt interpretieren muss.
|
||||||
|
* Die verwendeten TPHs werden in der Kette nach oben gereicht, so erhält der selbe GTV immer den selben TPH
|
||||||
|
* @param forType
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
private static List<UnifyPair> getSuperTypes(ClassOrInterface forType, Collection<ClassOrInterface> availableClasses) throws ClassNotFoundException {
|
||||||
|
return getSuperTypes(forType, availableClasses, new HashMap<>());
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO: implements Interface auch als superklassen beachten
|
||||||
|
private static List<UnifyPair> getSuperTypes(ClassOrInterface forType, Collection<ClassOrInterface> availableClasses, HashMap<String, UnifyType> gtvs) throws ClassNotFoundException {
|
||||||
|
List<UnifyType> params = new ArrayList<>();
|
||||||
|
//Die GTVs, die in forType hinzukommen:
|
||||||
|
HashMap<String, UnifyType> newGTVs = new HashMap<>();
|
||||||
|
//Generics mit gleichem Namen müssen den selben TPH bekommen
|
||||||
|
for(GenericTypeVar gtv : forType.getGenerics()){
|
||||||
|
if(!gtvs.containsKey(gtv.getParsedName())){
|
||||||
|
gtvs.put(gtv.getParsedName(), PlaceholderType.freshPlaceholder());
|
||||||
|
newGTVs.put(gtv.getParsedName(), PlaceholderType.freshPlaceholder());
|
||||||
|
}
|
||||||
|
params.add(gtvs.get(gtv.getParsedName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
Optional<ClassOrInterface> hasSuperclass = availableClasses.stream().filter(cl -> forType.getSuperClass().getName().equals(cl.getClassName())).findAny();
|
||||||
|
ClassOrInterface superClass;
|
||||||
|
if(!hasSuperclass.isPresent()) //Wenn es die Klasse in den available Klasses nicht gibt wird sie im Classpath gesucht. Ansonsten Exception
|
||||||
|
{
|
||||||
|
superClass = ASTFactory.createClass(ClassLoader.getSystemClassLoader().loadClass(forType.getSuperClass().getName().toString()));
|
||||||
|
}else{
|
||||||
|
superClass = hasSuperclass.get();
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
Die Parameter der superklasse müssen jetzt nach den Angaben in der Subklasse
|
||||||
|
modifiziert werden
|
||||||
|
Beispie: Matrix<A> extends Vector<Vector<A>>
|
||||||
|
Den ersten Parameter mit Vector<A> austauschen und dort alle Generics zu den Typplaceholdern in gtvs austauschen
|
||||||
|
*/
|
||||||
|
//Hier vermerken, welche Typen im der Superklasse ausgetauscht werden müssen
|
||||||
|
Iterator<GenericTypeVar> itGenParams = superClass.getGenerics().iterator();
|
||||||
|
Iterator<RefTypeOrTPHOrWildcardOrGeneric> itSetParams = forType.getSuperClass().getParaList().iterator();
|
||||||
|
while(itGenParams.hasNext()){
|
||||||
|
RefTypeOrTPHOrWildcardOrGeneric setType = itSetParams.next();
|
||||||
|
//In diesem Typ die GTVs durch TPHs und Einsetzungen austauschen:
|
||||||
|
UnifyType setSetType = setType.acceptTV(new TypeExchanger(gtvs));
|
||||||
|
newGTVs.put(itGenParams.next().getParsedName(), setSetType);
|
||||||
|
}
|
||||||
|
|
||||||
|
UnifyType superType = forType.getSuperClass().acceptTV(new TypeExchanger(newGTVs));
|
||||||
|
|
||||||
|
TypeParams paramList = new TypeParams(params);
|
||||||
|
UnifyType t1 = new ReferenceType(forType.getClassName().toString(), paramList);
|
||||||
|
UnifyType t2 = superType;
|
||||||
|
|
||||||
|
UnifyPair ret = UnifyTypeFactory.generateSmallerPair(t1, t2);
|
||||||
|
|
||||||
|
List<UnifyPair> superTypes;
|
||||||
|
//Rekursiver Aufruf. Abbruchbedingung ist Object als Superklasse:
|
||||||
|
if(superClass.getClassName().equals(ASTFactory.createObjectClass().getClassName())){
|
||||||
|
superTypes = Arrays.asList(UnifyTypeFactory.generateSmallerPair(UnifyTypeFactory.convert(ASTFactory.createObjectType()), UnifyTypeFactory.convert(ASTFactory.createObjectType())));
|
||||||
|
}else{
|
||||||
|
superTypes = getSuperTypes(superClass, availableClasses, newGTVs);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<UnifyPair> retList = new ArrayList<>();
|
||||||
|
retList.add(ret);
|
||||||
|
retList.addAll(superTypes);
|
||||||
|
|
||||||
|
return retList;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tauscht die GTVs in einem Typ gegen die entsprechenden Typen in der übergebenen Map aus.
|
||||||
|
*/
|
||||||
|
private static class TypeExchanger implements TypeVisitor<UnifyType>{
|
||||||
|
|
||||||
|
private final HashMap<String, UnifyType> gtvs;
|
||||||
|
|
||||||
|
TypeExchanger(HashMap<String, UnifyType> gtvs){
|
||||||
|
this.gtvs = gtvs;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UnifyType visit(RefType refType) {
|
||||||
|
List<UnifyType> params = new ArrayList<>();
|
||||||
|
for(RefTypeOrTPHOrWildcardOrGeneric param : refType.getParaList()){
|
||||||
|
params.add(param.acceptTV(this));
|
||||||
|
}
|
||||||
|
TypeParams paramList = new TypeParams(params);
|
||||||
|
UnifyType ret = new ReferenceType(refType.getName().toString(), paramList);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UnifyType visit(SuperWildcardType superWildcardType) {
|
||||||
|
throw new DebugException("Dieser Fall darf nicht auftreten");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UnifyType visit(TypePlaceholder typePlaceholder) {
|
||||||
|
throw new DebugException("Dieser Fall darf nicht auftreten");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UnifyType visit(ExtendsWildcardType extendsWildcardType) {
|
||||||
|
throw new DebugException("Dieser Fall darf nicht auftreten");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UnifyType visit(GenericRefType genericRefType) {
|
||||||
|
if(! gtvs.containsKey(genericRefType.getParsedName()))
|
||||||
|
throw new DebugException("Dieser Fall darf nicht auftreten");
|
||||||
|
return gtvs.get(genericRefType.getParsedName());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -6,6 +6,7 @@ import java.util.stream.Collectors;
|
|||||||
import de.dhbwstuttgart.exceptions.DebugException;
|
import de.dhbwstuttgart.exceptions.DebugException;
|
||||||
import de.dhbwstuttgart.exceptions.NotImplementedException;
|
import de.dhbwstuttgart.exceptions.NotImplementedException;
|
||||||
import de.dhbwstuttgart.parser.NullToken;
|
import de.dhbwstuttgart.parser.NullToken;
|
||||||
|
import de.dhbwstuttgart.parser.SyntaxTreeGenerator.FCGenerator;
|
||||||
import de.dhbwstuttgart.parser.scope.JavaClassName;
|
import de.dhbwstuttgart.parser.scope.JavaClassName;
|
||||||
import de.dhbwstuttgart.syntaxtree.ClassOrInterface;
|
import de.dhbwstuttgart.syntaxtree.ClassOrInterface;
|
||||||
import de.dhbwstuttgart.syntaxtree.GenericTypeVar;
|
import de.dhbwstuttgart.syntaxtree.GenericTypeVar;
|
||||||
@ -35,57 +36,7 @@ public class UnifyTypeFactory {
|
|||||||
Generell dürfen sie immer die gleichen Namen haben.
|
Generell dürfen sie immer die gleichen Namen haben.
|
||||||
TODO: die transitive Hülle bilden
|
TODO: die transitive Hülle bilden
|
||||||
*/
|
*/
|
||||||
HashSet<UnifyPair> pairs = new HashSet<>();
|
return new FiniteClosure(FCGenerator.toFC(fromClasses));
|
||||||
for(ClassOrInterface cly : fromClasses){
|
|
||||||
pairs.addAll(getSuperTypes(cly, fromClasses));
|
|
||||||
}
|
|
||||||
return new FiniteClosure(pairs);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Bildet eine Kette vom übergebenen Typ bis hin zum höchsten bekannten Typ
|
|
||||||
* Als Generics werden TPHs benutzt, welche der Unifikationsalgorithmus korrekt interpretieren muss.
|
|
||||||
* Die verwendeten TPHs werden in der Kette nach oben gereicht, so erhält der selbe GTV immer den selben TPH
|
|
||||||
* @param forType
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
private static List<UnifyPair> getSuperTypes(ClassOrInterface forType, List<ClassOrInterface> availableClasses) throws ClassNotFoundException {
|
|
||||||
return getSuperTypes(forType, availableClasses, new HashMap<>());
|
|
||||||
}
|
|
||||||
|
|
||||||
private static List<UnifyPair> getSuperTypes(ClassOrInterface forType, List<ClassOrInterface> availableClasses, HashMap<String, PlaceholderType> gtvs) throws ClassNotFoundException {
|
|
||||||
List<UnifyType> params = new ArrayList<>();
|
|
||||||
//Generics mit gleichem Namen müssen den selben TPH bekommen
|
|
||||||
for(GenericTypeVar gtv : forType.getGenerics()){
|
|
||||||
if(!gtvs.containsKey(gtv.getParsedName()))
|
|
||||||
gtvs.put(gtv.getParsedName(), PlaceholderType.freshPlaceholder());
|
|
||||||
params.add(gtvs.get(gtv.getParsedName()));
|
|
||||||
}
|
|
||||||
Optional<ClassOrInterface> hasSuperclass = availableClasses.stream().filter(cl -> forType.getSuperClass().getName().equals(cl.getClassName())).findAny();
|
|
||||||
ClassOrInterface superClass;
|
|
||||||
if(!hasSuperclass.isPresent()) //TODO: Wenn es die Object-Klasse ist, dann ist es in Ordnung, ansonsten Fehler ausgeben:
|
|
||||||
{
|
|
||||||
superClass = ASTFactory.createClass(ClassLoader.getSystemClassLoader().loadClass(forType.getSuperClass().getName().toString()));
|
|
||||||
}else{
|
|
||||||
superClass = hasSuperclass.get();
|
|
||||||
}
|
|
||||||
List<UnifyPair> superTypes;
|
|
||||||
if(superClass.getClassName().equals(ASTFactory.createObjectClass().getClassName())){
|
|
||||||
superTypes = Arrays.asList(generateSmallerPair(convert(ASTFactory.createObjectType()), convert(ASTFactory.createObjectType())));
|
|
||||||
}else{
|
|
||||||
superTypes = getSuperTypes(superClass, availableClasses, gtvs);
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeParams paramList = new TypeParams(params);
|
|
||||||
UnifyType t1 = new ReferenceType(forType.getClassName().toString(), paramList);
|
|
||||||
UnifyType t2 = superTypes.get(0).getLhsType();
|
|
||||||
|
|
||||||
UnifyPair ret = generateSmallerPair(t1, t2);
|
|
||||||
List<UnifyPair> retList = new ArrayList<>();
|
|
||||||
retList.add(ret);
|
|
||||||
retList.addAll(superTypes);
|
|
||||||
|
|
||||||
return retList;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UnifyPair generateSmallerPair(UnifyType tl, UnifyType tr){
|
public static UnifyPair generateSmallerPair(UnifyType tl, UnifyType tr){
|
||||||
|
Loading…
Reference in New Issue
Block a user