blob: 4aefe8db1e340cde025f3103b00c37d942decfe1 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2018 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.eclipse.jdt.internal.core.search.matching;
import org.eclipse.core.resources.IResource;
import org.eclipse.jdt.core.*;
import org.eclipse.jdt.core.compiler.CharOperation;
import org.eclipse.jdt.core.search.SearchDocument;
import org.eclipse.jdt.internal.compiler.ast.CompilationUnitDeclaration;
import org.eclipse.jdt.internal.compiler.env.IBinaryType;
import org.eclipse.jdt.internal.compiler.env.ICompilationUnit;
import org.eclipse.jdt.internal.compiler.lookup.TypeConstants;
import org.eclipse.jdt.internal.core.*;
import org.eclipse.jdt.internal.core.util.Util;
public class PossibleMatch implements ICompilationUnit {
public static final String NO_SOURCE_FILE_NAME = "NO SOURCE FILE NAME"; //$NON-NLS-1$
public static final char[] NO_SOURCE_FILE = new char[0];
public IResource resource;
public Openable openable;
public MatchingNodeSet nodeSet;
public char[][] compoundName;
CompilationUnitDeclaration parsedUnit;
public SearchDocument document;
private String sourceFileName;
private char[] source;
private PossibleMatch similarMatch;
public String autoModuleName;
public PossibleMatch(MatchLocator locator, IResource resource, Openable openable, SearchDocument document, boolean mustResolve) {
this.resource = resource;
this.openable = openable;
this.document = document;
this.nodeSet = new MatchingNodeSet(mustResolve);
char[] qualifiedName = getQualifiedName();
if (qualifiedName != null)
this.compoundName = CharOperation.splitOn('.', qualifiedName);
}
public void cleanUp() {
this.source = null;
if (this.parsedUnit != null) {
this.parsedUnit.cleanUp();
this.parsedUnit = null;
}
this.nodeSet = null;
}
@Override
public boolean equals(Object obj) {
if (this.compoundName == null) return super.equals(obj);
if (!(obj instanceof PossibleMatch)) return false;
// By using the compoundName of the source file, multiple .class files (A, A$M...) are considered equal
// Even .class files for secondary types and their nested types
return CharOperation.equals(this.compoundName, ((PossibleMatch) obj).compoundName);
}
@Override
public char[] getContents() {
char[] contents = (this.source == NO_SOURCE_FILE) ? null : this.source;
if (this.source == null) {
if (this.openable instanceof AbstractClassFile) {
String fileName = getSourceFileName();
if (fileName == NO_SOURCE_FILE_NAME) return CharOperation.NO_CHAR;
SourceMapper sourceMapper = this.openable.getSourceMapper();
if (sourceMapper != null) {
if (this.openable instanceof ClassFile) {
IType type = ((ClassFile) this.openable).getType();
contents = sourceMapper.findSource(type, fileName);
} else if (this.openable instanceof ModularClassFile) {
try {
IModuleDescription module = ((ModularClassFile) this.openable).getModule();
contents = module != null ? sourceMapper.findSource(module) : CharOperation.NO_CHAR; // FIXME(SHMOD)
} catch (JavaModelException e) {
return CharOperation.NO_CHAR;
}
}
}
} else if (this.autoModuleName != null) { // fab a module
contents = ("module " + this.autoModuleName + "{}").toCharArray(); //$NON-NLS-1$//$NON-NLS-2$
this.sourceFileName = "module-info.java"; //$NON-NLS-1$
this.compoundName = new char[][] { "module-info".toCharArray()}; //$NON-NLS-1$
} else {
contents = this.document.getCharContents();
}
this.source = (contents == null) ? NO_SOURCE_FILE : contents;
}
return contents;
}
/**
* The exact openable file name. In particular, will be the originating .class file for binary openable with attached
* source.
* @see org.eclipse.jdt.internal.compiler.env.IDependent#getFileName()
* @see PackageReferenceLocator#isDeclaringPackageFragment(IPackageFragment, org.eclipse.jdt.internal.compiler.lookup.ReferenceBinding)
*/
@Override
public char[] getFileName() {
return this.openable.getElementName().toCharArray();
}
@Override
public char[] getMainTypeName() {
// The file is no longer opened to get its name => remove fix for bug 32182
return this.compoundName[this.compoundName.length-1];
}
@Override
public char[][] getPackageName() {
int length = this.compoundName.length;
if (length <= 1) return CharOperation.NO_CHAR_CHAR;
return CharOperation.subarray(this.compoundName, 0, length - 1);
}
/*
* Returns the fully qualified name of the main type of the compilation unit
* or the main type of the .java file that defined the class file.
*/
private char[] getQualifiedName() {
if (this.openable instanceof CompilationUnit) {
// get file name
String fileName = this.openable.getElementName(); // working copy on a .class file may not have a resource, so use the element name
// get main type name
char[] mainTypeName = Util.getNameWithoutJavaLikeExtension(fileName).toCharArray();
CompilationUnit cu = (CompilationUnit) this.openable;
return cu.getType(new String(mainTypeName)).getFullyQualifiedName().toCharArray();
} else if (this.openable instanceof ClassFile) {
String fileName = getSourceFileName();
if (fileName == NO_SOURCE_FILE_NAME)
return ((ClassFile) this.openable).getType().getFullyQualifiedName('.').toCharArray();
// Class file may have a source file name with ".java" extension (see bug 73784)
int index = Util.indexOfJavaLikeExtension(fileName);
String simpleName = index==-1 ? fileName : fileName.substring(0, index);
PackageFragment pkg = (PackageFragment) this.openable.getParent();
return Util.concatWith(pkg.names, simpleName, '.').toCharArray();
} else if (this.openable instanceof ModularClassFile) {
// FIXME(SHMOD): not useful https://bugs.eclipse.org/501162#c30
String simpleName = TypeConstants.MODULE_INFO_NAME_STRING;
PackageFragment pkg = (PackageFragment) this.openable.getParent();
return Util.concatWith(pkg.names, simpleName, '.').toCharArray();
}
return null;
}
PossibleMatch getSimilarMatch() {
return this.similarMatch;
}
/*
* Returns the source file name of the class file.
* Returns NO_SOURCE_FILE_NAME if not found.
*/
private String getSourceFileName() {
if (this.sourceFileName != null) return this.sourceFileName;
this.sourceFileName = NO_SOURCE_FILE_NAME;
if (this.openable.getSourceMapper() != null) {
if (this.openable instanceof ClassFile) {
BinaryType type = (BinaryType) ((ClassFile) this.openable).getType();
IBinaryType reader = MatchLocator.classFileReader(type);
if (reader != null) {
String fileName = type.sourceFileName(reader);
this.sourceFileName = fileName == null ? NO_SOURCE_FILE_NAME : fileName;
}
} else if (this.openable instanceof ModularClassFile) {
// FIXME(SHMOD): premature https://bugs.eclipse.org/501162#c31
this.sourceFileName = TypeConstants.MODULE_INFO_FILE_NAME_STRING;
}
}
return this.sourceFileName;
}
boolean hasSimilarMatch() {
return this.similarMatch != null && (this.source == NO_SOURCE_FILE || isModuleInfo(this));
}
@Override
public int hashCode() {
if (this.compoundName == null) return super.hashCode();
int hashCode = 0;
for (int i = 0, length = this.compoundName.length; i < length; i++)
hashCode += CharOperation.hashCode(this.compoundName[i]);
return hashCode;
}
@Override
public boolean ignoreOptionalProblems() {
return false;
}
private boolean isModuleInfo(PossibleMatch possibleMatch) {
return CharOperation.equals(getMainTypeName(), TypeConstants.MODULE_INFO_NAME);
}
void setSimilarMatch(PossibleMatch possibleMatch) {
// source does not matter on similar match as it is read on
// the first stored possible match
possibleMatch.source = isModuleInfo(possibleMatch) ? null : NO_SOURCE_FILE;
this.similarMatch = possibleMatch;
}
@Override
public String toString() {
return this.openable == null ? "Fake PossibleMatch" : this.openable.toString(); //$NON-NLS-1$
}
@Override
public char[] getModuleName() {
if (this.openable instanceof CompilationUnit) {
return ((CompilationUnit) this.openable).getModuleName();
} else if (this.openable instanceof ClassFile) {
IModuleDescription moduleDescription = this.openable.getPackageFragmentRoot().getModuleDescription();
if (moduleDescription != null) {
return moduleDescription.getElementName().toCharArray();
}
}
return null;
}
}