blob: 05e7c9942198e7dc1822d4d2afde8837e9a24525 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2014 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
* Yevgen Kogan - Bug 403475 - Hot Code Replace drops too much frames in some cases
* Jacob Saoumi - Bug 434695 - Hot Code Replace drops some frames in case of anonymous classes
*******************************************************************************/
package org.eclipse.jdt.internal.debug.core.hcr;
import org.eclipse.jdt.core.Signature;
import org.eclipse.jdt.core.dom.ASTNode;
import org.eclipse.jdt.core.dom.ASTVisitor;
import org.eclipse.jdt.core.dom.AbstractTypeDeclaration;
import org.eclipse.jdt.core.dom.AnnotationTypeDeclaration;
import org.eclipse.jdt.core.dom.AnnotationTypeMemberDeclaration;
import org.eclipse.jdt.core.dom.AnonymousClassDeclaration;
import org.eclipse.jdt.core.dom.ArrayAccess;
import org.eclipse.jdt.core.dom.ArrayCreation;
import org.eclipse.jdt.core.dom.ArrayInitializer;
import org.eclipse.jdt.core.dom.ArrayType;
import org.eclipse.jdt.core.dom.AssertStatement;
import org.eclipse.jdt.core.dom.Assignment;
import org.eclipse.jdt.core.dom.Block;
import org.eclipse.jdt.core.dom.BlockComment;
import org.eclipse.jdt.core.dom.BooleanLiteral;
import org.eclipse.jdt.core.dom.BreakStatement;
import org.eclipse.jdt.core.dom.CastExpression;
import org.eclipse.jdt.core.dom.CatchClause;
import org.eclipse.jdt.core.dom.CharacterLiteral;
import org.eclipse.jdt.core.dom.ClassInstanceCreation;
import org.eclipse.jdt.core.dom.CompilationUnit;
import org.eclipse.jdt.core.dom.ConditionalExpression;
import org.eclipse.jdt.core.dom.ConstructorInvocation;
import org.eclipse.jdt.core.dom.ContinueStatement;
import org.eclipse.jdt.core.dom.DoStatement;
import org.eclipse.jdt.core.dom.EmptyStatement;
import org.eclipse.jdt.core.dom.EnhancedForStatement;
import org.eclipse.jdt.core.dom.EnumConstantDeclaration;
import org.eclipse.jdt.core.dom.EnumDeclaration;
import org.eclipse.jdt.core.dom.ExpressionStatement;
import org.eclipse.jdt.core.dom.FieldAccess;
import org.eclipse.jdt.core.dom.FieldDeclaration;
import org.eclipse.jdt.core.dom.ForStatement;
import org.eclipse.jdt.core.dom.IMethodBinding;
import org.eclipse.jdt.core.dom.ITypeBinding;
import org.eclipse.jdt.core.dom.IfStatement;
import org.eclipse.jdt.core.dom.ImportDeclaration;
import org.eclipse.jdt.core.dom.InfixExpression;
import org.eclipse.jdt.core.dom.Initializer;
import org.eclipse.jdt.core.dom.InstanceofExpression;
import org.eclipse.jdt.core.dom.Javadoc;
import org.eclipse.jdt.core.dom.LabeledStatement;
import org.eclipse.jdt.core.dom.LineComment;
import org.eclipse.jdt.core.dom.MarkerAnnotation;
import org.eclipse.jdt.core.dom.MemberRef;
import org.eclipse.jdt.core.dom.MemberValuePair;
import org.eclipse.jdt.core.dom.MethodDeclaration;
import org.eclipse.jdt.core.dom.MethodInvocation;
import org.eclipse.jdt.core.dom.MethodRef;
import org.eclipse.jdt.core.dom.MethodRefParameter;
import org.eclipse.jdt.core.dom.Modifier;
import org.eclipse.jdt.core.dom.NormalAnnotation;
import org.eclipse.jdt.core.dom.NullLiteral;
import org.eclipse.jdt.core.dom.NumberLiteral;
import org.eclipse.jdt.core.dom.PackageDeclaration;
import org.eclipse.jdt.core.dom.ParameterizedType;
import org.eclipse.jdt.core.dom.ParenthesizedExpression;
import org.eclipse.jdt.core.dom.PostfixExpression;
import org.eclipse.jdt.core.dom.PrefixExpression;
import org.eclipse.jdt.core.dom.PrimitiveType;
import org.eclipse.jdt.core.dom.QualifiedName;
import org.eclipse.jdt.core.dom.QualifiedType;
import org.eclipse.jdt.core.dom.ReturnStatement;
import org.eclipse.jdt.core.dom.SimpleName;
import org.eclipse.jdt.core.dom.SimpleType;
import org.eclipse.jdt.core.dom.SingleMemberAnnotation;
import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
import org.eclipse.jdt.core.dom.StringLiteral;
import org.eclipse.jdt.core.dom.SuperConstructorInvocation;
import org.eclipse.jdt.core.dom.SuperFieldAccess;
import org.eclipse.jdt.core.dom.SuperMethodInvocation;
import org.eclipse.jdt.core.dom.SwitchCase;
import org.eclipse.jdt.core.dom.SwitchStatement;
import org.eclipse.jdt.core.dom.SynchronizedStatement;
import org.eclipse.jdt.core.dom.TagElement;
import org.eclipse.jdt.core.dom.TextElement;
import org.eclipse.jdt.core.dom.ThisExpression;
import org.eclipse.jdt.core.dom.ThrowStatement;
import org.eclipse.jdt.core.dom.TryStatement;
import org.eclipse.jdt.core.dom.TypeDeclaration;
import org.eclipse.jdt.core.dom.TypeDeclarationStatement;
import org.eclipse.jdt.core.dom.TypeLiteral;
import org.eclipse.jdt.core.dom.TypeParameter;
import org.eclipse.jdt.core.dom.UnionType;
import org.eclipse.jdt.core.dom.VariableDeclarationExpression;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.core.dom.VariableDeclarationStatement;
import org.eclipse.jdt.core.dom.WhileStatement;
import org.eclipse.jdt.core.dom.WildcardType;
/**
* Visits an AST to find a method declaration
*/
public class MethodSearchVisitor extends ASTVisitor {
/**
* Class the method belongs to
*/
private String fClassName;
/**
* Method to search for
*/
private String fName;
private String[] fParameterTypes;
/**
* The search result, or <code>null</code> if none
*/
private MethodDeclaration fMatch;
/**
* Sets the signature of the method to search for
*
* @param methodName
* name of method to search for
* @param methodSignature
* signature of the method to search for
*/
public void setTargetMethod(String className, String methodName, String methodSignature) {
fClassName = className;
fName = methodName;
fParameterTypes = Signature.getParameterTypes(methodSignature);
// convert parameter types same format that we get from the AST type
// bindings
for (int i = 0; i < fParameterTypes.length; i++) {
String type = fParameterTypes[i];
type = type.replace('/', '.');
fParameterTypes[i] = type;
}
fMatch = null;
}
/**
* Returns the search result, or <code>null</code> if none
*
* @return matching method declartion or <code>null</code>
*/
public MethodDeclaration getMatch() {
return fMatch;
}
/*
* (non-Javadoc)
*
* @see org.eclipse.jdt.core.dom.ASTVisitor#visit(org.eclipse.jdt.core.dom.
* MethodDeclaration)
*/
@Override
public boolean visit(MethodDeclaration node) {
ITypeBinding binding = null;
IMethodBinding mbinding = node.resolveBinding();
if(mbinding != null) {
binding = mbinding.getDeclaringClass();
}
else {
ASTNode parent = node.getParent();
if(parent instanceof AbstractTypeDeclaration) {
binding = ((AbstractTypeDeclaration) parent).resolveBinding();
}
else if(parent instanceof AnonymousClassDeclaration) {
binding = ((AnonymousClassDeclaration) parent).resolveBinding();
}
}
String typeName = null;
if (binding != null) {
typeName = binding.getQualifiedName();
if ((typeName == null || "".equals(typeName)) && binding.getBinaryName() != null) { //$NON-NLS-1$
typeName = binding.getBinaryName().replace('$', '.');
}
}
// if no binding exists, the behaviour should be the same as without checking for type name
if (node.getName().getIdentifier().equals(fName) && (typeName == null || typeName.equals(fClassName))) {
IMethodBinding methodBinding = node.resolveBinding();
if (methodBinding != null) {
ITypeBinding[] typeBindings = methodBinding.getParameterTypes();
if (typeBindings.length == fParameterTypes.length) {
for (int i = 0; i < typeBindings.length; i++) {
ITypeBinding typeBinding = typeBindings[i];
String typeSignature = Signature.createTypeSignature(
typeBinding.getQualifiedName(), true);
if (!fParameterTypes[i].equals(typeSignature)) {
return true;
}
}
fMatch = node;
}
}
}
return isSearching();
}
/**
* Returns whether this visitor is still searching for a match.
*
* @return whether this visitor is still searching for a match
*/
private boolean isSearching() {
return fMatch == null;
}
@Override
public boolean visit(AnnotationTypeDeclaration node) {
return isSearching();
}
@Override
public boolean visit(AnnotationTypeMemberDeclaration node) {
return isSearching();
}
@Override
public boolean visit(AnonymousClassDeclaration node) {
return isSearching();
}
@Override
public boolean visit(ArrayAccess node) {
return isSearching();
}
@Override
public boolean visit(ArrayCreation node) {
return isSearching();
}
@Override
public boolean visit(ArrayInitializer node) {
return isSearching();
}
@Override
public boolean visit(ArrayType node) {
return isSearching();
}
@Override
public boolean visit(AssertStatement node) {
return isSearching();
}
@Override
public boolean visit(Assignment node) {
return isSearching();
}
@Override
public boolean visit(Block node) {
return isSearching();
}
@Override
public boolean visit(BlockComment node) {
return isSearching();
}
@Override
public boolean visit(BooleanLiteral node) {
return isSearching();
}
@Override
public boolean visit(BreakStatement node) {
return isSearching();
}
@Override
public boolean visit(CastExpression node) {
return isSearching();
}
@Override
public boolean visit(CatchClause node) {
return isSearching();
}
@Override
public boolean visit(CharacterLiteral node) {
return isSearching();
}
@Override
public boolean visit(ClassInstanceCreation node) {
return isSearching();
}
@Override
public boolean visit(CompilationUnit node) {
return isSearching();
}
@Override
public boolean visit(ConditionalExpression node) {
return isSearching();
}
@Override
public boolean visit(ConstructorInvocation node) {
return isSearching();
}
@Override
public boolean visit(ContinueStatement node) {
return isSearching();
}
@Override
public boolean visit(DoStatement node) {
return isSearching();
}
@Override
public boolean visit(EmptyStatement node) {
return isSearching();
}
@Override
public boolean visit(EnhancedForStatement node) {
return isSearching();
}
@Override
public boolean visit(EnumConstantDeclaration node) {
return isSearching();
}
@Override
public boolean visit(EnumDeclaration node) {
return isSearching();
}
@Override
public boolean visit(ExpressionStatement node) {
return isSearching();
}
@Override
public boolean visit(FieldAccess node) {
return isSearching();
}
@Override
public boolean visit(FieldDeclaration node) {
return isSearching();
}
@Override
public boolean visit(ForStatement node) {
return isSearching();
}
@Override
public boolean visit(IfStatement node) {
return isSearching();
}
@Override
public boolean visit(ImportDeclaration node) {
return isSearching();
}
@Override
public boolean visit(InfixExpression node) {
return isSearching();
}
@Override
public boolean visit(Initializer node) {
return isSearching();
}
@Override
public boolean visit(InstanceofExpression node) {
return isSearching();
}
@Override
public boolean visit(Javadoc node) {
return isSearching();
}
@Override
public boolean visit(LabeledStatement node) {
return isSearching();
}
@Override
public boolean visit(LineComment node) {
return isSearching();
}
@Override
public boolean visit(MarkerAnnotation node) {
return isSearching();
}
@Override
public boolean visit(MemberRef node) {
return isSearching();
}
@Override
public boolean visit(MemberValuePair node) {
return isSearching();
}
@Override
public boolean visit(MethodInvocation node) {
return isSearching();
}
@Override
public boolean visit(MethodRef node) {
return isSearching();
}
@Override
public boolean visit(MethodRefParameter node) {
return isSearching();
}
@Override
public boolean visit(Modifier node) {
return isSearching();
}
@Override
public boolean visit(NormalAnnotation node) {
return isSearching();
}
@Override
public boolean visit(NullLiteral node) {
return isSearching();
}
@Override
public boolean visit(NumberLiteral node) {
return isSearching();
}
@Override
public boolean visit(PackageDeclaration node) {
return isSearching();
}
@Override
public boolean visit(ParameterizedType node) {
return isSearching();
}
@Override
public boolean visit(ParenthesizedExpression node) {
return isSearching();
}
@Override
public boolean visit(PostfixExpression node) {
return isSearching();
}
@Override
public boolean visit(PrefixExpression node) {
return isSearching();
}
@Override
public boolean visit(PrimitiveType node) {
return isSearching();
}
@Override
public boolean visit(QualifiedName node) {
return isSearching();
}
@Override
public boolean visit(QualifiedType node) {
return isSearching();
}
@Override
public boolean visit(ReturnStatement node) {
return isSearching();
}
@Override
public boolean visit(SimpleName node) {
return isSearching();
}
@Override
public boolean visit(SimpleType node) {
return isSearching();
}
@Override
public boolean visit(SingleMemberAnnotation node) {
return isSearching();
}
@Override
public boolean visit(SingleVariableDeclaration node) {
return isSearching();
}
@Override
public boolean visit(StringLiteral node) {
return isSearching();
}
@Override
public boolean visit(SuperConstructorInvocation node) {
return isSearching();
}
@Override
public boolean visit(SuperFieldAccess node) {
return isSearching();
}
@Override
public boolean visit(SuperMethodInvocation node) {
return isSearching();
}
@Override
public boolean visit(SwitchCase node) {
return isSearching();
}
@Override
public boolean visit(SwitchStatement node) {
return isSearching();
}
@Override
public boolean visit(SynchronizedStatement node) {
return isSearching();
}
@Override
public boolean visit(TagElement node) {
return isSearching();
}
@Override
public boolean visit(TextElement node) {
return isSearching();
}
@Override
public boolean visit(ThisExpression node) {
return isSearching();
}
@Override
public boolean visit(ThrowStatement node) {
return isSearching();
}
@Override
public boolean visit(TryStatement node) {
return isSearching();
}
@Override
public boolean visit(TypeDeclaration node) {
return isSearching();
}
@Override
public boolean visit(TypeDeclarationStatement node) {
return isSearching();
}
@Override
public boolean visit(TypeLiteral node) {
return isSearching();
}
@Override
public boolean visit(TypeParameter node) {
return isSearching();
}
@Override
public boolean visit(UnionType node) {
return super.visit(node);
}
@Override
public boolean visit(VariableDeclarationExpression node) {
return isSearching();
}
@Override
public boolean visit(VariableDeclarationFragment node) {
return isSearching();
}
@Override
public boolean visit(VariableDeclarationStatement node) {
return isSearching();
}
@Override
public boolean visit(WhileStatement node) {
return isSearching();
}
@Override
public boolean visit(WildcardType node) {
return isSearching();
}
}