Package org.aspectj.org.eclipse.jdt.core.dom

Source Code of org.aspectj.org.eclipse.jdt.core.dom.MethodBinding

/*******************************************************************************
* Copyright (c) 2000, 2007 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
*     IBM Corporation - initial API and implementation
*******************************************************************************/

package org.aspectj.org.eclipse.jdt.core.dom;

import java.util.ArrayList;
import java.util.Iterator;

import org.aspectj.org.eclipse.jdt.core.IJavaElement;
import org.aspectj.org.eclipse.jdt.core.IMethod;
import org.aspectj.org.eclipse.jdt.core.IType;
import org.aspectj.org.eclipse.jdt.core.JavaModelException;
import org.aspectj.org.eclipse.jdt.core.Signature;
import org.aspectj.org.eclipse.jdt.core.compiler.CharOperation;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.ExtraCompilerModifiers;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.LookupEnvironment;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodVerifier;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.ParameterizedGenericMethodBinding;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.RawTypeBinding;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.ReferenceBinding;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.TypeBinding;
import org.aspectj.org.eclipse.jdt.internal.compiler.lookup.TypeVariableBinding;
import org.aspectj.org.eclipse.jdt.internal.compiler.problem.AbortCompilation;
import org.aspectj.org.eclipse.jdt.internal.core.JavaElement;
import org.aspectj.org.eclipse.jdt.internal.core.Member;
import org.aspectj.org.eclipse.jdt.internal.core.util.Util;

/**
* Internal implementation of method bindings.
*/
class MethodBinding implements IMethodBinding {

  private static final int VALID_MODIFIERS = Modifier.PUBLIC | Modifier.PROTECTED | Modifier.PRIVATE |
    Modifier.ABSTRACT | Modifier.STATIC | Modifier.FINAL | Modifier.SYNCHRONIZED | Modifier.NATIVE |
    Modifier.STRICTFP;
  private static final ITypeBinding[] NO_TYPE_BINDINGS = new ITypeBinding[0];
  private org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding binding;
  private BindingResolver resolver;
  private ITypeBinding[] parameterTypes;
  private ITypeBinding[] exceptionTypes;
  private String name;
  private ITypeBinding declaringClass;
  private ITypeBinding returnType;
  private String key;
  private ITypeBinding[] typeParameters;
  private ITypeBinding[] typeArguments;
  private IAnnotationBinding[] annotations;
  private IAnnotationBinding[] parameterAnnotations;

  MethodBinding(BindingResolver resolver, org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding binding) {
    this.resolver = resolver;
    this.binding = binding;
  }

  public boolean isAnnotationMember() {
    return getDeclaringClass().isAnnotation();
  }

  /**
   * @see IMethodBinding#isConstructor()
   */
  public boolean isConstructor() {
    return this.binding.isConstructor();
  }

  /**
   * @see IMethodBinding#isDefaultConstructor()
   * @since 3.0
   */
  public boolean isDefaultConstructor() {
    final ReferenceBinding declaringClassBinding = this.binding.declaringClass;
    if (declaringClassBinding.isRawType()) {
      RawTypeBinding rawTypeBinding = (RawTypeBinding) declaringClassBinding;
      if (rawTypeBinding.genericType().isBinaryBinding()) {
        return false;
      }
      return (this.binding.modifiers & ExtraCompilerModifiers.AccIsDefaultConstructor) != 0;
    }
    if (declaringClassBinding.isBinaryBinding()) {
      return false;
    }
    return (this.binding.modifiers & ExtraCompilerModifiers.AccIsDefaultConstructor) != 0;
  }

  /**
   * @see IBinding#getName()
   */
  public String getName() {
    if (name == null) {
      if (this.binding.isConstructor()) {
        name = this.getDeclaringClass().getName();
      } else {
        name = new String(this.binding.selector);
      }
    }
    return name;
  }

  public IAnnotationBinding[] getAnnotations() {
    if (this.annotations != null) {
      return this.annotations;
    }
    org.aspectj.org.eclipse.jdt.internal.compiler.lookup.AnnotationBinding[] annots = this.binding.getAnnotations();
    int length = annots == null ? 0 : annots.length;
    if (length == 0) {
      return this.annotations = AnnotationBinding.NoAnnotations;
    }
    IAnnotationBinding[] domInstances = new AnnotationBinding[length];
    for (int i = 0; i < length; i++) {
      final IAnnotationBinding annotationInstance = this.resolver.getAnnotationInstance(annots[i]);
      if (annotationInstance == null) {
        return this.annotations = AnnotationBinding.NoAnnotations;
      }
      domInstances[i] = annotationInstance;
    }
    return this.annotations = domInstances;
  }

  /**
   * @see IMethodBinding#getDeclaringClass()
   */
  public ITypeBinding getDeclaringClass() {
    if (this.declaringClass == null) {
      this.declaringClass = this.resolver.getTypeBinding(this.binding.declaringClass);
    }
    return declaringClass;
  }

