/* * Copyright (c) 2005, 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. */ package build.tools.buildmetaindex; import java.io.*; import java.util.*; import java.util.jar.*; /** Constructs a meta-index of the specified jar files. The meta-index contains prefixes of packages contained in these jars, indexed by the jar file name. It is intended to be consumed by the JVM to allow the boot class loader to be made lazier. For example, when class data sharing is enabled, the presence of the meta-index allows the JVM to skip opening rt.jar if all of the dependent classes of the application are in the shared archive. A similar mechanism could be useful at the application level as well, for example to make the extension class loader lazier. <p> The contents of the meta-index file for jre/lib look something like this: <PRE> % VERSION 2 # charsets.jar sun/ # jce.jar javax/ ! jsse.jar sun/ com/sun/net/ javax/ com/sun/security/ # management-agent.jar ! rt.jar org/w3c/ com/sun/image/ com/sun/org/ com/sun/imageio/ com/sun/accessibility/ javax/ ... </PRE> <p> It is a current invariant of the code in the JVM which consumes the meta-index that the meta-index indexes only jars in one directory. It is acceptable for jars in that directory to not be mentioned in the meta-index. The meta-index is designed more to be able to perform a quick rejection test of the presence of a particular class in a particular jar file than to be a precise index of the contents of the jar. */ public class BuildMetaIndex { public static void main(String[] args) throws IOException { /* The correct usage of this class is as following: * java BuildMetaIndex -o <meta-index> <a list of jar files> * So the argument length should be at least 3 and the first argument should * be '-o'. */ if (args.length < 3 || !args[0].equals("-o")) { printUsage(); System.exit(1); } try { PrintStream out = new PrintStream(new FileOutputStream(args[1])); out.println("% VERSION 2"); out.println("% WARNING: this file is auto-generated; do not edit"); out.println("% UNSUPPORTED: this file and its format may change and/or"); out.println("% may be removed in a future release"); for (int i = 2; i < args.length; i++) { String filename = args[i]; JarMetaIndex jmi = new JarMetaIndex(filename); HashSet<String> index = jmi.getMetaIndex(); if (index == null) { continue; } /* * meta-index file plays different role in JVM and JDK side. * On the JVM side, meta-index file is used to speed up locating the * class files only while on the JDK side, meta-index file is used to speed * up the resources file and class file. * To help the JVM and JDK code to better utilize the information in meta-index * file, we mark the jar file differently. Here is the current rule we use (See * JarFileKind.getMarkChar() method. ) * For jar file containing only class file, we put '!' before the jar file name; * for jar file containing only resources file, we put '@' before the jar file name; * for jar file containing both resources and class file, we put '#' before the jar name. * Notice the fact that every jar file contains at least the manifest file, so when * we say "jar file containing only class file", we don't include that file. */ out.println(jmi.getJarFileKind().getMarkerChar() + " " + filename); for (Iterator<String> iter = index.iterator(); iter.hasNext(); ) { out.println(iter.next()); } } out.flush(); out.close(); } catch (FileNotFoundException fnfe) { System.err.println("FileNotFoundException occurred"); System.exit(2); } } private static void printUsage() { String usage = "BuildMetaIndex is used to generate a meta index file for the jar files\n" + "you specified. The following is its usage:\n" + " java BuildMetaIndex -o <the output meta index file> <a list of jar files> \n" + " You can specify *.jar to refer to all the jar files in the current directory"; System.err.println(usage); } } enum JarFileKind { CLASSONLY ('!'), RESOURCEONLY ('@'), MIXED ('#'); private char markerChar; JarFileKind(char markerChar) { this.markerChar = markerChar; } public char getMarkerChar() { return markerChar; } } /* * JarMetaIndex associates the jar file with a set of what so called * "meta-index" of the jar file. Essentially, the meta-index is a list * of class prefixes and the plain files contained in META-INF directory ( * not include the manifest file itself). This will help sun.misc.URLClassPath * to quickly locate the resource file and hotspot VM to locate the class file. * */ class JarMetaIndex { private JarFile jar; private volatile HashSet<String> indexSet; /* * A hashmap contains a mapping from the prefix string to * a hashset which contains a set of the second level of prefix string. */ private HashMap<String, HashSet<String>> knownPrefixMap = new HashMap<String, HashSet<String>>(); /* * We add maximum 5 second level entries to "sun", "java" and * "javax" entries. Tune this parameter to get a balance on the * cold start and footprint. */ private static final int MAX_PKGS_WITH_KNOWN_PREFIX = 5; private JarFileKind jarFileKind; JarMetaIndex(String fileName) throws IOException { jar = new JarFile(fileName); knownPrefixMap.put("sun", new HashSet<String>()); knownPrefixMap.put("java", new HashSet<String>()); knownPrefixMap.put("javax", new HashSet<String>()); } /* Returns a HashSet contains the meta index string. */ HashSet<String> getMetaIndex() { if (indexSet == null) { synchronized(this) { if (indexSet == null) { indexSet = new HashSet<String>(); Enumeration entries = jar.entries(); boolean containsOnlyClass = true; boolean containsOnlyResource = true; while (entries.hasMoreElements()) { JarEntry entry = (JarEntry) entries.nextElement(); String name = entry.getName(); /* We only look at the non-directory entry. MANIFEST file is also skipped. */ if (entry.isDirectory() || name.equals("META-INF/MANIFEST.MF")) { continue; } /* Once containsOnlyResource or containsOnlyClass turns to false, no need to check the entry type. */ if (containsOnlyResource || containsOnlyClass) { if (name.endsWith(".class")) { containsOnlyResource = false; } else { containsOnlyClass = false; } } /* Add the full-qualified name of plain files under META-INF directory to the indexSet. */ if (name.startsWith("META-INF")) { indexSet.add(name); continue; } /* Add the prefix name to the knownPrefixMap if the name starts with any string in the knownPrefix list. */ if (isPrefixKnown(name)) { continue; } String[] pkgElements = name.split("/"); // Last one is the class name; definitely ignoring that if (pkgElements.length > 2) { String meta = null; // Need more information than just first two package // name elements to determine that classes in // deploy.jar are not in rt.jar if (pkgElements.length > 3 && pkgElements[0].equals("com") && pkgElements[1].equals("sun")) { // Need more precise information to disambiguate // (illegal) references from applications to // obsolete backported collections classes in // com/sun/java/util if (pkgElements.length > 4 && pkgElements[2].equals("java")) { int bound = 0; if (pkgElements[3].equals("util")) { // Take all of the packages bound = pkgElements.length - 1; } else { // Trim it somewhat more bound = 4; } meta = ""; for (int j = 0; j < bound; j++) { meta += pkgElements[j] + "/"; } } else { meta = pkgElements[0] + "/" + pkgElements[1] + "/" + pkgElements[2] + "/"; } } else { meta = pkgElements[0] + "/" + pkgElements[1] + "/"; } indexSet.add(meta); } } // end of "while" loop; // Add the second level package names to the indexSet for // the predefined names such as "sun", "java" and "javax". addKnownPrefix(); /* Set "jarFileKind" attribute. */ if (containsOnlyClass) { jarFileKind = JarFileKind.CLASSONLY; } else if (containsOnlyResource) { jarFileKind = JarFileKind.RESOURCEONLY; } else { jarFileKind = JarFileKind.MIXED; } } } } return indexSet; } /* * Checks to see whether the name starts with a string which is in the predefined * list. If it is among one of the predefined prefixes, add it to the knowPrefixMap * and returns true, otherwise, returns false. * Returns true if the name is in a predefined prefix list. Otherwise, returns false. */ boolean isPrefixKnown(String name) { int firstSlashIndex = name.indexOf("/"); if (firstSlashIndex == -1) { return false; } String firstPkgElement = name.substring(0, firstSlashIndex); HashSet<String> pkgSet = knownPrefixMap.get(firstPkgElement); /* The name does not starts with "sun", "java" or "javax". */ if (pkgSet == null) { return false; } String secondPkgElement = name.substring(firstSlashIndex + 1, name.indexOf("/", firstSlashIndex + 1)); /* Add the second level package name to the corresponding hashset. */ if (secondPkgElement != null) { pkgSet.add(secondPkgElement); } return true; } /* * Adds all the second level package elements for "sun", "java" and "javax" * if the corresponding jar file does not contain more than * MAX_PKGS_WITH_KNOWN_PREFIX such entries. */ void addKnownPrefix() { if (indexSet == null) { return; } /* Iterate through the hash map, add the second level package names * to the indexSet if has any. */ for (Iterator<String> keysIterator = knownPrefixMap.keySet().iterator(); keysIterator.hasNext();) { String key = keysIterator.next(); HashSet<String> pkgSetStartsWithKey = knownPrefixMap.get(key); int setSize = pkgSetStartsWithKey.size(); if (setSize == 0) { continue; } else if (setSize > JarMetaIndex.MAX_PKGS_WITH_KNOWN_PREFIX) { indexSet.add(key + "/"); } else { /* If the set contains less than MAX_PKGS_WITH_KNOWN_PREFIX, add * them to the indexSet of the MetaIndex object. */ for (Iterator<String> secondPkgElements = pkgSetStartsWithKey.iterator(); secondPkgElements.hasNext();) { indexSet.add(key + "/" + secondPkgElements.next()); } } } // end the outer "for" } JarFileKind getJarFileKind() { // Build meta index if it hasn't. if (indexSet == null) { indexSet = getMetaIndex(); } return jarFileKind; } }