blob: 07fa0fdf1bba25a350f96351f79bb62c8a53b617 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2005 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;
import java.util.ArrayList;
import java.util.HashSet;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.Path;
import org.eclipse.jdt.core.*;
import org.eclipse.jdt.core.IClasspathEntry;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.IJavaElementDelta;
import org.eclipse.jdt.core.IJavaModel;
import org.eclipse.jdt.core.IJavaProject;
import org.eclipse.jdt.core.IMember;
import org.eclipse.jdt.core.IOpenable;
import org.eclipse.jdt.core.IPackageFragmentRoot;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jdt.internal.compiler.env.AccessRuleSet;
import org.eclipse.jdt.internal.core.*;
import org.eclipse.jdt.internal.core.JavaElement;
import org.eclipse.jdt.internal.core.JavaModelManager;
import org.eclipse.jdt.internal.core.JavaProject;
import org.eclipse.jdt.internal.core.util.Util;
/**
* A Java-specific scope for searching relative to one or more java elements.
*/
public class JavaSearchScope extends AbstractSearchScope {
private ArrayList elements;
/* The paths of the resources in this search scope
(or the classpath entries' paths
if the resources are projects) */
private String[] paths;
private boolean[] pathWithSubFolders;
protected AccessRuleSet[] pathRestrictions;
private String[] containerPaths;
private int pathsCount;
private int threshold;
private IPath[] enclosingProjectsAndJars;
public final static AccessRuleSet NOT_ENCLOSED = new AccessRuleSet(null);
public JavaSearchScope() {
this(5);
}
private JavaSearchScope(int size) {
initialize(size);
//disabled for now as this could be expensive
//JavaModelManager.getJavaModelManager().rememberScope(this);
}
private void addEnclosingProjectOrJar(IPath path) {
int length = this.enclosingProjectsAndJars.length;
for (int i = 0; i < length; i++) {
if (this.enclosingProjectsAndJars[i].equals(path)) return;
}
System.arraycopy(
this.enclosingProjectsAndJars,
0,
this.enclosingProjectsAndJars = new IPath[length+1],
0,
length);
this.enclosingProjectsAndJars[length] = path;
}
/**
* Add java project all fragment roots to current java search scope.
* @see #add(JavaProject, IPath, int, HashSet, IClasspathEntry)
*/
public void add(JavaProject project, int includeMask, HashSet visitedProject) throws JavaModelException {
add(project, null, includeMask, visitedProject, null);
}
/**
* Add a path to current java search scope or all project fragment roots if null.
* Use project resolved classpath to retrieve and store access restriction on each classpath entry.
* Recurse if dependent projects are found.
* @param javaProject Project used to get resolved classpath entries
* @param pathToAdd Path to add in case of single element or null if user want to add all project package fragment roots
* @param includeMask Mask to apply on classpath entries
* @param visitedProjects Set to avoid infinite recursion
* @param referringEntry Project raw entry in referring project classpath
* @throws JavaModelException May happen while getting java model info
*/
void add(JavaProject javaProject, IPath pathToAdd, int includeMask, HashSet visitedProjects, IClasspathEntry referringEntry) throws JavaModelException {
IProject project = javaProject.getProject();
if (!project.isAccessible() || !visitedProjects.add(project)) return;
IPath projectPath = project.getFullPath();
String projectPathString = projectPath.toString();
this.addEnclosingProjectOrJar(projectPath);
IClasspathEntry[] entries = javaProject.getResolvedClasspath(true/*ignoreUnresolvedEntry*/, false/*don't generateMarkerOnError*/, false/*don't returnResolutionInProgress*/);
IJavaModel model = javaProject.getJavaModel();
JavaModelManager.PerProjectInfo perProjectInfo = javaProject.getPerProjectInfo();
for (int i = 0, length = entries.length; i < length; i++) {
IClasspathEntry entry = entries[i];
AccessRuleSet access = null;
ClasspathEntry cpEntry = (ClasspathEntry) entry;
if (referringEntry != null) {
// Add only exported entries.
// Source folder are implicitly exported.
if (!entry.isExported() && entry.getEntryKind() != IClasspathEntry.CPE_SOURCE) continue;
cpEntry = cpEntry.combineWith((ClasspathEntry)referringEntry);
// cpEntry = ((ClasspathEntry)referringEntry).combineWith(cpEntry);
}
access = cpEntry.getAccessRuleSet();
switch (entry.getEntryKind()) {
case IClasspathEntry.CPE_LIBRARY:
IClasspathEntry rawEntry = null;
if (perProjectInfo != null && perProjectInfo.resolvedPathToRawEntries != null) {
rawEntry = (IClasspathEntry) perProjectInfo.resolvedPathToRawEntries.get(entry.getPath());
}
if (rawEntry == null) break;
switch (rawEntry.getEntryKind()) {
case IClasspathEntry.CPE_LIBRARY:
case IClasspathEntry.CPE_VARIABLE:
if ((includeMask & APPLICATION_LIBRARIES) != 0) {
IPath path = entry.getPath();
if (pathToAdd == null || pathToAdd.equals(path)) {
String pathToString = path.getDevice() == null ? path.toString() : path.toOSString();
add("", pathToString, true, access); //$NON-NLS-1$
addEnclosingProjectOrJar(path);
}
}
break;
case IClasspathEntry.CPE_CONTAINER:
IClasspathContainer container = JavaCore.getClasspathContainer(rawEntry.getPath(), javaProject);
if (container == null) break;
if ((container.getKind() == IClasspathContainer.K_APPLICATION && (includeMask & APPLICATION_LIBRARIES) != 0)
|| (includeMask & SYSTEM_LIBRARIES) != 0) {
IPath path = entry.getPath();
if (pathToAdd == null || pathToAdd.equals(path)) {
String pathToString = path.getDevice() == null ? path.toString() : path.toOSString();
add("", pathToString, true, access); //$NON-NLS-1$
addEnclosingProjectOrJar(path);
}
}
break;
}
break;
case IClasspathEntry.CPE_PROJECT:
if ((includeMask & REFERENCED_PROJECTS) != 0) {
IPath path = entry.getPath();
if (pathToAdd == null || pathToAdd.equals(path)) {
add((JavaProject) model.getJavaProject(entry.getPath().lastSegment()), null, includeMask, visitedProjects, cpEntry);
}
}
break;
case IClasspathEntry.CPE_SOURCE:
if ((includeMask & SOURCES) != 0) {
IPath path = entry.getPath();
if (pathToAdd == null || pathToAdd.equals(path)) {
add(Util.relativePath(path,1/*remove project segment*/), projectPathString, true, access);
}
}
break;
}
}
}
/**
* Add an element to the java search scope.
* @param element The element we want to add to current java search scope
* @throws JavaModelException May happen if some Java Model info are not available
*/
public void add(IJavaElement element) throws JavaModelException {
IPath containerPath = null;
String containerPathToString = null;
int includeMask = SOURCES | APPLICATION_LIBRARIES | SYSTEM_LIBRARIES;
switch (element.getElementType()) {
case IJavaElement.JAVA_MODEL:
// a workspace sope should be used
break;
case IJavaElement.JAVA_PROJECT:
add((JavaProject)element, null, includeMask, new HashSet(2), null);
break;
case IJavaElement.PACKAGE_FRAGMENT_ROOT:
IPackageFragmentRoot root = (IPackageFragmentRoot)element;
IPath rootPath = root.getPath();
containerPath = root.getKind() == IPackageFragmentRoot.K_SOURCE ? root.getParent().getPath() : rootPath;
containerPathToString = containerPath.getDevice() == null ? containerPath.toString() : containerPath.toOSString();
IResource rootResource = root.getResource();
if (rootResource != null && rootResource.isAccessible()) {
String relativePath = Util.relativePath(rootResource.getFullPath(), containerPath.segmentCount());
add(relativePath, containerPathToString, true, null);
} else {
add("", containerPathToString, true, null); //$NON-NLS-1$
}
break;
case IJavaElement.PACKAGE_FRAGMENT:
root = (IPackageFragmentRoot)element.getParent();
if (root.isArchive()) {
String relativePath = Util.concatWith(((PackageFragment) element).names, '/');
containerPath = root.getPath();
containerPathToString = containerPath.getDevice() == null ? containerPath.toString() : containerPath.toOSString();
add(relativePath, containerPathToString, false, null);
} else {
IResource resource = element.getResource();
if (resource != null && resource.isAccessible()) {
containerPath = root.getKind() == IPackageFragmentRoot.K_SOURCE ? root.getParent().getPath() : root.getPath();
containerPathToString = containerPath.getDevice() == null ? containerPath.toString() : containerPath.toOSString();
String relativePath = Util.relativePath(resource.getFullPath(), containerPath.segmentCount());
add(relativePath, containerPathToString, false, null);
}
}
break;
default:
// remember sub-cu (or sub-class file) java elements
if (element instanceof IMember) {
if (this.elements == null) {
this.elements = new ArrayList();
}
this.elements.add(element);
}
root = (IPackageFragmentRoot) element.getAncestor(IJavaElement.PACKAGE_FRAGMENT_ROOT);
String relativePath;
if (root.getKind() == IPackageFragmentRoot.K_SOURCE) {
containerPath = root.getParent().getPath();
relativePath = Util.relativePath(getPath(element, false/*full path*/), 1/*remove project segmet*/);
} else {
containerPath = root.getPath();
relativePath = getPath(element, true/*relative path*/).toString();
}
containerPathToString = containerPath.getDevice() == null ? containerPath.toString() : containerPath.toOSString();
add(relativePath, containerPathToString, true, null);
}
if (containerPath != null)
addEnclosingProjectOrJar(containerPath);
}
/**
* Adds the given path to this search scope. Remember if subfolders need to be included
* and associated access restriction as well.
*/
private void add(String relativePath, String containerPath, boolean withSubFolders, AccessRuleSet access) {
int index = (containerPath.hashCode() & 0x7FFFFFFF) % this.paths.length;
String currentPath, currentContainerPath;
while ((currentPath = this.paths[index]) != null && (currentContainerPath = this.containerPaths[index]) != null) {
if (currentPath.equals(relativePath) && currentContainerPath.equals(containerPath))
return;
index = (index + 1) % this.paths.length;
}
this.paths[index] = relativePath;
this.containerPaths[index] = containerPath;
this.pathWithSubFolders[index] = withSubFolders;
if (this.pathRestrictions != null)
this.pathRestrictions[index] = access;
else if (access != null) {
this.pathRestrictions = new AccessRuleSet[this.paths.length];
this.pathRestrictions[index] = access;
}
// assumes the threshold is never equal to the size of the table
if (++this.pathsCount > this.threshold)
rehash();
}
/* (non-Javadoc)
* @see IJavaSearchScope#encloses(String)
*/
public boolean encloses(String resourcePathString) {
int separatorIndex = resourcePathString.indexOf(JAR_FILE_ENTRY_SEPARATOR);
if (separatorIndex != -1) {
return indexOf(resourcePathString.substring(separatorIndex+1), resourcePathString.substring(0, separatorIndex)) >= 0;
}
return indexOf(resourcePathString, null) >= 0;
}
/**
* Returns paths list index of given path or -1 if not found.
*/
private int indexOf(String relativePath, String containerPath) {
if (containerPath != null) {
// if container path is known, use the hash to get faster comparison
int index = (containerPath.hashCode()& 0x7FFFFFFF) % this.paths.length;
String currentContainerPath;
while ((currentContainerPath = this.containerPaths[index]) != null) {
if (currentContainerPath.equals(containerPath)) {
String scopePath = this.paths[index];
if (encloses(scopePath, relativePath, index))
return index;
}
index = (index + 1) % this.paths.length;
}
return -1;
}
// fallback to sequentially looking at all known paths
for (int i = 0, length = this.paths.length; i < length; i++) {
String scopePath = this.paths[i];
if (scopePath == null) continue;
if (encloses(this.containerPaths[i] + '/' + scopePath, relativePath, i))
return i;
}
return -1;
}
private boolean encloses(String scopePath, String path, int index) {
if (this.pathWithSubFolders[index]) {
// TODO (frederic) apply similar change also if not looking at subfolders
int pathLength = path.length();
int scopeLength = scopePath.length();
if (pathLength < scopeLength) {
return false;
}
if (scopeLength == 0) {
return true;
}
if (pathLength == scopeLength) {
return path.equals(scopePath);
}
if (path.startsWith(scopePath)) {
if (scopePath.charAt(scopeLength-1) == '/') scopeLength--;
return path.charAt(scopeLength) == '/';
}
} else {
// if not looking at subfolders, this scope encloses the given path
// if this path is a direct child of the scope's ressource
// or if this path is the scope's resource (see bug 13919 Declaration for package not found if scope is not project)
if (path.startsWith(scopePath)
&& ((scopePath.length() == path.lastIndexOf('/'))
|| (scopePath.length() == path.length()))) {
return true;
}
}
return false;
}
/* (non-Javadoc)
* @see IJavaSearchScope#encloses(IJavaElement)
*/
public boolean encloses(IJavaElement element) {
if (this.elements != null) {
for (int i = 0, length = this.elements.size(); i < length; i++) {
IJavaElement scopeElement = (IJavaElement)this.elements.get(i);
IJavaElement searchedElement = element;
while (searchedElement != null) {
if (searchedElement.equals(scopeElement))
return true;
searchedElement = searchedElement.getParent();
}
}
return false;
}
IPackageFragmentRoot root = (IPackageFragmentRoot) element.getAncestor(IJavaElement.PACKAGE_FRAGMENT_ROOT);
if (root != null && root.isArchive()) {
IPath rootPath = root.getPath();
String rootPathToString = rootPath.getDevice() == null ? rootPath.toString() : rootPath.toOSString();
IPath relativePath = getPath(element, true/*relative path*/);
return indexOf(relativePath.toString(), rootPathToString) >= 0;
}
return this.indexOf(getPath(element, false/*full path*/).toString(), null) >= 0;
}
/* (non-Javadoc)
* @see IJavaSearchScope#enclosingProjectsAndJars()
*/
public IPath[] enclosingProjectsAndJars() {
return this.enclosingProjectsAndJars;
}
private IPath getPath(IJavaElement element, boolean relativeToRoot) {
if (element instanceof IPackageFragmentRoot) {
if (relativeToRoot)
return Path.EMPTY;
return ((IPackageFragmentRoot)element).getPath();
}
IJavaElement parent = element.getParent();
IPath parentPath = parent == null ? null : getPath(parent, relativeToRoot);
IPath childPath;
if (element instanceof PackageFragment) {
String relativePath = Util.concatWith(((PackageFragment) element).names, '/');
childPath = new Path(relativePath);
} else if (element instanceof IOpenable) {
childPath = new Path(element.getElementName());
} else {
return parentPath;
}
return parentPath == null ? childPath : parentPath.append(childPath);
}
/**
* Get access rule set corresponding to a given path.
* @param relativePath The path user want to have restriction access
* @return The access rule set for given path or null if none is set for it.
* Returns specific uninit access rule set when scope does not enclose the given path.
*/
public AccessRuleSet getAccessRuleSet(String relativePath, String containerPath) {
int index = indexOf(relativePath, containerPath);
if (index == -1) {
// this search scope does not enclose given path
return NOT_ENCLOSED;
}
if (this.pathRestrictions == null)
return null;
return this.pathRestrictions[index];
}
protected void initialize(int size) {
this.pathsCount = 0;
this.threshold = size; // size represents the expected number of elements
int extraRoom = (int) (size * 1.75f);
if (this.threshold == extraRoom)
extraRoom++;
this.paths = new String[extraRoom];
this.containerPaths = new String[extraRoom];
this.pathWithSubFolders = new boolean[extraRoom];
this.pathRestrictions = null; // null to optimize case where no access rules are used
this.enclosingProjectsAndJars = new IPath[0];
}
/*
* @see AbstractSearchScope#processDelta(IJavaElementDelta)
*/
public void processDelta(IJavaElementDelta delta) {
switch (delta.getKind()) {
case IJavaElementDelta.CHANGED:
IJavaElementDelta[] children = delta.getAffectedChildren();
for (int i = 0, length = children.length; i < length; i++) {
IJavaElementDelta child = children[i];
this.processDelta(child);
}
break;
case IJavaElementDelta.REMOVED:
IJavaElement element = delta.getElement();
if (this.encloses(element)) {
if (this.elements != null) {
this.elements.remove(element);
}
IPath path = null;
switch (element.getElementType()) {
case IJavaElement.JAVA_PROJECT:
path = ((IJavaProject)element).getProject().getFullPath();
case IJavaElement.PACKAGE_FRAGMENT_ROOT:
if (path == null) {
path = ((IPackageFragmentRoot)element).getPath();
}
int toRemove = -1;
for (int i = 0; i < this.pathsCount; i++) {
if (this.paths[i].equals(path)) {
toRemove = i;
break;
}
}
if (toRemove != -1) {
this.paths[toRemove] = null;
rehash();
}
}
}
break;
}
}
private void rehash() {
JavaSearchScope newScope = new JavaSearchScope(this.pathsCount * 2); // double the number of expected elements
String currentPath;
for (int i = this.paths.length; --i >= 0;)
if ((currentPath = this.paths[i]) != null)
newScope.add(currentPath, this.containerPaths[i], this.pathWithSubFolders[i], this.pathRestrictions == null ? null : this.pathRestrictions[i]);
this.paths = newScope.paths;
this.containerPaths = newScope.containerPaths;
this.pathWithSubFolders = newScope.pathWithSubFolders;
this.pathRestrictions = newScope.pathRestrictions;
this.threshold = newScope.threshold;
}
public String toString() {
StringBuffer result = new StringBuffer("JavaSearchScope on "); //$NON-NLS-1$
if (this.elements != null) {
result.append("["); //$NON-NLS-1$
for (int i = 0, length = this.elements.size(); i < length; i++) {
JavaElement element = (JavaElement)this.elements.get(i);
result.append("\n\t"); //$NON-NLS-1$
result.append(element.toStringWithAncestors());
}
result.append("\n]"); //$NON-NLS-1$
} else {
if (this.pathsCount == 0) {
result.append("[empty scope]"); //$NON-NLS-1$
} else {
result.append("["); //$NON-NLS-1$
for (int i = 0; i < this.paths.length; i++) {
String path = this.paths[i];
if (path == null) continue;
result.append("\n\t"); //$NON-NLS-1$
result.append(this.containerPaths[i]);
if (path.length() > 0) {
result.append('/');
result.append(path);
}
}
result.append("\n]"); //$NON-NLS-1$
}
}
return result.toString();
}
}