Package com.strobel.reflection

Source Code of com.strobel.reflection.ReflectedType

/*
* Resolver.java
*
* Copyright (c) 2012 Mike Strobel
*
* This source code is subject to terms and conditions of the Apache License, Version 2.0.
* A copy of the license can be found in the License.html file at the root of this distribution.
* By using this source code in any fashion, you are agreeing to be bound by the terms of the
* Apache License, Version 2.0.
*
* You must not remove this notice, or any other, from this software.
*/

package com.strobel.reflection;

import com.strobel.core.ArrayUtilities;
import com.strobel.core.Comparer;
import com.strobel.core.VerifyArgument;
import com.strobel.util.ContractUtils;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.GenericArrayType;
import java.lang.reflect.GenericDeclaration;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.TypeVariable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Stack;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;

/**
* @author Mike Strobel
*/
@SuppressWarnings("unchecked")
final class Resolver {
    final static byte FLAG_RESOLVE_METHODS      = 0x01;
    final static byte FLAG_RESOLVE_FIELDS       = 0x02;
    final static byte FLAG_RESOLVE_CONSTRUCTORS = 0x04;
    final static byte FLAG_RESOLVE_NESTED_TYPES = 0x08;

    void resolveMembers(final ReflectedType<?> type, final int flags) {
        final Frame frame = new Frame(type, null);

        this.visit(type, frame, flags);
    }

    public final class Frame {
        private final ReflectedType<?>                  _type;
        private final java.lang.reflect.Type            _typeElement;
        private final Frame                             _previous;
        private final Map<java.lang.reflect.Type, Type> _elementTypeMap;
        private final Stack<ReflectedMethod>            _methods;

        private final ArrayList<Type<?>> _typeArguments = new ArrayList<>();

        public Frame(final java.lang.reflect.Type typeElement, final Frame previous) {
            _typeElement = VerifyArgument.notNull(typeElement, "typeElement");
            _previous = previous;
            _elementTypeMap = previous != null ? previous._elementTypeMap : new HashMap<java.lang.reflect.Type, Type>();
            _methods = previous != null ? previous._methods : new Stack<ReflectedMethod>();
            _type = new ReflectedType<>((Class<?>) typeElement);
            _elementTypeMap.put(typeElement, _type);
        }

        Frame(final ReflectedType<?> type, final Frame previous) {
            _type = VerifyArgument.notNull(type, "type");
            _typeElement = type;
            _previous = previous;
            _elementTypeMap = previous != null ? previous._elementTypeMap : new HashMap<java.lang.reflect.Type, Type>();
            _methods = previous != null ? previous._methods : new Stack<ReflectedMethod>();
            _elementTypeMap.put(_typeElement, _type);

            final Class<?> classType = type.getErasedClass();
            final Class<?> enclosingClass = classType.getEnclosingClass();

            if (enclosingClass != null && enclosingClass != classType) {
                final Frame ownerFrame = findFrame(enclosingClass);

                if (ownerFrame != null) {
                    if (ownerFrame._type.findNestedType(_type.getErasedClass()) == null) {
                        ownerFrame._type.addNestedType(_type);
                    }
                }
            }
        }

        public Type<?> getResult() {
            return _type;
        }

        void pushMethod(final ReflectedMethod method) {
            _methods.push(method);
        }

        ReflectedMethod popMethod() {
            return _methods.pop();
        }

        ReflectedMethod currentMethod() {
            return _methods.peek();
        }

        ReflectedType<?> getCurrentClass() {
            return _type;
        }

        java.lang.reflect.Type getCurrentType() {
            return _typeElement;
        }

        List<Type<?>> getTypeArguments() {
            return _typeArguments;
        }

        Type<?> findType(final java.lang.reflect.Type e) {
            Frame currentFrame = this;

            while (currentFrame != null) {
                Type result = currentFrame._elementTypeMap.get(e);

                if (result != null) {
                    return result;
                }

                if (e instanceof java.lang.reflect.TypeVariable) {
                    final TypeVariable typeVariable = (TypeVariable) e;

                    result = currentFrame._type.findGenericParameter(typeVariable);

                    if (result != null) {
                        return result;
                    }

                    if (!_typeArguments.isEmpty()) {
                        for (final Type<?> typeArgument : _typeArguments) {
                            if (!(typeArgument instanceof GenericParameter)) {
                                continue;
                            }
                            if (((GenericParameter) typeArgument).getRawTypeVariable() == typeVariable) {
                                return typeArgument;
                            }
                        }
                    }
                }

                currentFrame = currentFrame._previous;
            }

            return null;
        }

        Type<?> resolveType(final java.lang.reflect.Type t) {
            final Type<?> result = resolveTypeCore(t);

/*
            if (result != null && result.isGenericType()) {
                if (t instanceof Class<?> &&
                    ArrayUtilities.isNullOrEmpty(((Class)t).getTypeParameters())) {
                    return result.getErasedType();
                }
            }
*/

            if (result != null) {
                _elementTypeMap.put(t, result);
                Type.CACHE.add(result);
            }

            return result;
        }

