• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package signature.compare;
18 
19 import signature.compare.model.IAnnotationDelta;
20 import signature.compare.model.IAnnotationElementDelta;
21 import signature.compare.model.IAnnotationFieldDelta;
22 import signature.compare.model.IApiDelta;
23 import signature.compare.model.IClassDefinitionDelta;
24 import signature.compare.model.IConstructorDelta;
25 import signature.compare.model.IDelta;
26 import signature.compare.model.IEnumConstantDelta;
27 import signature.compare.model.IFieldDelta;
28 import signature.compare.model.IGenericDeclarationDelta;
29 import signature.compare.model.IMethodDelta;
30 import signature.compare.model.IModifierDelta;
31 import signature.compare.model.IPackageDelta;
32 import signature.compare.model.IParameterDelta;
33 import signature.compare.model.IParameterizedTypeDelta;
34 import signature.compare.model.IPrimitiveTypeDelta;
35 import signature.compare.model.ITypeReferenceDelta;
36 import signature.compare.model.ITypeVariableDefinitionDelta;
37 import signature.compare.model.IUpperBoundsDelta;
38 import signature.compare.model.IValueDelta;
39 import signature.compare.model.IWildcardTypeDelta;
40 import signature.compare.model.impl.SigAnnotationDelta;
41 import signature.compare.model.impl.SigAnnotationElementDelta;
42 import signature.compare.model.impl.SigAnnotationFieldDelta;
43 import signature.compare.model.impl.SigApiDelta;
44 import signature.compare.model.impl.SigArrayTypeDelta;
45 import signature.compare.model.impl.SigClassDefinitionDelta;
46 import signature.compare.model.impl.SigClassReferenceDelta;
47 import signature.compare.model.impl.SigConstructorDelta;
48 import signature.compare.model.impl.SigEnumConstantDelta;
49 import signature.compare.model.impl.SigFieldDelta;
50 import signature.compare.model.impl.SigGenericDeclarationDelta;
51 import signature.compare.model.impl.SigMethodDelta;
52 import signature.compare.model.impl.SigModifierDelta;
53 import signature.compare.model.impl.SigPackageDelta;
54 import signature.compare.model.impl.SigParameterDelta;
55 import signature.compare.model.impl.SigParameterizedTypeDelta;
56 import signature.compare.model.impl.SigPrimitiveTypeDelta;
57 import signature.compare.model.impl.SigTypeDelta;
58 import signature.compare.model.impl.SigTypeVariableDefinitionDelta;
59 import signature.compare.model.impl.SigTypeVariableReferenceDelta;
60 import signature.compare.model.impl.SigUpperBoundsDelta;
61 import signature.compare.model.impl.SigValueDelta;
62 import signature.compare.model.impl.SigWildcardTypeDelta;
63 import signature.compare.model.subst.ClassProjection;
64 import signature.compare.model.subst.ViewpointAdapter;
65 import signature.model.IAnnotation;
66 import signature.model.IAnnotationElement;
67 import signature.model.IAnnotationField;
68 import signature.model.IApi;
69 import signature.model.IArrayType;
70 import signature.model.IClassDefinition;
71 import signature.model.IClassReference;
72 import signature.model.IConstructor;
73 import signature.model.IEnumConstant;
74 import signature.model.IExecutableMember;
75 import signature.model.IField;
76 import signature.model.IGenericDeclaration;
77 import signature.model.IMethod;
78 import signature.model.IPackage;
79 import signature.model.IParameter;
80 import signature.model.IParameterizedType;
81 import signature.model.IPrimitiveType;
82 import signature.model.ITypeReference;
83 import signature.model.ITypeVariableDefinition;
84 import signature.model.ITypeVariableReference;
85 import signature.model.IWildcardType;
86 import signature.model.Kind;
87 import signature.model.Modifier;
88 import signature.model.impl.SigAnnotationElement;
89 import signature.model.impl.SigArrayType;
90 
91 import java.util.Arrays;
92 import java.util.HashMap;
93 import java.util.HashSet;
94 import java.util.Iterator;
95 import java.util.LinkedList;
96 import java.util.List;
97 import java.util.Set;
98 
99 /**
100  * {@code ApiComparator} takes two signature models as input and creates a delta
101  * model describing the differences between those.
102  */
103 public class ApiComparator implements IApiComparator {
104 
compare(IApi from, IApi to)105     public IApiDelta compare(IApi from, IApi to) {
106         assert from.getVisibility() == to.getVisibility();
107 
108         Set<IPackage> fromPackages = from.getPackages();
109         Set<IPackage> toPackages = to.getPackages();
110 
111         Set<IPackageDelta> packageDeltas = compareSets(fromPackages,
112                 toPackages, new SigComparator<IPackage, IPackageDelta>() {
113                     public IPackageDelta createChangedDelta(IPackage from,
114                             IPackage to) {
115                         return comparePackage(from, to);
116                     }
117 
118                     public IPackageDelta createAddRemoveDelta(IPackage from,
119                             IPackage to) {
120                         return new SigPackageDelta(from, to);
121                     }
122 
123                     public boolean considerEqualElement(IPackage from,
124                             IPackage to) {
125                         return from.getName().equals(to.getName());
126                     }
127                 });
128 
129         SigApiDelta delta = null;
130         if (packageDeltas != null) {
131             delta = new SigApiDelta(from, to);
132             delta.setPackageDeltas(packageDeltas);
133         }
134         return delta;
135     }
136 
comparePackage(IPackage from, IPackage to)137     private IPackageDelta comparePackage(IPackage from, IPackage to) {
138         assert from.getName().equals(to.getName());
139 
140         Set<IClassDefinition> fromClasses = from.getClasses();
141         Set<IClassDefinition> toClasses = to.getClasses();
142 
143         Set<IClassDefinitionDelta> classDeltas = compareSets(fromClasses,
144                 toClasses,
145                 new SigComparator<IClassDefinition, IClassDefinitionDelta>() {
146                     public boolean considerEqualElement(IClassDefinition from,
147                             IClassDefinition to) {
148                         return sameClassDefinition(from, to);
149                     }
150 
151                     public IClassDefinitionDelta createChangedDelta(
152                             IClassDefinition from, IClassDefinition to) {
153                         return compareClass(from, to);
154                     }
155 
156                     public IClassDefinitionDelta createAddRemoveDelta(
157                             IClassDefinition from, IClassDefinition to) {
158                         return new SigClassDefinitionDelta(from, to);
159                     }
160                 });
161 
162         SigPackageDelta delta = null;
163         if (classDeltas != null) {
164             delta = new SigPackageDelta(from, to);
165             delta.setClassDeltas(classDeltas);
166         }
167 
168         // Annotations
169         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
170                 .getAnnotations(), to.getAnnotations());
171         if (annotationDeltas != null) {
172             if (delta != null) {
173                 delta = new SigPackageDelta(from, to);
174             }
175             delta.setAnnotationDeltas(annotationDeltas);
176         }
177         return delta;
178     }
179 
compareClass(IClassDefinition from, IClassDefinition to)180     private IClassDefinitionDelta compareClass(IClassDefinition from,
181             IClassDefinition to) {
182         assert from.getKind() == to.getKind();
183         assert from.getName().equals(to.getName());
184         assert from.getPackageName().equals(to.getPackageName());
185 
186         SigClassDefinitionDelta classDelta = null;
187 
188         // modifiers
189         Set<IModifierDelta> modifierDeltas = compareModifiers(from
190                 .getModifiers(), to.getModifiers());
191         if (modifierDeltas != null) {
192             if (classDelta == null) {
193                 classDelta = new SigClassDefinitionDelta(from, to);
194             }
195             classDelta.setModifierDeltas(modifierDeltas);
196         }
197 
198         // super class
199         ITypeReferenceDelta<?> superTypeDelta = compareType(from
200                 .getSuperClass(), to.getSuperClass(), false);
201         if (superTypeDelta != null) {
202             if (classDelta == null) {
203                 classDelta = new SigClassDefinitionDelta(from, to);
204             }
205             classDelta.setSuperClassDelta(superTypeDelta);
206         }
207 
208         // interfaces
209         Set<ITypeReferenceDelta<?>> interfaceDeltas = compareInterfaces(from,
210                 to);
211         if (interfaceDeltas != null) {
212             if (classDelta == null) {
213                 classDelta = new SigClassDefinitionDelta(from, to);
214             }
215             classDelta.setInterfaceDeltas(interfaceDeltas);
216         }
217 
218         // type variables
219         Set<ITypeVariableDefinitionDelta> typeVariableDeltas =
220                 compareTypeVariableSequence(from.getTypeParameters(),
221                         to.getTypeParameters());
222         if (typeVariableDeltas != null) {
223             if (classDelta == null) {
224                 classDelta = new SigClassDefinitionDelta(from, to);
225             }
226             classDelta.setTypeVariableDeltas(typeVariableDeltas);
227         }
228 
229         // constructors
230         Set<IConstructorDelta> constructorDeltas = compareConstructors(from
231                 .getConstructors(), to.getConstructors());
232         if (constructorDeltas != null) {
233             if (classDelta == null) {
234                 classDelta = new SigClassDefinitionDelta(from, to);
235             }
236             classDelta.setConstructorDeltas(constructorDeltas);
237         }
238 
239         // methods
240         Set<IMethodDelta> methodDeltas = compareMethods(from, to);
241         if (methodDeltas != null) {
242             if (classDelta == null) {
243                 classDelta = new SigClassDefinitionDelta(from, to);
244             }
245             classDelta.setMethodDeltas(methodDeltas);
246         }
247 
248         // fields
249         Set<IFieldDelta> fieldDeltas = compareFields(from.getFields(), to
250                 .getFields());
251         if (fieldDeltas != null) {
252             if (classDelta == null) {
253                 classDelta = new SigClassDefinitionDelta(from, to);
254             }
255             classDelta.setFieldDeltas(fieldDeltas);
256         }
257 
258         // enum constants
259         if (from.getKind() == Kind.ENUM) {
260             Set<IEnumConstantDelta> enumDeltas = compareEnumConstants(from
261                     .getEnumConstants(), to.getEnumConstants());
262             if (enumDeltas != null) {
263                 if (classDelta == null) {
264                     classDelta = new SigClassDefinitionDelta(from, to);
265                 }
266                 classDelta.setEnumConstantDeltas(enumDeltas);
267             }
268         } else if (from.getKind() == Kind.ANNOTATION) {
269             Set<IAnnotationFieldDelta> annotationFieldDeltas =
270                     compareAnnotationFields(from.getAnnotationFields(),
271                             to.getAnnotationFields());
272             if (annotationFieldDeltas != null) {
273                 if (classDelta == null) {
274                     classDelta = new SigClassDefinitionDelta(from, to);
275                 }
276                 classDelta.setAnnotationFieldDeltas(annotationFieldDeltas);
277             }
278         }
279 
280         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
281                 .getAnnotations(), to.getAnnotations());
282         if (annotationDeltas != null) {
283             if (classDelta == null) {
284                 classDelta = new SigClassDefinitionDelta(from, to);
285             }
286             classDelta.setAnnotationDeltas(annotationDeltas);
287         }
288         return classDelta;
289     }
290 
compareInterfaces( IClassDefinition from, IClassDefinition to)291     private Set<ITypeReferenceDelta<?>> compareInterfaces(
292             IClassDefinition from, IClassDefinition to) {
293         Set<ITypeReference> fromClosure = getInterfaceClosure(from);
294         Set<ITypeReference> toClosure = getInterfaceClosure(to);
295 
296         Set<ITypeReference> fromInterfaces = from.getInterfaces();
297         Set<ITypeReference> toInterfaces = to.getInterfaces();
298 
299         Set<ITypeReferenceDelta<?>> deltas =
300                 new HashSet<ITypeReferenceDelta<?>>();
301 
302         // check whether all from interfaces are directly or indirectly
303         // implemented by the to method
304         for (ITypeReference type : fromInterfaces) {
305             if (!containsType(type, toInterfaces)) {
306                 if (!(containsType(type, toClosure) /*
307                                                      * && !containsType(type,
308                                                      * toInterfaces)
309                                                      */)) {
310                     deltas.add(new SigTypeDelta<ITypeReference>(type, null));
311                 }
312             }
313         }
314 
315         // check whether all interfaces to are directly or indirectly
316         // implemented by the from method
317         for (ITypeReference type : toInterfaces) {
318             if (!containsType(type, fromInterfaces)) {
319                 if (!(containsType(type, fromClosure) /*
320                                                        * && !containsType(type,
321                                                        * fromInterfaces)
322                                                        */)) {
323                     deltas.add(new SigTypeDelta<ITypeReference>(null, type));
324                 }
325             }
326         }
327         return deltas.isEmpty() ? null : deltas;
328     }
329 
330 
containsType(ITypeReference type, Set<ITypeReference> setOfTypes)331     private boolean containsType(ITypeReference type,
332             Set<ITypeReference> setOfTypes) {
333         for (ITypeReference other : setOfTypes) {
334             if (compareType(type, other, false) == null) {
335                 return true;
336             }
337         }
338         return false;
339     }
340 
getInterfaceClosure(IClassDefinition clazz)341     private Set<ITypeReference> getInterfaceClosure(IClassDefinition clazz) {
342         Set<ITypeReference> closure = new HashSet<ITypeReference>();
343         collectInterfaceClosure(ViewpointAdapter.getReferenceTo(clazz),
344                 closure);
345         return closure;
346     }
347 
collectInterfaceClosure(ITypeReference clazz, Set<ITypeReference> closure)348     private void collectInterfaceClosure(ITypeReference clazz,
349             Set<ITypeReference> closure) {
350 
351         IClassDefinition classDefinition = getClassDefinition(clazz);
352         Set<ITypeReference> interfaces = classDefinition.getInterfaces();
353         if (interfaces == null) {
354             return;
355         }
356         for (ITypeReference interfaze : interfaces) {
357             closure.add(interfaze);
358         }
359 
360         ITypeReference superclass = classDefinition.getSuperClass();
361         if (superclass != null) {
362             if (superclass instanceof IParameterizedType) {
363                 collectInterfaceClosure(((IParameterizedType) superclass)
364                         .getRawType(), closure);
365             } else {
366                 collectInterfaceClosure(superclass, closure);
367             }
368         }
369         for (ITypeReference interfaze : interfaces) {
370             if (interfaze instanceof IParameterizedType) {
371                 collectInterfaceClosure(((IParameterizedType) interfaze)
372                         .getRawType(), closure);
373             } else {
374                 collectInterfaceClosure(interfaze, closure);
375             }
376         }
377     }
378 
compareAnnotations(Set<IAnnotation> from, Set<IAnnotation> to)379     private Set<IAnnotationDelta> compareAnnotations(Set<IAnnotation> from,
380             Set<IAnnotation> to) {
381         return compareSets(from, to,
382                 new SigComparator<IAnnotation, IAnnotationDelta>() {
383                     public IAnnotationDelta createAddRemoveDelta(
384                             IAnnotation from, IAnnotation to) {
385                         return new SigAnnotationDelta(from, to);
386                     }
387 
388                     public boolean considerEqualElement(IAnnotation from,
389                             IAnnotation to) {
390                         return sameClassDefinition(from.getType()
391                                 .getClassDefinition(), to.getType()
392                                 .getClassDefinition());
393                     }
394 
395                     public IAnnotationDelta createChangedDelta(
396                             IAnnotation from, IAnnotation to) {
397                         return compareAnnotation(from, to);
398                     }
399                 });
400     }
401 
402     private Set<IAnnotationFieldDelta> compareAnnotationFields(
403             Set<IAnnotationField> from, Set<IAnnotationField> to) {
404         return compareSets(from, to,
405                 new SigComparator<IAnnotationField, IAnnotationFieldDelta>() {
406                     public boolean considerEqualElement(IAnnotationField from,
407                             IAnnotationField to) {
408                         return from.getName().equals(to.getName());
409                     }
410 
411                     public IAnnotationFieldDelta createAddRemoveDelta(
412                             IAnnotationField from, IAnnotationField to) {
413                         return new SigAnnotationFieldDelta(from, to);
414                     }
415 
416                     public IAnnotationFieldDelta createChangedDelta(
417                             IAnnotationField from, IAnnotationField to) {
418                         return compareAnnotationField(from, to);
419                     }
420                 });
421     }
422 
423     private Set<IEnumConstantDelta> compareEnumConstants(
424             Set<IEnumConstant> from, Set<IEnumConstant> to) {
425         return compareSets(from, to,
426                 new SigComparator<IEnumConstant, IEnumConstantDelta>() {
427                     public boolean considerEqualElement(IEnumConstant from,
428                             IEnumConstant to) {
429                         return from.getName().equals(to.getName());
430                     }
431 
432                     public IEnumConstantDelta createAddRemoveDelta(
433                             IEnumConstant from, IEnumConstant to) {
434                         return new SigEnumConstantDelta(from, to);
435                     }
436 
437                     public IEnumConstantDelta createChangedDelta(
438                             IEnumConstant from, IEnumConstant to) {
439                         return compareEnumConstant(from, to);
440                     }
441                 });
442     }
443 
444     private Set<IFieldDelta> compareFields(Set<IField> from, Set<IField> to) {
445         return compareSets(from, to, new SigComparator<IField, IFieldDelta>() {
446             public boolean considerEqualElement(IField from, IField to) {
447                 return from.getName().equals(to.getName());
448             }
449 
450             public IFieldDelta createAddRemoveDelta(IField from, IField to) {
451                 return new SigFieldDelta(from, to);
452             }
453 
454             public IFieldDelta createChangedDelta(IField from, IField to) {
455                 return compareField(from, to);
456             }
457         });
458     }
459 
460     private Set<IMethodDelta> compareMethods(IClassDefinition from,
461             IClassDefinition to) {
462         assert from != null;
463         assert to != null;
464 
465         Set<IMethod> toMethods = new HashSet<IMethod>(to.getMethods());
466         Set<IMethod> toClosure = getMethodClosure(to);
467         Set<IMethod> fromMethods = new HashSet<IMethod>(from.getMethods());
468         Set<IMethod> fromClosure = getMethodClosure(from);
469 
470         Set<IMethodDelta> deltas = new HashSet<IMethodDelta>();
471 
472         for (IMethod method : fromMethods) {
473             IMethod compatibleMethod = findCompatibleMethod(method, toMethods);
474             if (compatibleMethod == null) {
475                 compatibleMethod = findCompatibleMethod(method, toClosure);
476                 if (compatibleMethod == null) {
477                     deltas.add(new SigMethodDelta(method, null));
478                 }
479             }
480 
481             if (compatibleMethod != null) {
482                 IMethodDelta delta = compareMethod(method, compatibleMethod);
483                 if (delta != null) {
484                     deltas.add(delta);
485                 }
486             }
487         }
488 
489         for (IMethod method : toMethods) {
490             IMethod compatibleMethod = findCompatibleMethod(method, fromMethods);
491             if (compatibleMethod == null) {
492                 compatibleMethod = findCompatibleMethod(method, fromClosure);
493                 if (compatibleMethod == null) {
494                     deltas.add(new SigMethodDelta(null, method));
495                 }
496             }
497         }
498         return deltas.isEmpty() ? null : deltas;
499     }
500 
501     private IMethod findCompatibleMethod(IMethod method, Set<IMethod> set) {
502         for (IMethod methodFromSet : set) {
503             if (equalsSignature(method, methodFromSet)) {
504                 return methodFromSet;
505             }
506         }
507         return null;
508     }
509 
510 
511     private Set<IMethod> getMethodClosure(IClassDefinition clazz) {
512         Set<IMethod> closure = new HashSet<IMethod>();
513         collectMethods(new ClassProjection(clazz,
514                 new HashMap<ITypeVariableDefinition, ITypeReference>()),
515                 closure);
516         return closure;
517     }
518 
519     private void collectMethods(IClassDefinition clazz, Set<IMethod> closure) {
520         if (clazz == null) {
521             return;
522         }
523         if (clazz.getMethods() != null) {
524             closure.addAll(clazz.getMethods());
525         }
526         if (clazz.getSuperClass() != null) {
527             collectMethods(getClassDefinition(clazz.getSuperClass()), closure);
528         }
529         if (clazz.getInterfaces() != null) {
530             for (ITypeReference interfaze : clazz.getInterfaces()) {
531                 collectMethods(getClassDefinition(interfaze), closure);
532             }
533         }
534     }
535 
536     private Set<IConstructorDelta> compareConstructors(Set<IConstructor> from,
537             Set<IConstructor> to) {
538         return compareSets(from, to,
539                 new SigComparator<IConstructor, IConstructorDelta>() {
540                     public boolean considerEqualElement(IConstructor from,
541                             IConstructor to) {
542                         return equalsSignature(from, to);
543                     }
544 
545                     public IConstructorDelta createAddRemoveDelta(
546                             IConstructor from, IConstructor to) {
547                         return new SigConstructorDelta(from, to);
548                     }
549 
550                     public IConstructorDelta createChangedDelta(
551                             IConstructor from, IConstructor to) {
552                         return compareConstructor(from, to);
553                     }
554                 });
555     }
556 
557     // compares names and parameter types
558     private boolean equalsSignature(IExecutableMember from,
559             IExecutableMember to) {
560         if (from.getName().equals(to.getName())) {
561             return compareTypeSequence(getParameterList(from.getParameters()),
562                     getParameterList(to.getParameters()), true) == null;
563         }
564         return false;
565     }
566 
567     private List<ITypeReference> getParameterList(List<IParameter> parameters) {
568         List<ITypeReference> parameterTypes = new LinkedList<ITypeReference>();
569         for (IParameter parameter : parameters) {
570             parameterTypes.add(parameter.getType());
571         }
572         return parameterTypes;
573     }
574 
575     private IAnnotationDelta compareAnnotation(IAnnotation from,
576             IAnnotation to) {
577         assert sameClassDefinition(from.getType().getClassDefinition(), to
578                 .getType().getClassDefinition());
579 
580         Set<IAnnotationElement> fromAnnotationElement =
581                 getNormalizedAnnotationElements(from);
582         Set<IAnnotationElement> toAnnotationElement =
583                 getNormalizedAnnotationElements(to);
584 
585         Set<IAnnotationElementDelta> annotationElementDeltas =
586                 compareAnnotationElements(
587                         fromAnnotationElement, toAnnotationElement);
588         SigAnnotationDelta delta = null;
589 
590         if (annotationElementDeltas != null) {
591             delta = new SigAnnotationDelta(from, to);
592             delta.setAnnotationElementDeltas(annotationElementDeltas);
593         }
594         return delta;
595     }
596 
597     /**
598      * Returns the annotation elements for the given annotation. The returned
599      * set contains all declared elements plus all elements with default values.
600      *
601      * @param annotation
602      *            the annotation to return the elements for
603      * @return the default enriched annotation elements
604      */
605     private Set<IAnnotationElement> getNormalizedAnnotationElements(
606             IAnnotation annotation) {
607         Set<IAnnotationElement> elements = new HashSet<IAnnotationElement>(
608                 annotation.getElements());
609 
610         Set<String> names = new HashSet<String>();
611         for (IAnnotationElement annotationElement : elements) {
612             names.add(annotationElement.getDeclaringField().getName());
613         }
614 
615         for (IAnnotationField field : annotation.getType().getClassDefinition()
616                 .getAnnotationFields()) {
617             if (!names.contains(field.getName())) {
618                 SigAnnotationElement sigAnnotationElement =
619                         new SigAnnotationElement();
620                 sigAnnotationElement.setDeclaringField(field);
621                 sigAnnotationElement.setValue(field.getDefaultValue());
622                 elements.add(sigAnnotationElement);
623             }
624         }
625         return elements;
626     }
627 
628     private Set<IAnnotationElementDelta> compareAnnotationElements(
629             Set<IAnnotationElement> from, Set<IAnnotationElement> to) {
630         return compareSets(from, to,
631                 new SigComparator<IAnnotationElement, IAnnotationElementDelta>() {
632                     public boolean considerEqualElement(
633                             IAnnotationElement from, IAnnotationElement to) {
634                         return from.getDeclaringField().getName().equals(
635                                 to.getDeclaringField().getName());
636                     }
637 
638                     public IAnnotationElementDelta createAddRemoveDelta(
639                             IAnnotationElement from, IAnnotationElement to) {
640                         return new SigAnnotationElementDelta(from, to);
641                     }
642 
643                     public IAnnotationElementDelta createChangedDelta(
644                             IAnnotationElement from, IAnnotationElement to) {
645                         return compareAnnotationElement(from, to);
646                     }
647                 });
648     }
649 
650     private IAnnotationElementDelta compareAnnotationElement(
651             IAnnotationElement from, IAnnotationElement to) {
652         SigAnnotationElementDelta delta = null;
653         SigValueDelta valueDelta = compareValue(from.getValue(), to.getValue());
654 
655         if (valueDelta != null) {
656             delta = new SigAnnotationElementDelta(from, to);
657             delta.setValueDelta(valueDelta);
658         }
659         return delta;
660     }
661 
662     /**
663      * Removes the {@link Modifier#ABSTRACT} modifier.
664      */
665     private Set<Modifier> prepareMethodModifiers(IMethod method) {
666         Set<Modifier> modifierCopy = new HashSet<Modifier>(method
667                 .getModifiers());
668         modifierCopy.remove(Modifier.ABSTRACT);
669         return modifierCopy;
670     }
671 
672     private IMethodDelta compareMethod(IMethod from, IMethod to) {
673         assert from != null && to != null;
674 
675         SigMethodDelta methodDelta = null;
676         Set<IModifierDelta> modiferDeltas = compareModifiers(
677                 prepareMethodModifiers(from), prepareMethodModifiers(to));
678         if (modiferDeltas != null) {
679             methodDelta = new SigMethodDelta(from, to);
680             methodDelta.setModifierDeltas(modiferDeltas);
681         }
682 
683         Set<IParameterDelta> parameterDeltas = compareParameterSequence(from
684                 .getParameters(), to.getParameters());
685         if (parameterDeltas != null) {
686             if (methodDelta == null) {
687                 methodDelta = new SigMethodDelta(from, to);
688             }
689             methodDelta.setParameterDeltas(parameterDeltas);
690         }
691 
692         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
693                 .getAnnotations(), to.getAnnotations());
694         if (annotationDeltas != null) {
695             if (methodDelta == null) {
696                 methodDelta = new SigMethodDelta(from, to);
697             }
698             methodDelta.setAnnotationDeltas(annotationDeltas);
699         }
700 
701         Set<ITypeVariableDefinitionDelta> typeParameterDeltas =
702                 compareTypeVariableSequence(from.getTypeParameters(),
703                         to.getTypeParameters());
704         if (typeParameterDeltas != null) {
705             if (methodDelta == null) {
706                 methodDelta = new SigMethodDelta(from, to);
707             }
708             methodDelta.setTypeVariableDeltas(typeParameterDeltas);
709         }
710 
711         Set<ITypeReferenceDelta<?>> exceptionDeltas = compareTypes(
712                 normalizeExceptions(from.getExceptions()),
713                 normalizeExceptions(to.getExceptions()));
714         if (exceptionDeltas != null) {
715             if (methodDelta == null) {
716                 methodDelta = new SigMethodDelta(from, to);
717             }
718             methodDelta.setExceptionDeltas(exceptionDeltas);
719         }
720 
721         ITypeReferenceDelta<?> returnTypeDelta = compareType(from
722                 .getReturnType(), to.getReturnType(), false);
723         if (returnTypeDelta != null) {
724             if (methodDelta == null) {
725                 methodDelta = new SigMethodDelta(from, to);
726             }
727             methodDelta.setReturnTypeDelta(returnTypeDelta);
728         }
729 
730         return methodDelta;
731     }
732 
733     // remove runtime exceptions,
734     // remove sub types of containing exception
735     private Set<ITypeReference> normalizeExceptions(
736             Set<ITypeReference> exceptions) {
737         Set<ITypeReference> exceptionCopy = new HashSet<ITypeReference>(
738                 exceptions);
739 
740         Iterator<ITypeReference> iterator = exceptionCopy.iterator();
741         while (iterator.hasNext()) {
742             ITypeReference exception = iterator.next();
743             if (isRuntimeExceptionOrErrorSubtype(exception)) {
744                 iterator.remove();
745             }
746         }
747         exceptionCopy = removeSpecializations(exceptionCopy);
748         return exceptionCopy;
749     }
750 
751     private Set<ITypeReference> removeSpecializations(
752             Set<ITypeReference> exceptions) {
753         Set<ITypeReference> exceptionCopy = new HashSet<ITypeReference>(
754                 exceptions);
755         for (ITypeReference type : exceptions) {
756             Iterator<ITypeReference> it = exceptionCopy.iterator();
757             while (it.hasNext()) {
758                 ITypeReference subType = it.next();
759                 if (isSuperClass(getClassDefinition(type),
760                         getClassDefinition(subType))) {
761                     it.remove();
762                 }
763             }
764         }
765         return exceptionCopy;
766     }
767 
768     /**
769      * Returns true if superC is a super class of subC.
770      */
771     private boolean isSuperClass(IClassDefinition superC,
772             IClassDefinition subC) {
773         if (superC == null || subC == null) {
774             return false;
775         }
776 
777         if (subC.getSuperClass() == null) {
778             return false;
779         } else {
780             if (getClassDefinition(subC.getSuperClass()).equals(superC)) {
781                 return true;
782             } else {
783                 return isSuperClass(superC, getClassDefinition(subC
784                         .getSuperClass()));
785             }
786         }
787     }
788 
789     private boolean isSuperInterface(IClassDefinition superClass,
790             IClassDefinition subClass) {
791         if (superClass == null || subClass == null) {
792             return false;
793         }
794 
795         if (subClass.getInterfaces() == null) {
796             return false;
797         } else {
798             if (getClassDefinitions(subClass.getInterfaces()).contains(
799                     superClass)) {
800                 return true;
801             } else {
802                 for (ITypeReference subType : subClass.getInterfaces()) {
803                     if (isSuperInterface(superClass,
804                             getClassDefinition(subType))) {
805                         return true;
806                     }
807                 }
808                 return false;
809             }
810         }
811     }
812 
813     private Set<IClassDefinition> getClassDefinitions(
814             Set<ITypeReference> references) {
815         Set<IClassDefinition> definitions = new HashSet<IClassDefinition>();
816         for (ITypeReference ref : references) {
817             definitions.add(getClassDefinition(ref));
818         }
819         return definitions;
820     }
821 
822     /**
823      * Returns null if type is not one of:
824      * <ul>
825      * <li>IClassReference</li>
826      * <li>IParameterizedType</li>
827      * </ul>
828      */
829     private IClassDefinition getClassDefinition(ITypeReference type) {
830         assert type != null;
831 
832         IClassDefinition returnValue = null;
833         if (type instanceof IClassReference) {
834             returnValue = ((IClassReference) type).getClassDefinition();
835         } else if (type instanceof IParameterizedType) {
836             returnValue = ((IParameterizedType) type).getRawType()
837                     .getClassDefinition();
838         }
839         return returnValue;
840     }
841 
842     private boolean isRuntimeExceptionOrErrorSubtype(ITypeReference exception) {
843 
844         IClassDefinition clazz = getClassDefinition(exception);
845         if (clazz != null) {
846             if (isRuntimeExceptionOrError(clazz)) {
847                 return true;
848             } else if (clazz.getSuperClass() != null) {
849                 return isRuntimeExceptionOrErrorSubtype(clazz.getSuperClass());
850             } else {
851                 return false;
852             }
853         }
854         return false;
855     }
856 
857     private boolean isRuntimeExceptionOrError(IClassDefinition exception) {
858         if (exception == null) {
859             return false;
860         }
861         String packageName = exception.getPackageName();
862         String className = exception.getName();
863 
864         if (packageName != null && className != null
865                 && "java.lang".equals(packageName)) {
866             return "RuntimeException".equals(className)
867                     || "Error".equals(className);
868         }
869         return false;
870     }
871 
872     private IConstructorDelta compareConstructor(IConstructor from,
873             IConstructor to) {
874         SigConstructorDelta constructorDelta = null;
875         Set<IModifierDelta> modiferDeltas = compareModifiers(from
876                 .getModifiers(), to.getModifiers());
877         if (modiferDeltas != null) {
878             constructorDelta = new SigConstructorDelta(from, to);
879             constructorDelta.setModifierDeltas(modiferDeltas);
880         }
881 
882         Set<IParameterDelta> parameterDeltas = compareParameterSequence(from
883                 .getParameters(), to.getParameters());
884         if (parameterDeltas != null) {
885             if (constructorDelta == null) {
886                 constructorDelta = new SigConstructorDelta(from, to);
887             }
888             constructorDelta.setParameterDeltas(parameterDeltas);
889         }
890 
891         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
892                 .getAnnotations(), to.getAnnotations());
893         if (annotationDeltas != null) {
894             if (constructorDelta == null) {
895                 constructorDelta = new SigConstructorDelta(from, to);
896             }
897             constructorDelta.setAnnotationDeltas(annotationDeltas);
898         }
899 
900         Set<ITypeVariableDefinitionDelta> typeParameterDeltas =
901                 compareTypeVariableSequence(from.getTypeParameters(),
902                         to.getTypeParameters());
903         if (typeParameterDeltas != null) {
904             if (constructorDelta == null) {
905                 constructorDelta = new SigConstructorDelta(from, to);
906             }
907             constructorDelta.setTypeVariableDeltas(typeParameterDeltas);
908         }
909 
910         Set<ITypeReferenceDelta<?>> exceptionDeltas = compareTypes(
911                 normalizeExceptions(from.getExceptions()),
912                 normalizeExceptions(to.getExceptions()));
913         if (exceptionDeltas != null) {
914             if (constructorDelta == null) {
915                 constructorDelta = new SigConstructorDelta(from, to);
916             }
917             constructorDelta.setExceptionDeltas(exceptionDeltas);
918         }
919         return constructorDelta;
920     }
921 
922     private Set<IParameterDelta> compareParameterSequence(
923             List<IParameter> from, List<IParameter> to) {
924         assert from.size() == to.size();
925         Set<IParameterDelta> deltas = new HashSet<IParameterDelta>();
926         Iterator<IParameter> fromIterator = from.iterator();
927         Iterator<IParameter> toIterator = to.iterator();
928         while (fromIterator.hasNext() && toIterator.hasNext()) {
929             IParameterDelta delta = compareParameter(fromIterator.next(),
930                     toIterator.next());
931             if (delta != null) {
932                 deltas.add(delta);
933             }
934         }
935         return deltas.isEmpty() ? null : deltas;
936     }
937 
938     private IParameterDelta compareParameter(IParameter from, IParameter to) {
939         SigParameterDelta delta = null;
940         ITypeReferenceDelta<?> typeDelta = compareType(from.getType(), to
941                 .getType(), false);
942         if (typeDelta != null) {
943             if (delta == null) {
944                 delta = new SigParameterDelta(from, to);
945             }
946             delta.setTypeDelta(typeDelta);
947         }
948 
949         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
950                 .getAnnotations(), to.getAnnotations());
951         if (annotationDeltas != null) {
952             if (delta == null) {
953                 delta = new SigParameterDelta(from, to);
954             }
955             delta.setAnnotationDeltas(annotationDeltas);
956         }
957         return delta;
958     }
959 
960     private Set<ITypeVariableDefinitionDelta> compareTypeVariableSequence(
961             List<ITypeVariableDefinition> from,
962             List<ITypeVariableDefinition> to) {
963         Set<ITypeVariableDefinitionDelta> deltas =
964                 new HashSet<ITypeVariableDefinitionDelta>();
965         if (from.size() != to.size()) {
966             for (ITypeVariableDefinition fromVariable : from) {
967                 deltas.add(new SigTypeVariableDefinitionDelta(fromVariable,
968                         null));
969             }
970             for (ITypeVariableDefinition toVariable : to) {
971                 deltas
972                         .add(new SigTypeVariableDefinitionDelta(null,
973                                 toVariable));
974             }
975         }
976 
977         Iterator<ITypeVariableDefinition> fromIterator = from.iterator();
978         Iterator<ITypeVariableDefinition> toIterator = to.iterator();
979         while (fromIterator.hasNext() && toIterator.hasNext()) {
980             ITypeVariableDefinitionDelta delta = compareTypeVariableDefinition(
981                     fromIterator.next(), toIterator.next());
982             if (delta != null) {
983                 deltas.add(delta);
984             }
985         }
986         return deltas.isEmpty() ? null : deltas;
987     }
988 
989     private ITypeVariableDefinitionDelta compareTypeVariableDefinition(
990             ITypeVariableDefinition from, ITypeVariableDefinition to) {
991         IGenericDeclarationDelta declarationDelta = compareGenericDeclaration(
992                 from, to);
993 
994         if (declarationDelta != null) {
995             SigTypeVariableDefinitionDelta delta =
996                     new SigTypeVariableDefinitionDelta(from, to);
997             delta.setGenericDeclarationDelta(declarationDelta);
998             return delta;
999         }
1000         IUpperBoundsDelta upperBoundDelta = compareUpperBounds(from
1001                 .getUpperBounds(), to.getUpperBounds());
1002 
1003         if (upperBoundDelta != null) {
1004             SigTypeVariableDefinitionDelta delta =
1005                     new SigTypeVariableDefinitionDelta(from, to);
1006             delta.setUpperBoundsDelta(upperBoundDelta);
1007             return delta;
1008         }
1009         return null;
1010     }
1011 
1012     private ITypeReferenceDelta<ITypeVariableReference> compareTypeVariableReference(
1013             ITypeVariableReference from, ITypeVariableReference to) {
1014         IGenericDeclarationDelta declarationDelta = compareGenericDeclaration(
1015                 from.getTypeVariableDefinition(), to
1016                         .getTypeVariableDefinition());
1017         if (declarationDelta != null) {
1018             SigTypeVariableReferenceDelta delta =
1019                     new SigTypeVariableReferenceDelta(from, to);
1020             delta.setGenericDeclarationDelta(declarationDelta);
1021             return delta;
1022         }
1023         return null;
1024     }
1025 
1026     private Set<IModifierDelta> compareModifiers(Set<Modifier> from,
1027             Set<Modifier> to) {
1028         return compareSets(from, to,
1029                 new SigComparator<Modifier, IModifierDelta>() {
1030                     public boolean considerEqualElement(Modifier from,
1031                             Modifier to) {
1032                         return from.equals(to);
1033                     }
1034 
1035                     public IModifierDelta createAddRemoveDelta(Modifier from,
1036                             Modifier to) {
1037                         return new SigModifierDelta(from, to);
1038                     }
1039 
1040                     public IModifierDelta createChangedDelta(Modifier from,
1041                             Modifier to) {
1042                         return null;
1043                     }
1044                 });
1045     }
1046 
1047 
1048     private IFieldDelta compareField(IField from, IField to) {
1049         SigFieldDelta fieldDelta = null;
1050 
1051         Set<IModifierDelta> modiferDeltas = compareModifiers(from
1052                 .getModifiers(), to.getModifiers());
1053         if (modiferDeltas != null) {
1054             fieldDelta = new SigFieldDelta(from, to);
1055             fieldDelta.setModifierDeltas(modiferDeltas);
1056         }
1057 
1058         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
1059                 .getAnnotations(), to.getAnnotations());
1060         if (annotationDeltas != null) {
1061             if (fieldDelta == null) {
1062                 fieldDelta = new SigFieldDelta(from, to);
1063             }
1064             fieldDelta.setAnnotationDeltas(annotationDeltas);
1065         }
1066 
1067         ITypeReferenceDelta<?> typeDelta = compareType(from.getType(), to
1068                 .getType(), false);
1069         if (typeDelta != null) {
1070             if (fieldDelta == null) {
1071                 fieldDelta = new SigFieldDelta(from, to);
1072             }
1073             fieldDelta.setTypeDelta(typeDelta);
1074         }
1075         return fieldDelta;
1076     }
1077 
1078     private IEnumConstantDelta compareEnumConstant(IEnumConstant from,
1079             IEnumConstant to) {
1080         SigEnumConstantDelta enumConstantDelta = null;
1081 
1082         Set<IModifierDelta> modiferDeltas = compareModifiers(from
1083                 .getModifiers(), to.getModifiers());
1084         if (modiferDeltas != null) {
1085             enumConstantDelta = new SigEnumConstantDelta(from, to);
1086             enumConstantDelta.setModifierDeltas(modiferDeltas);
1087         }
1088 
1089         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
1090                 .getAnnotations(), to.getAnnotations());
1091         if (annotationDeltas != null) {
1092             if (enumConstantDelta == null) {
1093                 enumConstantDelta = new SigEnumConstantDelta(from, to);
1094             }
1095             enumConstantDelta.setAnnotationDeltas(annotationDeltas);
1096         }
1097 
1098         ITypeReferenceDelta<?> typeDelta = compareType(from.getType(), to
1099                 .getType(), false);
1100         if (typeDelta != null) {
1101             if (enumConstantDelta == null) {
1102                 enumConstantDelta = new SigEnumConstantDelta(from, to);
1103             }
1104             enumConstantDelta.setTypeDelta(typeDelta);
1105         }
1106 
1107         // FIXME ordinal not supported in dex
1108         // ValueDelta ordinalDelta = compareValue(from.getOrdinal(),
1109         // to.getOrdinal());
1110         // if (ordinalDelta != null) {
1111         // if (enumConstantDelta == null) {
1112         // enumConstantDelta = new SigEnumConstantDelta(from, to);
1113         // }
1114         // enumConstantDelta.setOrdinalDelta(ordinalDelta);
1115         // }
1116 
1117         return enumConstantDelta;
1118     }
1119 
1120     private IAnnotationFieldDelta compareAnnotationField(IAnnotationField from,
1121             IAnnotationField to) {
1122         SigAnnotationFieldDelta annotationFieldDelta = null;
1123 
1124         Set<IModifierDelta> modiferDeltas = compareModifiers(from
1125                 .getModifiers(), to.getModifiers());
1126         if (modiferDeltas != null) {
1127             annotationFieldDelta = new SigAnnotationFieldDelta(from, to);
1128             annotationFieldDelta.setModifierDeltas(modiferDeltas);
1129         }
1130 
1131         Set<IAnnotationDelta> annotationDeltas = compareAnnotations(from
1132                 .getAnnotations(), to.getAnnotations());
1133         if (annotationDeltas != null) {
1134             if (annotationFieldDelta == null) {
1135                 annotationFieldDelta = new SigAnnotationFieldDelta(from, to);
1136             }
1137             annotationFieldDelta.setAnnotationDeltas(annotationDeltas);
1138         }
1139 
1140         ITypeReferenceDelta<?> typeDelta = compareType(from.getType(), to
1141                 .getType(), false);
1142         if (typeDelta != null) {
1143             if (annotationFieldDelta == null) {
1144                 annotationFieldDelta = new SigAnnotationFieldDelta(from, to);
1145             }
1146             annotationFieldDelta.setTypeDelta(typeDelta);
1147         }
1148 
1149         IValueDelta defaultValueDelta = compareValue(from.getDefaultValue(), to
1150                 .getDefaultValue());
1151         if (defaultValueDelta != null) {
1152             if (annotationFieldDelta == null) {
1153                 annotationFieldDelta = new SigAnnotationFieldDelta(from, to);
1154             }
1155             annotationFieldDelta.setDefaultValueDelta(defaultValueDelta);
1156         }
1157 
1158         return annotationFieldDelta;
1159     }
1160 
1161     private SigValueDelta compareValue(Object from, Object to) {
1162         // same value
1163         if (from == null && to == null) {
1164             return null;
1165         }
1166 
1167         // one of both is null and other is not
1168         if (from == null || to == null) {
1169             return new SigValueDelta(from, to);
1170         }
1171 
1172         SigValueDelta delta = null;
1173         // different types
1174         if (from.getClass() == to.getClass()) {
1175             if (from.getClass().isArray()) {
1176                 Object[] fromArray = (Object[]) from;
1177                 Object[] toArray = (Object[]) from;
1178                 if (!Arrays.equals(fromArray, toArray)) {
1179                     delta = new SigValueDelta(from, to);
1180                 }
1181             } else if (from instanceof IEnumConstant) {
1182                 IEnumConstantDelta enumConstantDelta = compareEnumConstant(
1183                         (IEnumConstant) from, (IEnumConstant) to);
1184                 if (enumConstantDelta != null) {
1185                     delta = new SigValueDelta(from, to);
1186                 }
1187             } else if (from instanceof IAnnotation) {
1188                 IAnnotationDelta annotationDelta = compareAnnotation(
1189                         (IAnnotation) from, (IAnnotation) to);
1190                 if (annotationDelta != null) {
1191                     delta = new SigValueDelta(from, to);
1192                 }
1193             } else if (from instanceof IField) {
1194                 IFieldDelta fieldDelta = compareField((IField) from,
1195                         (IField) to);
1196                 if (fieldDelta != null) {
1197                     delta = new SigValueDelta(from, to);
1198                 }
1199             } else if (from instanceof ITypeReference) {
1200                 ITypeReferenceDelta<? extends ITypeReference> typeDelta =
1201                         compareType((ITypeReference) from, (ITypeReference) to,
1202                                 false);
1203                 if (typeDelta != null) {
1204                     delta = new SigValueDelta(from, to);
1205                 }
1206             } else if (!from.equals(to)) {
1207                 delta = new SigValueDelta(from, to);
1208             }
1209 
1210         } else if (!(from == null && to == null)) {
1211             delta = new SigValueDelta(from, to);
1212         }
1213         return delta;
1214     }
1215 
1216     private boolean considerEqualTypes(ITypeReference from, ITypeReference to) {
1217         assert from != null && to != null;
1218 
1219         if (implementInterface(from, to, IPrimitiveType.class)) {
1220             return comparePrimitiveType((IPrimitiveType) from,
1221                     (IPrimitiveType) to) == null;
1222         }
1223         if (implementInterface(from, to, IClassReference.class)) {
1224             return sameClassDefinition(((IClassReference) from)
1225                     .getClassDefinition(), ((IClassReference) to)
1226                     .getClassDefinition());
1227         }
1228         if (implementInterface(from, to, IArrayType.class)) {
1229             return considerEqualTypes(((IArrayType) from).getComponentType(),
1230                     ((IArrayType) to).getComponentType());
1231         }
1232         if (implementInterface(from, to, IParameterizedType.class)) {
1233             return compareClassReference(((IParameterizedType) from)
1234                     .getRawType(), ((IParameterizedType) to)
1235                     .getRawType()) == null;
1236         }
1237         if (implementInterface(from, to, ITypeVariableReference.class)) {
1238             return compareTypeVariableReference((ITypeVariableReference) from,
1239                     (ITypeVariableReference) to) == null;
1240         }
1241 
1242         return false;
1243     }
1244 
1245     private Set<ITypeReference> fromComparison = new HashSet<ITypeReference>();
1246     private Set<ITypeReference> toComparison = new HashSet<ITypeReference>();
1247 
1248 
1249     private boolean areInComparison(ITypeReference from, ITypeReference to) {
1250         return fromComparison.contains(from) && toComparison.contains(to);
1251     }
1252 
1253     private void markInComparison(ITypeReference from, ITypeReference to) {
1254         fromComparison.add(from);
1255         toComparison.add(to);
1256     }
1257 
1258     private void markFinishedComparison(ITypeReference from,
1259             ITypeReference to) {
1260         fromComparison.remove(from);
1261         toComparison.remove(to);
1262     }
1263 
1264     private ITypeReferenceDelta<? extends ITypeReference> compareType(
1265             ITypeReference from, ITypeReference to, boolean acceptErasedTypes) {
1266 
1267         if (from == null && to == null) {
1268             return null;
1269         }
1270         if ((from == null && to != null) || (from != null && to == null)) {
1271             return new SigTypeDelta<ITypeReference>(from, to);
1272         }
1273         if (areInComparison(from, to)) {
1274             return null;
1275         }
1276         try {
1277             markInComparison(from, to);
1278 
1279             if (implementInterface(from, to, IPrimitiveType.class)) {
1280                 return comparePrimitiveType((IPrimitiveType) from,
1281                         (IPrimitiveType) to);
1282             }
1283             if (implementInterface(from, to, IClassReference.class)) {
1284                 return compareClassReference((IClassReference) from,
1285                         (IClassReference) to);
1286             }
1287             if (implementInterface(from, to, IArrayType.class)) {
1288                 return compareArrayType((IArrayType) from, (IArrayType) to);
1289             }
1290             if (implementInterface(from, to, IParameterizedType.class)) {
1291                 return compareParameterizedType((IParameterizedType) from,
1292                         (IParameterizedType) to, acceptErasedTypes);
1293             }
1294             if (implementInterface(from, to, ITypeVariableReference.class)) {
1295                 return compareTypeVariableReference(
1296                         (ITypeVariableReference) from,
1297                         (ITypeVariableReference) to);
1298             }
1299             if (implementInterface(from, to, IWildcardType.class)) {
1300                 return compareWildcardType((IWildcardType) from,
1301                         (IWildcardType) to);
1302             }
1303 
1304             if (acceptErasedTypes) {
1305                 if (isGeneric(from) && !isGeneric(to)) {
1306                     return compareType(getErasedType(from), to, false);
1307                 }
1308 
1309                 if (!isGeneric(from) && isGeneric(to)) {
1310                     return compareType(from, getErasedType(to), false);
1311                 }
1312             }
1313             return new SigTypeDelta<ITypeReference>(from, to);
1314         } finally {
1315             markFinishedComparison(from, to);
1316         }
1317     }
1318 
1319     private boolean isGeneric(ITypeReference reference) {
1320         if (reference instanceof IParameterizedType
1321                 || reference instanceof ITypeVariableReference
1322                 || reference instanceof IWildcardType) {
1323             return true;
1324         }
1325         if (reference instanceof IArrayType) {
1326             return isGeneric(((IArrayType) reference).getComponentType());
1327         }
1328         return false;
1329     }
1330 
1331     private ITypeReference getErasedType(ITypeReference reference) {
1332 
1333         if (reference instanceof IParameterizedType) {
1334             return ((IParameterizedType) reference).getRawType();
1335         }
1336         if (reference instanceof ITypeVariableReference) {
1337             ITypeVariableDefinition typeVariableDefinition =
1338                     ((ITypeVariableReference) reference)
1339                             .getTypeVariableDefinition();
1340             return getErasedType(
1341                     typeVariableDefinition.getUpperBounds().get(0));
1342         }
1343         if (reference instanceof IWildcardType) {
1344             return getErasedType(((IWildcardType) reference).getUpperBounds()
1345                     .get(0));
1346         }
1347         if (reference instanceof IArrayType) {
1348             // FIXME implement with erasure projection?
1349             return new SigArrayType(getErasedType(((IArrayType) reference)
1350                     .getComponentType()));
1351         }
1352         if (reference instanceof IPrimitiveType) {
1353             return reference;
1354         }
1355         if (reference instanceof IClassReference) {
1356             return reference;
1357         }
1358         throw new IllegalArgumentException("Unexpected type: " + reference);
1359     }
1360 
1361     private boolean implementInterface(ITypeReference from, ITypeReference to,
1362             Class<?> check) {
1363         return check.isAssignableFrom(from.getClass())
1364                 && check.isAssignableFrom(to.getClass());
1365     }
1366 
1367     private IWildcardTypeDelta compareWildcardType(IWildcardType from,
1368             IWildcardType to) {
1369         SigWildcardTypeDelta delta = null;
1370 
1371         ITypeReference fromLowerBound = from.getLowerBound();
1372         ITypeReference toLowerBound = to.getLowerBound();
1373 
1374         ITypeReferenceDelta<?> lowerBoundDelta = compareType(fromLowerBound,
1375                 toLowerBound, false);
1376         if (lowerBoundDelta != null) {
1377             delta = new SigWildcardTypeDelta(from, to);
1378             delta.setLowerBoundDelta(lowerBoundDelta);
1379         }
1380 
1381         IUpperBoundsDelta upperBoundsDelta = compareUpperBounds(from
1382                 .getUpperBounds(), to.getUpperBounds());
1383         if (upperBoundsDelta != null) {
1384             if (delta == null) {
1385                 delta = new SigWildcardTypeDelta(from, to);
1386             }
1387             delta.setUpperBoundDelta(upperBoundsDelta);
1388         }
1389         return delta;
1390     }
1391 
1392     private IGenericDeclarationDelta compareGenericDeclaration(
1393             ITypeVariableDefinition fromVariable,
1394             ITypeVariableDefinition toVariable) {
1395         IGenericDeclarationDelta delta = null;
1396 
1397         IGenericDeclaration from = fromVariable.getGenericDeclaration();
1398         IGenericDeclaration to = toVariable.getGenericDeclaration();
1399 
1400         if (from != null && to != null) {
1401 
1402             if (from.getClass() != to.getClass()) {
1403                 delta = new SigGenericDeclarationDelta(from, to);
1404             } else if (from instanceof IClassDefinition) {
1405                 IClassDefinition fromDeclaringClass = (IClassDefinition) from;
1406                 IClassDefinition toDeclaringClass = (IClassDefinition) to;
1407 
1408                 if (!sameClassDefinition(fromDeclaringClass,
1409                         toDeclaringClass)) {
1410                     delta = new SigGenericDeclarationDelta(from, to);
1411                 }
1412 
1413             } else if (from instanceof IConstructor) {
1414                 IConstructor fromConstructor = (IConstructor) from;
1415                 IConstructor toConstructor = (IConstructor) from;
1416 
1417                 String fromConstructorName = fromConstructor.getName();
1418                 String fromClassName = fromConstructor.getDeclaringClass()
1419                         .getQualifiedName();
1420 
1421                 String toConstructorName = toConstructor.getName();
1422                 String toClassName = toConstructor.getDeclaringClass()
1423                         .getQualifiedName();
1424 
1425                 if ((!fromConstructorName.equals(toConstructorName))
1426                         || (!fromClassName.equals(toClassName))) {
1427                     delta = new SigGenericDeclarationDelta(from, to);
1428                 }
1429 
1430             } else if (from instanceof IMethod) {
1431                 IMethod fromMethod = (IMethod) from;
1432                 IMethod toMethod = (IMethod) from;
1433 
1434                 String fromConstructorName = fromMethod.getName();
1435                 String fromClassName = fromMethod.getDeclaringClass()
1436                         .getQualifiedName();
1437 
1438                 String toConstructorName = toMethod.getName();
1439                 String toClassName = toMethod.getDeclaringClass()
1440                         .getQualifiedName();
1441 
1442                 if ((!fromConstructorName.equals(toConstructorName))
1443                         || (!fromClassName.equals(toClassName))) {
1444                     delta = new SigGenericDeclarationDelta(from, to);
1445                 }
1446             } else {
1447                 throw new IllegalStateException("Invlaid eclaration site: "
1448                         + from);
1449             }
1450 
1451             // check position
1452             int fromPosition = getPositionOf(fromVariable, from);
1453             int toPosition = getPositionOf(toVariable, to);
1454 
1455             if (fromPosition != toPosition) {
1456                 delta = new SigGenericDeclarationDelta(from, to);
1457             }
1458 
1459 
1460         } else {
1461             // one of both is null
1462             delta = new SigGenericDeclarationDelta(from, to);
1463         }
1464         return delta;
1465     }
1466 
1467     private int getPositionOf(ITypeVariableDefinition variable,
1468             IGenericDeclaration declaration) {
1469         return declaration.getTypeParameters().indexOf(variable);
1470     }
1471 
1472     private IUpperBoundsDelta compareUpperBounds(List<ITypeReference> from,
1473             List<ITypeReference> to) {
1474         if (from.isEmpty() && to.isEmpty()) {
1475             return null;
1476         }
1477         SigUpperBoundsDelta delta = null;
1478 
1479         ITypeReference fromFirstUpperBound = from.get(0);
1480         ITypeReference toFirstUpperBound = to.get(0);
1481 
1482         ITypeReferenceDelta<?> firstUpperBoundDelta = compareType(
1483                 fromFirstUpperBound, toFirstUpperBound, false);
1484         if (firstUpperBoundDelta != null) {
1485             delta = new SigUpperBoundsDelta(from, to);
1486             delta.setFirstUpperBoundDelta(firstUpperBoundDelta);
1487         } else {
1488             // normalize
1489             Set<ITypeReference> normalizedfrom = removeGeneralizations(
1490                     new HashSet<ITypeReference>(from));
1491             Set<ITypeReference> normalizedto = removeGeneralizations(
1492                     new HashSet<ITypeReference>(to));
1493 
1494             Set<ITypeReferenceDelta<?>> remainingUpperBoundsDelta =
1495                     compareTypes(normalizedfrom, normalizedto);
1496             if (remainingUpperBoundsDelta != null) {
1497                 delta = new SigUpperBoundsDelta(from, to);
1498                 delta.setRemainingUpperBoundDeltas(remainingUpperBoundsDelta);
1499             }
1500         }
1501         return delta;
1502     }
1503 
1504     private Set<ITypeReference> removeGeneralizations(
1505             Set<ITypeReference> bounds) {
1506         Set<ITypeReference> boundsCopy = new HashSet<ITypeReference>(bounds);
1507         for (ITypeReference type : bounds) {
1508             Iterator<ITypeReference> it = boundsCopy.iterator();
1509             while (it.hasNext()) {
1510                 ITypeReference superType = it.next();
1511                 if (isSuperClass(getClassDefinition(superType),
1512                         getClassDefinition(type))
1513                         || isSuperInterface(getClassDefinition(superType),
1514                                 getClassDefinition(type))) {
1515                     it.remove();
1516                 }
1517             }
1518         }
1519         return boundsCopy;
1520     }
1521 
1522     private IParameterizedTypeDelta compareParameterizedType(
1523             IParameterizedType from, IParameterizedType to,
1524             boolean ignoreTypeArguments) {
1525 
1526         SigParameterizedTypeDelta delta = null;
1527         // check raw type
1528         ITypeReferenceDelta<?> rawTypeDelta = compareType(from.getRawType(), to
1529                 .getRawType(), false);
1530         if (rawTypeDelta != null) {
1531             delta = new SigParameterizedTypeDelta(from, to);
1532             delta.setRawTypeDelta(rawTypeDelta);
1533         } else {
1534             // check owner type
1535             ITypeReferenceDelta<?> ownerTypeDelta = compareType(from
1536                     .getOwnerType(), to.getOwnerType(), false);
1537             if (ownerTypeDelta != null) {
1538                 delta = new SigParameterizedTypeDelta(from, to);
1539                 delta.setOwnerTypeDelta(ownerTypeDelta);
1540             } else {
1541                 // check argument type
1542                 if (!ignoreTypeArguments) {
1543                     Set<ITypeReferenceDelta<?>> argumentTypeDeltas =
1544                             compareTypeSequence(from.getTypeArguments(),
1545                                     to.getTypeArguments(), false);
1546                     if (argumentTypeDeltas != null) {
1547                         delta = new SigParameterizedTypeDelta(from, to);
1548                         delta.setArgumentTypeDeltas(argumentTypeDeltas);
1549                     }
1550                 }
1551             }
1552         }
1553         return delta;
1554     }
1555 
1556     private Set<ITypeReferenceDelta<? extends ITypeReference>> compareTypeSequence(
1557             List<ITypeReference> from, List<ITypeReference> to,
1558             boolean ignoreTypeArguments) {
1559         Set<ITypeReferenceDelta<?>> deltas =
1560                 new HashSet<ITypeReferenceDelta<?>>();
1561         if (from.size() != to.size()) {
1562 
1563             for (ITypeReference type : from) {
1564                 deltas.add(new SigTypeDelta<ITypeReference>(type, null));
1565             }
1566             for (ITypeReference type : to) {
1567                 deltas.add(new SigTypeDelta<ITypeReference>(null, type));
1568             }
1569             return deltas;
1570         }
1571 
1572         Iterator<? extends ITypeReference> fromIterator = from.iterator();
1573         Iterator<? extends ITypeReference> toIterator = to.iterator();
1574         while (fromIterator.hasNext() && toIterator.hasNext()) {
1575             ITypeReferenceDelta<?> delta = compareType(fromIterator.next(),
1576                     toIterator.next(), ignoreTypeArguments);
1577             if (delta != null) {
1578                 deltas.add(delta);
1579             }
1580         }
1581         return deltas.isEmpty() ? null : deltas;
1582     }
1583 
1584     private Set<ITypeReferenceDelta<? extends ITypeReference>> compareTypes(
1585             Set<ITypeReference> from, Set<ITypeReference> to) {
1586         return compareSets(from, to,
1587                 new SigComparator<ITypeReference, ITypeReferenceDelta<? extends ITypeReference>>() {
1588                     public ITypeReferenceDelta<? extends ITypeReference> createAddRemoveDelta(
1589                             ITypeReference from, ITypeReference to) {
1590                         return new SigTypeDelta<ITypeReference>(from, to);
1591                     }
1592 
1593                     public boolean considerEqualElement(ITypeReference from,
1594                             ITypeReference to) {
1595                         return considerEqualTypes(from, to);
1596                     }
1597 
1598                     public ITypeReferenceDelta<? extends ITypeReference> createChangedDelta(
1599                             ITypeReference from, ITypeReference to) {
1600                         return compareType(from, to, false);
1601                     }
1602                 });
1603     }
1604 
1605     private static interface SigComparator<T, S extends IDelta<? extends T>> {
1606         boolean considerEqualElement(T from, T to);
1607 
1608         S createChangedDelta(T from, T to);
1609 
1610         /**
1611          * If null is returned, it will be ignored.
1612          */
1613         S createAddRemoveDelta(T from, T to);
1614     }
1615 
1616 
1617     private <T, S extends IDelta<? extends T>> Set<S> compareSets(Set<T> from,
1618             Set<T> to, SigComparator<T, S> comparator) {
1619 
1620         Set<T> toCopy = new HashSet<T>(to);
1621         Set<S> deltas = new HashSet<S>();
1622 
1623         for (T fromType : from) {
1624             Iterator<T> toIterator = toCopy.iterator();
1625             boolean equals = false;
1626             boolean hasNext = toIterator.hasNext();
1627 
1628             while (hasNext && !equals) {
1629                 T toElement = toIterator.next();
1630                 equals = comparator.considerEqualElement(fromType, toElement);
1631                 if (equals) {
1632                     S compare = comparator.createChangedDelta(fromType,
1633                             toElement);
1634                     if (compare != null) {
1635                         deltas.add(compare);
1636                     }
1637                 }
1638                 hasNext = toIterator.hasNext();
1639             }
1640 
1641             if (equals) {
1642                 toIterator.remove();
1643             } else {
1644                 S delta = comparator.createAddRemoveDelta(fromType, null);
1645                 if (delta != null) {
1646                     deltas.add(delta);
1647                 }
1648             }
1649         }
1650 
1651         for (T type : toCopy) {
1652             S delta = comparator.createAddRemoveDelta(null, type);
1653             if (delta != null) {
1654                 deltas.add(delta);
1655             }
1656         }
1657         return deltas.isEmpty() ? null : deltas;
1658     }
1659 
1660 
1661     private ITypeReferenceDelta<?> compareArrayType(IArrayType from,
1662             IArrayType to) {
1663         ITypeReferenceDelta<?> componentTypeDelta = compareType(from
1664                 .getComponentType(), to.getComponentType(), false);
1665         if (componentTypeDelta != null) {
1666             SigArrayTypeDelta delta = new SigArrayTypeDelta(from, to);
1667             delta.setComponentTypeDelta(componentTypeDelta);
1668             return delta;
1669         }
1670         return null;
1671     }
1672 
1673     private ITypeReferenceDelta<IClassReference> compareClassReference(
1674             IClassReference fromRef, IClassReference toRef) {
1675         IClassDefinition from = fromRef.getClassDefinition();
1676         IClassDefinition to = toRef.getClassDefinition();
1677 
1678         if (!sameClassDefinition(from, to)) {
1679             return new SigClassReferenceDelta(fromRef, toRef);
1680         }
1681         return null;
1682     }
1683 
1684 
1685     private boolean sameClassDefinition(IClassDefinition from,
1686             IClassDefinition to) {
1687         boolean sameName = from.getName().equals(to.getName());
1688         boolean samePackage = from.getPackageName().equals(to.getPackageName());
1689 
1690         Kind fromKind = from.getKind();
1691         Kind toKind = to.getKind();
1692         boolean sameKind = (fromKind == null || toKind == null)
1693                 || fromKind.equals(toKind);
1694 
1695         return sameName && samePackage && sameKind;
1696     }
1697 
1698     private IPrimitiveTypeDelta comparePrimitiveType(IPrimitiveType from,
1699             IPrimitiveType to) {
1700         if (!from.equals(to)) {
1701             return new SigPrimitiveTypeDelta(from, to);
1702         }
1703         return null;
1704     }
1705 }
1706