1 /* 2 * ProGuard -- shrinking, optimization, obfuscation, and preverification 3 * of Java bytecode. 4 * 5 * Copyright (c) 2002-2014 Eric Lafortune (eric (at) graphics.cornell.edu) 6 * 7 * This program is free software; you can redistribute it and/or modify it 8 * under the terms of the GNU General Public License as published by the Free 9 * Software Foundation; either version 2 of the License, or (at your option) 10 * any later version. 11 * 12 * This program is distributed in the hope that it will be useful, but WITHOUT 13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 15 * more details. 16 * 17 * You should have received a copy of the GNU General Public License along 18 * with this program; if not, write to the Free Software Foundation, Inc., 19 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 20 */ 21 package proguard; 22 23 import proguard.classfile.ClassPool; 24 import proguard.classfile.util.WarningPrinter; 25 import proguard.classfile.visitor.*; 26 import proguard.io.*; 27 28 import java.io.IOException; 29 30 /** 31 * This class reads the input class files. 32 * 33 * @author Eric Lafortune 34 */ 35 public class InputReader 36 { 37 private final Configuration configuration; 38 39 40 /** 41 * Creates a new InputReader to read input class files as specified by the 42 * given configuration. 43 */ 44 public InputReader(Configuration configuration) 45 { 46 this.configuration = configuration; 47 } 48 49 50 /** 51 * Fills the given program class pool and library class pool by reading 52 * class files, based on the current configuration. 53 */ 54 public void execute(ClassPool programClassPool, 55 ClassPool libraryClassPool) throws IOException 56 { 57 WarningPrinter warningPrinter = new WarningPrinter(System.err, configuration.warn); 58 WarningPrinter notePrinter = new WarningPrinter(System.out, configuration.note); 59 60 DuplicateClassPrinter duplicateClassPrinter = new DuplicateClassPrinter(notePrinter); 61 62 // Android-added: Read the classes from the systemjars. 63 // These are read into the library ClassPool before the programjars so that any duplicates 64 // between these are programjars will result in the classes in the programjars being 65 // ignored. 66 // Read the system class files, if any. 67 if (configuration.systemJars != null) 68 { 69 // Prepare a data entry reader to filter all classes, 70 // which are then decoded to classes by a class reader, 71 // which are then filtered by removing any that already exist in the libraryClassPool. 72 // which are then put in the class pool by a class pool filler. 73 readInput("Reading system ", 74 configuration.systemJars, 75 new ClassFilter( 76 new ClassReader(true /* isLibrary */, 77 configuration.skipNonPublicLibraryClasses, 78 configuration.skipNonPublicLibraryClassMembers, 79 warningPrinter, 80 new ClassPresenceFilter(libraryClassPool, duplicateClassPrinter, 81 new ClassPoolFiller(libraryClassPool))))); 82 } 83 84 // Read the program class files. 85 // Prepare a data entry reader to filter all classes, 86 // which are then decoded to classes by a class reader, 87 // which are then put in the class pool by a class pool filler. 88 readInput("Reading program ", 89 configuration.programJars, 90 new ClassFilter( 91 new ClassReader(false, 92 configuration.skipNonPublicLibraryClasses, 93 configuration.skipNonPublicLibraryClassMembers, 94 warningPrinter, 95 new ClassPresenceFilter(programClassPool, duplicateClassPrinter, 96 // Android-changed: Filter out classes already read from systemjars. 97 new ClassPresenceFilter(libraryClassPool, duplicateClassPrinter, 98 new ClassPoolFiller(programClassPool)))))); 99 100 // Check if we have at least some input classes. 101 if (programClassPool.size() == 0) 102 { 103 throw new IOException("The input doesn't contain any classes. Did you specify the proper '-injars' options?"); 104 } 105 106 // Read the library class files, if any. 107 if (configuration.libraryJars != null) 108 { 109 // Prepare a data entry reader to filter all classes, 110 // which are then decoded to classes by a class reader, 111 // which are then put in the class pool by a class pool filler. 112 readInput("Reading library ", 113 configuration.libraryJars, 114 new ClassFilter( 115 new ClassReader(true, 116 configuration.skipNonPublicLibraryClasses, 117 configuration.skipNonPublicLibraryClassMembers, 118 warningPrinter, 119 new ClassPresenceFilter(programClassPool, duplicateClassPrinter, 120 new ClassPresenceFilter(libraryClassPool, duplicateClassPrinter, 121 new ClassPoolFiller(libraryClassPool)))))); 122 } 123 124 // Print out a summary of the notes, if necessary. 125 int noteCount = notePrinter.getWarningCount(); 126 if (noteCount > 0) 127 { 128 System.err.println("Note: there were " + noteCount + 129 " duplicate class definitions."); 130 System.err.println(" (http://proguard.sourceforge.net/manual/troubleshooting.html#duplicateclass)"); 131 } 132 133 // Print out a summary of the warnings, if necessary. 134 int warningCount = warningPrinter.getWarningCount(); 135 if (warningCount > 0) 136 { 137 System.err.println("Warning: there were " + warningCount + 138 " classes in incorrectly named files."); 139 System.err.println(" You should make sure all file names correspond to their class names."); 140 System.err.println(" The directory hierarchies must correspond to the package hierarchies."); 141 System.err.println(" (http://proguard.sourceforge.net/manual/troubleshooting.html#unexpectedclass)"); 142 143 if (!configuration.ignoreWarnings) 144 { 145 System.err.println(" If you don't mind the mentioned classes not being written out,"); 146 System.err.println(" you could try your luck using the '-ignorewarnings' option."); 147 throw new IOException("Please correct the above warnings first."); 148 } 149 } 150 } 151 152 153 /** 154 * Reads all input entries from the given class path. 155 */ 156 private void readInput(String messagePrefix, 157 ClassPath classPath, 158 DataEntryReader reader) throws IOException 159 { 160 readInput(messagePrefix, 161 classPath, 162 0, 163 classPath.size(), 164 reader); 165 } 166 167 168 /** 169 * Reads all input entries from the given section of the given class path. 170 */ 171 public void readInput(String messagePrefix, 172 ClassPath classPath, 173 int fromIndex, 174 int toIndex, 175 DataEntryReader reader) throws IOException 176 { 177 for (int index = fromIndex; index < toIndex; index++) 178 { 179 ClassPathEntry entry = classPath.get(index); 180 if (!entry.isOutput()) 181 { 182 readInput(messagePrefix, entry, reader); 183 } 184 } 185 } 186 187 188 /** 189 * Reads the given input class path entry. 190 */ 191 private void readInput(String messagePrefix, 192 ClassPathEntry classPathEntry, 193 DataEntryReader dataEntryReader) throws IOException 194 { 195 try 196 { 197 // Create a reader that can unwrap jars, wars, ears, and zips. 198 DataEntryReader reader = 199 DataEntryReaderFactory.createDataEntryReader(messagePrefix, 200 classPathEntry, 201 dataEntryReader); 202 203 // Create the data entry pump. 204 DirectoryPump directoryPump = 205 new DirectoryPump(classPathEntry.getFile()); 206 207 // Pump the data entries into the reader. 208 directoryPump.pumpDataEntries(reader); 209 } 210 catch (IOException ex) 211 { 212 throw (IOException)new IOException("Can't read [" + classPathEntry + "] (" + ex.getMessage() + ")").initCause(ex); 213 } 214 } 215 } 216