  public IAnnotationBinding[] getParameterAnnotations(int index) {
    if (this.parameterAnnotations != null) {
      return this.parameterAnnotations;
    }
    org.aspectj.org.eclipse.jdt.internal.compiler.lookup.AnnotationBinding[] annots = this.binding.getParameterAnnotations(index);
    int length = annots == null ? 0 : annots.length;
    if (length == 0) {
      return this.parameterAnnotations = AnnotationBinding.NoAnnotations;
    }
    IAnnotationBinding[] domInstances =new AnnotationBinding[length];
    for (int i = 0; i < length; i++) {
      final IAnnotationBinding annotationInstance = this.resolver.getAnnotationInstance(annots[i]);
      if (annotationInstance == null) {
        return this.parameterAnnotations = AnnotationBinding.NoAnnotations;
      }
      domInstances[i] = annotationInstance;
    }
    return this.parameterAnnotations = domInstances;
  }

  /**
   * @see IMethodBinding#getParameterTypes()
   */
  public ITypeBinding[] getParameterTypes() {
    if (this.parameterTypes != null) {
      return parameterTypes;
    }
    org.aspectj.org.eclipse.jdt.internal.compiler.lookup.TypeBinding[] parameters = this.binding.parameters;
    int length = parameters == null ? 0 : parameters.length;
    if (length == 0) {
      return this.parameterTypes = NO_TYPE_BINDINGS;
    } else {
      ITypeBinding[] paramTypes = new ITypeBinding[length];
      for (int i = 0; i < length; i++) {
        final TypeBinding parameterBinding = parameters[i];
        if (parameterBinding != null) {
          ITypeBinding typeBinding = this.resolver.getTypeBinding(parameterBinding);
          if (typeBinding == null) {
            return this.parameterTypes = NO_TYPE_BINDINGS;
          }
          paramTypes[i] = typeBinding;
        } else {
          // log error
          StringBuffer message = new StringBuffer("Report method binding where a parameter is null:\n")//$NON-NLS-1$
          message.append(this.toString());
          Util.log(new IllegalArgumentException(), message.toString());
          // report no binding since one or more parameter has no binding
          return this.parameterTypes = NO_TYPE_BINDINGS;
        }
      }
      return this.parameterTypes = paramTypes;
    }
  }

  /**
   * @see IMethodBinding#getReturnType()
   */
  public ITypeBinding getReturnType() {
    if (this.returnType == null) {
      this.returnType = this.resolver.getTypeBinding(this.binding.returnType);
    }
    return this.returnType;
  }

  public Object getDefaultValue() {
    if (isAnnotationMember())
      return MemberValuePairBinding.buildDOMValue(this.binding.getDefaultValue(), this.resolver);
    return null;
  }

  /**
   * @see IMethodBinding#getExceptionTypes()
   */
  public ITypeBinding[] getExceptionTypes() {
    if (this.exceptionTypes != null) {
      return exceptionTypes;
    }
    org.aspectj.org.eclipse.jdt.internal.compiler.lookup.TypeBinding[] exceptions = this.binding.thrownExceptions;
    int length = exceptions == null ? 0 : exceptions.length;
    if (length == 0) {
      return this.exceptionTypes = NO_TYPE_BINDINGS;
    }
    ITypeBinding[] exTypes = new ITypeBinding[length];
    for (int i = 0; i < length; i++) {
      ITypeBinding typeBinding = this.resolver.getTypeBinding(exceptions[i]);
      if (typeBinding == null) {
        return this.exceptionTypes = NO_TYPE_BINDINGS;
      }
      exTypes[i] = typeBinding;
    }
    return this.exceptionTypes = exTypes;
  }

  public IJavaElement getJavaElement() {
    JavaElement element = getUnresolvedJavaElement();
    if (element == null)
      return null;
    return element.resolved(this.binding);
  }

