/* * Copyright (C) 2009 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package signature.compare; import signature.compare.model.IAnnotationDelta; import signature.compare.model.IAnnotationElementDelta; import signature.compare.model.IAnnotationFieldDelta; import signature.compare.model.IApiDelta; import signature.compare.model.IClassDefinitionDelta; import signature.compare.model.IConstructorDelta; import signature.compare.model.IDelta; import signature.compare.model.IEnumConstantDelta; import signature.compare.model.IFieldDelta; import signature.compare.model.IGenericDeclarationDelta; import signature.compare.model.IMethodDelta; import signature.compare.model.IModifierDelta; import signature.compare.model.IPackageDelta; import signature.compare.model.IParameterDelta; import signature.compare.model.IParameterizedTypeDelta; import signature.compare.model.IPrimitiveTypeDelta; import signature.compare.model.ITypeReferenceDelta; import signature.compare.model.ITypeVariableDefinitionDelta; import signature.compare.model.IUpperBoundsDelta; import signature.compare.model.IValueDelta; import signature.compare.model.IWildcardTypeDelta; import signature.compare.model.impl.SigAnnotationDelta; import signature.compare.model.impl.SigAnnotationElementDelta; import signature.compare.model.impl.SigAnnotationFieldDelta; import signature.compare.model.impl.SigApiDelta; import signature.compare.model.impl.SigArrayTypeDelta; import signature.compare.model.impl.SigClassDefinitionDelta; import signature.compare.model.impl.SigClassReferenceDelta; import signature.compare.model.impl.SigConstructorDelta; import signature.compare.model.impl.SigEnumConstantDelta; import signature.compare.model.impl.SigFieldDelta; import signature.compare.model.impl.SigGenericDeclarationDelta; import signature.compare.model.impl.SigMethodDelta; import signature.compare.model.impl.SigModifierDelta; import signature.compare.model.impl.SigPackageDelta; import signature.compare.model.impl.SigParameterDelta; import signature.compare.model.impl.SigParameterizedTypeDelta; import signature.compare.model.impl.SigPrimitiveTypeDelta; import signature.compare.model.impl.SigTypeDelta; import signature.compare.model.impl.SigTypeVariableDefinitionDelta; import signature.compare.model.impl.SigTypeVariableReferenceDelta; import signature.compare.model.impl.SigUpperBoundsDelta; import signature.compare.model.impl.SigValueDelta; import signature.compare.model.impl.SigWildcardTypeDelta; import signature.compare.model.subst.ClassProjection; import signature.compare.model.subst.ViewpointAdapter; import signature.model.IAnnotation; import signature.model.IAnnotationElement; import signature.model.IAnnotationField; import signature.model.IApi; import signature.model.IArrayType; import signature.model.IClassDefinition; import signature.model.IClassReference; import signature.model.IConstructor; import signature.model.IEnumConstant; import signature.model.IExecutableMember; import signature.model.IField; import signature.model.IGenericDeclaration; import signature.model.IMethod; import signature.model.IPackage; import signature.model.IParameter; import signature.model.IParameterizedType; import signature.model.IPrimitiveType; import signature.model.ITypeReference; import signature.model.ITypeVariableDefinition; import signature.model.ITypeVariableReference; import signature.model.IWildcardType; import signature.model.Kind; import signature.model.Modifier; import signature.model.impl.SigAnnotationElement; import signature.model.impl.SigArrayType; import java.util.Arrays; import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import java.util.Set; /** * {@code ApiComparator} takes two signature models as input and creates a delta * model describing the differences between those. */ public class ApiComparator implements IApiComparator { public IApiDelta compare(IApi from, IApi to) { assert from.getVisibility() == to.getVisibility(); Set fromPackages = from.getPackages(); Set toPackages = to.getPackages(); Set packageDeltas = compareSets(fromPackages, toPackages, new SigComparator() { public IPackageDelta createChangedDelta(IPackage from, IPackage to) { return comparePackage(from, to); } public IPackageDelta createAddRemoveDelta(IPackage from, IPackage to) { return new SigPackageDelta(from, to); } public boolean considerEqualElement(IPackage from, IPackage to) { return from.getName().equals(to.getName()); } }); SigApiDelta delta = null; if (packageDeltas != null) { delta = new SigApiDelta(from, to); delta.setPackageDeltas(packageDeltas); } return delta; } private IPackageDelta comparePackage(IPackage from, IPackage to) { assert from.getName().equals(to.getName()); Set fromClasses = from.getClasses(); Set toClasses = to.getClasses(); Set classDeltas = compareSets(fromClasses, toClasses, new SigComparator() { public boolean considerEqualElement(IClassDefinition from, IClassDefinition to) { return sameClassDefinition(from, to); } public IClassDefinitionDelta createChangedDelta( IClassDefinition from, IClassDefinition to) { return compareClass(from, to); } public IClassDefinitionDelta createAddRemoveDelta( IClassDefinition from, IClassDefinition to) { return new SigClassDefinitionDelta(from, to); } }); SigPackageDelta delta = null; if (classDeltas != null) { delta = new SigPackageDelta(from, to); delta.setClassDeltas(classDeltas); } // Annotations Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (delta != null) { delta = new SigPackageDelta(from, to); } delta.setAnnotationDeltas(annotationDeltas); } return delta; } private IClassDefinitionDelta compareClass(IClassDefinition from, IClassDefinition to) { assert from.getKind() == to.getKind(); assert from.getName().equals(to.getName()); assert from.getPackageName().equals(to.getPackageName()); SigClassDefinitionDelta classDelta = null; // modifiers Set modifierDeltas = compareModifiers(from .getModifiers(), to.getModifiers()); if (modifierDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setModifierDeltas(modifierDeltas); } // super class ITypeReferenceDelta superTypeDelta = compareType(from .getSuperClass(), to.getSuperClass(), false); if (superTypeDelta != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setSuperClassDelta(superTypeDelta); } // interfaces Set> interfaceDeltas = compareInterfaces(from, to); if (interfaceDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setInterfaceDeltas(interfaceDeltas); } // type variables Set typeVariableDeltas = compareTypeVariableSequence(from.getTypeParameters(), to.getTypeParameters()); if (typeVariableDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setTypeVariableDeltas(typeVariableDeltas); } // constructors Set constructorDeltas = compareConstructors(from .getConstructors(), to.getConstructors()); if (constructorDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setConstructorDeltas(constructorDeltas); } // methods Set methodDeltas = compareMethods(from, to); if (methodDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setMethodDeltas(methodDeltas); } // fields Set fieldDeltas = compareFields(from.getFields(), to .getFields()); if (fieldDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setFieldDeltas(fieldDeltas); } // enum constants if (from.getKind() == Kind.ENUM) { Set enumDeltas = compareEnumConstants(from .getEnumConstants(), to.getEnumConstants()); if (enumDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setEnumConstantDeltas(enumDeltas); } } else if (from.getKind() == Kind.ANNOTATION) { Set annotationFieldDeltas = compareAnnotationFields(from.getAnnotationFields(), to.getAnnotationFields()); if (annotationFieldDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setAnnotationFieldDeltas(annotationFieldDeltas); } } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (classDelta == null) { classDelta = new SigClassDefinitionDelta(from, to); } classDelta.setAnnotationDeltas(annotationDeltas); } return classDelta; } private Set> compareInterfaces( IClassDefinition from, IClassDefinition to) { Set fromClosure = getInterfaceClosure(from); Set toClosure = getInterfaceClosure(to); Set fromInterfaces = from.getInterfaces(); Set toInterfaces = to.getInterfaces(); Set> deltas = new HashSet>(); // check whether all from interfaces are directly or indirectly // implemented by the to method for (ITypeReference type : fromInterfaces) { if (!containsType(type, toInterfaces)) { if (!(containsType(type, toClosure) /* * && !containsType(type, * toInterfaces) */)) { deltas.add(new SigTypeDelta(type, null)); } } } // check whether all interfaces to are directly or indirectly // implemented by the from method for (ITypeReference type : toInterfaces) { if (!containsType(type, fromInterfaces)) { if (!(containsType(type, fromClosure) /* * && !containsType(type, * fromInterfaces) */)) { deltas.add(new SigTypeDelta(null, type)); } } } return deltas.isEmpty() ? null : deltas; } private boolean containsType(ITypeReference type, Set setOfTypes) { for (ITypeReference other : setOfTypes) { if (compareType(type, other, false) == null) { return true; } } return false; } private Set getInterfaceClosure(IClassDefinition clazz) { Set closure = new HashSet(); collectInterfaceClosure(ViewpointAdapter.getReferenceTo(clazz), closure); return closure; } private void collectInterfaceClosure(ITypeReference clazz, Set closure) { IClassDefinition classDefinition = getClassDefinition(clazz); Set interfaces = classDefinition.getInterfaces(); if (interfaces == null) { return; } for (ITypeReference interfaze : interfaces) { closure.add(interfaze); } ITypeReference superclass = classDefinition.getSuperClass(); if (superclass != null) { if (superclass instanceof IParameterizedType) { collectInterfaceClosure(((IParameterizedType) superclass) .getRawType(), closure); } else { collectInterfaceClosure(superclass, closure); } } for (ITypeReference interfaze : interfaces) { if (interfaze instanceof IParameterizedType) { collectInterfaceClosure(((IParameterizedType) interfaze) .getRawType(), closure); } else { collectInterfaceClosure(interfaze, closure); } } } private Set compareAnnotations(Set from, Set to) { return compareSets(from, to, new SigComparator() { public IAnnotationDelta createAddRemoveDelta( IAnnotation from, IAnnotation to) { return new SigAnnotationDelta(from, to); } public boolean considerEqualElement(IAnnotation from, IAnnotation to) { return sameClassDefinition(from.getType() .getClassDefinition(), to.getType() .getClassDefinition()); } public IAnnotationDelta createChangedDelta( IAnnotation from, IAnnotation to) { return compareAnnotation(from, to); } }); } private Set compareAnnotationFields( Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement(IAnnotationField from, IAnnotationField to) { return from.getName().equals(to.getName()); } public IAnnotationFieldDelta createAddRemoveDelta( IAnnotationField from, IAnnotationField to) { return new SigAnnotationFieldDelta(from, to); } public IAnnotationFieldDelta createChangedDelta( IAnnotationField from, IAnnotationField to) { return compareAnnotationField(from, to); } }); } private Set compareEnumConstants( Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement(IEnumConstant from, IEnumConstant to) { return from.getName().equals(to.getName()); } public IEnumConstantDelta createAddRemoveDelta( IEnumConstant from, IEnumConstant to) { return new SigEnumConstantDelta(from, to); } public IEnumConstantDelta createChangedDelta( IEnumConstant from, IEnumConstant to) { return compareEnumConstant(from, to); } }); } private Set compareFields(Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement(IField from, IField to) { return from.getName().equals(to.getName()); } public IFieldDelta createAddRemoveDelta(IField from, IField to) { return new SigFieldDelta(from, to); } public IFieldDelta createChangedDelta(IField from, IField to) { return compareField(from, to); } }); } private Set compareMethods(IClassDefinition from, IClassDefinition to) { assert from != null; assert to != null; Set toMethods = new HashSet(to.getMethods()); Set toClosure = getMethodClosure(to); Set fromMethods = new HashSet(from.getMethods()); Set fromClosure = getMethodClosure(from); Set deltas = new HashSet(); for (IMethod method : fromMethods) { IMethod compatibleMethod = findCompatibleMethod(method, toMethods); if (compatibleMethod == null) { compatibleMethod = findCompatibleMethod(method, toClosure); if (compatibleMethod == null) { deltas.add(new SigMethodDelta(method, null)); } } if (compatibleMethod != null) { IMethodDelta delta = compareMethod(method, compatibleMethod); if (delta != null) { deltas.add(delta); } } } for (IMethod method : toMethods) { IMethod compatibleMethod = findCompatibleMethod(method, fromMethods); if (compatibleMethod == null) { compatibleMethod = findCompatibleMethod(method, fromClosure); if (compatibleMethod == null) { deltas.add(new SigMethodDelta(null, method)); } } } return deltas.isEmpty() ? null : deltas; } private IMethod findCompatibleMethod(IMethod method, Set set) { for (IMethod methodFromSet : set) { if (equalsSignature(method, methodFromSet)) { return methodFromSet; } } return null; } private Set getMethodClosure(IClassDefinition clazz) { Set closure = new HashSet(); collectMethods(new ClassProjection(clazz, new HashMap()), closure); return closure; } private void collectMethods(IClassDefinition clazz, Set closure) { if (clazz == null) { return; } if (clazz.getMethods() != null) { closure.addAll(clazz.getMethods()); } if (clazz.getSuperClass() != null) { collectMethods(getClassDefinition(clazz.getSuperClass()), closure); } if (clazz.getInterfaces() != null) { for (ITypeReference interfaze : clazz.getInterfaces()) { collectMethods(getClassDefinition(interfaze), closure); } } } private Set compareConstructors(Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement(IConstructor from, IConstructor to) { return equalsSignature(from, to); } public IConstructorDelta createAddRemoveDelta( IConstructor from, IConstructor to) { return new SigConstructorDelta(from, to); } public IConstructorDelta createChangedDelta( IConstructor from, IConstructor to) { return compareConstructor(from, to); } }); } // compares names and parameter types private boolean equalsSignature(IExecutableMember from, IExecutableMember to) { if (from.getName().equals(to.getName())) { return compareTypeSequence(getParameterList(from.getParameters()), getParameterList(to.getParameters()), true) == null; } return false; } private List getParameterList(List parameters) { List parameterTypes = new LinkedList(); for (IParameter parameter : parameters) { parameterTypes.add(parameter.getType()); } return parameterTypes; } private IAnnotationDelta compareAnnotation(IAnnotation from, IAnnotation to) { assert sameClassDefinition(from.getType().getClassDefinition(), to .getType().getClassDefinition()); Set fromAnnotationElement = getNormalizedAnnotationElements(from); Set toAnnotationElement = getNormalizedAnnotationElements(to); Set annotationElementDeltas = compareAnnotationElements( fromAnnotationElement, toAnnotationElement); SigAnnotationDelta delta = null; if (annotationElementDeltas != null) { delta = new SigAnnotationDelta(from, to); delta.setAnnotationElementDeltas(annotationElementDeltas); } return delta; } /** * Returns the annotation elements for the given annotation. The returned * set contains all declared elements plus all elements with default values. * * @param annotation * the annotation to return the elements for * @return the default enriched annotation elements */ private Set getNormalizedAnnotationElements( IAnnotation annotation) { Set elements = new HashSet( annotation.getElements()); Set names = new HashSet(); for (IAnnotationElement annotationElement : elements) { names.add(annotationElement.getDeclaringField().getName()); } for (IAnnotationField field : annotation.getType().getClassDefinition() .getAnnotationFields()) { if (!names.contains(field.getName())) { SigAnnotationElement sigAnnotationElement = new SigAnnotationElement(); sigAnnotationElement.setDeclaringField(field); sigAnnotationElement.setValue(field.getDefaultValue()); elements.add(sigAnnotationElement); } } return elements; } private Set compareAnnotationElements( Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement( IAnnotationElement from, IAnnotationElement to) { return from.getDeclaringField().getName().equals( to.getDeclaringField().getName()); } public IAnnotationElementDelta createAddRemoveDelta( IAnnotationElement from, IAnnotationElement to) { return new SigAnnotationElementDelta(from, to); } public IAnnotationElementDelta createChangedDelta( IAnnotationElement from, IAnnotationElement to) { return compareAnnotationElement(from, to); } }); } private IAnnotationElementDelta compareAnnotationElement( IAnnotationElement from, IAnnotationElement to) { SigAnnotationElementDelta delta = null; SigValueDelta valueDelta = compareValue(from.getValue(), to.getValue()); if (valueDelta != null) { delta = new SigAnnotationElementDelta(from, to); delta.setValueDelta(valueDelta); } return delta; } /** * Removes the {@link Modifier#ABSTRACT} modifier. */ private Set prepareMethodModifiers(IMethod method) { Set modifierCopy = new HashSet(method .getModifiers()); modifierCopy.remove(Modifier.ABSTRACT); return modifierCopy; } private IMethodDelta compareMethod(IMethod from, IMethod to) { assert from != null && to != null; SigMethodDelta methodDelta = null; Set modiferDeltas = compareModifiers( prepareMethodModifiers(from), prepareMethodModifiers(to)); if (modiferDeltas != null) { methodDelta = new SigMethodDelta(from, to); methodDelta.setModifierDeltas(modiferDeltas); } Set parameterDeltas = compareParameterSequence(from .getParameters(), to.getParameters()); if (parameterDeltas != null) { if (methodDelta == null) { methodDelta = new SigMethodDelta(from, to); } methodDelta.setParameterDeltas(parameterDeltas); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (methodDelta == null) { methodDelta = new SigMethodDelta(from, to); } methodDelta.setAnnotationDeltas(annotationDeltas); } Set typeParameterDeltas = compareTypeVariableSequence(from.getTypeParameters(), to.getTypeParameters()); if (typeParameterDeltas != null) { if (methodDelta == null) { methodDelta = new SigMethodDelta(from, to); } methodDelta.setTypeVariableDeltas(typeParameterDeltas); } Set> exceptionDeltas = compareTypes( normalizeExceptions(from.getExceptions()), normalizeExceptions(to.getExceptions())); if (exceptionDeltas != null) { if (methodDelta == null) { methodDelta = new SigMethodDelta(from, to); } methodDelta.setExceptionDeltas(exceptionDeltas); } ITypeReferenceDelta returnTypeDelta = compareType(from .getReturnType(), to.getReturnType(), false); if (returnTypeDelta != null) { if (methodDelta == null) { methodDelta = new SigMethodDelta(from, to); } methodDelta.setReturnTypeDelta(returnTypeDelta); } return methodDelta; } // remove runtime exceptions, // remove sub types of containing exception private Set normalizeExceptions( Set exceptions) { Set exceptionCopy = new HashSet( exceptions); Iterator iterator = exceptionCopy.iterator(); while (iterator.hasNext()) { ITypeReference exception = iterator.next(); if (isRuntimeExceptionOrErrorSubtype(exception)) { iterator.remove(); } } exceptionCopy = removeSpecializations(exceptionCopy); return exceptionCopy; } private Set removeSpecializations( Set exceptions) { Set exceptionCopy = new HashSet( exceptions); for (ITypeReference type : exceptions) { Iterator it = exceptionCopy.iterator(); while (it.hasNext()) { ITypeReference subType = it.next(); if (isSuperClass(getClassDefinition(type), getClassDefinition(subType))) { it.remove(); } } } return exceptionCopy; } /** * Returns true if superC is a super class of subC. */ private boolean isSuperClass(IClassDefinition superC, IClassDefinition subC) { if (superC == null || subC == null) { return false; } if (subC.getSuperClass() == null) { return false; } else { if (getClassDefinition(subC.getSuperClass()).equals(superC)) { return true; } else { return isSuperClass(superC, getClassDefinition(subC .getSuperClass())); } } } private boolean isSuperInterface(IClassDefinition superClass, IClassDefinition subClass) { if (superClass == null || subClass == null) { return false; } if (subClass.getInterfaces() == null) { return false; } else { if (getClassDefinitions(subClass.getInterfaces()).contains( superClass)) { return true; } else { for (ITypeReference subType : subClass.getInterfaces()) { if (isSuperInterface(superClass, getClassDefinition(subType))) { return true; } } return false; } } } private Set getClassDefinitions( Set references) { Set definitions = new HashSet(); for (ITypeReference ref : references) { definitions.add(getClassDefinition(ref)); } return definitions; } /** * Returns null if type is not one of: *
    *
  • IClassReference
  • *
  • IParameterizedType
  • *
*/ private IClassDefinition getClassDefinition(ITypeReference type) { assert type != null; IClassDefinition returnValue = null; if (type instanceof IClassReference) { returnValue = ((IClassReference) type).getClassDefinition(); } else if (type instanceof IParameterizedType) { returnValue = ((IParameterizedType) type).getRawType() .getClassDefinition(); } return returnValue; } private boolean isRuntimeExceptionOrErrorSubtype(ITypeReference exception) { IClassDefinition clazz = getClassDefinition(exception); if (clazz != null) { if (isRuntimeExceptionOrError(clazz)) { return true; } else if (clazz.getSuperClass() != null) { return isRuntimeExceptionOrErrorSubtype(clazz.getSuperClass()); } else { return false; } } return false; } private boolean isRuntimeExceptionOrError(IClassDefinition exception) { if (exception == null) { return false; } String packageName = exception.getPackageName(); String className = exception.getName(); if (packageName != null && className != null && "java.lang".equals(packageName)) { return "RuntimeException".equals(className) || "Error".equals(className); } return false; } private IConstructorDelta compareConstructor(IConstructor from, IConstructor to) { SigConstructorDelta constructorDelta = null; Set modiferDeltas = compareModifiers(from .getModifiers(), to.getModifiers()); if (modiferDeltas != null) { constructorDelta = new SigConstructorDelta(from, to); constructorDelta.setModifierDeltas(modiferDeltas); } Set parameterDeltas = compareParameterSequence(from .getParameters(), to.getParameters()); if (parameterDeltas != null) { if (constructorDelta == null) { constructorDelta = new SigConstructorDelta(from, to); } constructorDelta.setParameterDeltas(parameterDeltas); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (constructorDelta == null) { constructorDelta = new SigConstructorDelta(from, to); } constructorDelta.setAnnotationDeltas(annotationDeltas); } Set typeParameterDeltas = compareTypeVariableSequence(from.getTypeParameters(), to.getTypeParameters()); if (typeParameterDeltas != null) { if (constructorDelta == null) { constructorDelta = new SigConstructorDelta(from, to); } constructorDelta.setTypeVariableDeltas(typeParameterDeltas); } Set> exceptionDeltas = compareTypes( normalizeExceptions(from.getExceptions()), normalizeExceptions(to.getExceptions())); if (exceptionDeltas != null) { if (constructorDelta == null) { constructorDelta = new SigConstructorDelta(from, to); } constructorDelta.setExceptionDeltas(exceptionDeltas); } return constructorDelta; } private Set compareParameterSequence( List from, List to) { assert from.size() == to.size(); Set deltas = new HashSet(); Iterator fromIterator = from.iterator(); Iterator toIterator = to.iterator(); while (fromIterator.hasNext() && toIterator.hasNext()) { IParameterDelta delta = compareParameter(fromIterator.next(), toIterator.next()); if (delta != null) { deltas.add(delta); } } return deltas.isEmpty() ? null : deltas; } private IParameterDelta compareParameter(IParameter from, IParameter to) { SigParameterDelta delta = null; ITypeReferenceDelta typeDelta = compareType(from.getType(), to .getType(), false); if (typeDelta != null) { if (delta == null) { delta = new SigParameterDelta(from, to); } delta.setTypeDelta(typeDelta); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (delta == null) { delta = new SigParameterDelta(from, to); } delta.setAnnotationDeltas(annotationDeltas); } return delta; } private Set compareTypeVariableSequence( List from, List to) { Set deltas = new HashSet(); if (from.size() != to.size()) { for (ITypeVariableDefinition fromVariable : from) { deltas.add(new SigTypeVariableDefinitionDelta(fromVariable, null)); } for (ITypeVariableDefinition toVariable : to) { deltas .add(new SigTypeVariableDefinitionDelta(null, toVariable)); } } Iterator fromIterator = from.iterator(); Iterator toIterator = to.iterator(); while (fromIterator.hasNext() && toIterator.hasNext()) { ITypeVariableDefinitionDelta delta = compareTypeVariableDefinition( fromIterator.next(), toIterator.next()); if (delta != null) { deltas.add(delta); } } return deltas.isEmpty() ? null : deltas; } private ITypeVariableDefinitionDelta compareTypeVariableDefinition( ITypeVariableDefinition from, ITypeVariableDefinition to) { IGenericDeclarationDelta declarationDelta = compareGenericDeclaration( from, to); if (declarationDelta != null) { SigTypeVariableDefinitionDelta delta = new SigTypeVariableDefinitionDelta(from, to); delta.setGenericDeclarationDelta(declarationDelta); return delta; } IUpperBoundsDelta upperBoundDelta = compareUpperBounds(from .getUpperBounds(), to.getUpperBounds()); if (upperBoundDelta != null) { SigTypeVariableDefinitionDelta delta = new SigTypeVariableDefinitionDelta(from, to); delta.setUpperBoundsDelta(upperBoundDelta); return delta; } return null; } private ITypeReferenceDelta compareTypeVariableReference( ITypeVariableReference from, ITypeVariableReference to) { IGenericDeclarationDelta declarationDelta = compareGenericDeclaration( from.getTypeVariableDefinition(), to .getTypeVariableDefinition()); if (declarationDelta != null) { SigTypeVariableReferenceDelta delta = new SigTypeVariableReferenceDelta(from, to); delta.setGenericDeclarationDelta(declarationDelta); return delta; } return null; } private Set compareModifiers(Set from, Set to) { return compareSets(from, to, new SigComparator() { public boolean considerEqualElement(Modifier from, Modifier to) { return from.equals(to); } public IModifierDelta createAddRemoveDelta(Modifier from, Modifier to) { return new SigModifierDelta(from, to); } public IModifierDelta createChangedDelta(Modifier from, Modifier to) { return null; } }); } private IFieldDelta compareField(IField from, IField to) { SigFieldDelta fieldDelta = null; Set modiferDeltas = compareModifiers(from .getModifiers(), to.getModifiers()); if (modiferDeltas != null) { fieldDelta = new SigFieldDelta(from, to); fieldDelta.setModifierDeltas(modiferDeltas); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (fieldDelta == null) { fieldDelta = new SigFieldDelta(from, to); } fieldDelta.setAnnotationDeltas(annotationDeltas); } ITypeReferenceDelta typeDelta = compareType(from.getType(), to .getType(), false); if (typeDelta != null) { if (fieldDelta == null) { fieldDelta = new SigFieldDelta(from, to); } fieldDelta.setTypeDelta(typeDelta); } return fieldDelta; } private IEnumConstantDelta compareEnumConstant(IEnumConstant from, IEnumConstant to) { SigEnumConstantDelta enumConstantDelta = null; Set modiferDeltas = compareModifiers(from .getModifiers(), to.getModifiers()); if (modiferDeltas != null) { enumConstantDelta = new SigEnumConstantDelta(from, to); enumConstantDelta.setModifierDeltas(modiferDeltas); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (enumConstantDelta == null) { enumConstantDelta = new SigEnumConstantDelta(from, to); } enumConstantDelta.setAnnotationDeltas(annotationDeltas); } ITypeReferenceDelta typeDelta = compareType(from.getType(), to .getType(), false); if (typeDelta != null) { if (enumConstantDelta == null) { enumConstantDelta = new SigEnumConstantDelta(from, to); } enumConstantDelta.setTypeDelta(typeDelta); } // FIXME ordinal not supported in dex // ValueDelta ordinalDelta = compareValue(from.getOrdinal(), // to.getOrdinal()); // if (ordinalDelta != null) { // if (enumConstantDelta == null) { // enumConstantDelta = new SigEnumConstantDelta(from, to); // } // enumConstantDelta.setOrdinalDelta(ordinalDelta); // } return enumConstantDelta; } private IAnnotationFieldDelta compareAnnotationField(IAnnotationField from, IAnnotationField to) { SigAnnotationFieldDelta annotationFieldDelta = null; Set modiferDeltas = compareModifiers(from .getModifiers(), to.getModifiers()); if (modiferDeltas != null) { annotationFieldDelta = new SigAnnotationFieldDelta(from, to); annotationFieldDelta.setModifierDeltas(modiferDeltas); } Set annotationDeltas = compareAnnotations(from .getAnnotations(), to.getAnnotations()); if (annotationDeltas != null) { if (annotationFieldDelta == null) { annotationFieldDelta = new SigAnnotationFieldDelta(from, to); } annotationFieldDelta.setAnnotationDeltas(annotationDeltas); } ITypeReferenceDelta typeDelta = compareType(from.getType(), to .getType(), false); if (typeDelta != null) { if (annotationFieldDelta == null) { annotationFieldDelta = new SigAnnotationFieldDelta(from, to); } annotationFieldDelta.setTypeDelta(typeDelta); } IValueDelta defaultValueDelta = compareValue(from.getDefaultValue(), to .getDefaultValue()); if (defaultValueDelta != null) { if (annotationFieldDelta == null) { annotationFieldDelta = new SigAnnotationFieldDelta(from, to); } annotationFieldDelta.setDefaultValueDelta(defaultValueDelta); } return annotationFieldDelta; } private SigValueDelta compareValue(Object from, Object to) { // same value if (from == null && to == null) { return null; } // one of both is null and other is not if (from == null || to == null) { return new SigValueDelta(from, to); } SigValueDelta delta = null; // different types if (from.getClass() == to.getClass()) { if (from.getClass().isArray()) { Object[] fromArray = (Object[]) from; Object[] toArray = (Object[]) from; if (!Arrays.equals(fromArray, toArray)) { delta = new SigValueDelta(from, to); } } else if (from instanceof IEnumConstant) { IEnumConstantDelta enumConstantDelta = compareEnumConstant( (IEnumConstant) from, (IEnumConstant) to); if (enumConstantDelta != null) { delta = new SigValueDelta(from, to); } } else if (from instanceof IAnnotation) { IAnnotationDelta annotationDelta = compareAnnotation( (IAnnotation) from, (IAnnotation) to); if (annotationDelta != null) { delta = new SigValueDelta(from, to); } } else if (from instanceof IField) { IFieldDelta fieldDelta = compareField((IField) from, (IField) to); if (fieldDelta != null) { delta = new SigValueDelta(from, to); } } else if (from instanceof ITypeReference) { ITypeReferenceDelta typeDelta = compareType((ITypeReference) from, (ITypeReference) to, false); if (typeDelta != null) { delta = new SigValueDelta(from, to); } } else if (!from.equals(to)) { delta = new SigValueDelta(from, to); } } else if (!(from == null && to == null)) { delta = new SigValueDelta(from, to); } return delta; } private boolean considerEqualTypes(ITypeReference from, ITypeReference to) { assert from != null && to != null; if (implementInterface(from, to, IPrimitiveType.class)) { return comparePrimitiveType((IPrimitiveType) from, (IPrimitiveType) to) == null; } if (implementInterface(from, to, IClassReference.class)) { return sameClassDefinition(((IClassReference) from) .getClassDefinition(), ((IClassReference) to) .getClassDefinition()); } if (implementInterface(from, to, IArrayType.class)) { return considerEqualTypes(((IArrayType) from).getComponentType(), ((IArrayType) to).getComponentType()); } if (implementInterface(from, to, IParameterizedType.class)) { return compareClassReference(((IParameterizedType) from) .getRawType(), ((IParameterizedType) to) .getRawType()) == null; } if (implementInterface(from, to, ITypeVariableReference.class)) { return compareTypeVariableReference((ITypeVariableReference) from, (ITypeVariableReference) to) == null; } return false; } private Set fromComparison = new HashSet(); private Set toComparison = new HashSet(); private boolean areInComparison(ITypeReference from, ITypeReference to) { return fromComparison.contains(from) && toComparison.contains(to); } private void markInComparison(ITypeReference from, ITypeReference to) { fromComparison.add(from); toComparison.add(to); } private void markFinishedComparison(ITypeReference from, ITypeReference to) { fromComparison.remove(from); toComparison.remove(to); } private ITypeReferenceDelta compareType( ITypeReference from, ITypeReference to, boolean acceptErasedTypes) { if (from == null && to == null) { return null; } if ((from == null && to != null) || (from != null && to == null)) { return new SigTypeDelta(from, to); } if (areInComparison(from, to)) { return null; } try { markInComparison(from, to); if (implementInterface(from, to, IPrimitiveType.class)) { return comparePrimitiveType((IPrimitiveType) from, (IPrimitiveType) to); } if (implementInterface(from, to, IClassReference.class)) { return compareClassReference((IClassReference) from, (IClassReference) to); } if (implementInterface(from, to, IArrayType.class)) { return compareArrayType((IArrayType) from, (IArrayType) to); } if (implementInterface(from, to, IParameterizedType.class)) { return compareParameterizedType((IParameterizedType) from, (IParameterizedType) to, acceptErasedTypes); } if (implementInterface(from, to, ITypeVariableReference.class)) { return compareTypeVariableReference( (ITypeVariableReference) from, (ITypeVariableReference) to); } if (implementInterface(from, to, IWildcardType.class)) { return compareWildcardType((IWildcardType) from, (IWildcardType) to); } if (acceptErasedTypes) { if (isGeneric(from) && !isGeneric(to)) { return compareType(getErasedType(from), to, false); } if (!isGeneric(from) && isGeneric(to)) { return compareType(from, getErasedType(to), false); } } return new SigTypeDelta(from, to); } finally { markFinishedComparison(from, to); } } private boolean isGeneric(ITypeReference reference) { if (reference instanceof IParameterizedType || reference instanceof ITypeVariableReference || reference instanceof IWildcardType) { return true; } if (reference instanceof IArrayType) { return isGeneric(((IArrayType) reference).getComponentType()); } return false; } private ITypeReference getErasedType(ITypeReference reference) { if (reference instanceof IParameterizedType) { return ((IParameterizedType) reference).getRawType(); } if (reference instanceof ITypeVariableReference) { ITypeVariableDefinition typeVariableDefinition = ((ITypeVariableReference) reference) .getTypeVariableDefinition(); return getErasedType( typeVariableDefinition.getUpperBounds().get(0)); } if (reference instanceof IWildcardType) { return getErasedType(((IWildcardType) reference).getUpperBounds() .get(0)); } if (reference instanceof IArrayType) { // FIXME implement with erasure projection? return new SigArrayType(getErasedType(((IArrayType) reference) .getComponentType())); } if (reference instanceof IPrimitiveType) { return reference; } if (reference instanceof IClassReference) { return reference; } throw new IllegalArgumentException("Unexpected type: " + reference); } private boolean implementInterface(ITypeReference from, ITypeReference to, Class check) { return check.isAssignableFrom(from.getClass()) && check.isAssignableFrom(to.getClass()); } private IWildcardTypeDelta compareWildcardType(IWildcardType from, IWildcardType to) { SigWildcardTypeDelta delta = null; ITypeReference fromLowerBound = from.getLowerBound(); ITypeReference toLowerBound = to.getLowerBound(); ITypeReferenceDelta lowerBoundDelta = compareType(fromLowerBound, toLowerBound, false); if (lowerBoundDelta != null) { delta = new SigWildcardTypeDelta(from, to); delta.setLowerBoundDelta(lowerBoundDelta); } IUpperBoundsDelta upperBoundsDelta = compareUpperBounds(from .getUpperBounds(), to.getUpperBounds()); if (upperBoundsDelta != null) { if (delta == null) { delta = new SigWildcardTypeDelta(from, to); } delta.setUpperBoundDelta(upperBoundsDelta); } return delta; } private IGenericDeclarationDelta compareGenericDeclaration( ITypeVariableDefinition fromVariable, ITypeVariableDefinition toVariable) { IGenericDeclarationDelta delta = null; IGenericDeclaration from = fromVariable.getGenericDeclaration(); IGenericDeclaration to = toVariable.getGenericDeclaration(); if (from != null && to != null) { if (from.getClass() != to.getClass()) { delta = new SigGenericDeclarationDelta(from, to); } else if (from instanceof IClassDefinition) { IClassDefinition fromDeclaringClass = (IClassDefinition) from; IClassDefinition toDeclaringClass = (IClassDefinition) to; if (!sameClassDefinition(fromDeclaringClass, toDeclaringClass)) { delta = new SigGenericDeclarationDelta(from, to); } } else if (from instanceof IConstructor) { IConstructor fromConstructor = (IConstructor) from; IConstructor toConstructor = (IConstructor) from; String fromConstructorName = fromConstructor.getName(); String fromClassName = fromConstructor.getDeclaringClass() .getQualifiedName(); String toConstructorName = toConstructor.getName(); String toClassName = toConstructor.getDeclaringClass() .getQualifiedName(); if ((!fromConstructorName.equals(toConstructorName)) || (!fromClassName.equals(toClassName))) { delta = new SigGenericDeclarationDelta(from, to); } } else if (from instanceof IMethod) { IMethod fromMethod = (IMethod) from; IMethod toMethod = (IMethod) from; String fromConstructorName = fromMethod.getName(); String fromClassName = fromMethod.getDeclaringClass() .getQualifiedName(); String toConstructorName = toMethod.getName(); String toClassName = toMethod.getDeclaringClass() .getQualifiedName(); if ((!fromConstructorName.equals(toConstructorName)) || (!fromClassName.equals(toClassName))) { delta = new SigGenericDeclarationDelta(from, to); } } else { throw new IllegalStateException("Invlaid eclaration site: " + from); } // check position int fromPosition = getPositionOf(fromVariable, from); int toPosition = getPositionOf(toVariable, to); if (fromPosition != toPosition) { delta = new SigGenericDeclarationDelta(from, to); } } else { // one of both is null delta = new SigGenericDeclarationDelta(from, to); } return delta; } private int getPositionOf(ITypeVariableDefinition variable, IGenericDeclaration declaration) { return declaration.getTypeParameters().indexOf(variable); } private IUpperBoundsDelta compareUpperBounds(List from, List to) { if (from.isEmpty() && to.isEmpty()) { return null; } SigUpperBoundsDelta delta = null; ITypeReference fromFirstUpperBound = from.get(0); ITypeReference toFirstUpperBound = to.get(0); ITypeReferenceDelta firstUpperBoundDelta = compareType( fromFirstUpperBound, toFirstUpperBound, false); if (firstUpperBoundDelta != null) { delta = new SigUpperBoundsDelta(from, to); delta.setFirstUpperBoundDelta(firstUpperBoundDelta); } else { // normalize Set normalizedfrom = removeGeneralizations( new HashSet(from)); Set normalizedto = removeGeneralizations( new HashSet(to)); Set> remainingUpperBoundsDelta = compareTypes(normalizedfrom, normalizedto); if (remainingUpperBoundsDelta != null) { delta = new SigUpperBoundsDelta(from, to); delta.setRemainingUpperBoundDeltas(remainingUpperBoundsDelta); } } return delta; } private Set removeGeneralizations( Set bounds) { Set boundsCopy = new HashSet(bounds); for (ITypeReference type : bounds) { Iterator it = boundsCopy.iterator(); while (it.hasNext()) { ITypeReference superType = it.next(); if (isSuperClass(getClassDefinition(superType), getClassDefinition(type)) || isSuperInterface(getClassDefinition(superType), getClassDefinition(type))) { it.remove(); } } } return boundsCopy; } private IParameterizedTypeDelta compareParameterizedType( IParameterizedType from, IParameterizedType to, boolean ignoreTypeArguments) { SigParameterizedTypeDelta delta = null; // check raw type ITypeReferenceDelta rawTypeDelta = compareType(from.getRawType(), to .getRawType(), false); if (rawTypeDelta != null) { delta = new SigParameterizedTypeDelta(from, to); delta.setRawTypeDelta(rawTypeDelta); } else { // check owner type ITypeReferenceDelta ownerTypeDelta = compareType(from .getOwnerType(), to.getOwnerType(), false); if (ownerTypeDelta != null) { delta = new SigParameterizedTypeDelta(from, to); delta.setOwnerTypeDelta(ownerTypeDelta); } else { // check argument type if (!ignoreTypeArguments) { Set> argumentTypeDeltas = compareTypeSequence(from.getTypeArguments(), to.getTypeArguments(), false); if (argumentTypeDeltas != null) { delta = new SigParameterizedTypeDelta(from, to); delta.setArgumentTypeDeltas(argumentTypeDeltas); } } } } return delta; } private Set> compareTypeSequence( List from, List to, boolean ignoreTypeArguments) { Set> deltas = new HashSet>(); if (from.size() != to.size()) { for (ITypeReference type : from) { deltas.add(new SigTypeDelta(type, null)); } for (ITypeReference type : to) { deltas.add(new SigTypeDelta(null, type)); } return deltas; } Iterator fromIterator = from.iterator(); Iterator toIterator = to.iterator(); while (fromIterator.hasNext() && toIterator.hasNext()) { ITypeReferenceDelta delta = compareType(fromIterator.next(), toIterator.next(), ignoreTypeArguments); if (delta != null) { deltas.add(delta); } } return deltas.isEmpty() ? null : deltas; } private Set> compareTypes( Set from, Set to) { return compareSets(from, to, new SigComparator>() { public ITypeReferenceDelta createAddRemoveDelta( ITypeReference from, ITypeReference to) { return new SigTypeDelta(from, to); } public boolean considerEqualElement(ITypeReference from, ITypeReference to) { return considerEqualTypes(from, to); } public ITypeReferenceDelta createChangedDelta( ITypeReference from, ITypeReference to) { return compareType(from, to, false); } }); } private static interface SigComparator> { boolean considerEqualElement(T from, T to); S createChangedDelta(T from, T to); /** * If null is returned, it will be ignored. */ S createAddRemoveDelta(T from, T to); } private > Set compareSets(Set from, Set to, SigComparator comparator) { Set toCopy = new HashSet(to); Set deltas = new HashSet(); for (T fromType : from) { Iterator toIterator = toCopy.iterator(); boolean equals = false; boolean hasNext = toIterator.hasNext(); while (hasNext && !equals) { T toElement = toIterator.next(); equals = comparator.considerEqualElement(fromType, toElement); if (equals) { S compare = comparator.createChangedDelta(fromType, toElement); if (compare != null) { deltas.add(compare); } } hasNext = toIterator.hasNext(); } if (equals) { toIterator.remove(); } else { S delta = comparator.createAddRemoveDelta(fromType, null); if (delta != null) { deltas.add(delta); } } } for (T type : toCopy) { S delta = comparator.createAddRemoveDelta(null, type); if (delta != null) { deltas.add(delta); } } return deltas.isEmpty() ? null : deltas; } private ITypeReferenceDelta compareArrayType(IArrayType from, IArrayType to) { ITypeReferenceDelta componentTypeDelta = compareType(from .getComponentType(), to.getComponentType(), false); if (componentTypeDelta != null) { SigArrayTypeDelta delta = new SigArrayTypeDelta(from, to); delta.setComponentTypeDelta(componentTypeDelta); return delta; } return null; } private ITypeReferenceDelta compareClassReference( IClassReference fromRef, IClassReference toRef) { IClassDefinition from = fromRef.getClassDefinition(); IClassDefinition to = toRef.getClassDefinition(); if (!sameClassDefinition(from, to)) { return new SigClassReferenceDelta(fromRef, toRef); } return null; } private boolean sameClassDefinition(IClassDefinition from, IClassDefinition to) { boolean sameName = from.getName().equals(to.getName()); boolean samePackage = from.getPackageName().equals(to.getPackageName()); Kind fromKind = from.getKind(); Kind toKind = to.getKind(); boolean sameKind = (fromKind == null || toKind == null) || fromKind.equals(toKind); return sameName && samePackage && sameKind; } private IPrimitiveTypeDelta comparePrimitiveType(IPrimitiveType from, IPrimitiveType to) { if (!from.equals(to)) { return new SigPrimitiveTypeDelta(from, to); } return null; } }