        private Type<?> resolveTypeCore(final java.lang.reflect.Type t) {
            final Type<?> fromMap = findType(t);

            if (fromMap != null) {
                return fromMap;
            }

            if (t instanceof Class<?>) {
                final Class<?> classType = (Class<?>) t;

                if (classType.isPrimitive() || classType.isEnum() || classType == Void.TYPE) {
                    return Type.of(classType);
                }

                if (classType.isArray()) {
                    final Type<?> componentType = resolveType(classType.getComponentType());

                    if (componentType != null) {
                        return componentType.makeArrayType();
                    }

                    return null;
                }

                final Type cachedType = Type.CACHE.find((Class<?>) t);

                if (cachedType != null) {
                    return cachedType;
                }

                return visit(classType, this);
            }

            if (t instanceof GenericArrayType) {
                final java.lang.reflect.Type genericComponentType = ((GenericArrayType) t).getGenericComponentType();
                final Type<?> componentType = resolveType(genericComponentType);

                if (componentType != null) {
                    return componentType.makeArrayType();
                }

                return null;
            }

            if (t instanceof java.lang.reflect.WildcardType) {
                final java.lang.reflect.WildcardType w = (java.lang.reflect.WildcardType) t;
                final java.lang.reflect.Type[] upperBounds = w.getUpperBounds();
                final java.lang.reflect.Type[] lowerBounds = w.getLowerBounds();

                List<Type<?>> resolvedUpperBounds = null;
                List<Type<?>> resolvedLowerBounds = null;

                for (final java.lang.reflect.Type lowerBound : lowerBounds) {
                    final Type<?> resolvedLowerBound = resolveType(lowerBound);

                    if (resolvedLowerBound == null) {
                        continue;
                    }

                    if (resolvedLowerBounds == null) {
                        resolvedLowerBounds = new ArrayList<>();
                    }

                    resolvedLowerBounds.add(resolvedLowerBound);
                }

                if (resolvedLowerBounds != null && resolvedLowerBounds.size() != lowerBounds.length) {
                    return null;
                }

                for (final java.lang.reflect.Type upperBound : upperBounds) {
                    final Type<?> resolvedUpperBound = resolveType(upperBound);

                    if (resolvedUpperBound == null) {
                        continue;
                    }

                    if (resolvedUpperBounds == null) {
                        resolvedUpperBounds = new ArrayList<>();
                    }

                    resolvedUpperBounds.add(resolvedUpperBound);
                }

                if (resolvedUpperBounds != null && resolvedUpperBounds.size() != upperBounds.length) {
                    return null;
                }

                Type<?> upperBound = Types.Object;
                Type<?> lowerBound = Type.Bottom;

                if (resolvedUpperBounds != null) {
                    if (resolvedUpperBounds.size() == 1) {
                        upperBound = resolvedUpperBounds.get(0);
                    }
                    else {
                        upperBound = Type.makeCompoundType(Type.list(resolvedUpperBounds));
                    }
                }

                if (resolvedLowerBounds != null) {
                    if (resolvedLowerBounds.size() == 1) {
                        lowerBound = resolvedLowerBounds.get(0);
                    }
                    else {
                        lowerBound = Type.makeCompoundType(Type.list(resolvedLowerBounds));
                    }
                }

                return new WildcardType(upperBound, lowerBound);
            }

            // Now redundant to findType()...
            if (t instanceof java.lang.reflect.TypeVariable) {
                final java.lang.reflect.TypeVariable typeVariable = (java.lang.reflect.TypeVariable) t;

                Frame currentFrame = this;

                while (currentFrame != null) {
                    final GenericParameter genericParameter = currentFrame._type.findGenericParameter(typeVariable);

                    if (genericParameter != null) {
                        return genericParameter;
                    }

                    currentFrame = currentFrame._previous;
                }
            }

            if (t instanceof ParameterizedType) {
                final ParameterizedType type = (ParameterizedType) t;

                final Type<?> rawType = resolveType(type.getRawType());
                final java.lang.reflect.Type[] typeArguments = type.getActualTypeArguments();

                if (ArrayUtilities.isNullOrEmpty(typeArguments) || !rawType.isGenericType()) {
                    return rawType;
                }

                final Type<?>[] resolvedTypeArguments = new Type<?>[typeArguments.length];

                for (int i = 0, n = typeArguments.length; i < n; i++) {
                    final java.lang.reflect.Type typeArg = typeArguments[i];

                    if (typeArg instanceof java.lang.reflect.TypeVariable) {
                        final java.lang.reflect.TypeVariable typeVariable = (java.lang.reflect.TypeVariable) typeArg;
                        final GenericDeclaration genericDeclaration = typeVariable.getGenericDeclaration();
                        final Type<?> existingTypeArgument = findType(typeVariable);

                        if (existingTypeArgument != null) {
                            resolvedTypeArguments[i] = existingTypeArgument;
                            continue;
                        }

                        if (genericDeclaration instanceof Method) {
                            final ReflectedType<?> declaringType = (ReflectedType<?>) resolveType(((Method) genericDeclaration).getDeclaringClass());
                            final Method method = (Method) genericDeclaration;
                            final int position = ArrayUtilities.indexOf(method.getTypeParameters(), typeVariable);
                            final ReflectedMethod declaredMethod = declaringType.findMethod(method);

                            resolvedTypeArguments[i] = declaredMethod.getTypeArguments().get(position);
                        }
                        else {
                            final Class<?> declaringClass = (Class<?>) genericDeclaration;
                            final Type<?> declaringType = resolveType(declaringClass);
                            final int position = ArrayUtilities.indexOf(declaringClass.getTypeParameters(), typeVariable);

                            if (declaringType instanceof ReflectedType<?>) {
                                resolvedTypeArguments[i] = ((ReflectedType<?>) declaringType).getGenericParameters().get(position);
                            }
                            else {
                                resolvedTypeArguments[i] = declaringType.getTypeArguments().get(position);
                            }
                        }
                    }
                    else {
                        if (typeArg == null) {
                            resolvedTypeArguments[i] = rawType.getGenericTypeParameters().get(i);
                        }
                        else {
                            resolvedTypeArguments[i] = resolveType(typeArg);
                        }
                    }
                }

                final Type fromCache = Type.CACHE.find(
                    Type.CACHE.key(
                        rawType.getErasedClass(),
                        Type.list(resolvedTypeArguments)
                    )
                );

                if (fromCache != null) {
                    return fromCache;
                }

                for (final Type<?> resolvedTypeArgument : resolvedTypeArguments) {
                    if (!resolvedTypeArgument.isGenericParameter() || resolvedTypeArgument.getDeclaringType() != rawType) {
                        return Type.CACHE.getGenericType(rawType, Type.list(resolvedTypeArguments));
                    }
                }

                return rawType;
            }

            return null;
        }

        Frame findFrame(final java.lang.reflect.Type e) {
            Frame current = this;

            while (current != null) {
                if (current._typeElement.equals(e)) {
                    return current;
                }

                current = current._previous;
            }

            return null;
        }

        void addTypeArgument(final GenericParameter genericParameter) {
            _typeArguments.add(genericParameter);
        }
    }

    public Type<?> resolve(final java.lang.reflect.Type type) {
        return resolveExisting(null, type, true);
    }

    private Type<?> visit(final java.lang.reflect.Type type, final Frame frame) {
        if (type instanceof ParameterizedType) {
            return visitParameterizedType((ParameterizedType) type, frame);
        }
        else if (type instanceof java.lang.reflect.WildcardType) {
            return visitWildcardType((java.lang.reflect.WildcardType) type, frame);
        }
        else if (type instanceof GenericArrayType) {
            return visitGenericArrayType((GenericArrayType) type, frame);
        }
        else if (type instanceof java.lang.reflect.TypeVariable) {
            return visitTypeVariable((java.lang.reflect.TypeVariable) type, frame);
        }
        else {
            return visitClass((Class<?>) type, frame);
        }
    }

    @SuppressWarnings("ConstantConditions")
    private Type<?> visitClass(final Class<?> c, final Frame frame) {
        final Method declaringMethod;
        final Constructor<?> declaringConstructor;
        final Class<?> enclosingClass = c.getEnclosingClass();

        if (enclosingClass != null) {
            if (c.isLocalClass()) {
                declaringMethod = c.getEnclosingMethod();
                declaringConstructor = c.getEnclosingConstructor();
            }
            else {
                if (c.isAnonymousClass()) {
                    return null;
                }
                declaringMethod = null;
                declaringConstructor = null;
            }
        }
        else {
            declaringMethod = null;
            declaringConstructor = null;
        }

        final Frame currentFrame = new Frame(c, frame);
        final ReflectedType<?> currentType = currentFrame.getCurrentClass();

        if (enclosingClass != null) {
            final Type declaringType = currentFrame.resolveType(enclosingClass);

            currentType.setDeclaringType(declaringType);

            if (declaringMethod != null) {
                for (final MethodInfo method : declaringType.getMethods(BindingFlags.AllDeclared)) {
                    if (declaringMethod.equals(method.getRawMethod())) {
                        currentType.setDeclaringMethod(method);
                        break;
                    }
                }
            }
            else if (declaringConstructor != null) {
                for (final ConstructorInfo constructor : declaringType.getConstructors(BindingFlags.AllDeclared)) {
                    if (declaringConstructor.equals(constructor.getRawConstructor())) {
                        currentType.setDeclaringMethod(constructor);
                        break;
                    }
                }
            }

            // Redundant: Frame constructor adds nested type to its declaring type.
            if (declaringType instanceof ReflectedType<?>) {
                final ReflectedType<?> owner = (ReflectedType<?>) declaringType;
                if (owner.findNestedType(c) == null) {
                    owner.addNestedType(currentType);
                }
            }
        }

        final TypeVariable<?>[] typeParameters = c.getTypeParameters();

        if (!ArrayUtilities.isNullOrEmpty(typeParameters)) {
            for (int i = 0, n = typeParameters.length; i < n; i++) {
                final TypeVariable typeVariable = typeParameters[i];

                currentType.addGenericParameter(
                    new GenericParameter(
                        typeVariable.getName(),
                        currentType,
                        null,
                        i
                    )
                );
            }
        }

        final Type<?> baseType;

        java.lang.reflect.Type baseClass;

        if (!c.isInterface()) {
            baseClass = c.getGenericSuperclass();

            if (baseClass == null) {
                baseClass = c.getSuperclass();
            }

            if (baseClass != null && baseClass != c) {
                baseType = currentFrame.resolveType(baseClass);

                if (baseType != null) {
                    currentType.setBaseType(baseType);
                }
                else {
                    return null;
                }
            }
        }

        List<Type<?>> interfaceList = null;

        java.lang.reflect.Type[] interfaces = c.getGenericInterfaces();

        if (ArrayUtilities.isNullOrEmpty(interfaces)) {
            interfaces = c.getInterfaces();
        }

        if (!ArrayUtilities.isNullOrEmpty(interfaces)) {
            for (final java.lang.reflect.Type t : interfaces) {
                final Type<?> interfaceType = currentFrame.resolveType(t);

                if (interfaceType == null) {
                    return null;
                }

                if (interfaceList == null) {
                    interfaceList = new ArrayList<>();
                }

                interfaceList.add(interfaceType);
            }
        }

        if (interfaceList != null) {
            currentType.setInterfaces(Type.list(interfaceList));
        }

/*
        for (final Class<?> ee : c.getDeclaredClasses()) {
            this.visit(ee, currentFrame);
        }
*/

        for (final GenericParameter genericParameter : currentType.getGenericParameters()) {
            final java.lang.reflect.Type[] bounds = genericParameter.getRawTypeVariable().getBounds();

            if (ArrayUtilities.isNullOrEmpty(bounds)) {
                continue;
            }

            final Type<?> boundType;

            if (bounds.length == 1) {
                boundType = currentFrame.resolveType(bounds[0]);
            }
            else {
                final Type<?>[] resolvedBounds = new Type<?>[bounds.length];

                for (int i = 0, n = bounds.length; i < n; i++) {
                    resolvedBounds[i] = currentFrame.resolveType(bounds[i]);
                }

                boundType = Type.makeCompoundType(Type.list(resolvedBounds));
            }

            genericParameter.setUpperBound(boundType);
        }

        currentType.complete();

        if (!java.lang.reflect.Modifier.isPrivate(currentType.getModifiers())) {
            Type.CACHE.add(currentType);
        }

        return currentType;
    }

    private Type<?> visitTypeVariable(final java.lang.reflect.TypeVariable type, final Frame frame) {
        return frame.findType(type);
    }

    private Type<?> visitGenericArrayType(final GenericArrayType type, final Frame frame) {
        final Type<?> elementType = frame.resolveType(type.getGenericComponentType());

        if (elementType != null) {
            return elementType.makeArrayType();
        }

        return null;
    }

    private Type<?> visitWildcardType(final java.lang.reflect.WildcardType w, final Frame frame) {
        final java.lang.reflect.Type[] upperBounds = w.getUpperBounds();
        final java.lang.reflect.Type[] lowerBounds = w.getLowerBounds();

        List<Type<?>> resolvedUpperBounds = null;
        List<Type<?>> resolvedLowerBounds = null;

        for (final java.lang.reflect.Type lowerBound : lowerBounds) {
            final Type<?> resolvedLowerBound = visit(lowerBound, frame);

            if (resolvedLowerBound == null) {
                continue;
            }

            if (resolvedLowerBounds == null) {
                resolvedLowerBounds = new ArrayList<>();
            }

            resolvedLowerBounds.add(resolvedLowerBound);
        }

        if (resolvedLowerBounds != null && resolvedLowerBounds.size() != lowerBounds.length) {
            return null;
        }

        for (final java.lang.reflect.Type upperBound : upperBounds) {
            final Type<?> resolvedUpperBound = visit(upperBound, frame);

            if (resolvedUpperBound == null) {
                continue;
            }

            if (resolvedUpperBounds == null) {
                resolvedUpperBounds = new ArrayList<>();
            }

            resolvedUpperBounds.add(resolvedUpperBound);
        }

        if (resolvedUpperBounds != null && resolvedUpperBounds.size() != upperBounds.length) {
            return null;
        }

        Type<?> upperBound = Types.Object;
        Type<?> lowerBound = Type.Bottom;

        if (resolvedUpperBounds != null) {
            if (resolvedUpperBounds.size() == 1) {
                upperBound = resolvedUpperBounds.get(0);
            }
            else {
                upperBound = Type.makeCompoundType(Type.list(resolvedUpperBounds));
            }
        }

        if (resolvedLowerBounds != null) {
            if (resolvedLowerBounds.size() == 1) {
                lowerBound = resolvedLowerBounds.get(0);
            }
            else {
                lowerBound = Type.makeCompoundType(Type.list(resolvedLowerBounds));
            }
        }

        return new WildcardType(upperBound, lowerBound);
    }

    private Type<?> visitParameterizedType(final ParameterizedType type, final Frame frame) {
        final Type<?> rawType = frame.resolveType(type.getRawType());

        if (rawType == null) {
            return null;
        }

        final java.lang.reflect.Type[] typeArguments = type.getActualTypeArguments();
        final Type<?>[] resolvedTypeArguments = new Type<?>[typeArguments.length];

        for (int i = 0, n = typeArguments.length; i < n; i++) {
            resolvedTypeArguments[i] = frame.resolveType(typeArguments[i]);

            if (resolvedTypeArguments[i] == null) {
                return null;
            }
        }

        return rawType.makeGenericType(resolvedTypeArguments);
    }

    private Type<?> resolveExisting(final Frame frame, final java.lang.reflect.Type type, final boolean resolve) {
        final Type<?> result;

        Type<?> fromCacheOrFrame = Type.tryFind(type);

        if (fromCacheOrFrame != null) {
            result = fromCacheOrFrame;
        }
        else {
            if (frame != null) {
                fromCacheOrFrame = frame.resolveType(type);
            }

            if (!resolve) {
                return fromCacheOrFrame;
            }

            if (fromCacheOrFrame != null) {
                result = fromCacheOrFrame;
            }
            else if (type instanceof Class<?>) {
                result = visitClass((Class) type, frame);
            }
            else {
                result = visit(type, frame);
            }
        }

        return result;
    }

    private Type<?> visit(final ReflectedType<?> type, final Frame frame, final int flags) {
        final Class<?> erasedClass = type.getErasedClass();

        if ((flags & FLAG_RESOLVE_NESTED_TYPES) == FLAG_RESOLVE_NESTED_TYPES) {
            for (final Class<?> nestedClass : erasedClass.getDeclaredClasses()) {
                if (type.findNestedType(nestedClass) == null) {
                    visit(nestedClass, frame);
                }
            }
        }

        if ((flags & FLAG_RESOLVE_FIELDS) == FLAG_RESOLVE_FIELDS) {
            for (final Field field : erasedClass.getDeclaredFields()) {
                if (type.findField(field) == null) {
                    visitField(field, frame);
                }
            }
        }

        if ((flags & FLAG_RESOLVE_METHODS) == FLAG_RESOLVE_METHODS) {
            for (final Method method : erasedClass.getDeclaredMethods()) {
                if (type.findMethod(method) == null) {
                    visitMethod(method, frame);
                }
            }
        }

        if ((flags & FLAG_RESOLVE_CONSTRUCTORS) == FLAG_RESOLVE_CONSTRUCTORS) {
            for (final Constructor<?> constructor : erasedClass.getDeclaredConstructors()) {
                if (type.findConstructor(constructor) == null) {
                    visitConstructor(constructor, frame);
                }
            }
        }

        return type;
    }

    public Type<?> visitField(final Field field, final Frame frame) {
        final ReflectedType<?> declaringType = frame.getCurrentClass();

        final ReflectedField reflectedField = new ReflectedField(
            declaringType,
            field,
            frame.resolveType(field.getType())
        );

        declaringType.addField(reflectedField);

        return declaringType;
    }

    @SuppressWarnings("ConstantConditions")
    private Type<?> visitMethod(final Method m, final Frame frame) {
        java.lang.reflect.Type[] parameterTypes = m.getGenericParameterTypes();
        java.lang.reflect.Type[] thrownTypes = m.getGenericExceptionTypes();
        java.lang.reflect.Type returnType = m.getGenericReturnType();

        if (parameterTypes == null) {
            parameterTypes = m.getParameterTypes();
        }

        if (thrownTypes == null) {
            thrownTypes = m.getExceptionTypes();
        }

        if (returnType == null) {
            returnType = m.getReturnType();
        }

        final Type<?>[] resolvedThrownTypes = new Type<?>[thrownTypes.length];
        final ParameterInfo[] parameters = new ParameterInfo[parameterTypes.length];
        final TypeVariable<Method>[] typeParameters = m.getTypeParameters();
        final TypeBindings genericParameters;

        if (ArrayUtilities.isNullOrEmpty(typeParameters)) {
            genericParameters = TypeBindings.empty();
        }
        else {
            final Type<?>[] resolvedTypeParameters = new Type<?>[typeParameters.length];

            for (int i = 0; i < typeParameters.length; i++) {
                final TypeVariable<Method> typeVariable = typeParameters[i];
                final GenericParameter genericParameter = new GenericParameter(typeVariable.getName(), typeVariable, i);

                resolvedTypeParameters[i] = genericParameter;
                frame.addTypeArgument(genericParameter);
            }

            genericParameters = TypeBindings.createUnbound(Type.list(resolvedTypeParameters));
        }

        for (int i = 0; i < parameterTypes.length; i++) {
            final Type<?> resolvedParameterType = frame.resolveType(parameterTypes[i]);

            if (resolvedParameterType == null) {
                return null;
            }

            parameters[i] = new ParameterInfo("p" + i, i, resolvedParameterType);
        }

        for (int i = 0; i < thrownTypes.length; i++) {
            final Type<?> resolvedThrownType = frame.resolveType(thrownTypes[i]);

            if (resolvedThrownType == null) {
                return null;
            }

            resolvedThrownTypes[i] = resolvedThrownType;
        }

        final Type<?> resolvedReturnType = frame.resolveType(returnType);

        if (resolvedReturnType == null) {
            return null;
        }

        final ReflectedType<?> declaringType = frame.getCurrentClass();

        final ReflectedMethod method = new ReflectedMethod(
            declaringType,
            m,
            new ParameterList(parameters),
            resolvedReturnType,
            Type.list(resolvedThrownTypes),
            genericParameters
        );

        if (!genericParameters.isEmpty()) {
            for (final Type type : genericParameters.getGenericParameters()) {
                final GenericParameter genericParameter = (GenericParameter) type;

                genericParameter.setDeclaringMethod(method);

                final java.lang.reflect.Type[] bounds = genericParameter.getRawTypeVariable().getBounds();

                if (ArrayUtilities.isNullOrEmpty(bounds)) {
                    continue;
                }

                final Type<?> boundType;

                if (bounds.length == 1) {
                    boundType = frame.resolveType(bounds[0]);
                }
                else {
                    final Type<?>[] resolvedBounds = new Type<?>[bounds.length];

                    for (int i = 0, n = bounds.length; i < n; i++) {
                        resolvedBounds[i] = frame.resolveType(bounds[i]);
                    }

                    boundType = Type.makeCompoundType(Type.list(resolvedBounds));
                }

                genericParameter.setUpperBound(boundType);

            }
        }

        declaringType.addMethod(method);

        return declaringType;
    }

    private Type<?> visitConstructor(final Constructor<?> c, final Frame frame) {
        java.lang.reflect.Type[] parameterTypes = c.getGenericParameterTypes();
        java.lang.reflect.Type[] thrownTypes = c.getGenericExceptionTypes();

        if (parameterTypes == null) {
            parameterTypes = c.getParameterTypes();
        }

        if (thrownTypes == null) {
            thrownTypes = c.getExceptionTypes();
        }

        final Type<?>[] resolvedThrownTypes = new Type<?>[thrownTypes.length];
        final ParameterInfo[] parameters = new ParameterInfo[parameterTypes.length];

        for (int i = 0; i < parameterTypes.length; i++) {
            final Type<?> resolvedParameterType = frame.resolveType(parameterTypes[i]);

            if (resolvedParameterType == null) {
                return null;
            }

            parameters[i] = new ParameterInfo("p" + i, i, resolvedParameterType);
        }

        for (int i = 0; i < thrownTypes.length; i++) {
            final Type<?> resolvedThrownType = frame.resolveType(thrownTypes[i]);

            if (resolvedThrownType == null) {
                return null;
            }

            resolvedThrownTypes[i] = resolvedThrownType;
        }

        final ReflectedType<?> declaringType = frame.getCurrentClass();

        final ReflectedConstructor constructor = new ReflectedConstructor(
            declaringType,
            c,
            new ParameterList(parameters),
            Type.list(resolvedThrownTypes)
        );

        declaringType.addConstructor(constructor);

        return declaringType;
    }
}

@SuppressWarnings({ "unchecked", "UnusedDeclaration" })
class ReflectedType<T> extends Type<T> {
    private final static AtomicIntegerFieldUpdater<ReflectedType> FLAGS_UPDATER = AtomicIntegerFieldUpdater.newUpdater(
        ReflectedType.class,
        "_flags"
    );

    private final static byte FLAG_METHODS_RESOLVED      = 0x01;
    private final static byte FLAG_FIELDS_RESOLVED       = 0x02;
    private final static byte FLAG_CONSTRUCTORS_RESOLVED = 0x04;
    private final static byte FLAG_NESTED_TYPES_RESOLVED = 0x08;
    private final static byte FLAG_ALL_MEMBERS_RESOLVED  = 0x0F;
    private final static byte FLAG_RESOLVING_MEMBERS     = 0x10;

    private final    String          _name;
    private final    String          _simpleName;
    private final    Class<T>        _rawClass;
    private          Type<? super T> _baseType;
    private          TypeList        _interfaces;
    private volatile int             _flags;
    private          boolean         _completed;
    private          Type<?>         _declaringType;
    private          MethodBase      _declaringMethod;

    private List<GenericParameter<?>>  _genericParameters = null;
    private List<ReflectedType<?>>     _nestedTypes       = null;
    private List<ReflectedMethod>      _methods           = null;
    private List<ReflectedField>       _fields            = null;
    private List<ReflectedConstructor> _constructors      = null;

    private TypeBindings _typeBindings;

    ReflectedType(final Class<T> rawClass) {
        _rawClass = VerifyArgument.notNull(rawClass, "rawClass");
        _name = rawClass.isLocalClass() ? rawClass.getName() : rawClass.getCanonicalName();
        _simpleName = rawClass.getSimpleName();
    }

    @Override
    public String getFullName() {
        return _name;
    }

    private boolean checkFlags(final int flags) {
        return (_flags & flags) == flags;
    }

    private void setFlags(final int flags) {
        while (true) {
            final int oldFlags = _flags;
            if (FLAGS_UPDATER.compareAndSet(this, oldFlags, oldFlags | flags)) {
                return;
            }
        }
    }

    private void clearFlags(final int flags) {
        while (true) {
            final int oldFlags = _flags;
            if (FLAGS_UPDATER.compareAndSet(this, oldFlags, oldFlags & ~flags)) {
                return;
            }
        }
    }

    void setBaseType(final Type<?> baseType) {
        _baseType = (Type<? super T>) baseType;
    }

    void setInterfaces(final TypeList interfaces) {
        _interfaces = VerifyArgument.notNull(interfaces, "interfaces");
    }

    List<GenericParameter<?>> getGenericParameters() {
        if (_genericParameters == null) {
            return Collections.emptyList();
        }
        return _genericParameters;
    }

    ReflectedField findField(final Field rawField) {
        if (_fields == null) {
            return null;
        }
        for (final ReflectedField field : _fields) {
            if (Comparer.equals(field.getRawField(), rawField)) {
                return field;
            }
        }
        return null;
    }

