📄 singleimplementationinliner.java
字号:
/* $Id: SingleImplementationInliner.java,v 1.7 2004/12/11 16:35:23 eric Exp $ * * ProGuard -- shrinking, optimization, and obfuscation of Java class files. * * Copyright (c) 2002-2004 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.optimize.peephole;import proguard.classfile.*;import proguard.classfile.attribute.*;import proguard.classfile.attribute.annotation.*;import proguard.classfile.editor.*;import proguard.classfile.instruction.*;import proguard.classfile.util.*;import proguard.classfile.visitor.*;/** * This MemberInfoVisitor and AttrInfoVisitor replaces all references to * interfaces that have single implementations by references to those * implementations. * * @author Eric Lafortune */public class SingleImplementationInlinerimplements MemberInfoVisitor, AttrInfoVisitor, InstructionVisitor, CpInfoVisitor, LocalVariableInfoVisitor, LocalVariableTypeInfoVisitor{ private MemberFinder memberFinder = new MemberFinder(); private ConstantPoolEditor constantPoolEditor = new ConstantPoolEditor(); private CodeAttrInfoEditor codeAttrInfoEditor = new CodeAttrInfoEditor(1024); // Return values of the single implementation inliner. private int cpIndex; private ClassFile singleImplementationClassFile; // Implementations for MemberInfoVisitor. public void visitProgramFieldInfo(ProgramClassFile programClassFile, ProgramFieldInfo programFieldInfo) { visitMemberInfo(programClassFile, programFieldInfo); } public void visitProgramMethodInfo(ProgramClassFile programClassFile, ProgramMethodInfo programMethodInfo) { visitMemberInfo(programClassFile, programMethodInfo); } private void visitMemberInfo(ProgramClassFile programClassFile, ProgramMemberInfo programMemberInfo) { // Update the member info if any of its referenced classes // is an interface with a single implementation. ClassFile[] referencedClassFiles = updateReferencedClassFiles(programMemberInfo.referencedClassFiles); // Update the descriptor if necessary. if (referencedClassFiles != null) { programMemberInfo.u2descriptorIndex = constantPoolEditor.addUtf8CpInfo(programClassFile, newDescriptor(programMemberInfo.getDescriptor(programClassFile), referencedClassFiles)); programMemberInfo.referencedClassFiles = referencedClassFiles; } } public void visitLibraryFieldInfo(LibraryClassFile libraryClassFile, LibraryFieldInfo libraryFieldInfo) {} public void visitLibraryMethodInfo(LibraryClassFile libraryClassFile, LibraryMethodInfo libraryMethodInfo) {} // Implementations for AttrInfoVisitor. public void visitUnknownAttrInfo(ClassFile classFile, UnknownAttrInfo unknownAttrInfo) {} public void visitInnerClassesAttrInfo(ClassFile classFile, InnerClassesAttrInfo innerClassesAttrInfo) {} public void visitEnclosingMethodAttrInfo(ClassFile classFile, EnclosingMethodAttrInfo enclosingMethodAttrInfo) {} public void visitSourceFileAttrInfo(ClassFile classFile, SourceFileAttrInfo sourceFileAttrInfo) {} public void visitSourceDirAttrInfo(ClassFile classFile, SourceDirAttrInfo sourceDirAttrInfo) {} public void visitConstantValueAttrInfo(ClassFile classFile, FieldInfo fieldInfo, ConstantValueAttrInfo constantValueAttrInfo) {} public void visitExceptionsAttrInfo(ClassFile classFile, MethodInfo methodInfo, ExceptionsAttrInfo exceptionsAttrInfo) {} public void visitLineNumberTableAttrInfo(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, LineNumberTableAttrInfo lineNumberTableAttrInfo) {} public void visitDeprecatedAttrInfo(ClassFile classFile, DeprecatedAttrInfo deprecatedAttrInfo) {} public void visitSyntheticAttrInfo(ClassFile classFile, SyntheticAttrInfo syntheticAttrInfo) {} public void visitCodeAttrInfo(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo) { // Reset the code changes. codeAttrInfoEditor.reset(codeAttrInfo.u4codeLength); // Update the instructions that refer to interfaces with a single // implementation. codeAttrInfo.instructionsAccept(classFile, methodInfo, this); // Apply the code changes. codeAttrInfoEditor.visitCodeAttrInfo(classFile, methodInfo, codeAttrInfo); //codeAttrInfo.attributesAccept(classFile, methodInfo, this); } public void visitLocalVariableTableAttrInfo(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, LocalVariableTableAttrInfo localVariableTableAttrInfo) { // Rename the types of the local variables. localVariableTableAttrInfo.localVariablesAccept(classFile, methodInfo, codeAttrInfo, this); } public void visitLocalVariableTypeTableAttrInfo(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, LocalVariableTypeTableAttrInfo localVariableTypeTableAttrInfo) { // Rename the signatures of the local variables. localVariableTypeTableAttrInfo.localVariablesAccept(classFile, methodInfo, codeAttrInfo, this); } public void visitSignatureAttrInfo(ClassFile classFile, SignatureAttrInfo signatureAttrInfo) { // TODO: Update signature attribute. } public void visitRuntimeVisibleAnnotationAttrInfo(ClassFile classFile, RuntimeVisibleAnnotationsAttrInfo runtimeVisibleAnnotationsAttrInfo) { // TODO: Update runtime visible annotation attribute. //runtimeVisibleAnnotationsAttrInfo.annotationsAccept(classFile, this); } public void visitRuntimeInvisibleAnnotationAttrInfo(ClassFile classFile, RuntimeInvisibleAnnotationsAttrInfo runtimeInvisibleAnnotationsAttrInfo) { // TODO: Update runtime invisible annotation attribute. //runtimeInvisibleAnnotationsAttrInfo.annotationsAccept(classFile, this); } public void visitRuntimeVisibleParameterAnnotationAttrInfo(ClassFile classFile, RuntimeVisibleParameterAnnotationsAttrInfo runtimeVisibleParameterAnnotationsAttrInfo) { // TODO: Update runtime visible parameter annotation attribute. //runtimeVisibleParameterAnnotationsAttrInfo.annotationsAccept(classFile, this); } public void visitRuntimeInvisibleParameterAnnotationAttrInfo(ClassFile classFile, RuntimeInvisibleParameterAnnotationsAttrInfo runtimeInvisibleParameterAnnotationsAttrInfo) { // TODO: Update runtime invisible parameter annotation attribute. //runtimeInvisibleParameterAnnotationsAttrInfo.annotationsAccept(classFile, this); } public void visitAnnotationDefaultAttrInfo(ClassFile classFile, AnnotationDefaultAttrInfo annotationDefaultAttrInfo) { // TODO: Update annotation default attribute. //annotationDefaultAttrInfo.defaultValueAccept(classFile, this); } // Implementations for InstructionVisitor. public void visitSimpleInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, SimpleInstruction simpleInstruction) {} public void visitVariableInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, VariableInstruction variableInstruction) {} public void visitBranchInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, BranchInstruction branchInstruction) {} public void visitTableSwitchInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, TableSwitchInstruction tableSwitchInstruction) {} public void visitLookUpSwitchInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, LookUpSwitchInstruction lookUpSwitchInstruction) {} public void visitCpInstruction(ClassFile classFile, MethodInfo methodInfo, CodeAttrInfo codeAttrInfo, int offset, CpInstruction cpInstruction) { cpIndex = 0; classFile.constantPoolEntryAccept(cpInstruction.cpIndex, this); if (cpIndex != 0) { byte opcode = cpInstruction.opcode; // Does this instruction now invoke a single implementation? if (opcode == InstructionConstants.OP_INVOKEINTERFACE && singleImplementationClassFile != null) { // Replace the interface invocation by an ordinary invocation. opcode = InstructionConstants.OP_INVOKEVIRTUAL; } // Replace the instruction by an updated instruction. Instruction replacementInstruction = new CpInstruction(opcode, cpIndex, cpInstruction.constant).shrink(); codeAttrInfoEditor.replaceInstruction(offset, replacementInstruction); } } // Implementations for CpInfoVisitor. public void visitIntegerCpInfo(ClassFile classFile, IntegerCpInfo integerCpInfo) {} public void visitLongCpInfo(ClassFile classFile, LongCpInfo longCpInfo) {} public void visitFloatCpInfo(ClassFile classFile, FloatCpInfo floatCpInfo) {} public void visitDoubleCpInfo(ClassFile classFile, DoubleCpInfo doubleCpInfo) {} public void visitUtf8CpInfo(ClassFile classFile, Utf8CpInfo utf8CpInfo) {} public void visitStringCpInfo(ClassFile classFile, StringCpInfo stringCpInfo) { // Create a new string entry if its referenced class is an interface with // a single implementation. singleImplementationClassFile = SingleImplementationMarker.singleImplementation(stringCpInfo.referencedClassFile); if (singleImplementationClassFile != null) { // Create a new string entry. cpIndex = constantPoolEditor.addStringCpInfo((ProgramClassFile)classFile, singleImplementationClassFile.getName(), singleImplementationClassFile); } } public void visitFieldrefCpInfo(ClassFile classFile, FieldrefCpInfo fieldrefCpInfo) { // Create a new field reference entry if its type has changed. cpIndex = 0; classFile.constantPoolEntryAccept(fieldrefCpInfo.getNameAndTypeIndex(), this); int nameAndTypeIndex = cpIndex; if (nameAndTypeIndex != 0) { // Create a new field reference entry. cpIndex = constantPoolEditor.addFieldrefCpInfo((ProgramClassFile)classFile, fieldrefCpInfo.getClassIndex(), nameAndTypeIndex, fieldrefCpInfo.referencedClassFile, fieldrefCpInfo.referencedMemberInfo); } } public void visitMethodrefCpInfo(ClassFile classFile, MethodrefCpInfo methodrefCpInfo) { // Create a new method reference entry if its type has changed. cpIndex = 0; classFile.constantPoolEntryAccept(methodrefCpInfo.getNameAndTypeIndex(), this);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -