/* * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /** * @test * @bug 7101822 * @summary Verify that the processing of classes in TypeEnter runs in the correct order. * @library /tools/lib * @build annotations.TriggersComplete annotations.TriggersCompleteRepeat annotations.Phase * @build DependenciesTest * @run main DependenciesTest */ import java.io.IOException; import java.net.URI; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.Objects; import java.util.Set; import java.util.Stack; import java.util.stream.Stream; import javax.lang.model.element.AnnotationMirror; import javax.lang.model.element.AnnotationValue; import javax.lang.model.element.Element; import javax.lang.model.element.ExecutableElement; import javax.lang.model.element.Name; import javax.lang.model.element.TypeElement; import javax.lang.model.type.DeclaredType; import javax.lang.model.type.TypeMirror; import javax.lang.model.util.Elements; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import annotations.*; import com.sun.source.tree.AnnotationTree; import com.sun.source.tree.ClassTree; import com.sun.source.tree.CompilationUnitTree; import com.sun.source.tree.ImportTree; import com.sun.source.tree.Tree; import com.sun.source.util.JavacTask; import com.sun.source.util.SourcePositions; import com.sun.source.util.TreePathScanner; import com.sun.source.util.Trees; import com.sun.tools.javac.api.JavacTool; import com.sun.tools.javac.api.JavacTrees; import com.sun.tools.javac.code.Symbol.ClassSymbol; import com.sun.tools.javac.file.JavacFileManager; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.util.Context; import com.sun.tools.javac.util.Context.Factory; import com.sun.tools.javac.util.Dependencies; public class DependenciesTest { public static void main(String... args) throws IOException { new DependenciesTest().run(); } void run() throws IOException { Path src = Paths.get(System.getProperty("test.src"), "tests"); try (Stream tests = Files.list(src)) { tests.map(p -> Files.isRegularFile(p) ? Stream.of(p) : silentWalk(p)) .forEach(this :: runTest); } } Stream silentWalk(Path src) { try { return Files.walk(src).filter(Files :: isRegularFile); } catch (IOException ex) { throw new IllegalStateException(ex); } } void runTest(Stream inputs) { JavacTool tool = JavacTool.create(); try (JavacFileManager fm = tool.getStandardFileManager(null, null, null)) { Path classes = Paths.get(System.getProperty("test.classes")); Iterable reconFiles = fm.getJavaFileObjectsFromFiles(inputs.sorted().map(p -> p.toFile()) :: iterator); List options = Arrays.asList("-classpath", classes.toAbsolutePath().toString()); JavacTask reconTask = tool.getTask(null, fm, null, options, null, reconFiles); Iterable reconUnits = reconTask.parse(); JavacTrees reconTrees = JavacTrees.instance(reconTask); SearchAnnotations scanner = new SearchAnnotations(reconTrees, reconTask.getElements()); List validateFiles = new ArrayList<>(); reconTask.analyze(); scanner.scan(reconUnits, null); for (CompilationUnitTree cut : reconUnits) { validateFiles.add(ClearAnnotations.clearAnnotations(reconTrees, cut)); } Context validateContext = new Context(); TestDependencies.preRegister(validateContext); JavacTask validateTask = tool.getTask(null, fm, null, options, null, validateFiles, validateContext); validateTask.analyze(); TestDependencies deps = (TestDependencies) Dependencies.instance(validateContext); if (!scanner.topLevel2Expected.equals(deps.topLevel2Completing)) { throw new IllegalStateException( "expected=" + scanner.topLevel2Expected + "; actual=" + deps.topLevel2Completing); } } catch (IOException ex) { throw new IllegalStateException(ex); } finally { inputs.close(); } } static final class TestDependencies extends Dependencies { public static void preRegister(Context context) { context.put(dependenciesKey, (Factory) TestDependencies :: new); } public TestDependencies(Context context) { super(context); } final Stack inProcess = new Stack<>(); String topLevelMemberEnter; Map> topLevel2Completing = new HashMap<>(); @Override public void push(ClassSymbol s, CompletionCause phase) { String flatname = s.flatName().toString(); for (Phase p : Phase.values()) { if (phase == p.cause) { inProcess.push(new PhaseDescription(flatname, p)); return ; } } if (phase == CompletionCause.MEMBER_ENTER) { if (inProcess.isEmpty()) { topLevelMemberEnter = flatname; } else { for (PhaseDescription running : inProcess) { if (running == null) continue; Set completing = topLevel2Completing.computeIfAbsent(running.flatname, $ -> new HashSet<>()); completing.add(new PhaseDescription(flatname, running.phase)); } } } inProcess.push(null); } @Override public void push(AttributionKind ak, JCTree t) { inProcess.push(null); } @Override public void pop() { inProcess.pop(); } } static final class SearchAnnotations extends TreePathScanner { final Trees trees; final Elements elements; final TypeElement triggersCompleteAnnotation; final TypeElement triggersCompleteRepeatAnnotation; final Map> topLevel2Expected = new HashMap<>(); public SearchAnnotations(Trees trees, Elements elements) { this.trees = trees; this.elements = elements; this.triggersCompleteAnnotation = elements.getTypeElement(TriggersComplete.class.getName()); this.triggersCompleteRepeatAnnotation = elements.getTypeElement(TriggersCompleteRepeat.class.getName()); } @Override public Void visitClass(ClassTree node, Void p) { TypeElement te = (TypeElement) trees.getElement(getCurrentPath()); Set expected = new HashSet<>(); for (AnnotationMirror am : getTriggersCompleteAnnotation(te)) { TypeMirror of = (TypeMirror) findAttribute(am, "of").getValue(); Name ofName = elements.getBinaryName((TypeElement) ((DeclaredType) of).asElement()); Element at = (Element) findAttribute(am, "at").getValue(); Phase phase = Phase.valueOf(at.getSimpleName().toString()); expected.add(new PhaseDescription(ofName.toString(), phase)); } if (!expected.isEmpty()) topLevel2Expected.put(elements.getBinaryName(te).toString(), expected); return super.visitClass(node, p); } Collection getTriggersCompleteAnnotation(TypeElement te) { for (AnnotationMirror am : te.getAnnotationMirrors()) { if (triggersCompleteAnnotation.equals(am.getAnnotationType().asElement())) { return Collections.singletonList(am); } if (triggersCompleteRepeatAnnotation.equals(am.getAnnotationType().asElement())) { return (Collection) findAttribute(am, "value").getValue(); } } return Collections.emptyList(); } AnnotationValue findAttribute(AnnotationMirror mirror, String name) { for (Entry e : mirror.getElementValues().entrySet()) { if (e.getKey().getSimpleName().contentEquals(name)) { return e.getValue(); } } throw new IllegalStateException("Could not find " + name + " in " + mirror); } } static final class ClearAnnotations extends TreePathScanner { final SourcePositions positions; final List spans2Clear = new ArrayList<>(); ClearAnnotations(Trees trees) { this.positions = trees.getSourcePositions(); } @Override public Void visitAnnotation(AnnotationTree node, Void p) { removeCurrentNode(); return null; } @Override public Void visitImport(ImportTree node, Void p) { if (node.getQualifiedIdentifier().toString().startsWith("annotations.")) { removeCurrentNode(); return null; } return super.visitImport(node, p); } void removeCurrentNode() { CompilationUnitTree topLevel = getCurrentPath().getCompilationUnit(); Tree node = getCurrentPath().getLeaf(); spans2Clear.add(new int[] {(int) positions.getStartPosition(topLevel, node), (int) positions.getEndPosition(topLevel, node)}); } static JavaFileObject clearAnnotations(Trees trees, CompilationUnitTree cut) throws IOException { ClearAnnotations a = new ClearAnnotations(trees); a.scan(cut, null); Collections.sort(a.spans2Clear, (s1, s2) -> s2[0] - s1[0]); StringBuilder result = new StringBuilder(cut.getSourceFile().getCharContent(true)); for (int[] toClear : a.spans2Clear) { result.delete(toClear[0], toClear[1]); } return new TestJavaFileObject(cut.getSourceFile().toUri(), result.toString()); } } static final class PhaseDescription { final String flatname; final Phase phase; public PhaseDescription(String flatname, Phase phase) { this.flatname = flatname; this.phase = phase; } @Override public String toString() { return "@annotations.TriggersComplete(of=" + flatname + ".class," + "at=annotations.Phase." + phase + ')'; } @Override public int hashCode() { int hash = 7; hash = 89 * hash + Objects.hashCode(this.flatname); hash = 89 * hash + Objects.hashCode(this.phase); return hash; } @Override public boolean equals(Object obj) { if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } final PhaseDescription other = (PhaseDescription) obj; if (!Objects.equals(this.flatname, other.flatname)) { return false; } if (this.phase != other.phase) { return false; } return true; } } static final class TestJavaFileObject extends SimpleJavaFileObject { private final String content; public TestJavaFileObject(URI uri, String content) { super(uri, Kind.SOURCE); this.content = content; } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException { return content; } } }