proguard.optimize.info.ReferenceEscapeChecker 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.Level;
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.constant.visitor.ConstantVisitor;
import proguard.classfile.editor.ClassEstimates;
import proguard.classfile.instruction.*;
import proguard.classfile.instruction.visitor.InstructionVisitor;
import proguard.classfile.util.*;
import proguard.evaluation.*;
import proguard.evaluation.value.*;
import proguard.optimize.evaluation.*;
import proguard.util.ArrayUtil;
/**
* This AttributeVisitor can tell whether reference parameters and instances
* are escaping, are modified, or are returned.
*
* @see ParameterEscapeMarker
* @author Eric Lafortune
*/
public class ReferenceEscapeChecker
implements AttributeVisitor,
InstructionVisitor,
ConstantVisitor
{
private static final Logger logger = LogManager.getLogger(ReferenceEscapeChecker.class);
private boolean[] instanceEscaping = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
private boolean[] instanceReturned = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
private boolean[] instanceModified = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
private boolean[] externalInstance = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
// private boolean[] exceptionEscaping = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
// private boolean[] exceptionReturned = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
// private boolean[] exceptionModified = new boolean[ClassEstimates.TYPICAL_CODE_LENGTH];
private final PartialEvaluator partialEvaluator;
private final boolean runPartialEvaluator;
// Parameters and values for visitor methods.
private Method referencingMethod;
private int referencingOffset;
private int referencingPopCount;
/**
* Creates a new ReferenceEscapeChecker.
*/
public ReferenceEscapeChecker()
{
this(new ReferenceTracingValueFactory(new BasicValueFactory()));
}
/**
* Creates a new ReferenceEscapeChecker. This private constructor gets around
* the constraint that it's not allowed to add statements before calling
* 'this'.
*/
private ReferenceEscapeChecker(ReferenceTracingValueFactory referenceTracingValueFactory)
{
this(
PartialEvaluator.Builder.create()
.setValueFactory(referenceTracingValueFactory)
.setInvocationUnit(new ParameterTracingInvocationUnit(new BasicInvocationUnit(referenceTracingValueFactory)))
.setEvaluateAllCode(true)
.setExtraInstructionVisitor(referenceTracingValueFactory)
.build(),
true);
}
/**
* Creates a new ReferenceEscapeChecker.
* @param partialEvaluator the evaluator to be used for the analysis.
* @param runPartialEvaluator specifies whether to run this evaluator on
* every code attribute that is visited.
*/
public ReferenceEscapeChecker(PartialEvaluator partialEvaluator,
boolean runPartialEvaluator)
{
this.partialEvaluator = partialEvaluator;
this.runPartialEvaluator = runPartialEvaluator;
}
/**
* Returns whether the instance created or retrieved at the specified
* instruction offset is escaping.
*/
public boolean isInstanceEscaping(int instructionOffset)
{
return instanceEscaping[instructionOffset];
}
/**
* Returns whether the instance created or retrieved at the specified
* instruction offset is being returned.
*/
public boolean isInstanceReturned(int instructionOffset)
{
return instanceReturned[instructionOffset];
}
/**
* Returns whether the instance created or retrieved at the specified
* instruction offset is being modified.
*/
public boolean isInstanceModified(int instructionOffset)
{
return instanceModified[instructionOffset];
}
/**
* Returns whether the instance created or retrieved at the specified
* instruction offset is external to this method and its invoked methods.
*/
public boolean isInstanceExternal(int instructionOffset)
{
return externalInstance[instructionOffset];
}
// Implementations for AttributeVisitor.
public void visitAnyAttribute(Clazz clazz, Attribute attribute) {}
public void visitCodeAttribute(Clazz clazz, Method method, CodeAttribute codeAttribute)
{
// Evaluate the method.
if (runPartialEvaluator)
{
partialEvaluator.visitCodeAttribute(clazz, method, codeAttribute);
}
int codeLength = codeAttribute.u4codeLength;
// Initialize the global arrays.
instanceEscaping = ArrayUtil.ensureArraySize(instanceEscaping, codeLength, false);
instanceReturned = ArrayUtil.ensureArraySize(instanceReturned, codeLength, false);
instanceModified = ArrayUtil.ensureArraySize(instanceModified, codeLength, false);
externalInstance = ArrayUtil.ensureArraySize(externalInstance, codeLength, false);
// Mark the parameters and instances that are escaping from the code.
codeAttribute.instructionsAccept(clazz, method, partialEvaluator.tracedInstructionFilter(this));
logger.debug("ReferenceEscapeChecker: [{}.{}{}]",
clazz.getName(),
method.getName(clazz),
method.getDescriptor(clazz)
);
if (logger.getLevel().isLessSpecificThan(Level.DEBUG))
{
for (int index = 0; index < codeLength; index++) {
if (partialEvaluator.isInstruction(index)) {
logger.debug(" {}{}{}{} {}",
instanceEscaping[index] ? 'E' : '.',
instanceReturned[index] ? 'R' : '.',
instanceModified[index] ? 'M' : '.',
externalInstance[index] ? 'X' : '.',
InstructionFactory.create(codeAttribute.code, index).toString(index)
);
}
}
}
}
// Implementations for InstructionVisitor.
public void visitAnyInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, Instruction instruction) {}
public void visitSimpleInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, SimpleInstruction simpleInstruction)
{
switch (simpleInstruction.opcode)
{
case Instruction.OP_AASTORE:
// Mark array reference values whose element is modified.
markModifiedReferenceValues(offset,
simpleInstruction.stackPopCount(clazz) - 1);
// Mark reference values that are put in the array.
markEscapingReferenceValues(offset, 0);
break;
case Instruction.OP_IASTORE:
case Instruction.OP_LASTORE:
case Instruction.OP_FASTORE:
case Instruction.OP_DASTORE:
case Instruction.OP_BASTORE:
case Instruction.OP_CASTORE:
case Instruction.OP_SASTORE:
// Mark array reference values whose element is modified.
markModifiedReferenceValues(offset,
simpleInstruction.stackPopCount(clazz) - 1);
break;
case Instruction.OP_ARETURN:
// Mark the returned reference values.
markReturnedReferenceValues(offset, 0);
break;
case Instruction.OP_ATHROW:
// Mark the escaping reference values.
markEscapingReferenceValues(offset, 0);
break;
}
}
public void visitConstantInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, ConstantInstruction constantInstruction)
{
switch (constantInstruction.opcode)
{
case Instruction.OP_GETSTATIC:
case Instruction.OP_GETFIELD:
// Mark external reference values.
markExternalReferenceValue(offset);
break;
case Instruction.OP_PUTSTATIC:
// Mark reference values that are put in the field.
markEscapingReferenceValues(offset, 0);
break;
case Instruction.OP_PUTFIELD:
// Mark reference reference values whose field is modified.
markModifiedReferenceValues(offset,
constantInstruction.stackPopCount(clazz) - 1);
// Mark reference values that are put in the field.
markEscapingReferenceValues(offset, 0);
break;
case Instruction.OP_INVOKEVIRTUAL:
case Instruction.OP_INVOKESPECIAL:
case Instruction.OP_INVOKESTATIC:
case Instruction.OP_INVOKEINTERFACE:
// Mark reference reference values that are modified as parameters
// of the invoked method.
// Mark reference values that are escaping as parameters
// of the invoked method.
// Mark escaped reference reference values in the invoked method.
referencingMethod = method;
referencingOffset = offset;
referencingPopCount = constantInstruction.stackPopCount(clazz);
clazz.constantPoolEntryAccept(constantInstruction.constantIndex, this);
break;
}
}
// Implementations for ConstantVisitor.
public void visitAnyConstant(Clazz clazz, Constant constant) {}
public void visitFieldrefConstant(Clazz clazz, FieldrefConstant fieldrefConstant)
{
clazz.constantPoolEntryAccept(fieldrefConstant.u2classIndex, this);
}
public void visitAnyMethodrefConstant(Clazz clazz, AnyMethodrefConstant anyMethodrefConstant)
{
Method referencedMethod = (Method)anyMethodrefConstant.referencedMethod;
// Mark reference reference values that are passed to the method.
for (int index = 0; index < referencingPopCount; index++)
{
int stackEntryIndex = referencingPopCount - index - 1;
TracedStack stackBefore = partialEvaluator.getStackBefore(referencingOffset);
Value stackEntry = stackBefore.getTop(stackEntryIndex);
if (stackEntry.computationalType() == Value.TYPE_REFERENCE)
{
// Is the parameter escaping from the referenced method?
if (referencedMethod == null ||
ParameterEscapeMarker.isParameterEscaping(referencedMethod, index))
{
markEscapingReferenceValues(referencingOffset,
stackEntryIndex);
}
// Is the parameter being modified in the referenced method?
if (referencedMethod == null ||
ParameterEscapeMarker.isParameterModified(referencedMethod, index))
{
markModifiedReferenceValues(referencingOffset,
stackEntryIndex);
}
}
}
// Is the return value from the referenced method external?
String returnType =
ClassUtil.internalMethodReturnType(anyMethodrefConstant.getType(clazz));
if (referencedMethod == null ||
((ClassUtil.isInternalClassType(returnType) ||
ClassUtil.isInternalArrayType(returnType)) &&
ParameterEscapeMarker.returnsExternalValues(referencedMethod)))
{
markExternalReferenceValue(referencingOffset);
}
}
// Small utility methods.
/**
* Marks the producing offsets of the specified stack entry at the given
* instruction offset.
*/
private void markEscapingReferenceValues(int instructionOffset,
int stackEntryIndex)
{
TracedStack stackBefore = partialEvaluator.getStackBefore(instructionOffset);
Value stackEntry = stackBefore.getTop(stackEntryIndex);
if (stackEntry.computationalType() == Value.TYPE_REFERENCE)
{
ReferenceValue referenceValue = stackEntry.referenceValue();
// The null reference value may not have a trace value.
if (referenceValue.isNull() != Value.ALWAYS)
{
markEscapingReferenceValues(referenceValue);
}
}
}
/**
* Marks the producing offsets of the given traced reference value.
*/
private void markEscapingReferenceValues(ReferenceValue referenceValue)
{
TracedReferenceValue tracedReferenceValue = (TracedReferenceValue)referenceValue;
InstructionOffsetValue instructionOffsetValue = tracedReferenceValue.getTraceValue().instructionOffsetValue();
int parameterCount = instructionOffsetValue.instructionOffsetCount();
for (int index = 0; index < parameterCount; index++)
{
if (!instructionOffsetValue.isMethodParameter(index))
{
instanceEscaping[instructionOffsetValue.instructionOffset(index)] = true;
}
}
}
/**
* Marks the producing offsets of the specified stack entry at the given
* instruction offset.
*/
private void markReturnedReferenceValues(int instructionOffset,
int stackEntryIndex)
{
TracedStack stackBefore = partialEvaluator.getStackBefore(instructionOffset);
ReferenceValue referenceValue = stackBefore.getTop(stackEntryIndex).referenceValue();
// The null reference value may not have a trace value.
if (referenceValue.isNull() != Value.ALWAYS)
{
markReturnedReferenceValues(referenceValue);
}
}
/**
* Marks the producing offsets of the given traced reference value.
*/
private void markReturnedReferenceValues(ReferenceValue referenceValue)
{
TracedReferenceValue tracedReferenceValue = (TracedReferenceValue)referenceValue;
InstructionOffsetValue instructionOffsetValue = tracedReferenceValue.getTraceValue().instructionOffsetValue();
int parameterCount = instructionOffsetValue.instructionOffsetCount();
for (int index = 0; index < parameterCount; index++)
{
if (!instructionOffsetValue.isMethodParameter(index))
{
instanceReturned[instructionOffsetValue.instructionOffset(index)] = true;
}
}
}
/**
* Marks the producing offsets of the specified stack entry at the given
* instruction offset.
*/
private void markModifiedReferenceValues(int instructionOffset,
int stackEntryIndex)
{
TracedStack stackBefore = partialEvaluator.getStackBefore(instructionOffset);
ReferenceValue referenceValue = stackBefore.getTop(stackEntryIndex).referenceValue();
// The null reference value may not have a trace value.
if (referenceValue.isNull() != Value.ALWAYS)
{
markModifiedReferenceValues(referenceValue);
}
}
/**
* Marks the producing offsets of the given traced reference value.
*/
private void markModifiedReferenceValues(ReferenceValue referenceValue)
{
TracedReferenceValue tracedReferenceValue = (TracedReferenceValue)referenceValue;
InstructionOffsetValue instructionOffsetValue = tracedReferenceValue.getTraceValue().instructionOffsetValue();
int parameterCount = instructionOffsetValue.instructionOffsetCount();
for (int index = 0; index < parameterCount; index++)
{
if (!instructionOffsetValue.isMethodParameter(index))
{
instanceModified[instructionOffsetValue.instructionOffset(index)] = true;
}
}
}
/**
* Marks the producing offsets of the specified stack entry at the given
* instruction offset.
*/
private void markExternalReferenceValue(int offset)
{
externalInstance[offset] = true;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy