From 353c37c985f62c5bf9e184ad637ee6db4cc7083e Mon Sep 17 00:00:00 2001 From: Rob Stryker Date: Mon, 9 Dec 2024 15:27:27 -0500 Subject: [PATCH] Fixes #3428 - Ensure calls to ASTParser.createBindings use the correct unit resolver Signed-off-by: Rob Stryker --- .../org/eclipse/jdt/core/dom/ASTParser.java | 97 +++++++++++++++++-- .../jdt/core/dom/CompilationUnitResolver.java | 86 +++------------- 2 files changed, 101 insertions(+), 82 deletions(-) diff --git a/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/ASTParser.java b/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/ASTParser.java index a156cd00975..09fec8ade0b 100644 --- a/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/ASTParser.java +++ b/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/ASTParser.java @@ -27,6 +27,7 @@ import org.eclipse.jdt.core.*; import org.eclipse.jdt.core.compiler.CategorizedProblem; import org.eclipse.jdt.core.compiler.CharOperation; +import org.eclipse.jdt.core.dom.CompilationUnitResolver.IntArrayList; import org.eclipse.jdt.internal.compiler.ast.ConstructorDeclaration; import org.eclipse.jdt.internal.compiler.ast.ExplicitConstructorCall; import org.eclipse.jdt.internal.compiler.batch.FileSystem.Classpath; @@ -38,16 +39,13 @@ import org.eclipse.jdt.internal.compiler.parser.RecoveryScanner; import org.eclipse.jdt.internal.compiler.parser.RecoveryScannerData; import org.eclipse.jdt.internal.compiler.parser.Scanner; +import org.eclipse.jdt.internal.compiler.util.HashtableOfObjectToInt; import org.eclipse.jdt.internal.compiler.util.SuffixConstants; -import org.eclipse.jdt.internal.core.BasicCompilationUnit; -import org.eclipse.jdt.internal.core.BinaryType; -import org.eclipse.jdt.internal.core.ClassFileWorkingCopy; -import org.eclipse.jdt.internal.core.DefaultWorkingCopyOwner; -import org.eclipse.jdt.internal.core.JavaModelManager; -import org.eclipse.jdt.internal.core.PackageFragment; +import org.eclipse.jdt.internal.core.*; import org.eclipse.jdt.internal.core.dom.ICompilationUnitResolver; import org.eclipse.jdt.internal.core.dom.util.DOMASTUtil; import org.eclipse.jdt.internal.core.util.CodeSnippetParsingUtil; +import org.eclipse.jdt.internal.core.util.DOMFinder; import org.eclipse.jdt.internal.core.util.RecordedParsingInformation; import org.eclipse.jdt.internal.core.util.Util; @@ -1167,13 +1165,98 @@ public IBinding[] createBindings(IJavaElement[] elements, IProgressMonitor monit if ((this.bits & CompilationUnitResolver.IGNORE_METHOD_BODIES) != 0) { flags |= ICompilationUnit.IGNORE_METHOD_BODIES; } - return CompilationUnitResolver.resolve(elements, this.apiLevel, this.compilerOptions, this.project, this.workingCopyOwner, flags, monitor); + + return resolve(elements, this.apiLevel, this.compilerOptions, this.project, this.workingCopyOwner, flags, this.unitResolver, monitor); } finally { // reset to defaults to allow reuse (and avoid leaking) initializeDefaults(); } } + + static IBinding[] resolve( + final IJavaElement[] elements, + int apiLevel, + Map compilerOptions, + IJavaProject javaProject, + WorkingCopyOwner owner, + int flags, + ICompilationUnitResolver unitResolver, + IProgressMonitor monitor) { + + final int length = elements.length; + final HashMap sourceElementPositions = new HashMap(); // a map from ICompilationUnit to int[] (positions in elements) + int cuNumber = 0; + final HashtableOfObjectToInt binaryElementPositions = new HashtableOfObjectToInt(); // a map from String (binding key) to int (position in elements) + for (int i = 0; i < length; i++) { + IJavaElement element = elements[i]; + if (!(element instanceof SourceRefElement)) + throw new IllegalStateException(element + " is not part of a compilation unit or class file"); //$NON-NLS-1$ + Object cu = element.getAncestor(IJavaElement.COMPILATION_UNIT); + if (cu != null) { + // source member + IntArrayList intList = (IntArrayList) sourceElementPositions.get(cu); + if (intList == null) { + sourceElementPositions.put(cu, intList = new IntArrayList()); + cuNumber++; + } + intList.add(i); + } else { + // binary member or method argument + try { + String key; + if (element instanceof BinaryMember) + key = ((BinaryMember) element).getKey(true/*open to get resolved info*/); + else if (element instanceof LocalVariable) + key = ((LocalVariable) element).getKey(true/*open to get resolved info*/); + else if (element instanceof org.eclipse.jdt.internal.core.TypeParameter) + key = ((org.eclipse.jdt.internal.core.TypeParameter) element).getKey(true/*open to get resolved info*/); + else if (element instanceof BinaryModule) + key = ((BinaryModule) element).getKey(true); + else + throw new IllegalArgumentException(element + " has an unexpected type"); //$NON-NLS-1$ + binaryElementPositions.put(key, i); + } catch (JavaModelException e) { + throw new IllegalArgumentException(element + " does not exist", e); //$NON-NLS-1$ + } + } + } + ICompilationUnit[] cus = new ICompilationUnit[cuNumber]; + sourceElementPositions.keySet().toArray(cus); + + int bindingKeyNumber = binaryElementPositions.size(); + String[] bindingKeys = new String[bindingKeyNumber]; + binaryElementPositions.keysToArray(bindingKeys); + + class Requestor extends ASTRequestor { + IBinding[] bindings = new IBinding[length]; + @Override + public void acceptAST(ICompilationUnit source, CompilationUnit ast) { + // TODO (jerome) optimize to visit the AST only once + IntArrayList intList = (IntArrayList) sourceElementPositions.get(source); + for (int i = 0; i < intList.length; i++) { + final int index = intList.list[i]; + SourceRefElement element = (SourceRefElement) elements[index]; + DOMFinder finder = new DOMFinder(ast, element, true/*resolve binding*/); + try { + finder.search(); + } catch (JavaModelException e) { + throw new IllegalArgumentException(element + " does not exist", e); //$NON-NLS-1$ + } + this.bindings[index] = finder.foundBinding; + } + } + @Override + public void acceptBinding(String bindingKey, IBinding binding) { + int index = binaryElementPositions.get(bindingKey); + this.bindings[index] = binding; + } + } + Requestor requestor = new Requestor(); + unitResolver.resolve(cus, bindingKeys, requestor, apiLevel, compilerOptions, javaProject, owner, flags, monitor); + return requestor.bindings; + } + private ASTNode internalCreateAST(IProgressMonitor monitor) { return JavaModelManager.cacheZipFiles(() -> internalCreateASTCached(monitor)); } diff --git a/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/CompilationUnitResolver.java b/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/CompilationUnitResolver.java index 050b7ec0af8..27a51053bd8 100644 --- a/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/CompilationUnitResolver.java +++ b/org.eclipse.jdt.core/dom/org/eclipse/jdt/core/dom/CompilationUnitResolver.java @@ -19,7 +19,6 @@ import java.io.File; import java.io.IOException; import java.util.ArrayList; -import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; @@ -60,14 +59,18 @@ import org.eclipse.jdt.internal.compiler.problem.DefaultProblemFactory; import org.eclipse.jdt.internal.compiler.problem.ProblemReporter; import org.eclipse.jdt.internal.compiler.util.HashtableOfObject; -import org.eclipse.jdt.internal.compiler.util.HashtableOfObjectToInt; import org.eclipse.jdt.internal.compiler.util.Messages; import org.eclipse.jdt.internal.compiler.util.Util; -import org.eclipse.jdt.internal.core.*; +import org.eclipse.jdt.internal.core.CancelableNameEnvironment; +import org.eclipse.jdt.internal.core.CancelableProblemFactory; +import org.eclipse.jdt.internal.core.ClasspathEntry; +import org.eclipse.jdt.internal.core.INameEnvironmentWithProgress; +import org.eclipse.jdt.internal.core.JavaProject; +import org.eclipse.jdt.internal.core.NameLookup; +import org.eclipse.jdt.internal.core.SourceTypeElementInfo; import org.eclipse.jdt.internal.core.dom.ICompilationUnitResolver; import org.eclipse.jdt.internal.core.util.BindingKeyResolver; import org.eclipse.jdt.internal.core.util.CommentRecorderParser; -import org.eclipse.jdt.internal.core.util.DOMFinder; @SuppressWarnings({ "rawtypes", "unchecked" }) class CompilationUnitResolver extends Compiler { @@ -866,6 +869,7 @@ public static CompilationUnitDeclaration resolve( } } } + public static IBinding[] resolve( final IJavaElement[] elements, int apiLevel, @@ -874,79 +878,11 @@ public static IBinding[] resolve( WorkingCopyOwner owner, int flags, IProgressMonitor monitor) { + // Should not be called anymore? Candidate for deprecation + return ASTParser.resolve(elements, apiLevel, compilerOptions, javaProject, owner, flags, getInstance(), monitor); + } - final int length = elements.length; - final HashMap sourceElementPositions = new HashMap(); // a map from ICompilationUnit to int[] (positions in elements) - int cuNumber = 0; - final HashtableOfObjectToInt binaryElementPositions = new HashtableOfObjectToInt(); // a map from String (binding key) to int (position in elements) - for (int i = 0; i < length; i++) { - IJavaElement element = elements[i]; - if (!(element instanceof SourceRefElement)) - throw new IllegalStateException(element + " is not part of a compilation unit or class file"); //$NON-NLS-1$ - Object cu = element.getAncestor(IJavaElement.COMPILATION_UNIT); - if (cu != null) { - // source member - IntArrayList intList = (IntArrayList) sourceElementPositions.get(cu); - if (intList == null) { - sourceElementPositions.put(cu, intList = new IntArrayList()); - cuNumber++; - } - intList.add(i); - } else { - // binary member or method argument - try { - String key; - if (element instanceof BinaryMember) - key = ((BinaryMember) element).getKey(true/*open to get resolved info*/); - else if (element instanceof LocalVariable) - key = ((LocalVariable) element).getKey(true/*open to get resolved info*/); - else if (element instanceof org.eclipse.jdt.internal.core.TypeParameter) - key = ((org.eclipse.jdt.internal.core.TypeParameter) element).getKey(true/*open to get resolved info*/); - else if (element instanceof BinaryModule) - key = ((BinaryModule) element).getKey(true); - else - throw new IllegalArgumentException(element + " has an unexpected type"); //$NON-NLS-1$ - binaryElementPositions.put(key, i); - } catch (JavaModelException e) { - throw new IllegalArgumentException(element + " does not exist", e); //$NON-NLS-1$ - } - } - } - ICompilationUnit[] cus = new ICompilationUnit[cuNumber]; - sourceElementPositions.keySet().toArray(cus); - - int bindingKeyNumber = binaryElementPositions.size(); - String[] bindingKeys = new String[bindingKeyNumber]; - binaryElementPositions.keysToArray(bindingKeys); - class Requestor extends ASTRequestor { - IBinding[] bindings = new IBinding[length]; - @Override - public void acceptAST(ICompilationUnit source, CompilationUnit ast) { - // TODO (jerome) optimize to visit the AST only once - IntArrayList intList = (IntArrayList) sourceElementPositions.get(source); - for (int i = 0; i < intList.length; i++) { - final int index = intList.list[i]; - SourceRefElement element = (SourceRefElement) elements[index]; - DOMFinder finder = new DOMFinder(ast, element, true/*resolve binding*/); - try { - finder.search(); - } catch (JavaModelException e) { - throw new IllegalArgumentException(element + " does not exist", e); //$NON-NLS-1$ - } - this.bindings[index] = finder.foundBinding; - } - } - @Override - public void acceptBinding(String bindingKey, IBinding binding) { - int index = binaryElementPositions.get(bindingKey); - this.bindings[index] = binding; - } - } - Requestor requestor = new Requestor(); - resolve(cus, bindingKeys, requestor, apiLevel, compilerOptions, javaProject, owner, flags, monitor); - return requestor.bindings; - } /* * When unit result is about to be accepted, removed back pointers * to unresolved bindings