proguard.optimize.info.WrapperClassMarker Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of proguard-base Show documentation
Show all versions of proguard-base Show documentation
ProGuard is a free shrinker, optimizer, obfuscator, and preverifier for Java bytecode
The newest version!
/*
* ProGuard -- shrinking, optimization, obfuscation, and preverification
* of Java bytecode.
*
* Copyright (c) 2002-2020 Guardsquare NV
*
* 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.info;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import proguard.classfile.*;
import proguard.classfile.attribute.*;
import proguard.classfile.attribute.visitor.AttributeVisitor;
import proguard.classfile.constant.*;
import proguard.classfile.instruction.*;
import proguard.classfile.util.*;
import proguard.classfile.visitor.*;
import proguard.evaluation.value.*;
import proguard.optimize.evaluation.StoringInvocationUnit;
/**
* This ClassVisitor marks all program classes that are a simple wrapper for a
* single non-null instance of another class.
*
* A wrapper class has
* - exactly one non-static field, which references an object,
* - exactly one initializer, with a single parameter that is never null,
* that initializes the field,
* - no subclasses.
*
* @see StoringInvocationUnit
* @author Eric Lafortune
*/
public class WrapperClassMarker
implements ClassVisitor,
MemberVisitor,
AttributeVisitor
{
private static final Logger logger = LogManager.getLogger(WrapperClassMarker.class);
private final Constant[] INITIALIZER_CONSTANTS = new Constant[]
{
new FieldrefConstant(InstructionSequenceMatcher.A,
InstructionSequenceMatcher.B, null, null),
};
// Instruction pattern:
// this.x = arg0;
// super.;
// return;
private final Instruction[] INITIALIZER_INSTRUCTIONS = new Instruction[]
{
new VariableInstruction(Instruction.OP_ALOAD_0, 0),
new VariableInstruction(Instruction.OP_ALOAD_1, 1),
new ConstantInstruction(Instruction.OP_PUTFIELD, 0),
new VariableInstruction(Instruction.OP_ALOAD_0, 0),
new ConstantInstruction(Instruction.OP_INVOKESPECIAL, InstructionSequenceMatcher.X),
new SimpleInstruction(Instruction.OP_RETURN),
};
private final InstructionSequenceMatcher INITIALIZER_MATCHER = new InstructionSequenceMatcher(INITIALIZER_CONSTANTS, INITIALIZER_INSTRUCTIONS);
// Fields acting as parameters and return values for the visitor methods.
private Clazz wrappedClass;
private int wrapCounter;
// Implementations for ClassVisitor.
@Override
public void visitAnyClass(Clazz clazz) { }
@Override
public void visitProgramClass(ProgramClass programClass)
{
if (programClass.subClassCount == 0)
{
wrappedClass = null;
// Can we find one non-static field with a class type?
wrapCounter = 0;
programClass.fieldsAccept(this);
if (wrapCounter == 1)
{
// Can we find exactly one initializer that initializes this
// field?
wrapCounter = 0;
programClass.methodsAccept(this);
if (wrapCounter == 1)
{
setWrappedClass(programClass, wrappedClass);
}
}
}
}
// Implementations for MemberVisitor.
public void visitProgramField(ProgramClass programClass, ProgramField programField)
{
// Is the field non-static and of a class type?
if ((programField.getAccessFlags() & AccessConstants.STATIC) == 0 &&
ClassUtil.isInternalClassType(programField.getDescriptor(programClass)) &&
!ClassUtil.isInternalArrayType(programField.getDescriptor(programClass)))
{
wrappedClass = programField.referencedClass;
if (wrappedClass != null)
{
wrapCounter++;
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
public void visitProgramMethod(ProgramClass programClass, ProgramMethod programMethod)
{
// Is the method an initializer?
if (ClassUtil.isInitializer(programMethod.getName(programClass)))
{
// Does it have exactly one parameter?
if (ClassUtil.internalMethodParameterCount(programMethod.getDescriptor(programClass)) == 1)
{
// Is the parameter a non-null reference?
Value value =
StoringInvocationUnit.getMethodParameterValue(programMethod, 1);
if (value != null &&
value.computationalType() == Value.TYPE_REFERENCE &&
value.referenceValue().isNotNull() == Value.ALWAYS)
{
// Does the method initialize the field?
programMethod.attributesAccept(programClass, this);
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
}
// Implementations for AttributeVisitor.
public void visitAnyAttribute(Clazz clazz, Attribute attribute) {}
public void visitCodeAttribute(Clazz clazz, Method method, CodeAttribute codeAttribute)
{
// Is the initializer initializing the field?
if (codeAttribute.u4codeLength == 10)
{
INITIALIZER_MATCHER.reset();
codeAttribute.instructionsAccept(clazz, method, INITIALIZER_MATCHER);
if (INITIALIZER_MATCHER.isMatching())
{
String initializerClassName = clazz.getName();
String fieldClassName = clazz.getClassName(INITIALIZER_MATCHER.matchedConstantIndex(InstructionSequenceMatcher.A));
if (fieldClassName.equals(initializerClassName))
{
wrapCounter++;
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
else
{
wrapCounter = Integer.MIN_VALUE;
}
}
// Small utility methods.
private static void setWrappedClass(Clazz clazz, Clazz wrappedClass)
{
logger.debug("WrapperClassMarker: [{}] wraps [{}]", clazz.getName(), wrappedClass.getName());
ProgramClassOptimizationInfo.getProgramClassOptimizationInfo(clazz).setWrappedClass(wrappedClass);
}
public static Clazz getWrappedClass(Clazz clazz)
{
return ClassOptimizationInfo.getClassOptimizationInfo(clazz).getWrappedClass();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy