/* * ProGuard -- shrinking, optimization, obfuscation, and preverification * of Java bytecode. * * Copyright (c) 2002-2009 Eric Lafortune (eric@graphics.cornell.edu) * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the Free * Software Foundation; either version 2 of the License, or (at your option) * any later version. * * This program 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 for * more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ package proguard.shrink; import proguard.classfile.*; import proguard.classfile.attribute.*; import proguard.classfile.attribute.annotation.*; import proguard.classfile.attribute.annotation.visitor.*; import proguard.classfile.attribute.visitor.AttributeVisitor; import proguard.classfile.constant.*; import proguard.classfile.editor.*; import proguard.classfile.util.*; import proguard.classfile.visitor.*; /** * This ClassVisitor removes constant pool entries and class members that * are not marked as being used. * * @see UsageMarker * * @author Eric Lafortune */ public class ClassShrinker extends SimplifiedVisitor implements ClassVisitor, MemberVisitor, AttributeVisitor, AnnotationVisitor, ElementValueVisitor { private final UsageMarker usageMarker; private int[] constantIndexMap = new int[ClassConstants.TYPICAL_CONSTANT_POOL_SIZE]; private final ConstantPoolRemapper constantPoolRemapper = new ConstantPoolRemapper(); /** * Creates a new ClassShrinker. * @param usageMarker the usage marker that is used to mark the classes * and class members. */ public ClassShrinker(UsageMarker usageMarker) { this.usageMarker = usageMarker; } // Implementations for ClassVisitor. public void visitProgramClass(ProgramClass programClass) { // Shrink the arrays for constant pool, interfaces, fields, methods, // and class attributes. programClass.u2interfacesCount = shrinkConstantIndexArray(programClass.constantPool, programClass.u2interfaces, programClass.u2interfacesCount); // Shrinking the constant pool also sets up an index map. programClass.u2constantPoolCount = shrinkConstantPool(programClass.constantPool, programClass.u2constantPoolCount); programClass.u2fieldsCount = shrinkArray(programClass.fields, programClass.u2fieldsCount); programClass.u2methodsCount = shrinkArray(programClass.methods, programClass.u2methodsCount); programClass.u2attributesCount = shrinkArray(programClass.attributes, programClass.u2attributesCount); // Compact the remaining fields, methods, and attributes, // and remap their references to the constant pool. programClass.fieldsAccept(this); programClass.methodsAccept(this); programClass.attributesAccept(this); // Remap all constant pool references. constantPoolRemapper.setConstantIndexMap(constantIndexMap); constantPoolRemapper.visitProgramClass(programClass); // Remove the unused interfaces from the class signature. programClass.attributesAccept(new SignatureShrinker()); // Compact the extra field pointing to the subclasses of this class. programClass.subClasses = shrinkToNewArray(programClass.subClasses); } public void visitLibraryClass(LibraryClass libraryClass) { // Library classes are left unchanged. // Compact the extra field pointing to the subclasses of this class. libraryClass.subClasses = shrinkToNewArray(libraryClass.subClasses); } // Implementations for MemberVisitor. public void visitProgramMember(ProgramClass programClass, ProgramMember programMember) { // Shrink the attributes array. programMember.u2attributesCount = shrinkArray(programMember.attributes, programMember.u2attributesCount); // Shrink any attributes. programMember.attributesAccept(programClass, this); } // Implementations for AttributeVisitor. public void visitAnyAttribute(Clazz clazz, Attribute attribute) {} public void visitInnerClassesAttribute(Clazz clazz, InnerClassesAttribute innerClassesAttribute) { // Shrink the array of InnerClassesInfo objects. innerClassesAttribute.u2classesCount = shrinkArray(innerClassesAttribute.classes, innerClassesAttribute.u2classesCount); } public void visitEnclosingMethodAttribute(Clazz clazz, EnclosingMethodAttribute enclosingMethodAttribute) { // Sometimes, a class is still referenced (apparently as a dummy class), // but its enclosing method is not. Then remove the reference to // the enclosing method. // E.g. the anonymous inner class javax.swing.JList$1 is defined inside // a constructor of javax.swing.JList, but it is also referenced as a // dummy argument in a constructor of javax.swing.JList$ListSelectionHandler. if (enclosingMethodAttribute.referencedMethod != null && !usageMarker.isUsed(enclosingMethodAttribute.referencedMethod)) { enclosingMethodAttribute.u2nameAndTypeIndex = 0; enclosingMethodAttribute.referencedMethod = null; } } public void visitCodeAttribute(Clazz clazz, Method method, CodeAttribute codeAttribute) { // Shrink the attributes array. codeAttribute.u2attributesCount = shrinkArray(codeAttribute.attributes, codeAttribute.u2attributesCount); } public void visitAnyAnnotationsAttribute(Clazz clazz, AnnotationsAttribute annotationsAttribute) { // Shrink the annotations array. annotationsAttribute.u2annotationsCount = shrinkArray(annotationsAttribute.annotations, annotationsAttribute.u2annotationsCount); // Shrink the annotations themselves. annotationsAttribute.annotationsAccept(clazz, this); } public void visitAnyParameterAnnotationsAttribute(Clazz clazz, Method method, ParameterAnnotationsAttribute parameterAnnotationsAttribute) { // Loop over all parameters. for (int parameterIndex = 0; parameterIndex < parameterAnnotationsAttribute.u2parametersCount; parameterIndex++) { // Shrink the parameter annotations array. parameterAnnotationsAttribute.u2parameterAnnotationsCount[parameterIndex] = shrinkArray(parameterAnnotationsAttribute.parameterAnnotations[parameterIndex], parameterAnnotationsAttribute.u2parameterAnnotationsCount[parameterIndex]); } // Shrink the annotations themselves. parameterAnnotationsAttribute.annotationsAccept(clazz, method, this); } // Implementations for AnnotationVisitor. public void visitAnnotation(Clazz clazz, Annotation annotation) { // Shrink the element values array. annotation.u2elementValuesCount = shrinkArray(annotation.elementValues, annotation.u2elementValuesCount); // Shrink the element values themselves. annotation.elementValuesAccept(clazz, this); } /** * This AttributeVisitor updates the Utf8 constants of class signatures, * removing any unused interfaces. */ private class SignatureShrinker extends SimplifiedVisitor implements AttributeVisitor { public void visitAnyAttribute(Clazz clazz, Attribute attribute) {} public void visitSignatureAttribute(Clazz clazz, SignatureAttribute signatureAttribute) { Clazz[] referencedClasses = signatureAttribute.referencedClasses; if (referencedClasses != null) { // Go over the generic definitions, superclass and implemented interfaces. String signature = clazz.getString(signatureAttribute.u2signatureIndex); InternalTypeEnumeration internalTypeEnumeration = new InternalTypeEnumeration(signature); StringBuffer newSignatureBuffer = new StringBuffer(); int referencedClassIndex = 0; int newReferencedClassIndex = 0; while (internalTypeEnumeration.hasMoreTypes()) { // Consider the classes referenced by this signature. String type = internalTypeEnumeration.nextType(); int classCount = new DescriptorClassEnumeration(type).classCount(); Clazz referencedClass = referencedClasses[referencedClassIndex]; if (referencedClass == null || usageMarker.isUsed(referencedClass)) { // Append the superclass or interface. newSignatureBuffer.append(type); // Copy the referenced classes. for (int counter = 0; counter < classCount; counter++) { referencedClasses[newReferencedClassIndex++] = referencedClasses[referencedClassIndex++]; } } else { // Skip the referenced classes. referencedClassIndex += classCount; } } if (newReferencedClassIndex < referencedClassIndex) { // Update the signature. ((Utf8Constant)((ProgramClass)clazz).constantPool[signatureAttribute.u2signatureIndex]).setString(newSignatureBuffer.toString()); // Clear the unused entries. while (newReferencedClassIndex < referencedClassIndex) { referencedClasses[newReferencedClassIndex++] = null; } } } } } // Implementations for ElementValueVisitor. public void visitAnyElementValue(Clazz clazz, Annotation annotation, ElementValue elementValue) {} public void visitAnnotationElementValue(Clazz clazz, Annotation annotation, AnnotationElementValue annotationElementValue) { // Shrink the contained annotation. annotationElementValue.annotationAccept(clazz, this); } public void visitArrayElementValue(Clazz clazz, Annotation annotation, ArrayElementValue arrayElementValue) { // Shrink the element values array. arrayElementValue.u2elementValuesCount = shrinkArray(arrayElementValue.elementValues, arrayElementValue.u2elementValuesCount); // Shrink the element values themselves. arrayElementValue.elementValuesAccept(clazz, annotation, this); } // Small utility methods. /** * Removes all entries that are not marked as being used from the given * constant pool. * @return the new number of entries. */ private int shrinkConstantPool(Constant[] constantPool, int length) { if (constantIndexMap.length < length) { constantIndexMap = new int[length]; } int counter = 1; boolean isUsed = false; // Shift the used constant pool entries together. for (int index = 1; index < length; index++) { constantIndexMap[index] = counter; Constant constant = constantPool[index]; // Don't update the flag if this is the second half of a long entry. if (constant != null) { isUsed = usageMarker.isUsed(constant); } if (isUsed) { constantPool[counter++] = constant; } } // Clear the remaining constant pool elements. for (int index = counter; index < length; index++) { constantPool[index] = null; } return counter; } /** * Removes all indices that point to unused constant pool entries * from the given array. * @return the new number of indices. */ private int shrinkConstantIndexArray(Constant[] constantPool, int[] array, int length) { int counter = 0; // Shift the used objects together. for (int index = 0; index < length; index++) { if (usageMarker.isUsed(constantPool[array[index]])) { array[counter++] = array[index]; } } // Clear the remaining array elements. for (int index = counter; index < length; index++) { array[index] = 0; } return counter; } /** * Removes all Clazz objects that are not marked as being used * from the given array and returns the remaining objects in a an array * of the right size. * @return the new array. */ private Clazz[] shrinkToNewArray(Clazz[] array) { if (array == null) { return null; } // Shrink the given array in-place. int length = shrinkArray(array, array.length); if (length == 0) { return null; } // Return immediately if the array is of right size already. if (length == array.length) { return array; } // Copy the remaining elements into a new array of the right size. Clazz[] newArray = new Clazz[length]; System.arraycopy(array, 0, newArray, 0, length); return newArray; } /** * Removes all VisitorAccepter objects that are not marked as being used * from the given array. * @return the new number of VisitorAccepter objects. */ private int shrinkArray(VisitorAccepter[] array, int length) { int counter = 0; // Shift the used objects together. for (int index = 0; index < length; index++) { if (usageMarker.isUsed(array[index])) { array[counter++] = array[index]; } } // Clear the remaining array elements. for (int index = counter; index < length; index++) { array[index] = null; } return counter; } }