diff --git a/groovy-eclipse.setup b/groovy-eclipse.setup index 5766cfbad5..48bd28bc68 100644 --- a/groovy-eclipse.setup +++ b/groovy-eclipse.setup @@ -301,7 +301,7 @@ + url="https://download.eclipse.org/eclipse/updates/4.26-I-builds/I20221123-1800"/> diff --git a/jdt-patch/e426/org.eclipse.jdt.core/compiler/org/eclipse/jdt/internal/compiler/lookup/TypeSystem.java b/jdt-patch/e426/org.eclipse.jdt.core/compiler/org/eclipse/jdt/internal/compiler/lookup/TypeSystem.java index 98c5ec9a6d..bb0ba7a319 100644 --- a/jdt-patch/e426/org.eclipse.jdt.core/compiler/org/eclipse/jdt/internal/compiler/lookup/TypeSystem.java +++ b/jdt-patch/e426/org.eclipse.jdt.core/compiler/org/eclipse/jdt/internal/compiler/lookup/TypeSystem.java @@ -28,47 +28,47 @@ /* TypeSystem: An abstraction responsible for keeping track of types that undergo "derivation" of some sort and the derived types produced thus. Here we use the term derivation in the Pascal sense and not per object oriented parlance. - + As of Java SE8, a type can undergo derivation in a bunch of ways: - + - By being created arrays out of, - By being parameterized, - By being created raw forms of, - By being the generic type which a wildcard type or an intersection type parameterizes, - By being annotated. - + It is the responsibility of the TypeSystem to serve as the factory and ensure that unique types are created and maintained. Most of the compiler depends on object identity given the derivation parameters are the same. E.g: If we dole out non-unique ParameterizedTypeBinding's for two attempts to create List, then one cannot be assigned to the other. - + Till Java SE7, we could manage to create a single binding for a type - not so with annotations coming into the picture. In order for two uses of the same type to be annotated differently, the bindings for them need to be distinct and cannot be shared. If we start doling out different bindings, then validating type identity and equivalence becomes an issue. - + What we do to solve the problem is produce different bindings when they need to be annotated differently, but stamp them with the same id (TypeBinding#id). Thus types that fail == or != could quickly be ascertained to be mere annotation variants by comparing the id field. - - This class is responsible for id stamping unique types. Only those types that are "derived from" in some form or participate in the - derivation in some form (by being type arguments say) get tracked and id'd here. A type which is not thus derived from in one form or + + This class is responsible for id stamping unique types. Only those types that are "derived from" in some form or participate in the + derivation in some form (by being type arguments say) get tracked and id'd here. A type which is not thus derived from in one form or the other or participate in the derivation thus - we are completely oblivious to. - - TypeBinding.id computation: For primitive types and certain "well known" types, id assignment happens elsewhere. Here we start with an - id value that is suitably high and proceed monotonically upwards so we will not accidentally collide with the id space in use already. - id assignments happens in such a way that a naked type and its annotated variants - variously annotated - would all share the same id. - Example: @T1 Map<@T2 String, @T3 Object> and Map<@T4 String, @T5 Object> and @T6 Map and @T7 Map and - Map would all share the same id since the unadorned naked type in each case is the same: Map. None - of this would share the id with Map. Briefly put, if you take a certain annotated type and strip it of all annotations - to come up with the naked type, that naked type and the annotated type would have the same id. Alternately, if you take a certain naked - type and arrive at the universe of all differently annotated types, they would all share the same id while their bindings could be different - + + TypeBinding.id computation: For primitive types and certain "well known" types, id assignment happens elsewhere. Here we start with an + id value that is suitably high and proceed monotonically upwards so we will not accidentally collide with the id space in use already. + id assignments happens in such a way that a naked type and its annotated variants - variously annotated - would all share the same id. + Example: @T1 Map<@T2 String, @T3 Object> and Map<@T4 String, @T5 Object> and @T6 Map and @T7 Map and + Map would all share the same id since the unadorned naked type in each case is the same: Map. None + of this would share the id with Map. Briefly put, if you take a certain annotated type and strip it of all annotations + to come up with the naked type, that naked type and the annotated type would have the same id. Alternately, if you take a certain naked + type and arrive at the universe of all differently annotated types, they would all share the same id while their bindings could be different - would be different unless they are identically annotated. - + Thus subsystems that are annotation agnostic could quickly ascertain binding equality by comparing the id field. */ public class TypeSystem { - + public final class HashedParameterizedTypes { - + private final class PTBKey extends ReferenceBinding { // extends ReferenceBinding so it can be used as wrapper protected ReferenceBinding type; // must ensure the type is resolved public TypeBinding[] arguments; @@ -111,7 +111,7 @@ public void swapUnresolved(UnresolvedReferenceBinding unresolvedType, ReferenceB } @Override public boolean equals(Object other) { - PTBKey that = (PTBKey) other; // homogeneous container. + PTBKey that = (PTBKey) other; // homogeneous container. return this.type == that.type && this.enclosingType == that.enclosingType && Util.effectivelyEqual(this.arguments, that.arguments); //$IDENTITY-COMPARISON$ } final int hash(TypeBinding b) { @@ -134,11 +134,11 @@ public int hashCode() { return hashCode; } } - + HashMap hashedParameterizedTypes = new HashMap<>(256); ParameterizedTypeBinding get(ReferenceBinding genericType, TypeBinding[] typeArguments, ReferenceBinding enclosingType, AnnotationBinding[] annotations) { - + ReferenceBinding unannotatedGenericType = (ReferenceBinding) getUnannotatedType(genericType); int typeArgumentsLength = typeArguments == null ? 0: typeArguments.length; TypeBinding [] unannotatedTypeArguments = typeArguments == null ? null : new TypeBinding[typeArgumentsLength]; @@ -146,7 +146,7 @@ ParameterizedTypeBinding get(ReferenceBinding genericType, TypeBinding[] typeArg unannotatedTypeArguments[i] = getUnannotatedType(typeArguments[i]); } ReferenceBinding unannotatedEnclosingType = enclosingType == null ? null : (ReferenceBinding) getUnannotatedType(enclosingType); - + PTBKey key = new PTBKey(unannotatedGenericType, unannotatedTypeArguments, unannotatedEnclosingType, null); ReferenceBinding genericTypeToMatch = unannotatedGenericType, enclosingTypeToMatch = unannotatedEnclosingType; TypeBinding [] typeArgumentsToMatch = unannotatedTypeArguments; @@ -162,7 +162,7 @@ ParameterizedTypeBinding get(ReferenceBinding genericType, TypeBinding[] typeArg continue; } if (parameterizedType.enclosingType != enclosingTypeToMatch //$IDENTITY-COMPARISON$ - || !Util.effectivelyEqual(parameterizedType.typeArguments(), typeArgumentsToMatch)) + || !Util.effectivelyEqual(parameterizedType.typeArguments(), typeArgumentsToMatch)) continue; if (Util.effectivelyEqual(annotations, parameterizedType.getTypeAnnotations())) return parameterizedType; @@ -179,34 +179,34 @@ void put (ReferenceBinding genericType, TypeBinding[] typeArguments, ReferenceBi unannotatedTypeArguments[i] = getUnannotatedType(typeArguments[i]); } ReferenceBinding unannotatedEnclosingType = enclosingType == null ? null : (ReferenceBinding) getUnannotatedType(enclosingType); - + PTBKey key = new PTBKey(unannotatedGenericType, unannotatedTypeArguments, unannotatedEnclosingType, TypeSystem.this.environment); - + ParameterizedTypeBinding [] parameterizedTypeBindings = this.hashedParameterizedTypes.get(key); int slot; if (parameterizedTypeBindings == null) { slot = 0; parameterizedTypeBindings = new ParameterizedTypeBinding[1]; - } else { + } else { slot = parameterizedTypeBindings.length; System.arraycopy(parameterizedTypeBindings, 0, parameterizedTypeBindings = new ParameterizedTypeBinding[slot + 1], 0, slot); } parameterizedTypeBindings[slot] = parameterizedType; this.hashedParameterizedTypes.put(key, parameterizedTypeBindings); } - } - + } + private int typeid = TypeIds.T_LastWellKnownTypeId; - private TypeBinding [][] types; + private TypeBinding [][] types; protected HashedParameterizedTypes parameterizedTypes; // auxiliary fast lookup table for parameterized types. private SimpleLookupTable annotationTypes; // cannot store in types, since AnnotationBinding is not a TypeBinding and we don't want types to operate at Binding level. LookupEnvironment environment; - + public TypeSystem(LookupEnvironment environment) { this.environment = environment; this.annotationTypes = new SimpleLookupTable(16); this.typeid = TypeIds.T_LastWellKnownTypeId; - this.types = new TypeBinding[TypeIds.T_LastWellKnownTypeId * 2][]; + this.types = new TypeBinding[TypeIds.T_LastWellKnownTypeId * 2][]; this.parameterizedTypes = new HashedParameterizedTypes(); } @@ -240,7 +240,7 @@ public final TypeBinding getUnannotatedType(TypeBinding type) { if (urb != null && urb.id == TypeIds.NoId) urb.id = type.id; } - + return this.types[type.id][0] = type; } @@ -271,9 +271,9 @@ public TypeBinding[] getAnnotatedTypes(TypeBinding type) { /* Note: parameters will not have type type annotations if lookup environment directly uses TypeSystem as its typeSystem. When ATS is used however they may be annotated and we need to materialize the unannotated versions and work on them. - + See ArrayBinding.swapUnresolved for further special case handling if incoming leafType is a URB that would resolve to a raw type later. - */ + */ public ArrayBinding getArrayType(TypeBinding leafType, int dimensions) { if (leafType instanceof ArrayBinding) { dimensions += leafType.dimensions(); @@ -284,7 +284,7 @@ public ArrayBinding getArrayType(TypeBinding leafType, int dimensions) { int i, length = derivedTypes.length; for (i = 0; i < length; i++) { TypeBinding derivedType = derivedTypes[i]; - if (derivedType == null) + if (derivedType == null) break; if (!derivedType.isArrayType() || derivedType.hasTypeAnnotations()) continue; @@ -302,7 +302,7 @@ public ArrayBinding getArrayType(TypeBinding leafType, int dimensions) { this.types[this.typeid] = new TypeBinding[1]; return (ArrayBinding) (this.types[arrayType.id = this.typeid++][0] = arrayType); } - + public ArrayBinding getArrayType(TypeBinding leafComponentType, int dimensions, AnnotationBinding[] annotations) { return getArrayType(leafComponentType, dimensions); } @@ -313,7 +313,7 @@ public ReferenceBinding getMemberType(ReferenceBinding memberType, ReferenceBind /* Note: parameters will not have type type annotations if lookup environment directly uses TypeSystem. When AnnotatableTypeSystem is in use they may and we need to materialize the unannotated versions and work on them. - */ + */ public ParameterizedTypeBinding getParameterizedType(ReferenceBinding genericType, TypeBinding[] typeArguments, ReferenceBinding enclosingType) { ReferenceBinding unannotatedGenericType = (ReferenceBinding) getUnannotatedType(genericType); // getUnannotatedType may have replaced URB by resolvedType @@ -329,7 +329,7 @@ public ParameterizedTypeBinding getParameterizedType(ReferenceBinding genericTyp ReferenceBinding unannotatedEnclosingType = enclosingType == null ? null : (ReferenceBinding) getUnannotatedType(enclosingType); ParameterizedTypeBinding parameterizedType = this.parameterizedTypes.get(unannotatedGenericType, unannotatedTypeArguments, unannotatedEnclosingType, Binding.NO_ANNOTATIONS); - if (parameterizedType != null) + if (parameterizedType != null) return parameterizedType; parameterizedType = new ParameterizedTypeBinding(unannotatedGenericType, unannotatedTypeArguments, unannotatedEnclosingType, this.environment); @@ -348,19 +348,19 @@ public ParameterizedTypeBinding getParameterizedType(ReferenceBinding genericTyp /* Note: Parameters will not have type type annotations if lookup environment directly uses TypeSystem. However when AnnotatableTypeSystem is in use, they may and we need to materialize the unannotated versions and work on them. - */ + */ public RawTypeBinding getRawType(ReferenceBinding genericType, ReferenceBinding enclosingType) { if (!genericType.hasEnclosingInstanceContext() && enclosingType != null) { enclosingType = (ReferenceBinding) enclosingType.original(); } ReferenceBinding unannotatedGenericType = (ReferenceBinding) getUnannotatedType(genericType); ReferenceBinding unannotatedEnclosingType = enclosingType == null ? null : (ReferenceBinding) getUnannotatedType(enclosingType); - + TypeBinding[] derivedTypes = this.types[unannotatedGenericType.id]; int i, length = derivedTypes.length; for (i = 0; i < length; i++) { TypeBinding derivedType = derivedTypes[i]; - if (derivedType == null) + if (derivedType == null) break; if (!derivedType.isRawType() || derivedType.actualType() != unannotatedGenericType || derivedType.hasTypeAnnotations()) //$IDENTITY-COMPARISON$ continue; @@ -372,7 +372,7 @@ public RawTypeBinding getRawType(ReferenceBinding genericType, ReferenceBinding System.arraycopy(derivedTypes, 0, derivedTypes = new TypeBinding[length * 2], 0, length); this.types[unannotatedGenericType.id] = derivedTypes; } - + TypeBinding rawTytpe = derivedTypes[i] = new RawTypeBinding(unannotatedGenericType, unannotatedEnclosingType, this.environment); int typesLength = this.types.length; if (this.typeid == typesLength) @@ -380,18 +380,18 @@ public RawTypeBinding getRawType(ReferenceBinding genericType, ReferenceBinding this.types[this.typeid] = new TypeBinding[1]; return (RawTypeBinding) (this.types[rawTytpe.id = this.typeid++][0] = rawTytpe); } - + public RawTypeBinding getRawType(ReferenceBinding genericType, ReferenceBinding enclosingType, AnnotationBinding[] annotations) { return getRawType(genericType, enclosingType); } /* Parameters will not have type type annotations if lookup environment directly uses TypeSystem. When AnnotatableTypeSystem is in use, they may and we need to materialize the unannotated versions and work on them. - */ + */ public WildcardBinding getWildcard(ReferenceBinding genericType, int rank, TypeBinding bound, TypeBinding[] otherBounds, int boundKind) { if (genericType == null) // pseudo wildcard denoting composite bounds for lub computation genericType = ReferenceBinding.LUB_GENERIC; - + ReferenceBinding unannotatedGenericType = (ReferenceBinding) getUnannotatedType(genericType); int otherBoundsLength = otherBounds == null ? 0: otherBounds.length; TypeBinding [] unannotatedOtherBounds = otherBounds == null ? null : new TypeBinding[otherBoundsLength]; @@ -406,7 +406,7 @@ public WildcardBinding getWildcard(ReferenceBinding genericType, int rank, TypeB int i, length = derivedTypes.length; for (i = 0; i < length; i++) { TypeBinding derivedType = derivedTypes[i]; - if (derivedType == null) + if (derivedType == null) break; if (!derivedType.isWildcard() || derivedType.actualType() != unannotatedGenericType || derivedType.hasTypeAnnotations()) //$IDENTITY-COMPARISON$ continue; @@ -415,29 +415,29 @@ public WildcardBinding getWildcard(ReferenceBinding genericType, int rank, TypeB if (Util.effectivelyEqual(derivedType.additionalBounds(), unannotatedOtherBounds)) return (WildcardBinding) derivedType; } - + if (i == length) { System.arraycopy(derivedTypes, 0, derivedTypes = new TypeBinding[length * 2], 0, length); this.types[useDerivedTypesOfBound ? unannotatedBound.id :unannotatedGenericType.id] = derivedTypes; } TypeBinding wildcard = derivedTypes[i] = new WildcardBinding(unannotatedGenericType, rank, unannotatedBound, unannotatedOtherBounds, boundKind, this.environment); - + int typesLength = this.types.length; if (this.typeid == typesLength) System.arraycopy(this.types, 0, this.types = new TypeBinding[typesLength * 2][], 0, typesLength); this.types[this.typeid] = new TypeBinding[1]; return (WildcardBinding) (this.types[wildcard.id = this.typeid++][0] = wildcard); } - + // No need for an override in ATS, since interning is position specific and either the wildcard there is annotated or not. public final CaptureBinding getCapturedWildcard(WildcardBinding wildcard, ReferenceBinding contextType, int start, int end, ASTNode cud, int id) { - + WildcardBinding unannotatedWildcard = (WildcardBinding) getUnannotatedType(wildcard); TypeBinding[] derivedTypes = this.types[unannotatedWildcard.id]; // by construction, cachedInfo != null now. int i, length = derivedTypes.length; - + /* Search backwards looking at recent captures, if we encounter a capture from a different compilation unit, this is a fresh uninterned capture. - While compiling one file, we may reach into another file to build structure, we should not compile method bodies there, so we expect to see + While compiling one file, we may reach into another file to build structure, we should not compile method bodies there, so we expect to see all captures from the same file together without being interleaved by captures from other files. */ int nullSlot = length; @@ -447,7 +447,7 @@ public final CaptureBinding getCapturedWildcard(WildcardBinding wildcard, Refere break; } TypeBinding derivedType = derivedTypes[i]; - if (derivedType == null) { + if (derivedType == null) { nullSlot = i; continue; } @@ -462,7 +462,7 @@ public final CaptureBinding getCapturedWildcard(WildcardBinding wildcard, Refere continue; return prior; } - + if (i == length) { System.arraycopy(derivedTypes, 0, derivedTypes = new TypeBinding[length * 2], 0, length); this.types[unannotatedWildcard.id] = derivedTypes; @@ -470,7 +470,7 @@ public final CaptureBinding getCapturedWildcard(WildcardBinding wildcard, Refere return (CaptureBinding) (derivedTypes[i] = new CaptureBinding(wildcard, contextType, start, end, cud, id)); // the above constructor already registers the capture, don't repeat that here } - + public WildcardBinding getWildcard(ReferenceBinding genericType, int rank, TypeBinding bound, TypeBinding[] otherBounds, int boundKind, AnnotationBinding[] annotations) { return getWildcard(genericType, rank, bound, otherBounds, boundKind); } @@ -478,16 +478,16 @@ public WildcardBinding getWildcard(ReferenceBinding genericType, int rank, TypeB public TypeBinding getAnnotatedType(TypeBinding type, AnnotationBinding[][] annotations) { return type; // Nothing to do for plain vanilla type system. } - + protected final TypeBinding /* @NonNull */ [] getDerivedTypes(TypeBinding keyType) { keyType = getUnannotatedType(keyType); return this.types[keyType.id]; } - + private TypeBinding cacheDerivedType(TypeBinding keyType, TypeBinding derivedType) { if (keyType == null || derivedType == null || keyType.id == TypeIds.NoId) throw new IllegalStateException(); - + TypeBinding[] derivedTypes = this.types[keyType.id]; // binary search for the *earliest* slot with a null reference. By design and construction, a null value will never be followed by a valid derived type. int first, last,length = derivedTypes.length; @@ -498,7 +498,7 @@ private TypeBinding cacheDerivedType(TypeBinding keyType, TypeBinding derivedTyp if (i == first || i > 0 && derivedTypes[i - 1] != null) break; last = i - 1; - } else { + } else { first = i + 1; } i = (first + last) / 2; @@ -509,19 +509,19 @@ private TypeBinding cacheDerivedType(TypeBinding keyType, TypeBinding derivedTyp } return derivedTypes[i] = derivedType; } - + protected final TypeBinding cacheDerivedType(TypeBinding keyType, TypeBinding nakedType, TypeBinding derivedType) { - + /* Cache the derived type, tagging it as a derivative of both the key type and the naked type. E.g: int @NonNull [] would be tagged as a derived type of both int and int []. This is not needed for correctness, but for annotated object reuse. We provide two alternate ways to - annotate a type: - + annotate a type: + Taking parameterized types as an example, a call to getParamaterizedType can be made with annotations to create @NonNull List<@NonNull String> in one stroke. Or a parameterized type can be created first and then annotated via getAnnotatedType. In the former case, the tables get looked up with List as the key, in the latter with List as the key. - + Binary vs source, substitutions, annotation re-attribution from SE7 locations etc trigger these alternate code paths. Unless care is exercised, we will end up with duplicate objects (that share the same TypeBinding.id => correctness is not an issue, but memory wastage is) @@ -532,16 +532,16 @@ alternate code paths. Unless care is exercised, we will end up with duplicate ob } return derivedType; } - + /* Return a unique annotation binding for an annotation with either no or all default element-value pairs. - We may return a resolved annotation when requested for unresolved one, but not vice versa. + We may return a resolved annotation when requested for unresolved one, but not vice versa. */ public final AnnotationBinding getAnnotationType(ReferenceBinding annotationType, boolean requiredResolved) { AnnotationBinding annotation = (AnnotationBinding) this.annotationTypes.get(annotationType); if (annotation == null) { if (requiredResolved) annotation = new AnnotationBinding(annotationType, Binding.NO_ELEMENT_VALUE_PAIRS); - else + else annotation = new UnresolvedAnnotationBinding(annotationType, Binding.NO_ELEMENT_VALUE_PAIRS, this.environment); this.annotationTypes.put(annotationType, annotation); } @@ -571,7 +571,7 @@ public void reset() { this.types = new TypeBinding[TypeIds.T_LastWellKnownTypeId * 2][]; this.parameterizedTypes = new HashedParameterizedTypes(); } - + public void updateCaches(UnresolvedReferenceBinding unresolvedType, ReferenceBinding resolvedType) { final int unresolvedTypeId = unresolvedType.id; if (resolvedType.id != TypeIds.NoId) { @@ -608,13 +608,13 @@ public final TypeBinding getIntersectionType18(ReferenceBinding[] intersectingTy TypeBinding keyType = intersectingTypes[0]; if (keyType == null || intersectingTypesLength == 1) return keyType; - + TypeBinding[] derivedTypes = getDerivedTypes(keyType); int i, length = derivedTypes.length; next: for (i = 0; i < length; i++) { TypeBinding derivedType = derivedTypes[i]; - if (derivedType == null) + if (derivedType == null) break; if (!derivedType.isIntersectionType18()) continue; @@ -624,12 +624,12 @@ public final TypeBinding getIntersectionType18(ReferenceBinding[] intersectingTy for (int j = 0; j < intersectingTypesLength; j++) { if (intersectingTypes[j] != priorIntersectingTypes[j]) //$IDENTITY-COMPARISON$ continue next; - } + } return derivedType; } return cacheDerivedType(keyType, new IntersectionTypeBinding18(intersectingTypes, this.environment)); } - + /** * If a TVB was created with a dummy declaring element and needs to be fixed now, * make sure that this update affects all early clones, too. diff --git a/pom.xml b/pom.xml index c4e38faba9..74d74cd1fe 100644 --- a/pom.xml +++ b/pom.xml @@ -129,7 +129,7 @@ milestone p2 - https://download.eclipse.org/eclipse/updates/4.26-I-builds/I20221117-1330 + https://download.eclipse.org/eclipse/updates/4.26-I-builds/I20221123-1800