blob: bc9b95e6a6122ab69885a9e059afe18083304a2f [file] [log] [blame]
// Copyright (c) 2017, 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.ir.optimize;
import com.android.tools.r8.graph.AppInfoWithSubtyping;
import com.android.tools.r8.graph.DexCode;
import com.android.tools.r8.graph.DexEncodedMethod;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.graph.GraphLense;
import com.android.tools.r8.ir.code.BasicBlock;
import com.android.tools.r8.ir.code.DominatorTree;
import com.android.tools.r8.ir.code.IRCode;
import com.android.tools.r8.ir.code.Instruction;
import com.android.tools.r8.ir.code.InstructionIterator;
import com.android.tools.r8.ir.code.InstructionListIterator;
import com.android.tools.r8.ir.code.Invoke;
import com.android.tools.r8.ir.code.InvokeMethod;
import com.android.tools.r8.ir.code.Value;
import com.android.tools.r8.ir.code.ValueNumberGenerator;
import com.android.tools.r8.ir.conversion.CallGraph;
import com.android.tools.r8.ir.conversion.LensCodeRewriter;
import com.android.tools.r8.logging.Log;
import com.android.tools.r8.utils.InternalOptions;
import com.google.common.collect.Sets;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Set;
public class Inliner {
private static final int INLINING_INSTRUCTION_LIMIT = 5;
protected final AppInfoWithSubtyping appInfo;
private final GraphLense graphLense;
private final InternalOptions options;
// State for inlining methods which are known to be called twice.
public boolean applyDoubleInlining = false;
public final Set<DexEncodedMethod> doubleInlineCallers = Sets.newIdentityHashSet();
public final Set<DexEncodedMethod> doubleInlineSelectedTargets = Sets.newIdentityHashSet();
public final Map<DexEncodedMethod, DexEncodedMethod> doubleInlineeCandidates = new HashMap<>();
public Inliner(AppInfoWithSubtyping appInfo, GraphLense graphLense, InternalOptions options) {
this.appInfo = appInfo;
this.graphLense = graphLense;
this.options = options;
}
private InliningConstraint instructionAllowedForInlining(
DexEncodedMethod method, Instruction instruction) {
InliningConstraint result = instruction.inliningConstraint(appInfo, method.method.holder);
if ((result == InliningConstraint.NEVER) && instruction.isDebugInstruction()) {
return InliningConstraint.ALWAYS;
}
return result;
}
public InliningConstraint identifySimpleMethods(IRCode code, DexEncodedMethod method) {
DexCode dex = method.getCode().asDexCode();
// We have generated code for a method and we want to figure out whether the method is a
// candidate for inlining. The code is the final IR after optimizations.
if (dex.instructions.length > INLINING_INSTRUCTION_LIMIT) {
return InliningConstraint.NEVER;
}
InliningConstraint result = InliningConstraint.ALWAYS;
ListIterator<BasicBlock> iterator = code.listIterator();
assert iterator.hasNext();
BasicBlock block = iterator.next();
BasicBlock nextBlock;
do {
nextBlock = iterator.hasNext() ? iterator.next() : null;
InstructionListIterator it = block.listIterator();
while (it.hasNext()) {
Instruction instruction = it.next();
InliningConstraint state = instructionAllowedForInlining(method, instruction);
if (state == InliningConstraint.NEVER) {
return InliningConstraint.NEVER;
}
if (state.ordinal() < result.ordinal()) {
result = state;
}
}
block = nextBlock;
} while (block != null);
return result;
}
protected boolean hasInliningAccess(DexEncodedMethod method, DexEncodedMethod target) {
if (target.accessFlags.isPublic()) {
return true;
}
DexType methodHolder = method.method.getHolder();
DexType targetHolder = target.method.getHolder();
if (target.accessFlags.isPrivate()) {
return methodHolder == targetHolder;
}
if (target.accessFlags.isProtected() &&
methodHolder.isSubtypeOf(targetHolder, appInfo)) {
return true;
}
return methodHolder.isSamePackage(targetHolder);
}
public enum InliningConstraint {
// The ordinal values are important so please do not reorder.
NEVER, // Never inline this.
PRIVATE, // Only inline this into methods with same holder.
PACKAGE, // Only inline this into methods with holders from same package.
ALWAYS, // No restrictions for inlining this.
}
static public class InlineAction {
public final DexEncodedMethod target;
public final Invoke invoke;
public final boolean forceInline;
public InlineAction(
DexEncodedMethod target, Invoke invoke, boolean forceInline) {
this.target = target;
this.invoke = invoke;
this.forceInline = forceInline;
}
public InlineAction(DexEncodedMethod target, Invoke invoke) {
this.target = target;
this.invoke = invoke;
this.forceInline = target.getOptimizationInfo().forceInline();
}
public IRCode buildIR(ValueNumberGenerator generator, AppInfoWithSubtyping appInfo,
GraphLense graphLense, InternalOptions options) {
if (target.isProcessed()) {
assert target.getCode().isDexCode();
return target.buildIR(generator, options);
} else {
// Build the IR for a yet not processed method, and perform minimal IR processing.
IRCode code;
if (target.getCode().isJarCode()) {
code = target.getCode().asJarCode().buildIR(target, generator, options);
} else {
code = target.getCode().asDexCode().buildIR(target, generator, options);
}
new LensCodeRewriter(graphLense, appInfo).rewrite(code, target);
return code;
}
}
}
private int numberOfInstructions(IRCode code) {
int numOfInstructions = 0;
for (BasicBlock block : code.blocks) {
numOfInstructions += block.getInstructions().size();
}
return numOfInstructions;
}
private boolean legalConstructorInline(DexEncodedMethod method, IRCode code) {
// In the Java VM Specification section "4.10.2.4. Instance Initialization Methods and
// Newly Created Objects" it says:
//
// Before that method invokes another instance initialization method of myClass or its direct
// superclass on this, the only operation the method can perform on this is assigning fields
// declared within myClass.
//
// Allow inlining a constructor into a constructor, as the constructor code is expected to
// adhere to the VM specification.
if (method.accessFlags.isConstructor()) {
return true;
}
// Don't allow inlining a constructor into a non-constructor if the first use of the
// un-initialized object is not an argument of an invoke of <init>.
InstructionIterator iterator = code.instructionIterator();
Instruction instruction = iterator.next();
// A constructor always has the un-initialized object as the first argument.
assert instruction.isArgument();
Value unInitializedObject = instruction.outValue();
while (iterator.hasNext()) {
instruction = iterator.next();
if (instruction.inValues().contains(unInitializedObject)) {
return instruction.isInvokeDirect()
&& appInfo.dexItemFactory
.isConstructor(instruction.asInvokeDirect().getInvokedMethod());
}
}
assert false : "Execution should never reach this point";
return false;
}
/// Computer the receiver value for the holder method.
private Value receiverValue(DexEncodedMethod method, IRCode code) {
// Ignore static methods.
if (method.accessFlags.isStatic()) {
return null;
}
// Find the outValue of the first argument instruction in the first block.
return code.collectArguments().get(0);
}
public void performInlining(DexEncodedMethod method, IRCode code, CallGraph callGraph) {
int instruction_allowance = 1500;
instruction_allowance -= numberOfInstructions(code);
if (instruction_allowance < 0) {
return;
}
computeReceiverMustBeNonNull(code);
Value receiver = receiverValue(method, code);
InliningOracle oracle = new InliningOracle(this, method, receiver, callGraph);
List<BasicBlock> blocksToRemove = new ArrayList<>();
ListIterator<BasicBlock> blockIterator = code.listIterator();
while (blockIterator.hasNext() && (instruction_allowance >= 0)) {
BasicBlock block = blockIterator.next();
if (blocksToRemove.contains(block)) {
continue;
}
InstructionListIterator iterator = block.listIterator();
while (iterator.hasNext() && (instruction_allowance >= 0)) {
Instruction current = iterator.next();
if (current.isInvokeMethod()) {
InvokeMethod invoke = current.asInvokeMethod();
InlineAction result = invoke.computeInlining(oracle);
if (result != null) {
DexEncodedMethod target = appInfo.lookup(invoke.getType(), invoke.getInvokedMethod());
if (target == null) {
// The declared target cannot be found so skip inlining.
continue;
}
boolean forceInline = target.getOptimizationInfo().forceInline();
if (!target.isProcessed() && !forceInline) {
// Do not inline code that was not processed unless we have to force inline.
continue;
}
IRCode inlinee = result
.buildIR(code.valueNumberGenerator, appInfo, graphLense, options);
if (inlinee != null) {
// TODO(sgjesse): Get rid of this additional check by improved inlining.
if (block.hasCatchHandlers() && inlinee.getNormalExitBlock() == null) {
continue;
}
// If this code did not go through the full pipeline, apply inlining to make sure
// that force inline targets get processed.
if (!target.isProcessed()) {
assert forceInline;
if (Log.ENABLED) {
Log.verbose(getClass(), "Forcing extra inline on " + target.toSourceString());
}
performInlining(target, inlinee, callGraph);
}
// Make sure constructor inlining is legal.
if (target.accessFlags.isConstructor() && !legalConstructorInline(method, inlinee)) {
continue;
}
DexType downcast = null;
if (invoke.isInvokeMethodWithReceiver()) {
// If the invoke has a receiver but the declared method holder is different
// from the computed target holder, inlining requires a downcast of the receiver.
if (result.target.method.getHolder() != target.method.getHolder()) {
downcast = result.target.method.getHolder();
}
}
// Inline the inlinee code in place of the invoke instruction
// Back up before the invoke instruction.
iterator.previous();
instruction_allowance -= numberOfInstructions(inlinee);
if (instruction_allowance >= 0 || result.forceInline) {
iterator.inlineInvoke(code, inlinee, blockIterator, blocksToRemove, downcast);
}
// If we inlined the invoke from a bridge method, it is no longer a bridge method.
if (method.accessFlags.isBridge()) {
method.accessFlags.unsetSynthetic();
method.accessFlags.unsetBridge();
}
}
}
}
}
}
oracle.finish();
code.removeBlocks(blocksToRemove);
assert code.isConsistentSSA();
}
// Determine whether the receiver of an invocation is guaranteed to be non-null based on
// the dominator tree. If a method call is dominated by another method call with the same
// receiver, the receiver most be non-null when we reach the dominated call.
//
// We bail out for exception handling. If an invoke is covered by a try block we cannot use
// dominance to determine that the receiver is non-null at a dominated call:
//
// Object o;
// try {
// o.m();
// } catch (NullPointerException e) {
// o.f(); // Dominated by other call with receiver o, but o is null.
// }
//
private void computeReceiverMustBeNonNull(IRCode code) {
DominatorTree dominatorTree = new DominatorTree(code);
InstructionIterator it = code.instructionIterator();
while (it.hasNext()) {
Instruction instruction = it.next();
if (instruction.isInvokeMethodWithReceiver()) {
Value receiverValue = instruction.inValues().get(0);
for (Instruction user : receiverValue.uniqueUsers()) {
if (user.isInvokeMethodWithReceiver() &&
user.inValues().get(0) == receiverValue &&
!user.getBlock().hasCatchHandlers() &&
dominatorTree.strictlyDominatedBy(instruction.getBlock(), user.getBlock())) {
instruction.asInvokeMethodWithReceiver().setIsDominatedByCallWithSameReceiver();
break;
}
}
}
}
}
}