blob: 47f01c130f4784d613e7cde4ca22e9d1800d7dd6 [file] [log] [blame]
// Copyright (c) 2021, the R8 project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.optimize.argumentpropagation.reprocessingcriteria;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.graph.ProgramMethod;
import com.android.tools.r8.ir.analysis.type.DynamicType;
import com.android.tools.r8.ir.analysis.type.Nullability;
import com.android.tools.r8.ir.optimize.info.ConcreteCallSiteOptimizationInfo;
import com.android.tools.r8.optimize.argumentpropagation.utils.WideningUtils;
import com.android.tools.r8.shaking.AppInfoWithLiveness;
/**
* Represents that a parameter should be reprocessed under certain conditions if we have non-trivial
* information about it (e.g., abstract value, dynamic type, nullability).
*
* <p>Example: If we determine that a parameter should not be reprocessed if we only have
* non-trivial information about its dynamic type, then an instance of this class is used which
* returns false in {@link #shouldReprocessDueToDynamicType()}.
*/
public class NonTrivialParameterReprocessingCriteria extends ParameterReprocessingCriteria {
public NonTrivialParameterReprocessingCriteria(boolean reprocessDueToDynamicType) {
assert !reprocessDueToDynamicType;
}
@Override
public boolean shouldReprocess(
AppView<AppInfoWithLiveness> appView,
ProgramMethod method,
ConcreteCallSiteOptimizationInfo methodState,
int parameterIndex,
DexType parameterType) {
if (parameterType.isReferenceType()) {
return shouldReprocessReferenceParameter(
appView, method, methodState, parameterIndex, parameterType);
} else {
assert parameterType.isPrimitiveType();
return shouldReprocessPrimitiveParameter(methodState, parameterIndex);
}
}
@Override
public boolean shouldReprocessDueToAbstractValue() {
return true;
}
@Override
public boolean shouldReprocessDueToDynamicType() {
return false;
}
@Override
public boolean shouldReprocessDueToNullability() {
return true;
}
private boolean shouldReprocessPrimitiveParameter(
ConcreteCallSiteOptimizationInfo methodState, int parameterIndex) {
return methodState.getAbstractArgumentValue(parameterIndex).isNonTrivial();
}
private boolean shouldReprocessReferenceParameter(
AppView<AppInfoWithLiveness> appView,
ProgramMethod method,
ConcreteCallSiteOptimizationInfo methodState,
int parameterIndex,
DexType parameterType) {
if (shouldReprocessDueToAbstractValue()
&& !methodState.getAbstractArgumentValue(parameterIndex).isUnknown()) {
return true;
}
if (shouldReprocessDueToDynamicType()) {
DynamicType widenedDynamicType =
WideningUtils.widenDynamicNonReceiverType(
appView,
methodState.getDynamicType(parameterIndex).withNullability(Nullability.maybeNull()),
parameterType);
if (!widenedDynamicType.isUnknown()) {
return true;
}
}
boolean isReceiverParameter = parameterIndex == 0 && method.getDefinition().isInstance();
if (shouldReprocessDueToNullability()
&& !isReceiverParameter
&& !methodState.getNullability(parameterIndex).isUnknown()) {
return true;
}
return false;
}
}