    ReflectedMethod findMethod(final Method rawMethod) {
        if (_methods == null) {
            return null;
        }
        for (final ReflectedMethod method : _methods) {
            if (Comparer.equals(method.getRawMethod(), rawMethod)) {
                return method;
            }
        }
        return null;
    }

    ReflectedConstructor findConstructor(final Constructor rawConstructor) {
        if (_constructors == null) {
            return null;
        }
        for (final ReflectedConstructor constructor : _constructors) {
            if (Comparer.equals(constructor.getRawConstructor(), rawConstructor)) {
                return constructor;
            }
        }
        return null;
    }

    ReflectedType<?> findNestedType(final Class<?> rawClass) {
        if (_nestedTypes == null) {
            return null;
        }
        for (final ReflectedType<?> nestedType : _nestedTypes) {
            if (Comparer.equals(nestedType.getErasedClass(), rawClass)) {
                return nestedType;
            }
        }
        return null;
    }

    GenericParameter findGenericParameter(final java.lang.reflect.TypeVariable typeVariable) {
        return GenericParameterFinder.visit(this, typeVariable);
    }

    void setDeclaringType(final Type<?> declaringType) {
        _declaringType = VerifyArgument.notNull(declaringType, "declaringType");
    }

    void setDeclaringMethod(final MethodBase declaringMethod) {
        _declaringMethod = VerifyArgument.notNull(declaringMethod, "declaringMethod");
    }

    void addGenericParameter(final GenericParameter genericParameter) {
        VerifyArgument.notNull(genericParameter, "typeParameter");
        _completed = false;
        if (_genericParameters == null) {
            _genericParameters = new ArrayList<>();
        }
        _genericParameters.add(genericParameter);
    }

    void addNestedType(final ReflectedType<?> nestedType) {
        VerifyArgument.notNull(nestedType, "nestedType");

        if (_nestedTypes == null) {
            _nestedTypes = new ArrayList<>();
        }

        _nestedTypes.add(nestedType);

        if (!checkFlags(FLAG_RESOLVING_MEMBERS)) {
            clearFlags(FLAG_NESTED_TYPES_RESOLVED);
        }
    }

    void addMethod(final ReflectedMethod method) {
        VerifyArgument.notNull(method, "method");

        if (_methods == null) {
            _methods = new ArrayList<>();
        }

        _methods.add(method);

        if (!checkFlags(FLAG_RESOLVING_MEMBERS)) {
            clearFlags(FLAG_METHODS_RESOLVED);
        }
    }

    void addConstructor(final ReflectedConstructor constructor) {
        VerifyArgument.notNull(constructor, "constructor");

        if (_constructors == null) {
            _constructors = new ArrayList<>();
        }

        _constructors.add(constructor);

        if (!checkFlags(FLAG_RESOLVING_MEMBERS)) {
            clearFlags(FLAG_CONSTRUCTORS_RESOLVED);
        }
    }

    void addField(final ReflectedField field) {
        VerifyArgument.notNull(field, "field");

        if (_fields == null) {
            _fields = new ArrayList<>();
        }

        _fields.add(field);

        if (!checkFlags(FLAG_RESOLVING_MEMBERS)) {
            clearFlags(FLAG_FIELDS_RESOLVED);
        }
    }

    private void completeIfNecessary() {
        if (!_completed) {
            synchronized (CACHE_LOCK) {
                if (!_completed) {
                    complete();
                }
            }
        }
    }

    private void ensureMembersResolved() {
        while (true) {
            if (checkFlags(FLAG_ALL_MEMBERS_RESOLVED)) {
                return;
            }
            synchronized (CACHE_LOCK) {
                if (!checkFlags(FLAG_ALL_MEMBERS_RESOLVED)) {
                    setFlags(FLAG_RESOLVING_MEMBERS);

                    final int oldFlags = _flags;

                    RESOLVER.resolveMembers(this, oldFlags ^ FLAG_ALL_MEMBERS_RESOLVED);

                    if (_nestedTypes != null) {
                        for (final ReflectedType<?> nestedType : _nestedTypes) {
                            nestedType.complete();
                        }
                    }

                    if (FLAGS_UPDATER.compareAndSet(this, oldFlags, FLAG_ALL_MEMBERS_RESOLVED)) {
                        return;
                    }
                }
            }
        }
    }

    @Override
    public MethodBase getDeclaringMethod() {
        return _declaringMethod;
    }

    @Override
    protected ConstructorList getDeclaredConstructors() {
        ensureMembersResolved();
        if (_constructors == null) {
            return ConstructorList.empty();
        }
        return new ConstructorList(_constructors);
    }

    @Override
    protected MethodList getDeclaredMethods() {
        ensureMembersResolved();
        if (_methods == null) {
            return MethodList.empty();
        }
        return new MethodList(_methods);
    }

    @Override
    protected FieldList getDeclaredFields() {
        ensureMembersResolved();
        if (_fields == null) {
            return FieldList.empty();
        }
        return new FieldList(_fields);
    }

    @Override
    protected TypeList getDeclaredTypes() {
        ensureMembersResolved();
        if (_nestedTypes == null) {
            return TypeList.empty();
        }
        return new TypeList(_nestedTypes);
    }

