| /* |
| * Copyright (c) 2006, 2012, 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. Oracle designates this |
| * particular file as subject to the "Classpath" exception as provided |
| * by Oracle in the LICENSE file that accompanied this code. |
| * |
| * 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 8003967 |
| * @summary detect and remove all mutable implicit static enum fields in langtools |
| * @run main DetectMutableStaticFields |
| */ |
| |
| import java.io.File; |
| import java.io.IOException; |
| import java.net.URI; |
| import java.net.URISyntaxException; |
| import java.util.ArrayList; |
| import java.util.Arrays; |
| import java.util.EnumSet; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| import javax.tools.JavaCompiler; |
| import javax.tools.JavaFileManager; |
| import javax.tools.JavaFileObject; |
| import javax.tools.StandardJavaFileManager; |
| import javax.tools.StandardLocation; |
| import javax.tools.ToolProvider; |
| import com.sun.tools.classfile.ClassFile; |
| import com.sun.tools.classfile.ConstantPoolException; |
| import com.sun.tools.classfile.Descriptor; |
| import com.sun.tools.classfile.Descriptor.InvalidDescriptor; |
| import com.sun.tools.classfile.Field; |
| |
| import static javax.tools.JavaFileObject.Kind.CLASS; |
| import static com.sun.tools.classfile.AccessFlags.ACC_ENUM; |
| import static com.sun.tools.classfile.AccessFlags.ACC_FINAL; |
| import static com.sun.tools.classfile.AccessFlags.ACC_STATIC; |
| |
| public class DetectMutableStaticFields { |
| |
| private static final String keyResource = |
| "com/sun/tools/javac/tree/JCTree.class"; |
| |
| private String[] packagesToSeekFor = new String[] { |
| "javax.tools", |
| "javax.lang.model", |
| "com.sun.javadoc", |
| "com.sun.source", |
| "com.sun.tools.classfile", |
| "com.sun.tools.doclets", |
| "com.sun.tools.javac", |
| "com.sun.tools.javadoc", |
| "com.sun.tools.javah", |
| "com.sun.tools.javap", |
| }; |
| |
| private static final Map<String, List<String>> classFieldsToIgnoreMap = new HashMap<>(); |
| |
| static { |
| classFieldsToIgnoreMap. |
| put("javax/tools/ToolProvider", |
| Arrays.asList("instance")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javah/JavahTask", |
| Arrays.asList("versionRB")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/classfile/Dependencies$DefaultFilter", |
| Arrays.asList("instance")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javap/JavapTask", |
| Arrays.asList("versionRB")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/doclets/formats/html/HtmlDoclet", |
| Arrays.asList("docletToStart")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/util/JCDiagnostic", |
| Arrays.asList("fragmentFormatter")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/util/JavacMessages", |
| Arrays.asList("defaultBundle", "defaultMessages")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/file/ZipFileIndexCache", |
| Arrays.asList("sharedInstance")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/main/JavaCompiler", |
| Arrays.asList("versionRB")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/code/Type", |
| Arrays.asList("moreInfo")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/util/SharedNameTable", |
| Arrays.asList("freelist")); |
| classFieldsToIgnoreMap. |
| put("com/sun/tools/javac/util/Log", |
| Arrays.asList("useRawMessages")); |
| } |
| |
| private List<String> errors = new ArrayList<>(); |
| |
| public static void main(String[] args) { |
| try { |
| new DetectMutableStaticFields().run(); |
| } catch (Exception ex) { |
| throw new AssertionError( |
| "Exception during test execution with cause ", |
| ex.getCause()); |
| } |
| } |
| |
| private void run() |
| throws |
| IOException, |
| ConstantPoolException, |
| InvalidDescriptor, |
| URISyntaxException { |
| |
| URI resource = findResource(keyResource); |
| if (resource == null) { |
| throw new AssertionError("Resource " + keyResource + |
| "not found in the class path"); |
| } |
| analyzeResource(resource); |
| |
| if (errors.size() > 0) { |
| for (String error: errors) { |
| System.err.println(error); |
| } |
| throw new AssertionError("There are mutable fields, " |
| + "please check output"); |
| } |
| } |
| |
| URI findResource(String className) throws URISyntaxException { |
| URI uri = getClass().getClassLoader().getResource(className).toURI(); |
| if (uri.getScheme().equals("jar")) { |
| String ssp = uri.getRawSchemeSpecificPart(); |
| int sep = ssp.lastIndexOf("!"); |
| uri = new URI(ssp.substring(0, sep)); |
| } else if (uri.getScheme().equals("file")) { |
| uri = new URI(uri.getPath().substring(0, |
| uri.getPath().length() - keyResource.length())); |
| } |
| return uri; |
| } |
| |
| boolean shouldAnalyzePackage(String packageName) { |
| for (String aPackage: packagesToSeekFor) { |
| if (packageName.contains(aPackage)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| void analyzeResource(URI resource) |
| throws |
| IOException, |
| ConstantPoolException, |
| InvalidDescriptor { |
| JavaCompiler tool = ToolProvider.getSystemJavaCompiler(); |
| StandardJavaFileManager fm = tool.getStandardFileManager(null, null, null); |
| JavaFileManager.Location location = |
| StandardLocation.locationFor(resource.getPath()); |
| fm.setLocation(location, com.sun.tools.javac.util.List.of( |
| new File(resource.getPath()))); |
| |
| for (JavaFileObject file : fm.list(location, "", EnumSet.of(CLASS), true)) { |
| String className = fm.inferBinaryName(location, file); |
| int index = className.lastIndexOf('.'); |
| String pckName = index == -1 ? "" : className.substring(0, index); |
| if (shouldAnalyzePackage(pckName)) { |
| analyzeClassFile(ClassFile.read(file.openInputStream())); |
| } |
| } |
| } |
| |
| List<String> currentFieldsToIgnore; |
| |
| boolean ignoreField(String field) { |
| if (currentFieldsToIgnore != null) { |
| for (String fieldToIgnore : currentFieldsToIgnore) { |
| if (field.equals(fieldToIgnore)) { |
| return true; |
| } |
| } |
| } |
| return false; |
| } |
| |
| void analyzeClassFile(ClassFile classFileToCheck) |
| throws |
| IOException, |
| ConstantPoolException, |
| Descriptor.InvalidDescriptor { |
| boolean enumClass = |
| (classFileToCheck.access_flags.flags & ACC_ENUM) != 0; |
| boolean nonFinalStaticEnumField; |
| boolean nonFinalStaticField; |
| |
| currentFieldsToIgnore = |
| classFieldsToIgnoreMap.get(classFileToCheck.getName()); |
| |
| for (Field field : classFileToCheck.fields) { |
| if (ignoreField(field.getName(classFileToCheck.constant_pool))) { |
| continue; |
| } |
| nonFinalStaticEnumField = |
| (field.access_flags.flags & (ACC_ENUM | ACC_FINAL)) == 0; |
| nonFinalStaticField = |
| (field.access_flags.flags & ACC_STATIC) != 0 && |
| (field.access_flags.flags & ACC_FINAL) == 0; |
| if (enumClass ? nonFinalStaticEnumField : nonFinalStaticField) { |
| errors.add("There is a mutable field named " + |
| field.getName(classFileToCheck.constant_pool) + |
| ", at class " + |
| classFileToCheck.getName()); |
| } |
| } |
| } |
| |
| } |