/* * Copyright (c) 2014, 2016, 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 8042251 * @summary Test that inner classes have in its inner classes attribute enclosing classes and its immediate members. * @library /tools/lib /tools/javac/lib ../lib * @modules jdk.compiler/com.sun.tools.javac.api * jdk.compiler/com.sun.tools.javac.main * java.base/jdk.internal.classfile * java.base/jdk.internal.classfile.attribute * java.base/jdk.internal.classfile.constantpool * java.base/jdk.internal.classfile.instruction * java.base/jdk.internal.classfile.components * java.base/jdk.internal.classfile.impl * @build toolbox.ToolBox InMemoryFileManager TestResult TestBase * @run main InnerClassesHierarchyTest */ import java.io.File; import java.io.FilenameFilter; import java.io.IOException; import java.lang.annotation.Annotation; import java.util.*; import java.util.stream.Collectors; import jdk.internal.classfile.*; import jdk.internal.classfile.attribute.*; import jdk.internal.classfile.constantpool.*; public class InnerClassesHierarchyTest extends TestResult { private final Map> innerClasses; private final String outerClassName; public InnerClassesHierarchyTest() throws IOException, ConstantPoolException { innerClasses = new HashMap<>(); outerClassName = InnerClassesHierarchyTest.class.getSimpleName(); File classDir = getClassDir(); FilenameFilter filter = (dir, name) -> name.matches(outerClassName + ".*\\.class"); for (File file : Arrays.asList(classDir.listFiles(filter))) { ClassModel classFile = readClassFile(file); String className = classFile.thisClass().name().stringValue(); for (int i = 1; i < classFile.constantPool().entryCount(); ++i) { if (classFile.constantPool().entryByIndex(i) instanceof ClassEntry classInfo && classInfo.asSymbol().isClassOrInterface()) { String cpClassName = classInfo.asInternalName(); if (isInnerClass(cpClassName)) { get(className).add(cpClassName); } } } } } private boolean isInnerClass(String cpClassName) { return cpClassName.contains("$"); } private Set get(String className) { if (!innerClasses.containsKey(className)) { innerClasses.put(className, new HashSet<>()); } return innerClasses.get(className); } public static void main(String[] args) throws IOException, ConstantPoolException, TestFailedException { new InnerClassesHierarchyTest().test(); } private void test() throws TestFailedException { addTestCase("Source file is InnerClassesHierarchyTest.java"); try { Queue queue = new LinkedList<>(); Set visitedClasses = new HashSet<>(); queue.add(outerClassName); while (!queue.isEmpty()) { String currentClassName = queue.poll(); if (!currentClassName.startsWith(outerClassName)) { continue; } ClassModel cf = readClassFile(currentClassName); InnerClassesAttribute attr = cf.findAttribute(Attributes.INNER_CLASSES).orElse(null); checkNotNull(attr, "Class should not contain " + "inner classes attribute : " + currentClassName); checkTrue(innerClasses.containsKey(currentClassName), "map contains class name : " + currentClassName); Set setClasses = innerClasses.get(currentClassName); if (setClasses == null) { continue; } checkEquals(attr.classes().size(), setClasses.size(), "Check number of inner classes : " + setClasses); for (InnerClassInfo info : attr.classes()) { if (!info.innerClass().asSymbol().isClassOrInterface()) continue; String innerClassName = info .innerClass().asInternalName(); checkTrue(setClasses.contains(innerClassName), currentClassName + " contains inner class : " + innerClassName); if (visitedClasses.add(innerClassName)) { queue.add(innerClassName); } } } Set allClasses = innerClasses.entrySet().stream() .flatMap(entry -> entry.getValue().stream()) .collect(Collectors.toSet()); Set a_b = removeAll(visitedClasses, allClasses); Set b_a = removeAll(allClasses, visitedClasses); checkEquals(visitedClasses, allClasses, "All classes are found\n" + "visited - all classes : " + a_b + "\nall classes - visited : " + b_a); } catch (Exception e) { addFailure(e); } finally { checkStatus(); } } private Set removeAll(Set set1, Set set2) { Set set = new HashSet<>(set1); set.removeAll(set2); return set; } public static class A1 { public class B1 { } public enum B2 { } public interface B3 { } public @interface B4 { } public void f() { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } } Runnable r = () -> { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } }; } public enum A2 {; public class B1 { } public enum B2 { } public interface B3 { } public @interface B4 { } public void a2() { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } } Runnable r = () -> { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } }; } public interface A3 { public class B1 { } public enum B2 { } public interface B3 { } public @interface B4 { } default void a1() { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } } static void a2() { new B1() { }; new B3() { }; new B4() { @Override public Class annotationType() { return null; } }; class B5 { } } } public @interface A4 { public class B1 { } public enum B2 { } public interface B3 { } public @interface B4 { } } { new A1() { class B1 { } public void a2() { new B1() { }; class B5 { } } }; new A3() { class B1 { } public void a3() { new B1() { }; class B5 { } } }; new A4() { @Override public Class annotationType() { return null; } class B1 { } public void a4() { new B1() { }; class B5 { } } }; Runnable r = () -> { new A1() { }; new A3() { }; new A4() { @Override public Class annotationType() { return null; } }; class B5 { } }; } static { new A1() { class B1 { } public void a2() { new B1() { }; class B5 { } } }; new A3() { class B1 { } public void a3() { new B1() { }; class B5 { } } }; new A4() { @Override public Class annotationType() { return null; } class B1 { } public void a4() { new B1() { }; class B5 { } } }; Runnable r = () -> { new A1() { }; new A3() { }; new A4() { @Override public Class annotationType() { return null; } }; class B5 { } }; } public void a5() { class A5 { class B1 { } public void a5() { new B1() { }; class B5 { } } } Runnable r = () -> { new A1() { }; new A3() { }; new A4() { @Override public Class annotationType() { return null; } }; class B5 { } }; } }