diff options
-rw-r--r-- | weaver/src/org/aspectj/weaver/UnresolvedType.java | 1202 |
1 files changed, 645 insertions, 557 deletions
diff --git a/weaver/src/org/aspectj/weaver/UnresolvedType.java b/weaver/src/org/aspectj/weaver/UnresolvedType.java index 2aca3954e..d1704380b 100644 --- a/weaver/src/org/aspectj/weaver/UnresolvedType.java +++ b/weaver/src/org/aspectj/weaver/UnresolvedType.java @@ -25,63 +25,48 @@ import org.aspectj.apache.bcel.classfile.Signature.ClassSignature; import org.aspectj.weaver.tools.Traceable; /** - * A UnresolvedType represents a type to the weaver. It has a basic signature that knows - * nothing about type variables, type parameters, etc.. UnresolvedTypes are resolved in some World - * (a repository of types). When a UnresolvedType is resolved it turns into a - * ResolvedType which may be a primitive type, or a ReferenceType. - * ReferenceTypes may refer to simple, generic, parameterized or type-variable - * based reference types. A ReferenceType is backed by a delegate that provides - * information about the type based on some repository (currently either BCEL - * or an EclipseSourceType, but in the future we probably need to support - * java.lang.reflect based delegates too). + * A UnresolvedType represents a type to the weaver. It has a basic signature that knows nothing about type variables, type + * parameters, etc.. UnresolvedTypes are resolved in some World (a repository of types). When a UnresolvedType is resolved it turns + * into a ResolvedType which may be a primitive type, or a ReferenceType. ReferenceTypes may refer to simple, generic, parameterized + * or type-variable based reference types. A ReferenceType is backed by a delegate that provides information about the type based on + * some repository (currently either BCEL or an EclipseSourceType, but in the future we probably need to support java.lang.reflect + * based delegates too). * - * Every UnresolvedType has a signature, the unique key for the type in the world. - * + * Every UnresolvedType has a signature, the unique key for the type in the world. * - * TypeXs are fully aware of their complete type information (there is no - * erasure in the UnresolvedType world). To achieve this, the signature of TypeXs - * combines the basic Java signature and the generic signature information - * into one complete signature. + * + * TypeXs are fully aware of their complete type information (there is no erasure in the UnresolvedType world). To achieve this, the + * signature of TypeXs combines the basic Java signature and the generic signature information into one complete signature. * * The format of a UnresolvedType signature is as follows: * - * a simple (non-generic, non-parameterized) type has the as its signature - * the Java signature. - * e.g. Ljava/lang/String; + * a simple (non-generic, non-parameterized) type has the as its signature the Java signature. e.g. Ljava/lang/String; * - * a generic type has signature: - * TypeParamsOpt ClassSig SuperClassSig SuperIntfListOpt + * a generic type has signature: TypeParamsOpt ClassSig SuperClassSig SuperIntfListOpt * - * following the Generic signature grammar in the JVM spec., but with the - * addition of the ClassSignature (which is not in the generic signature). In addition - * type variable names are replaced by a simple number which represents their - * declaration order in the type declaration. + * following the Generic signature grammar in the JVM spec., but with the addition of the ClassSignature (which is not in the + * generic signature). In addition type variable names are replaced by a simple number which represents their declaration order in + * the type declaration. * - * e.g. public class Foo<T extends Number> would have signature: - * <1:Ljava/lang/Number>Lorg.xyz.Foo;Ljava/lang/Object; + * e.g. public class Foo<T extends Number> would have signature: <1:Ljava/lang/Number>Lorg.xyz.Foo;Ljava/lang/Object; * - * A parameterized type is a distinct type in the world with its own signature - * following the grammar: + * A parameterized type is a distinct type in the world with its own signature following the grammar: * * TypeParamsOpt ClassSig<ParamSigList>; * - * but with the L for the class sig replaced by "P". For example List<String> has - * signature + * but with the L for the class sig replaced by "P". For example List<String> has signature * * Pjava/util/List<Ljava/lang/String>; * - * and List<T> in the following class : - * class Foo<T> { List<T> lt; } + * and List<T> in the following class : class Foo<T> { List<T> lt; } * - * has signature: - * <1:>Pjava/util/List<T1;>; + * has signature: <1:>Pjava/util/List<T1;>; * - * A typex that represents a type variable has its own unique signature, - * following the grammar for a FormalTypeParameter in the JVM spec. + * A typex that represents a type variable has its own unique signature, following the grammar for a FormalTypeParameter in the JVM + * spec. * - * A generic typex has its true signature and also an erasure signature. - * Both of these are keys pointing to the same UnresolvedType in the world. For example - * List has signature: + * A generic typex has its true signature and also an erasure signature. Both of these are keys pointing to the same UnresolvedType + * in the world. For example List has signature: * * <1:>Ljava/util/List;Ljava/lang/Object; * @@ -89,241 +74,274 @@ import org.aspectj.weaver.tools.Traceable; * * Ljava/util/List; * - * Generics wildcards introduce their own special signatures for type parameters. - * The wildcard ? has signature * - * The wildcard ? extends Foo has signature +LFoo; - * The wildcard ? super Foo has signature -LFoo; + * Generics wildcards introduce their own special signatures for type parameters. The wildcard ? has signature * The wildcard ? + * extends Foo has signature +LFoo; The wildcard ? super Foo has signature -LFoo; */ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { // common types referred to by the weaver - public static final UnresolvedType[] NONE = new UnresolvedType[0]; - public static final UnresolvedType OBJECT = forSignature("Ljava/lang/Object;"); - public static final UnresolvedType OBJECTARRAY = forSignature("[Ljava/lang/Object;"); - public static final UnresolvedType CLONEABLE = forSignature("Ljava/lang/Cloneable;"); - public static final UnresolvedType SERIALIZABLE = forSignature("Ljava/io/Serializable;"); - public static final UnresolvedType THROWABLE = forSignature("Ljava/lang/Throwable;"); - public static final UnresolvedType RUNTIME_EXCEPTION = forSignature("Ljava/lang/RuntimeException;"); - public static final UnresolvedType ERROR = forSignature("Ljava/lang/Error;"); - public static final UnresolvedType AT_INHERITED = forSignature("Ljava/lang/annotation/Inherited;"); - public static final UnresolvedType AT_RETENTION = forSignature("Ljava/lang/annotation/Retention;"); - public static final UnresolvedType ENUM = forSignature("Ljava/lang/Enum;"); - public static final UnresolvedType ANNOTATION = forSignature("Ljava/lang/annotation/Annotation;"); - public static final UnresolvedType JAVA_LANG_CLASS = forSignature("Ljava/lang/Class;"); - public static final UnresolvedType JAVA_LANG_STRING = forSignature("Ljava/lang/String;"); - public static final UnresolvedType JAVA_LANG_EXCEPTION = forSignature("Ljava/lang/Exception;"); - public static final UnresolvedType JAVA_LANG_REFLECT_METHOD = forSignature("Ljava/lang/reflect/Method;"); - public static final UnresolvedType SUPPRESS_AJ_WARNINGS = forSignature("Lorg/aspectj/lang/annotation/SuppressAjWarnings;"); - public static final UnresolvedType AT_TARGET = forSignature("Ljava/lang/annotation/Target;"); - public static final UnresolvedType SOMETHING = new UnresolvedType("?"); - public static final UnresolvedType[] ARRAY_WITH_JUST_OBJECT = new UnresolvedType[]{OBJECT}; - public static final UnresolvedType JOINPOINT_STATICPART = forSignature("Lorg/aspectj/lang/JoinPoint$StaticPart;"); - public static final UnresolvedType JOINPOINT_ENCLOSINGSTATICPART = forSignature("Lorg/aspectj/lang/JoinPoint$EnclosingStaticPart;"); - - // this doesn't belong here and will get moved to ResolvedType later in the refactoring + public static final UnresolvedType[] NONE = new UnresolvedType[0]; + public static final UnresolvedType OBJECT = forSignature("Ljava/lang/Object;"); + public static final UnresolvedType OBJECTARRAY = forSignature("[Ljava/lang/Object;"); + public static final UnresolvedType CLONEABLE = forSignature("Ljava/lang/Cloneable;"); + public static final UnresolvedType SERIALIZABLE = forSignature("Ljava/io/Serializable;"); + public static final UnresolvedType THROWABLE = forSignature("Ljava/lang/Throwable;"); + public static final UnresolvedType RUNTIME_EXCEPTION = forSignature("Ljava/lang/RuntimeException;"); + public static final UnresolvedType ERROR = forSignature("Ljava/lang/Error;"); + public static final UnresolvedType AT_INHERITED = forSignature("Ljava/lang/annotation/Inherited;"); + public static final UnresolvedType AT_RETENTION = forSignature("Ljava/lang/annotation/Retention;"); + public static final UnresolvedType ENUM = forSignature("Ljava/lang/Enum;"); + public static final UnresolvedType ANNOTATION = forSignature("Ljava/lang/annotation/Annotation;"); + public static final UnresolvedType JAVA_LANG_CLASS = forSignature("Ljava/lang/Class;"); + public static final UnresolvedType JAVA_LANG_CLASS_ARRAY = forSignature("[Ljava/lang/Class;"); + public static final UnresolvedType JAVA_LANG_STRING = forSignature("Ljava/lang/String;"); + public static final UnresolvedType JAVA_LANG_EXCEPTION = forSignature("Ljava/lang/Exception;"); + public static final UnresolvedType JAVA_LANG_REFLECT_METHOD = forSignature("Ljava/lang/reflect/Method;"); + public static final UnresolvedType JAVA_LANG_ANNOTATION = UnresolvedType.forSignature("Ljava.lang.annotation.Annotation;"); + public static final UnresolvedType SUPPRESS_AJ_WARNINGS = forSignature("Lorg/aspectj/lang/annotation/SuppressAjWarnings;"); + public static final UnresolvedType AT_TARGET = forSignature("Ljava/lang/annotation/Target;"); + public static final UnresolvedType SOMETHING = new UnresolvedType("?"); + public static final UnresolvedType[] ARRAY_WITH_JUST_OBJECT = new UnresolvedType[] { OBJECT }; + public static final UnresolvedType JOINPOINT_STATICPART = forSignature("Lorg/aspectj/lang/JoinPoint$StaticPart;"); + public static final UnresolvedType JOINPOINT_ENCLOSINGSTATICPART = forSignature("Lorg/aspectj/lang/JoinPoint$EnclosingStaticPart;"); + + // this doesn't belong here and will get moved to ResolvedType later in the refactoring public static final String MISSING_NAME = "@missing@"; - // OPTIMIZE I dont think you can ask something unresolved what kind of type it is, how can it always know? Push down into resolvedtype + // OPTIMIZE I dont think you can ask something unresolved what kind of type it is, how can it always know? Push down into + // resolvedtype // that will force references to resolvedtypes to be correct rather than relying on unresolvedtypes to answer questions - protected TypeKind typeKind = TypeKind.SIMPLE; // what kind of type am I? + protected TypeKind typeKind = TypeKind.SIMPLE; // what kind of type am I? /** * THE SIGNATURE - see the comments above for how this is defined */ - protected String signature; + protected String signature; - /** - * The erasure of the signature. Contains only the Java signature of the type - * with all supertype, superinterface, type variable, and parameter information - * removed. - */ + /** + * The erasure of the signature. Contains only the Java signature of the type with all supertype, superinterface, type variable, + * and parameter information removed. + */ protected String signatureErasure; - /** - * Iff isParameterized(), then these are the type parameters - */ - protected UnresolvedType[] typeParameters; + /** + * Iff isParameterized(), then these are the type parameters + */ + protected UnresolvedType[] typeParameters; /** - * Iff isGeneric(), then these are the type variables declared on the type - * Iff isParameterized(), then these are the type variables bound as parameters - * in the type + * Iff isGeneric(), then these are the type variables declared on the type Iff isParameterized(), then these are the type + * variables bound as parameters in the type */ // OPTIMIZE should be no state in here that will damage whether equals() is correct... protected TypeVariable[] typeVariables; - - public boolean isPrimitiveType() { return typeKind == TypeKind.PRIMITIVE; } - public boolean isSimpleType() { return typeKind == TypeKind.SIMPLE; } - public boolean isRawType() { return typeKind == TypeKind.RAW; } - public boolean isGenericType() { return typeKind == TypeKind.GENERIC; } - public boolean isParameterizedType() { return typeKind == TypeKind.PARAMETERIZED; } - public boolean isTypeVariableReference() { return typeKind == TypeKind.TYPE_VARIABLE; } - public boolean isGenericWildcard() { return typeKind == TypeKind.WILDCARD; } - public TypeKind getTypekind() { return typeKind;} - - // for any reference type, we can get some extra information... - public final boolean isArray() { - return signature.length() > 0 && signature.charAt(0) == '['; - } - - /** - * Equality is checked based on the underlying signature. - * {@link ResolvedType} objects' equals is by reference. - */ - public boolean equals(Object other) { - if (! (other instanceof UnresolvedType)) return false; - return signature.equals(((UnresolvedType) other).signature); - } - - /** - * Equality is checked based on the underlying signature, so the hash code - * of a particular type is the hash code of its signature string. - */ - public final int hashCode() { - return signature.hashCode(); - } - - - protected UnresolvedType(String signature) { - this.signature = signature; - this.signatureErasure = signature; - } - - protected UnresolvedType(String signature, String signatureErasure) { - this.signature = signature; - this.signatureErasure = signatureErasure; - } - - // called from TypeFactory - public UnresolvedType(String signature, String signatureErasure, UnresolvedType[] typeParams) { - this.signature = signature; - this.signatureErasure = signatureErasure; - this.typeParameters = typeParams; - if (typeParams != null) this.typeKind = TypeKind.PARAMETERIZED; - } - - // ---- Things we can do without a world - - /** - * This is the size of this type as used in JVM. - */ - public int getSize() { - return 1; - } - - - public static UnresolvedType makeArray(UnresolvedType base, int dims) { - StringBuffer sig = new StringBuffer(); - for (int i=0; i < dims; i++) sig.append("["); - sig.append(base.getSignature()); - return UnresolvedType.forSignature(sig.toString()); - } - - /** - * NOTE: Use forSignature() if you can, it'll be cheaper ! - * Constructs a UnresolvedType for a java language type name. For example: - * - * <blockquote><pre> - * UnresolvedType.forName("java.lang.Thread[]") - * UnresolvedType.forName("int") - * </pre></blockquote> - * - * Types may equivalently be produced by this or by {@link #forSignature(String)}. - * - * <blockquote><pre> - * UnresolvedType.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") - * UnresolvedType.forName("int").equals(Type.forSignature("I")) - * </pre></blockquote> - * - * @param name the java language type name in question. - * @return a type object representing that java language type. - */ - // OPTIMIZE change users of this to use forSignature, especially for simple cases - public static UnresolvedType forName(String name) { - return forSignature(nameToSignature(name)); - } - - /** Constructs a UnresolvedType for each java language type name in an incoming array. - * - * @param names an array of java language type names. - * @return an array of UnresolvedType objects. - * @see #forName(String) - */ - public static UnresolvedType[] forNames(String[] names) { - UnresolvedType[] ret = new UnresolvedType[names.length]; - for (int i = 0, len = names.length; i < len; i++) { - ret[i] = UnresolvedType.forName(names[i]); - } - return ret; - } - - public static UnresolvedType forGenericType(String name,TypeVariable[] tvbs,String genericSig) { + public boolean isPrimitiveType() { + return typeKind == TypeKind.PRIMITIVE; + } + + public boolean isSimpleType() { + return typeKind == TypeKind.SIMPLE; + } + + public boolean isRawType() { + return typeKind == TypeKind.RAW; + } + + public boolean isGenericType() { + return typeKind == TypeKind.GENERIC; + } + + public boolean isParameterizedType() { + return typeKind == TypeKind.PARAMETERIZED; + } + + public boolean isTypeVariableReference() { + return typeKind == TypeKind.TYPE_VARIABLE; + } + + public boolean isGenericWildcard() { + return typeKind == TypeKind.WILDCARD; + } + + public TypeKind getTypekind() { + return typeKind; + } + + // for any reference type, we can get some extra information... + public final boolean isArray() { + return signature.length() > 0 && signature.charAt(0) == '['; + } + + /** + * Equality is checked based on the underlying signature. {@link ResolvedType} objects' equals is by reference. + */ + public boolean equals(Object other) { + if (!(other instanceof UnresolvedType)) + return false; + return signature.equals(((UnresolvedType) other).signature); + } + + /** + * Equality is checked based on the underlying signature, so the hash code of a particular type is the hash code of its + * signature string. + */ + public final int hashCode() { + return signature.hashCode(); + } + + protected UnresolvedType(String signature) { + this.signature = signature; + this.signatureErasure = signature; + } + + protected UnresolvedType(String signature, String signatureErasure) { + this.signature = signature; + this.signatureErasure = signatureErasure; + } + + // called from TypeFactory + public UnresolvedType(String signature, String signatureErasure, UnresolvedType[] typeParams) { + this.signature = signature; + this.signatureErasure = signatureErasure; + this.typeParameters = typeParams; + if (typeParams != null) + this.typeKind = TypeKind.PARAMETERIZED; + } + + // ---- Things we can do without a world + + /** + * This is the size of this type as used in JVM. + */ + public int getSize() { + return 1; + } + + public static UnresolvedType makeArray(UnresolvedType base, int dims) { + StringBuffer sig = new StringBuffer(); + for (int i = 0; i < dims; i++) + sig.append("["); + sig.append(base.getSignature()); + return UnresolvedType.forSignature(sig.toString()); + } + + /** + * NOTE: Use forSignature() if you can, it'll be cheaper ! Constructs a UnresolvedType for a java language type name. For + * example: + * + * <blockquote> + * + * <pre> + * UnresolvedType.forName("java.lang.Thread[]") + * UnresolvedType.forName("int") + * </pre> + * + * </blockquote> + * + * Types may equivalently be produced by this or by {@link #forSignature(String)}. + * + * <blockquote> + * + * <pre> + * UnresolvedType.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") + * UnresolvedType.forName("int").equals(Type.forSignature("I")) + * </pre> + * + * </blockquote> + * + * @param name the java language type name in question. + * @return a type object representing that java language type. + */ + // OPTIMIZE change users of this to use forSignature, especially for simple cases + public static UnresolvedType forName(String name) { + return forSignature(nameToSignature(name)); + } + + /** + * Constructs a UnresolvedType for each java language type name in an incoming array. + * + * @param names an array of java language type names. + * @return an array of UnresolvedType objects. + * @see #forName(String) + */ + public static UnresolvedType[] forNames(String[] names) { + UnresolvedType[] ret = new UnresolvedType[names.length]; + for (int i = 0, len = names.length; i < len; i++) { + ret[i] = UnresolvedType.forName(names[i]); + } + return ret; + } + + public static UnresolvedType forGenericType(String name, TypeVariable[] tvbs, String genericSig) { // TODO asc generics needs a declared sig String sig = nameToSignature(name); UnresolvedType ret = UnresolvedType.forSignature(sig); - ret.typeKind=TypeKind.GENERIC; + ret.typeKind = TypeKind.GENERIC; ret.typeVariables = tvbs; ret.signatureErasure = sig; - return ret; - } - - public static UnresolvedType forGenericTypeSignature(String sig,String declaredGenericSig) { - UnresolvedType ret = UnresolvedType.forSignature(sig); - ret.typeKind=TypeKind.GENERIC; - - ClassSignature csig = new GenericSignatureParser().parseAsClassSignature(declaredGenericSig); - - Signature.FormalTypeParameter[] ftps = csig.formalTypeParameters; - ret.typeVariables = new TypeVariable[ftps.length]; - for (int i = 0; i < ftps.length; i++) { + return ret; + } + + public static UnresolvedType forGenericTypeSignature(String sig, String declaredGenericSig) { + UnresolvedType ret = UnresolvedType.forSignature(sig); + ret.typeKind = TypeKind.GENERIC; + + ClassSignature csig = new GenericSignatureParser().parseAsClassSignature(declaredGenericSig); + + Signature.FormalTypeParameter[] ftps = csig.formalTypeParameters; + ret.typeVariables = new TypeVariable[ftps.length]; + for (int i = 0; i < ftps.length; i++) { Signature.FormalTypeParameter parameter = ftps[i]; if (parameter.classBound instanceof Signature.ClassTypeSignature) { - Signature.ClassTypeSignature cts = (Signature.ClassTypeSignature)parameter.classBound; - ret.typeVariables[i]=new TypeVariable(ftps[i].identifier,UnresolvedType.forSignature(cts.outerType.identifier+";")); + Signature.ClassTypeSignature cts = (Signature.ClassTypeSignature) parameter.classBound; + ret.typeVariables[i] = new TypeVariable(ftps[i].identifier, UnresolvedType.forSignature(cts.outerType.identifier + + ";")); } else if (parameter.classBound instanceof Signature.TypeVariableSignature) { - Signature.TypeVariableSignature tvs = (Signature.TypeVariableSignature)parameter.classBound; - UnresolvedTypeVariableReferenceType utvrt = new UnresolvedTypeVariableReferenceType(new TypeVariable(tvs.typeVariableName)); - ret.typeVariables[i]=new TypeVariable(ftps[i].identifier,utvrt); + Signature.TypeVariableSignature tvs = (Signature.TypeVariableSignature) parameter.classBound; + UnresolvedTypeVariableReferenceType utvrt = new UnresolvedTypeVariableReferenceType(new TypeVariable( + tvs.typeVariableName)); + ret.typeVariables[i] = new TypeVariable(ftps[i].identifier, utvrt); } else { - throw new BCException("UnresolvedType.forGenericTypeSignature(): Do not know how to process type variable bound of type '"+ - parameter.classBound.getClass()+"'. Full signature is '"+sig+"'"); + throw new BCException( + "UnresolvedType.forGenericTypeSignature(): Do not know how to process type variable bound of type '" + + parameter.classBound.getClass() + "'. Full signature is '" + sig + "'"); } } - ret.signatureErasure = sig; - ret.signature = ret.signatureErasure; - return ret; - } - - public static UnresolvedType forGenericTypeVariables(String sig, TypeVariable[] tVars) { - UnresolvedType ret = UnresolvedType.forSignature(sig); - ret.typeKind=TypeKind.GENERIC; - ret.typeVariables = tVars; - ret.signatureErasure = sig; - ret.signature = ret.signatureErasure; - return ret; - } - + ret.signatureErasure = sig; + ret.signature = ret.signatureErasure; + return ret; + } + + public static UnresolvedType forGenericTypeVariables(String sig, TypeVariable[] tVars) { + UnresolvedType ret = UnresolvedType.forSignature(sig); + ret.typeKind = TypeKind.GENERIC; + ret.typeVariables = tVars; + ret.signatureErasure = sig; + ret.signature = ret.signatureErasure; + return ret; + } + public static UnresolvedType forRawTypeName(String name) { UnresolvedType ret = UnresolvedType.forName(name); ret.typeKind = TypeKind.RAW; return ret; } - + /** * Creates a new type array with a fresh type appended to the end. * * @param types the left hand side of the new array * @param end the right hand side of the new array */ - public static UnresolvedType[] add(UnresolvedType[] types, UnresolvedType end) { + public static UnresolvedType[] add(UnresolvedType[] types, UnresolvedType end) { int len = types.length; UnresolvedType[] ret = new UnresolvedType[len + 1]; System.arraycopy(types, 0, ret, 0, len); ret[len] = end; return ret; - } - + } + /** * Creates a new type array with a fresh type inserted at the beginning. * @@ -331,190 +349,226 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { * @param start the left hand side of the new array * @param types the right hand side of the new array */ - public static UnresolvedType[] insert(UnresolvedType start, UnresolvedType[] types) { + public static UnresolvedType[] insert(UnresolvedType start, UnresolvedType[] types) { int len = types.length; UnresolvedType[] ret = new UnresolvedType[len + 1]; ret[0] = start; System.arraycopy(types, 0, ret, 1, len); return ret; - } - - /** - * Constructs a Type for a JVM bytecode signature string. For example: - * - * <blockquote><pre> - * UnresolvedType.forSignature("[Ljava/lang/Thread;") - * UnresolvedType.forSignature("I"); - * </pre></blockquote> - * - * Types may equivalently be produced by this or by {@link #forName(String)}. - * - * <blockquote><pre> - * UnresolvedType.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") - * UnresolvedType.forName("int").equals(Type.forSignature("I")) - * </pre></blockquote> - * - * @param signature the JVM bytecode signature string for the desired type. - * @return a type object represnting that JVM bytecode signature. - */ - public static UnresolvedType forSignature(String signature) { - switch (signature.charAt(0)) { - case 'B': return ResolvedType.BYTE; - case 'C': return ResolvedType.CHAR; - case 'D': return ResolvedType.DOUBLE; - case 'F': return ResolvedType.FLOAT; - case 'I': return ResolvedType.INT; - case 'J': return ResolvedType.LONG; - case 'L': return TypeFactory.createTypeFromSignature(signature); - case 'P': return TypeFactory.createTypeFromSignature(signature); - case 'S': return ResolvedType.SHORT; - case 'V': return ResolvedType.VOID; - case 'Z': return ResolvedType.BOOLEAN; - case '[': return TypeFactory.createTypeFromSignature(signature); - case '+': return TypeFactory.createTypeFromSignature(signature); - case '-' : return TypeFactory.createTypeFromSignature(signature); - case '?' : return TypeFactory.createTypeFromSignature(signature); - case 'T' : return TypeFactory.createTypeFromSignature(signature); - default: throw new BCException("Bad type signature " + signature); - } - } - - /** Constructs a UnresolvedType for each JVM bytecode type signature in an incoming array. - * - * @param names an array of JVM bytecode type signatures - * @return an array of UnresolvedType objects. - * @see #forSignature(String) - */ - public static UnresolvedType[] forSignatures(String[] sigs) { - UnresolvedType[] ret = new UnresolvedType[sigs.length]; - for (int i = 0, len = sigs.length; i < len; i++) { - ret[i] = UnresolvedType.forSignature(sigs[i]); - } - return ret; - } - - /** - * Returns the name of this type in java language form (e.g. java.lang.Thread or boolean[]). - * This produces a more aesthetically pleasing string than {@link java.lang.Class#getName()}. - * - * @return the java language name of this type. - */ - public String getName() { - return signatureToName(signature); - } - - public String getSimpleName() { - String name = getRawName(); - int lastDot = name.lastIndexOf('.'); - if (lastDot != -1) { - name = name.substring(lastDot+1); - } - if (isParameterizedType()) { - StringBuffer sb = new StringBuffer(name); - sb.append("<"); - for (int i = 0; i < (typeParameters.length -1); i++) { + } + + /** + * Constructs a Type for a JVM bytecode signature string. For example: + * + * <blockquote> + * + * <pre> + * UnresolvedType.forSignature("[Ljava/lang/Thread;") + * UnresolvedType.forSignature("I"); + * </pre> + * + * </blockquote> + * + * Types may equivalently be produced by this or by {@link #forName(String)}. + * + * <blockquote> + * + * <pre> + * UnresolvedType.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") + * UnresolvedType.forName("int").equals(Type.forSignature("I")) + * </pre> + * + * </blockquote> + * + * @param signature the JVM bytecode signature string for the desired type. + * @return a type object represnting that JVM bytecode signature. + */ + public static UnresolvedType forSignature(String signature) { + switch (signature.charAt(0)) { + case 'B': + return ResolvedType.BYTE; + case 'C': + return ResolvedType.CHAR; + case 'D': + return ResolvedType.DOUBLE; + case 'F': + return ResolvedType.FLOAT; + case 'I': + return ResolvedType.INT; + case 'J': + return ResolvedType.LONG; + case 'L': + return TypeFactory.createTypeFromSignature(signature); + case 'P': + return TypeFactory.createTypeFromSignature(signature); + case 'S': + return ResolvedType.SHORT; + case 'V': + return ResolvedType.VOID; + case 'Z': + return ResolvedType.BOOLEAN; + case '[': + return TypeFactory.createTypeFromSignature(signature); + case '+': + return TypeFactory.createTypeFromSignature(signature); + case '-': + return TypeFactory.createTypeFromSignature(signature); + case '?': + return TypeFactory.createTypeFromSignature(signature); + case 'T': + return TypeFactory.createTypeFromSignature(signature); + default: + throw new BCException("Bad type signature " + signature); + } + } + + /** + * Constructs a UnresolvedType for each JVM bytecode type signature in an incoming array. + * + * @param names an array of JVM bytecode type signatures + * @return an array of UnresolvedType objects. + * @see #forSignature(String) + */ + public static UnresolvedType[] forSignatures(String[] sigs) { + UnresolvedType[] ret = new UnresolvedType[sigs.length]; + for (int i = 0, len = sigs.length; i < len; i++) { + ret[i] = UnresolvedType.forSignature(sigs[i]); + } + return ret; + } + + /** + * Returns the name of this type in java language form (e.g. java.lang.Thread or boolean[]). This produces a more aesthetically + * pleasing string than {@link java.lang.Class#getName()}. + * + * @return the java language name of this type. + */ + public String getName() { + return signatureToName(signature); + } + + public String getSimpleName() { + String name = getRawName(); + int lastDot = name.lastIndexOf('.'); + if (lastDot != -1) { + name = name.substring(lastDot + 1); + } + if (isParameterizedType()) { + StringBuffer sb = new StringBuffer(name); + sb.append("<"); + for (int i = 0; i < (typeParameters.length - 1); i++) { sb.append(typeParameters[i].getSimpleName()); sb.append(","); } - sb.append(typeParameters[typeParameters.length -1].getSimpleName()); - sb.append(">"); - name = sb.toString(); - } - return name; - } - - public String getRawName() { - return signatureToName((signatureErasure==null?signature:signatureErasure)); - } - + sb.append(typeParameters[typeParameters.length - 1].getSimpleName()); + sb.append(">"); + name = sb.toString(); + } + return name; + } + + public String getRawName() { + return signatureToName((signatureErasure == null ? signature : signatureErasure)); + } + public String getBaseName() { String name = getName(); if (isParameterizedType() || isGenericType()) { - if (typeParameters==null) return name; - else return name.substring(0,name.indexOf("<")); + if (typeParameters == null) + return name; + else + return name.substring(0, name.indexOf("<")); } else { return name; } } - + public String getSimpleBaseName() { - String name = getBaseName(); - int lastDot = name.lastIndexOf('.'); - if (lastDot != -1) { - name = name.substring(lastDot+1); - } - return name; - } - - /** - * Returns an array of strings representing the java langauge names of - * an array of types. - * - * @param types an array of UnresolvedType objects - * @return an array of Strings fo the java language names of types. - * @see #getName() - */ - public static String[] getNames(UnresolvedType[] types) { - String[] ret = new String[types.length]; - for (int i = 0, len = types.length; i < len; i++) { - ret[i] = types[i].getName(); - } - return ret; - } - - /** - * Returns the name of this type in JVM signature form. For all - * UnresolvedType t: - * - * <blockquote><pre> - * UnresolvedType.forSignature(t.getSignature()).equals(t) - * </pre></blockquote> - * - * and for all String s where s is a lexically valid JVM type signature string: - * - * <blockquote><pre> - * UnresolvedType.forSignature(s).getSignature().equals(s) - * </pre></blockquote> - * - * @return the java JVM signature string for this type. - */ - public String getSignature() { + String name = getBaseName(); + int lastDot = name.lastIndexOf('.'); + if (lastDot != -1) { + name = name.substring(lastDot + 1); + } + return name; + } + + /** + * Returns an array of strings representing the java langauge names of an array of types. + * + * @param types an array of UnresolvedType objects + * @return an array of Strings fo the java language names of types. + * @see #getName() + */ + public static String[] getNames(UnresolvedType[] types) { + String[] ret = new String[types.length]; + for (int i = 0, len = types.length; i < len; i++) { + ret[i] = types[i].getName(); + } + return ret; + } + + /** + * Returns the name of this type in JVM signature form. For all UnresolvedType t: + * + * <blockquote> + * + * <pre> + * UnresolvedType.forSignature(t.getSignature()).equals(t) + * </pre> + * + * </blockquote> + * + * and for all String s where s is a lexically valid JVM type signature string: + * + * <blockquote> + * + * <pre> + * UnresolvedType.forSignature(s).getSignature().equals(s) + * </pre> + * + * </blockquote> + * + * @return the java JVM signature string for this type. + */ + public String getSignature() { return signature; - } - - + } + /** * For parameterized types, return the signature for the raw type */ public String getErasureSignature() { - if (signatureErasure==null) return signature; + if (signatureErasure == null) + return signature; return signatureErasure; } - - private boolean needsModifiableDelegate =false; + + private boolean needsModifiableDelegate = false; + public boolean needsModifiableDelegate() { return needsModifiableDelegate; } - + public void setNeedsModifiableDelegate(boolean b) { - this.needsModifiableDelegate=b; + this.needsModifiableDelegate = b; } - + public UnresolvedType getRawType() { return UnresolvedType.forSignature(getErasureSignature()); } - - - /** - * Returns a UnresolvedType object representing the effective outermost enclosing type - * for a name type. For all other types, this will return the type itself. - * - * The only guarantee is given in JLS 13.1 where code generated according to - * those rules will have type names that can be split apart in this way. - * @return the outermost enclosing UnresolvedType object or this. - */ - public UnresolvedType getOutermostType() { - if (isArray() || isPrimitiveType()) return this; + + /** + * Returns a UnresolvedType object representing the effective outermost enclosing type for a name type. For all other types, + * this will return the type itself. + * + * The only guarantee is given in JLS 13.1 where code generated according to those rules will have type names that can be split + * apart in this way. + * + * @return the outermost enclosing UnresolvedType object or this. + */ + public UnresolvedType getOutermostType() { + if (isArray() || isPrimitiveType()) + return this; String sig = getErasureSignature(); int dollar = sig.indexOf('$'); if (dollar != -1) { @@ -522,134 +576,164 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { } else { return this; } - } - - /** - * Returns a UnresolvedType object representing the component type of this array, or - * null if this type does not represent an array type. - * - * @return the component UnresolvedType object, or null. - */ - public UnresolvedType getComponentType() { - if (isArray()) { - return forSignature(signature.substring(1)); - } else { - return null; - } - } - - /** - * Returns a java language string representation of this type. - */ - public String toString() { - return getName(); // + " - " + getKind(); - } - - public String toDebugString() { - return getName(); - } - - // ---- requires worlds - - /** + } + + /** + * Returns a UnresolvedType object representing the component type of this array, or null if this type does not represent an + * array type. + * + * @return the component UnresolvedType object, or null. + */ + public UnresolvedType getComponentType() { + if (isArray()) { + return forSignature(signature.substring(1)); + } else { + return null; + } + } + + /** + * Returns a java language string representation of this type. + */ + public String toString() { + return getName(); // + " - " + getKind(); + } + + public String toDebugString() { + return getName(); + } + + // ---- requires worlds + + /** * Returns a resolved version of this type according to a particular world. - * - * @param world thie {@link World} within which to resolve. - * @return a resolved type representing this type in the appropriate world. + * + * @param world thie {@link World} within which to resolve. + * @return a resolved type representing this type in the appropriate world. */ public ResolvedType resolve(World world) { return world.resolve(this); } - // ---- helpers - - private static String signatureToName(String signature) { - switch (signature.charAt(0)) { - case 'B': return "byte"; - case 'C': return "char"; - case 'D': return "double"; - case 'F': return "float"; - case 'I': return "int"; - case 'J': return "long"; - case 'L': - String name = signature.substring(1, signature.length() - 1).replace('/', '.'); - return name; - case 'T': - StringBuffer nameBuff2 = new StringBuffer(); - int colon = signature.indexOf(";"); - String tvarName = signature.substring(1,colon); - nameBuff2.append(tvarName); - return nameBuff2.toString(); - case 'P': // it's one of our parameterized type sigs - StringBuffer nameBuff = new StringBuffer(); - // signature for parameterized types is e.g. - // List<String> -> Ljava/util/List<Ljava/lang/String;>; - // Map<String,List<Integer>> -> Ljava/util/Map<java/lang/String;Ljava/util/List<Ljava/lang/Integer;>;>; - int paramNestLevel = 0; - for (int i = 1 ; i < signature.length(); i++) { - char c = signature.charAt(i); - switch (c) { - case '/' : nameBuff.append('.'); break; - case '<' : - nameBuff.append("<"); + // ---- helpers + + private static String signatureToName(String signature) { + switch (signature.charAt(0)) { + case 'B': + return "byte"; + case 'C': + return "char"; + case 'D': + return "double"; + case 'F': + return "float"; + case 'I': + return "int"; + case 'J': + return "long"; + case 'L': + String name = signature.substring(1, signature.length() - 1).replace('/', '.'); + return name; + case 'T': + StringBuffer nameBuff2 = new StringBuffer(); + int colon = signature.indexOf(";"); + String tvarName = signature.substring(1, colon); + nameBuff2.append(tvarName); + return nameBuff2.toString(); + case 'P': // it's one of our parameterized type sigs + StringBuffer nameBuff = new StringBuffer(); + // signature for parameterized types is e.g. + // List<String> -> Ljava/util/List<Ljava/lang/String;>; + // Map<String,List<Integer>> -> Ljava/util/Map<java/lang/String;Ljava/util/List<Ljava/lang/Integer;>;>; + int paramNestLevel = 0; + for (int i = 1; i < signature.length(); i++) { + char c = signature.charAt(i); + switch (c) { + case '/': + nameBuff.append('.'); + break; + case '<': + nameBuff.append("<"); + paramNestLevel++; + StringBuffer innerBuff = new StringBuffer(); + while (paramNestLevel > 0) { + c = signature.charAt(++i); + if (c == '<') paramNestLevel++; - StringBuffer innerBuff = new StringBuffer(); - while(paramNestLevel > 0) { - c = signature.charAt(++i); - if (c == '<') paramNestLevel++; - if (c == '>') paramNestLevel--; - if (paramNestLevel > 0) innerBuff.append(c); - if (c == ';' && paramNestLevel == 1) { - nameBuff.append(signatureToName(innerBuff.toString())); - if (signature.charAt(i+1) != '>') nameBuff.append(','); - innerBuff = new StringBuffer(); - } - } - nameBuff.append(">"); - break; - case ';' : break; - default: - nameBuff.append(c); + if (c == '>') + paramNestLevel--; + if (paramNestLevel > 0) + innerBuff.append(c); + if (c == ';' && paramNestLevel == 1) { + nameBuff.append(signatureToName(innerBuff.toString())); + if (signature.charAt(i + 1) != '>') + nameBuff.append(','); + innerBuff = new StringBuffer(); + } } + nameBuff.append(">"); + break; + case ';': + break; + default: + nameBuff.append(c); } - return nameBuff.toString(); - case 'S': return "short"; - case 'V': return "void"; - case 'Z': return "boolean"; - case '[': - return signatureToName(signature.substring(1, signature.length())) + "[]"; -// case '<': -// // its a generic! - // if (signature.charAt(1)=='>') return signatureToName(signature.substring(2)); - case '+' : return "? extends " + signatureToName(signature.substring(1, signature.length())); - case '-' : return "? super " + signatureToName(signature.substring(1, signature.length())); - case '?' : return "?"; - default: - throw new BCException("Bad type signature: " + signature); - } - } - - private static String nameToSignature(String name) { - if (name.equals("byte")) return "B"; - if (name.equals("char")) return "C"; - if (name.equals("double")) return "D"; - if (name.equals("float")) return "F"; - if (name.equals("int")) return "I"; - if (name.equals("long")) return "J"; - if (name.equals("short")) return "S"; - if (name.equals("boolean")) return "Z"; - if (name.equals("void")) return "V"; - if (name.equals("?")) return name; - if (name.endsWith("[]")) - return "[" + nameToSignature(name.substring(0, name.length() - 2)); - if (name.length() != 0) { - // lots more tests could be made here... - - // check if someone is calling us with something that is a signature already - if (name.charAt(0)=='[') { - throw new BCException("Do not call nameToSignature with something that looks like a signature (descriptor): '"+name+"'"); - } - + } + return nameBuff.toString(); + case 'S': + return "short"; + case 'V': + return "void"; + case 'Z': + return "boolean"; + case '[': + return signatureToName(signature.substring(1, signature.length())) + "[]"; + // case '<': + // // its a generic! + // if (signature.charAt(1)=='>') return signatureToName(signature.substring(2)); + case '+': + return "? extends " + signatureToName(signature.substring(1, signature.length())); + case '-': + return "? super " + signatureToName(signature.substring(1, signature.length())); + case '?': + return "?"; + default: + throw new BCException("Bad type signature: " + signature); + } + } + + private static String nameToSignature(String name) { + if (name.equals("byte")) + return "B"; + if (name.equals("char")) + return "C"; + if (name.equals("double")) + return "D"; + if (name.equals("float")) + return "F"; + if (name.equals("int")) + return "I"; + if (name.equals("long")) + return "J"; + if (name.equals("short")) + return "S"; + if (name.equals("boolean")) + return "Z"; + if (name.equals("void")) + return "V"; + if (name.equals("?")) + return name; + if (name.endsWith("[]")) + return "[" + nameToSignature(name.substring(0, name.length() - 2)); + if (name.length() != 0) { + // lots more tests could be made here... + + // check if someone is calling us with something that is a signature already + if (name.charAt(0) == '[') { + throw new BCException("Do not call nameToSignature with something that looks like a signature (descriptor): '" + + name + "'"); + } + if (name.indexOf("<") == -1) { // not parameterised return "L" + name.replace('.', '/') + ";"; @@ -660,44 +744,49 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { for (int i = 0; i < name.length(); i++) { char c = name.charAt(i); switch (c) { - case '.' : nameBuff.append('/'); break; - case '<' : + case '.': + nameBuff.append('/'); + break; + case '<': nameBuff.append("<"); nestLevel++; StringBuffer innerBuff = new StringBuffer(); - while(nestLevel > 0) { + while (nestLevel > 0) { c = name.charAt(++i); - if (c == '<') nestLevel++; - if (c == '>') nestLevel--; + if (c == '<') + nestLevel++; + if (c == '>') + nestLevel--; if (c == ',' && nestLevel == 1) { nameBuff.append(nameToSignature(innerBuff.toString())); innerBuff = new StringBuffer(); } else { - if (nestLevel > 0) innerBuff.append(c); + if (nestLevel > 0) + innerBuff.append(c); } } nameBuff.append(nameToSignature(innerBuff.toString())); nameBuff.append('>'); break; - case '>' : + case '>': throw new IllegalStateException("Should by matched by <"); - case ',' : + case ',': throw new IllegalStateException("Should only happen inside <...>"); - default: nameBuff.append(c); + default: + nameBuff.append(c); } } nameBuff.append(";"); return nameBuff.toString(); } - } - else - throw new BCException("Bad type name: " + name); - } - + } else + throw new BCException("Bad type name: " + name); + } + public void write(DataOutputStream s) throws IOException { s.writeUTF(getSignature()); } - + public static UnresolvedType read(DataInputStream s) throws IOException { String sig = s.readUTF(); if (sig.equals(MISSING_NAME)) { @@ -707,26 +796,26 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { return ret; } } - + public static void writeArray(UnresolvedType[] types, DataOutputStream s) throws IOException { int len = types.length; s.writeShort(len); - for (int i=0; i < len; i++) { + for (int i = 0; i < len; i++) { types[i].write(s); } } - + public static UnresolvedType[] readArray(DataInputStream s) throws IOException { int len = s.readShort(); - if (len==0) return UnresolvedType.NONE; + if (len == 0) + return UnresolvedType.NONE; UnresolvedType[] types = new UnresolvedType[len]; - for (int i=0; i < len; i++) { + for (int i = 0; i < len; i++) { types[i] = UnresolvedType.read(s); } return types; } - public String getNameAsIdentifier() { return getName().replace('.', '_'); } @@ -734,60 +823,59 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { public String getPackageNameAsIdentifier() { String name = getName(); int index = name.lastIndexOf('.'); - if (index == -1) { - return ""; + if (index == -1) { + return ""; } else { return name.substring(0, index).replace('.', '_'); } } - + public String getPackageName() { String name = getName(); - if (name.indexOf("<")!=-1) { - name = name.substring(0,name.indexOf("<")); + if (name.indexOf("<") != -1) { + name = name.substring(0, name.indexOf("<")); } int index = name.lastIndexOf('.'); - if (index == -1) { - return null; + if (index == -1) { + return null; } else { return name.substring(0, index); } } - + public UnresolvedType[] getTypeParameters() { return typeParameters == null ? UnresolvedType.NONE : typeParameters; } - + /** * Doesn't include the package */ public String getClassName() { String name = getName(); int index = name.lastIndexOf('.'); - if (index == -1) { - return name; + if (index == -1) { + return name; } else { - return name.substring(index+1); + return name.substring(index + 1); } } - + public TypeVariable[] getTypeVariables() { return typeVariables; } - public static class TypeKind { // Note: It is not sufficient to say that a parameterized type with no type parameters in fact // represents a raw type - a parameterized type with no type parameters can represent // an inner type of a parameterized type that specifies no type parameters of its own. - public final static TypeKind PRIMITIVE = new TypeKind("primitive"); - public final static TypeKind SIMPLE = new TypeKind("simple"); // a type with NO type parameters/vars - public final static TypeKind RAW = new TypeKind("raw"); // the erasure of a generic type - public final static TypeKind GENERIC = new TypeKind("generic"); // a generic type - public final static TypeKind PARAMETERIZED= new TypeKind("parameterized"); // a parameterized type - public final static TypeKind TYPE_VARIABLE= new TypeKind("type_variable"); // a type variable - public final static TypeKind WILDCARD = new TypeKind("wildcard"); // a generic wildcard type - + public final static TypeKind PRIMITIVE = new TypeKind("primitive"); + public final static TypeKind SIMPLE = new TypeKind("simple"); // a type with NO type parameters/vars + public final static TypeKind RAW = new TypeKind("raw"); // the erasure of a generic type + public final static TypeKind GENERIC = new TypeKind("generic"); // a generic type + public final static TypeKind PARAMETERIZED = new TypeKind("parameterized"); // a parameterized type + public final static TypeKind TYPE_VARIABLE = new TypeKind("type_variable"); // a type variable + public final static TypeKind WILDCARD = new TypeKind("wildcard"); // a generic wildcard type + public String toString() { return type; } @@ -795,16 +883,18 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { private TypeKind(String type) { this.type = type; } - + private final String type; } - + public TypeVariable getTypeVariableNamed(String name) { TypeVariable[] vars = getTypeVariables(); - if (vars==null || vars.length==0) return null; + if (vars == null || vars.length == 0) + return null; for (int i = 0; i < vars.length; i++) { TypeVariable aVar = vars[i]; - if (aVar.getName().equals(name)) return aVar; + if (aVar.getName().equals(name)) + return aVar; } return null; } @@ -812,15 +902,13 @@ public class UnresolvedType implements Traceable, TypeVariableDeclaringElement { public String toTraceString() { return getClass().getName() + "[" + getName() + "]"; } - - /** - * Return a version of this parameterized type in which any type parameters - * that are type variable references are replaced by their matching type variable - * binding. - */ - // OPTIMIZE methods like this just allow callers to be lazy and not ensure they are working with the right (resolved) subtype - public UnresolvedType parameterize(Map typeBindings) { - throw new UnsupportedOperationException("unable to parameterize unresolved type: " + signature); - } -} + /** + * Return a version of this parameterized type in which any type parameters that are type variable references are replaced by + * their matching type variable binding. + */ + // OPTIMIZE methods like this just allow callers to be lazy and not ensure they are working with the right (resolved) subtype + public UnresolvedType parameterize(Map typeBindings) { + throw new UnsupportedOperationException("unable to parameterize unresolved type: " + signature); + } +} |