proguard.optimize.info.ParameterUsageMarker 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-2021 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.instruction.*;
import proguard.classfile.instruction.visitor.InstructionVisitor;
import proguard.classfile.util.*;
import proguard.classfile.visitor.MemberVisitor;
import proguard.evaluation.PartialEvaluator;
import proguard.evaluation.value.*;
/**
* This MemberVisitor counts the parameters and marks the used parameters
* of the methods that it visits. It also marks the 'this' parameters of
* methods that have hierarchies.
*
* @author Eric Lafortune
*/
public class ParameterUsageMarker
implements MemberVisitor,
AttributeVisitor,
InstructionVisitor
{
private static final Logger logger = LogManager.getLogger(ParameterUsageMarker.class);
private final boolean markThisParameter;
private final boolean markAllParameters;
private final boolean analyzeCode;
private final PartialEvaluator partialEvaluator = PartialEvaluator.Builder.create().build();
/**
* Creates a new ParameterUsageMarker.
*/
public ParameterUsageMarker()
{
this(false, false);
}
/**
* Creates a new ParameterUsageMarker that optionally marks all parameters.
* @param markThisParameter specifies whether all 'this' parameters should
* be marked as being used.
* @param markAllParameters specifies whether all other parameters should
* be marked as being used.
*/
public ParameterUsageMarker(boolean markThisParameter,
boolean markAllParameters)
{
this(markThisParameter, markAllParameters, true);
}
/**
* Creates a new ParameterUsageMarker that optionally marks all parameters.
* @param markThisParameter specifies whether all 'this' parameters should
* be marked as being used.
* @param markAllParameters specifies whether all other parameters should
* be marked as being used.
* @param analyzeCode specifies whether the code of visited methods
* should be analyzed for used parameters.
*/
public ParameterUsageMarker(boolean markThisParameter,
boolean markAllParameters,
boolean analyzeCode)
{
this.markThisParameter = markThisParameter;
this.markAllParameters = markAllParameters;
this.analyzeCode = analyzeCode;
}
// Implementations for MemberVisitor.
public void visitProgramMethod(ProgramClass programClass, ProgramMethod programMethod)
{
int parameterSize =
ClassUtil.internalMethodParameterSize(programMethod.getDescriptor(programClass),
programMethod.getAccessFlags());
if (parameterSize > 0)
{
int accessFlags = programMethod.getAccessFlags();
// Must we mark the 'this' parameter?
if (markThisParameter &&
(accessFlags & AccessConstants.STATIC) == 0)
{
// Mark the 'this' parameter.
markParameterUsed(programMethod, 0);
}
// Must we mark all other parameters?
if (markAllParameters)
{
// Mark all parameters, without the 'this' parameter.
markUsedParameters(programMethod,
(accessFlags & AccessConstants.STATIC) != 0 ?
-1L : -2L);
}
if (programMethod.processingInfo instanceof ProgramMethodOptimizationInfo
&& parameterSize >= 64) {
int parameterSizesCummulative = 0;
for (int index = 0; parameterSizesCummulative < 64; index++) {
boolean isCategory2 =
((ProgramMethodOptimizationInfo) programMethod.processingInfo).getParameterSize(index)
== 2;
if (parameterSizesCummulative == 63 && isCategory2) {
markParameterUsed(programMethod, 63);
}
parameterSizesCummulative +=
((ProgramMethodOptimizationInfo) programMethod.processingInfo)
.getParameterSize(index);
}
}
// Is it a native method?
if ((accessFlags & AccessConstants.NATIVE) != 0)
{
// Mark all parameters.
markUsedParameters(programMethod, -1L);
}
// Is it an abstract method?
else if ((accessFlags & AccessConstants.ABSTRACT) != 0)
{
// Mark the 'this' parameter.
markParameterUsed(programMethod, 0);
}
// Is it a non-native, concrete method?
else
{
// Is the method not static, but synchronized, or can it have
// other implementations, or is it a class instance initializer?
if ((accessFlags & AccessConstants.STATIC) == 0 &&
((accessFlags & AccessConstants.SYNCHRONIZED) != 0 ||
programClass.mayHaveImplementations(programMethod) ||
programMethod.getName(programClass).equals(ClassConstants.METHOD_NAME_INIT)))
{
// Mark the 'this' parameter.
markParameterUsed(programMethod, 0);
}
if (analyzeCode)
{
// Mark the parameters that are used by the code.
programMethod.attributesAccept(programClass, this);
}
}
logger.debug("{}",
() -> {
StringBuilder debugMessage = new StringBuilder(String.format("ParameterUsageMarker: [%s.%s%s]: ",
programClass.getName(),
programMethod.getName(programClass),
programMethod.getDescriptor(programClass)));
for (int variableIndex = 0; variableIndex < parameterSize; variableIndex++) {
debugMessage.append(isParameterUsed(programMethod, variableIndex) ? '+' : '-');
}
return debugMessage.toString();
}
);
}
// Set the parameter size.
setParameterSize(programMethod, parameterSize);
}
public void visitLibraryMethod(LibraryClass libraryClass, LibraryMethod libraryMethod)
{
// Can the method have other implementations?
if (libraryClass.mayHaveImplementations(libraryMethod))
{
// All implementations must keep all parameters of this method,
// including the 'this' parameter.
markUsedParameters(libraryMethod, -1L);
}
}
// Implementations for AttributeVisitor.
public void visitAnyAttribute(Clazz clazz, Attribute attribute) {}
public void visitCodeAttribute(Clazz clazz, Method method, CodeAttribute codeAttribute)
{
// Evaluate the code.
partialEvaluator.visitCodeAttribute(clazz, method, codeAttribute);
// Mark the parameters that are used by the code.
codeAttribute.instructionsAccept(clazz, method, this);
}
// Implementations for InstructionVisitor.
public void visitAnyInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, Instruction instruction) {}
public void visitVariableInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, VariableInstruction variableInstruction)
{
if (partialEvaluator.isTraced(offset) &&
variableInstruction.isLoad())
{
int variableIndex = variableInstruction.variableIndex;
if (variableIndex < codeAttribute.u2maxLocals)
{
// The parameter indices stored in the producer values are
// parameter offsets, taking into account Category 2 types,
// and therefore compatible with variable indices.
Value producer =
partialEvaluator.getVariablesBefore(offset).getProducerValue(variableIndex);
if (producer != null &&
producer.instructionOffsetValue().contains(variableIndex | InstructionOffsetValue.METHOD_PARAMETER))
{
// Mark the variable.
markParameterUsed(method, variableIndex);
// Account for Category 2 instructions, which take up two entries.
if (variableInstruction.stackPopCount(clazz) == 2 ||
variableInstruction.stackPushCount(clazz) == 2)
{
markParameterUsed(method, variableIndex + 1);
}
}
}
}
}
// Small utility methods.
/**
* Sets the total size of the parameters.
*/
private static void setParameterSize(Method method, int parameterSize)
{
ProgramMethodOptimizationInfo.getProgramMethodOptimizationInfo(method).setParameterSize(parameterSize);
}
/**
* Returns the total size of the parameters.
*/
public static int getParameterSize(Method method)
{
return MethodOptimizationInfo.getMethodOptimizationInfo(method).getParameterSize();
}
/**
* Marks the given parameter as being used.
*/
public static void markParameterUsed(Method method, int variableIndex)
{
ProgramMethodOptimizationInfo.getProgramMethodOptimizationInfo(method).setParameterUsed(variableIndex);
}
/**
* Marks the given parameters as being used.
*/
private static void markUsedParameters(Method method, long usedParameters)
{
ProgramMethodOptimizationInfo.getProgramMethodOptimizationInfo(method).updateUsedParameters(usedParameters);
}
/**
* Returns whether the given method has any unused parameters.
*/
public static boolean hasUnusedParameters(Method method)
{
return MethodOptimizationInfo.getMethodOptimizationInfo(method).hasUnusedParameters();
}
/**
* Returns whether the given parameter is being used.
*/
public static boolean isParameterUsed(Method method, int variableIndex)
{
return MethodOptimizationInfo.getMethodOptimizationInfo(method).isParameterUsed(variableIndex);
}
/**
* Returns which parameters are being used.
*/
public static long getUsedParameters(Method method)
{
return MethodOptimizationInfo.getMethodOptimizationInfo(method).getUsedParameters();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy