forked from JavaTX/JavaCompilerCore
refactored standard unification
This commit is contained in:
parent
64247b689b
commit
5dd90cb30c
@ -42,27 +42,38 @@ public class MartelliMontanariUnify implements IUnify {
|
|||||||
int idx = 0;
|
int idx = 0;
|
||||||
while(idx < termsQ.size()) {
|
while(idx < termsQ.size()) {
|
||||||
UnifyPair pair = termsQ.get(idx);
|
UnifyPair pair = termsQ.get(idx);
|
||||||
|
UnifyType rhsType = pair.getRhsType();
|
||||||
|
UnifyType lhsType = pair.getLhsType();
|
||||||
|
TypeParams rhsTypeParams = rhsType.getTypeParams();
|
||||||
|
TypeParams lhsTypeParams = lhsType.getTypeParams();
|
||||||
|
|
||||||
if(delete(pair)) {
|
// DELETE
|
||||||
|
if(pair.getRhsType().equals(pair.getLhsType())) {
|
||||||
termsQ.remove(idx);
|
termsQ.remove(idx);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<Set<UnifyPair>> optSet = decompose(pair);
|
// REDUCE
|
||||||
|
if(!(rhsType instanceof PlaceholderType) && !(lhsType instanceof PlaceholderType)
|
||||||
if(optSet == null)
|
&& (rhsTypeParams.size() != 0 || lhsTypeParams.size() != 0)) {
|
||||||
return Optional.empty(); // Unification failed
|
Set<UnifyPair> result = new HashSet<>();
|
||||||
|
|
||||||
if(optSet.isPresent()) {
|
if(!rhsType.getName().equals(lhsType.getName()))
|
||||||
termsQ.addAll(optSet.get());
|
return Optional.empty(); // conflict
|
||||||
|
if(rhsTypeParams.size() != lhsTypeParams.size())
|
||||||
|
return Optional.empty(); // conflict
|
||||||
|
|
||||||
|
for(int i = 0; i < rhsTypeParams.size(); i++)
|
||||||
|
result.add(new UnifyPair(rhsTypeParams.get(i), lhsTypeParams.get(i), PairOperator.EQUALSDOT));
|
||||||
|
|
||||||
|
termsQ.addAll(result);
|
||||||
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<UnifyPair> optPair = swap(pair);
|
// SWAP
|
||||||
|
if(!(lhsType instanceof PlaceholderType) && (rhsType instanceof PlaceholderType)) {
|
||||||
if(optPair.isPresent()) {
|
termsQ.add(new UnifyPair(rhsType, lhsType, PairOperator.EQUALSDOT));
|
||||||
termsQ.add(optPair.get());
|
|
||||||
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -72,11 +83,9 @@ public class MartelliMontanariUnify implements IUnify {
|
|||||||
&& pair.getRhsType().getTypeParams().occurs((PlaceholderType) pair.getLhsType()))
|
&& pair.getRhsType().getTypeParams().occurs((PlaceholderType) pair.getLhsType()))
|
||||||
return Optional.empty();
|
return Optional.empty();
|
||||||
|
|
||||||
Optional<Entry<PlaceholderType, UnifyType>> optUni = eliminate(pair);
|
// SUBST
|
||||||
|
if(lhsType instanceof PlaceholderType) {
|
||||||
if(optUni.isPresent()) {
|
mgu.Add((PlaceholderType) lhsType, rhsType);
|
||||||
Entry<PlaceholderType, UnifyType> substitution = optUni.get();
|
|
||||||
mgu.Add(substitution.getKey(), substitution.getValue());
|
|
||||||
termsQ = termsQ.stream().map(mgu::apply).collect(Collectors.toCollection(ArrayList::new));
|
termsQ = termsQ.stream().map(mgu::apply).collect(Collectors.toCollection(ArrayList::new));
|
||||||
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
idx = idx+1 == termsQ.size() ? 0 : idx+1;
|
||||||
continue;
|
continue;
|
||||||
@ -87,55 +96,4 @@ public class MartelliMontanariUnify implements IUnify {
|
|||||||
|
|
||||||
return Optional.of(mgu);
|
return Optional.of(mgu);
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean delete(UnifyPair pair) {
|
|
||||||
return pair.getRhsType().equals(pair.getLhsType());
|
|
||||||
}
|
|
||||||
|
|
||||||
private Optional<Set<UnifyPair>> decompose(UnifyPair pair) {
|
|
||||||
Set<UnifyPair> result = new HashSet<>();
|
|
||||||
|
|
||||||
UnifyType rhs = pair.getRhsType();
|
|
||||||
UnifyType lhs = pair.getLhsType();
|
|
||||||
|
|
||||||
TypeParams rhsTypeParams = rhs.getTypeParams();
|
|
||||||
TypeParams lhsTypeParams = lhs.getTypeParams();
|
|
||||||
|
|
||||||
if(!(rhs instanceof PlaceholderType) && !(lhs instanceof PlaceholderType)) {
|
|
||||||
if(!rhs.getName().equals(lhs.getName()))
|
|
||||||
return null; // conflict
|
|
||||||
if(rhsTypeParams.size() != lhsTypeParams.size())
|
|
||||||
return null; // conflict;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(rhsTypeParams.size() == 0 || lhsTypeParams.size() == 0)
|
|
||||||
return Optional.empty();
|
|
||||||
|
|
||||||
for(int i = 0; i < rhsTypeParams.size(); i++)
|
|
||||||
result.add(new UnifyPair(rhsTypeParams.get(i), lhsTypeParams.get(i), PairOperator.EQUALSDOT));
|
|
||||||
|
|
||||||
return Optional.of(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Optional<UnifyPair> swap(UnifyPair pair) {
|
|
||||||
UnifyType rhs = pair.getRhsType();
|
|
||||||
UnifyType lhs = pair.getLhsType();
|
|
||||||
|
|
||||||
if(!(lhs instanceof PlaceholderType) && (rhs instanceof PlaceholderType))
|
|
||||||
return Optional.of(new UnifyPair(rhs, lhs, PairOperator.EQUALSDOT));
|
|
||||||
|
|
||||||
return Optional.empty();
|
|
||||||
}
|
|
||||||
|
|
||||||
private Optional<Entry<PlaceholderType, UnifyType>> eliminate(UnifyPair pair) {
|
|
||||||
UnifyType rhs = pair.getRhsType();
|
|
||||||
UnifyType lhs = pair.getLhsType();
|
|
||||||
|
|
||||||
// TODO only apply when lhs is element of vars(termsQ)?
|
|
||||||
|
|
||||||
if(!(lhs instanceof PlaceholderType))
|
|
||||||
return Optional.empty();
|
|
||||||
|
|
||||||
return Optional.of(new AbstractMap.SimpleImmutableEntry<PlaceholderType, UnifyType>((PlaceholderType) lhs, rhs));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -347,7 +347,7 @@ public class Unify {
|
|||||||
* TODO Optimierungsmöglichkeit:
|
* TODO Optimierungsmöglichkeit:
|
||||||
*
|
*
|
||||||
* An dieser Stelle gibt es Raum für Optimierung.
|
* An dieser Stelle gibt es Raum für Optimierung.
|
||||||
* Z.B. resultiert (a <. C<Integer>) durch Permutation der Parameter mit grArg in:
|
* Z.B. resultiert (a <. C<Integer>) durch Permutation der Parameter mit grArg und smaller in:
|
||||||
* (a = C<b'>, b' <.? ? extends Number)
|
* (a = C<b'>, b' <.? ? extends Number)
|
||||||
* (a = C<b'>, b' <.? ? extends Integer)
|
* (a = C<b'>, b' <.? ? extends Integer)
|
||||||
* (a = C<b'>, b' <.? Integer)
|
* (a = C<b'>, b' <.? Integer)
|
||||||
@ -357,7 +357,9 @@ public class Unify {
|
|||||||
* rekursiven Aufruf des Unify. Es würde reichen nur den allgemeinsten Fall zu betrachten da dieser den Lösungraum
|
* rekursiven Aufruf des Unify. Es würde reichen nur den allgemeinsten Fall zu betrachten da dieser den Lösungraum
|
||||||
* der anderen Fälle miteinschließt.
|
* der anderen Fälle miteinschließt.
|
||||||
*
|
*
|
||||||
* (Gibt es einen einzigen maximalen fall? Wsl. ? super x (x möglichst klein) und ? ext y (y möglichst groß))
|
* Prüfen:
|
||||||
|
* Gibt es einen einzigen maximalen Fall?
|
||||||
|
* Wahrscheinlich gibt es 2: ? super x (x möglichst klein) und ? ext y (y möglichst groß))
|
||||||
*/
|
*/
|
||||||
|
|
||||||
for(UnifyType param : cParams)
|
for(UnifyType param : cParams)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user