  private JavaElement getUnresolvedJavaElement() {
    IType declaringType = (IType) getDeclaringClass().getJavaElement();
    if (declaringType == null) return null;
    if (!(this.resolver instanceof DefaultBindingResolver)) return null;
    ASTNode node = (ASTNode) ((DefaultBindingResolver) this.resolver).bindingsToAstNodes.get(this);
    if (node != null && declaringType.getParent().getElementType() != IJavaElement.CLASS_FILE) {
      if (node instanceof MethodDeclaration) {
        MethodDeclaration methodDeclaration = (MethodDeclaration) node;
        ArrayList parameterSignatures = new ArrayList();
        Iterator iterator = methodDeclaration.parameters().iterator();
        while (iterator.hasNext()) {
          SingleVariableDeclaration parameter = (SingleVariableDeclaration) iterator.next();
          Type type = parameter.getType();
          String typeSig = Util.getSignature(type);
          int arrayDim = parameter.getExtraDimensions();
          if (parameter.getAST().apiLevel() >= AST.JLS3 && parameter.isVarargs()) {
            arrayDim++;
          }
          if (arrayDim > 0) {
            typeSig = Signature.createArraySignature(typeSig, arrayDim);
          }
          parameterSignatures.add(typeSig);
        }
        int parameterCount = parameterSignatures.size();
        String[] parameters = new String[parameterCount];
        parameterSignatures.toArray(parameters);
        return (JavaElement) declaringType.getMethod(getName(), parameters);
      } else {
        // annotation type member declaration
        AnnotationTypeMemberDeclaration typeMemberDeclaration = (AnnotationTypeMemberDeclaration) node;
        return (JavaElement) declaringType.getMethod(typeMemberDeclaration.getName().getIdentifier(), CharOperation.NO_STRINGS); // annotation type members don't have parameters
      }
    } else {
      // case of method not in the created AST, or a binary method
      org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding original = this.binding.original();
      String selector = original.isConstructor() ? declaringType.getElementName() : new String(original.selector);
      boolean isBinary = declaringType.isBinary();
      ReferenceBinding enclosingType = original.declaringClass.enclosingType();
      boolean isInnerBinaryTypeConstructor = isBinary && original.isConstructor() && enclosingType != null;
      TypeBinding[] parameters = original.parameters;
      int length = parameters == null ? 0 : parameters.length;
      int declaringIndex = isInnerBinaryTypeConstructor ? 1 : 0;
      String[] parameterSignatures = new String[declaringIndex + length];
      if (isInnerBinaryTypeConstructor)
        parameterSignatures[0] = new String(enclosingType.genericTypeSignature()).replace('/', '.');
      for (int i = 0;  i < length; i++) {
        parameterSignatures[declaringIndex + i] = new String(parameters[i].genericTypeSignature()).replace('/', '.');
      }
      IMethod result = declaringType.getMethod(selector, parameterSignatures);
      if (isBinary)
        return (JavaElement) result;
      IMethod[] methods = null;
      try {
        methods = declaringType.getMethods();
      } catch (JavaModelException e) {
        // declaring type doesn't exist
        return null;
      }
      IMethod[] candidates = Member.findMethods(result, methods);
      if (candidates == null || candidates.length == 0)
        return null;
      return (JavaElement) candidates[0];
    }
  }

  /**
   * @see IBinding#getKind()
   */
  public int getKind() {
    return IBinding.METHOD;
  }

  /**
   * @see IBinding#getModifiers()
   */
  public int getModifiers() {
    return this.binding.getAccessFlags() & VALID_MODIFIERS;
  }

  /**
   * @see IBinding#isDeprecated()
   */
  public boolean isDeprecated() {
    return this.binding.isDeprecated();
  }

  /**
   * @see IBinding#isRecovered()
   */
  public boolean isRecovered() {
    return false;
  }

  /**
   * @see IBinding#isSynthetic()
   */
  public boolean isSynthetic() {
    return this.binding.isSynthetic();
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#isVarargs()
   * @since 3.1
   */
  public boolean isVarargs() {
    return this.binding.isVarargs();
  }

  /**
   * @see IBinding#getKey()
   */
  public String getKey() {
    if (this.key == null) {
      this.key = new String(this.binding.computeUniqueKey());
    }
    return this.key;
  }

  /**
   * @see IBinding#isEqualTo(IBinding)
   * @since 3.1
   */
  public boolean isEqualTo(IBinding other) {
    if (other == this) {
      // identical binding - equal (key or no key)
      return true;
    }
    if (other == null) {
      // other binding missing
      return false;
    }
    if (!(other instanceof MethodBinding)) {
      return false;
    }
    org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding otherBinding = ((MethodBinding) other).binding;
    return BindingComparator.isEqual(this.binding, otherBinding);
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#getTypeParameters()
   */
  public ITypeBinding[] getTypeParameters() {
    if (this.typeParameters != null) {
      return this.typeParameters;
    }
    TypeVariableBinding[] typeVariableBindings = this.binding.typeVariables();
    int typeVariableBindingsLength = typeVariableBindings == null ? 0 : typeVariableBindings.length;
    if (typeVariableBindingsLength == 0) {
      return this.typeParameters = NO_TYPE_BINDINGS;
    }
    ITypeBinding[] tParameters = new ITypeBinding[typeVariableBindingsLength];
    for (int i = 0; i < typeVariableBindingsLength; i++) {
      ITypeBinding typeBinding = this.resolver.getTypeBinding(typeVariableBindings[i]);
      if (typeBinding == null) {
        return this.typeParameters = NO_TYPE_BINDINGS;
      }
      tParameters[i] = typeBinding;
    }
    return this.typeParameters = tParameters;
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#isGenericMethod()
   * @since 3.1
   */
  public boolean isGenericMethod() {
    // equivalent to return getTypeParameters().length > 0;
    if (this.typeParameters != null) {
      return this.typeParameters.length > 0;
    }
    TypeVariableBinding[] typeVariableBindings = this.binding.typeVariables();
    return (typeVariableBindings != null && typeVariableBindings.length > 0);
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#getTypeArguments()
   */
  public ITypeBinding[] getTypeArguments() {
    if (this.typeArguments != null) {
      return this.typeArguments;
    }

    if (this.binding instanceof ParameterizedGenericMethodBinding) {
      ParameterizedGenericMethodBinding genericMethodBinding = (ParameterizedGenericMethodBinding) this.binding;
      org.aspectj.org.eclipse.jdt.internal.compiler.lookup.TypeBinding[] typeArgumentsBindings = genericMethodBinding.typeArguments;
      int typeArgumentsLength = typeArgumentsBindings == null ? 0 : typeArgumentsBindings.length;
      if (typeArgumentsLength != 0) {
        ITypeBinding[] tArguments = new ITypeBinding[typeArgumentsLength];
        for (int i = 0; i < typeArgumentsLength; i++) {
          ITypeBinding typeBinding = this.resolver.getTypeBinding(typeArgumentsBindings[i]);
          if (typeBinding == null) {
            return this.typeArguments = NO_TYPE_BINDINGS;
          }
          tArguments[i] = typeBinding;
        }
        return this.typeArguments = tArguments;
      }
    }
    return this.typeArguments = NO_TYPE_BINDINGS;
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#isParameterizedMethod()
   */
  public boolean isParameterizedMethod() {
    return (this.binding instanceof ParameterizedGenericMethodBinding)
      && !((ParameterizedGenericMethodBinding) this.binding).isRaw;
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#isRawMethod()
   */
  public boolean isRawMethod() {
    return (this.binding instanceof ParameterizedGenericMethodBinding)
      && ((ParameterizedGenericMethodBinding) this.binding).isRaw;
  }

  public boolean isSubsignature(IMethodBinding otherMethod) {
    try {
      org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding other = ((MethodBinding) otherMethod).binding;
      if (!CharOperation.equals(this.binding.selector, other.selector))
        return false;
      return this.binding.areParameterErasuresEqual(other) && this.binding.areTypeVariableErasuresEqual(other);
    } catch (AbortCompilation e) {
      // don't surface internal exception to clients
      // see https://bugs.eclipse.org/bugs/show_bug.cgi?id=143013
      return false;
    }
  }

  /**
   * @see org.aspectj.org.eclipse.jdt.core.dom.IMethodBinding#getMethodDeclaration()
   */
  public IMethodBinding getMethodDeclaration() {
    return this.resolver.getMethodBinding(this.binding.original());
  }

  /**
   * @see IMethodBinding#overrides(IMethodBinding)
   */
  public boolean overrides(IMethodBinding overridenMethod) {
    try {
      org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding overridenCompilerBinding = ((MethodBinding) overridenMethod).binding;
      if (this.binding == overridenCompilerBinding
          || overridenCompilerBinding.isStatic()
          || overridenCompilerBinding.isPrivate()
          || this.binding.isStatic())
        return false;
      char[] selector = this.binding.selector;
      if (!CharOperation.equals(selector, overridenCompilerBinding.selector))
        return false;
      TypeBinding match = this.binding.declaringClass.findSuperTypeWithSameErasure(overridenCompilerBinding.declaringClass);
      if (!(match instanceof ReferenceBinding)) return false;

      org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding[] superMethods = ((ReferenceBinding)match).getMethods(selector);
      for (int i = 0, length = superMethods.length; i < length; i++) {
        if (superMethods[i].original() == overridenCompilerBinding) {
          LookupEnvironment lookupEnvironment = this.resolver.lookupEnvironment();
          if (lookupEnvironment == null) return false;
          MethodVerifier methodVerifier = lookupEnvironment.methodVerifier();
          org.aspectj.org.eclipse.jdt.internal.compiler.lookup.MethodBinding superMethod = superMethods[i];
          return !(superMethod.isDefault() && (superMethod.declaringClass.getPackage()) != this.binding.declaringClass.getPackage())
            && methodVerifier.doesMethodOverride(this.binding, superMethod);
        }
      }
      return false;
    } catch (AbortCompilation e) {
      // don't surface internal exception to clients
      // see https://bugs.eclipse.org/bugs/show_bug.cgi?id=143013
      return false;
    }
  }

  /**
   * For debugging purpose only.
   * @see java.lang.Object#toString()
   */
  public String toString() {
    return this.binding.toString();
  }
}
TOP

Related Classes of org.aspectj.org.eclipse.jdt.core.dom.MethodBinding

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.