    void complete() {
        if (_completed) {
            return;
        }

        _completed = true;

        if (_genericParameters == null || _genericParameters.isEmpty()) {
            _typeBindings = TypeBindings.empty();
        }
        else {
            _typeBindings = TypeBindings.createUnbound(list(_genericParameters));
        }
    }

    @Override
    public Type<? super T> getBaseType() {
        return _baseType;
    }

    @Override
    public TypeList getExplicitInterfaces() {
        if (_interfaces == null) {
            return TypeList.empty();
        }
        return _interfaces;
    }

    @Override
    protected TypeBindings getTypeBindings() {
        completeIfNecessary();
        return _typeBindings;
    }

    @Override
    public Type getGenericTypeDefinition() {
        if (!isGenericType()) {
            throw Error.notGenericType(this);
        }
        if (!getTypeBindings().hasBoundParameters()) {
            return this;
        }
        throw ContractUtils.unreachable();
    }

    @Override
    protected Type makeGenericTypeCore(final TypeList typeArguments) {
        synchronized (CACHE_LOCK) {
            return CACHE.getGenericType(getGenericTypeDefinition(), typeArguments);
        }
    }

    @Override
    public Class<T> getErasedClass() {
        completeIfNecessary();
        return _rawClass;
    }

    @Override
    public MemberType getMemberType() {
        return MemberType.TypeInfo;
    }

    @Override
    public Type getDeclaringType() {
        return _declaringType;
    }

    @Override
    public int getModifiers() {
        return _rawClass.getModifiers();
    }

    @Override
    public <P, R> R accept(final TypeVisitor<P, R> visitor, final P parameter) {
        return visitor.visitClassType(this, parameter);
    }

    @Override
    protected StringBuilder _appendClassName(final StringBuilder sb, final boolean fullName, final boolean dottedName) {
        if (!fullName) {
            return sb.append(_simpleName);
        }
        if (dottedName) {
            return sb.append(_name);
        }
        return super._appendClassName(sb, fullName, dottedName);
    }

    private final static SimpleVisitor<java.lang.reflect.Type, GenericParameter> GenericParameterFinder =
        new SimpleVisitor<java.lang.reflect.Type, GenericParameter>() {
            public GenericParameter visit(final TypeList types, final java.lang.reflect.Type s) {
                for (final Type type : types) {
                    final GenericParameter result = visit(type, s);
                    if (result != null) {
                        return result;
                    }
                }
                return null;
            }

            @Override
            public GenericParameter visitCapturedType(final Type<?> t, final java.lang.reflect.Type s) {
                return null;
            }

            @Override
            public GenericParameter visitClassType(final Type<?> type, final java.lang.reflect.Type parameter) {
                GenericParameter result;

                if (type.isGenericType()) {
                    result = visit(type.getGenericTypeParameters(), parameter);

                    if (result != null) {
                        return result;
                    }
                }

                if (type instanceof ReflectedType) {
                    final ReflectedType reflectedType = (ReflectedType) type;
                    if (reflectedType._methods != null) {
                        for (final Object o : reflectedType._methods) {
                            final ReflectedMethod method = (ReflectedMethod) o;
                            if (method.isGenericMethod()) {
                                result = visit(method.getGenericMethodParameters(), parameter);

                                if (result != null) {
                                    return result;
                                }
                            }
                        }
                    }
                }

                final Type declaringType = type.getDeclaringType();

                if (declaringType != null && declaringType != NullType) {
                    return visitClassType(declaringType, parameter);
                }

                return null;
            }

            @Override
            public GenericParameter visitPrimitiveType(final Type<?> type, final java.lang.reflect.Type parameter) {
                return super.visitPrimitiveType(type, parameter);
            }

            @Override
            public GenericParameter visitTypeParameter(final Type<?> type, final java.lang.reflect.Type parameter) {
                if (parameter instanceof TypeVariable<?>) {
                    return visitTypeParameter(type, (TypeVariable) parameter);
                }
                return null;
            }

            public GenericParameter visitTypeParameter(final Type<?> type, final TypeVariable typeVariable) {
                if (!(type instanceof GenericParameter)) {
                    return null;
                }

                final GenericParameter genericParameter = (GenericParameter) type;

                final int position = ArrayUtilities.indexOf(
                    typeVariable.getGenericDeclaration().getTypeParameters(),
                    typeVariable
                );

                if (position != genericParameter.getGenericParameterPosition()) {
                    return null;
                }

                final Type declaringType = genericParameter.getDeclaringType();

                if (declaringType != null && declaringType.getErasedClass() == typeVariable.getGenericDeclaration()) {
                    return genericParameter;
                }

                final MethodInfo method = genericParameter.getDeclaringMethod();

                if (method != null && method.getRawMethod() == typeVariable.getGenericDeclaration()) {
                    return genericParameter;
                }

                return null;
            }

            @Override
            public GenericParameter visitWildcardType(final Type<?> type, final java.lang.reflect.Type parameter) {
                return null;
            }

            @Override
            public GenericParameter visitArrayType(final Type<?> type, final java.lang.reflect.Type parameter) {
                return null;
            }
        };
}
TOP

Related Classes of com.strobel.reflection.ReflectedType

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.