blob: 2340a8929977df13586fff3d46297a305cc7f691 [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.cf.code;
import com.android.tools.r8.cf.CfPrinter;
import com.android.tools.r8.errors.Unreachable;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.CfCode;
import com.android.tools.r8.graph.CfCompareHelper;
import com.android.tools.r8.graph.DexItemFactory;
import com.android.tools.r8.graph.DexMethod;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.graph.GraphLens;
import com.android.tools.r8.graph.InitClassLens;
import com.android.tools.r8.graph.ProgramMethod;
import com.android.tools.r8.ir.code.If;
import com.android.tools.r8.ir.code.If.Type;
import com.android.tools.r8.ir.code.ValueType;
import com.android.tools.r8.ir.conversion.CfSourceCode;
import com.android.tools.r8.ir.conversion.CfState;
import com.android.tools.r8.ir.conversion.IRBuilder;
import com.android.tools.r8.ir.conversion.LensCodeRewriterUtils;
import com.android.tools.r8.ir.optimize.Inliner.ConstraintWithTarget;
import com.android.tools.r8.ir.optimize.InliningConstraints;
import com.android.tools.r8.naming.NamingLens;
import com.android.tools.r8.utils.structural.CompareToVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
public class CfIfCmp extends CfInstruction {
private final If.Type kind;
private final ValueType type;
private final CfLabel target;
public CfIfCmp(If.Type kind, ValueType type, CfLabel target) {
this.kind = kind;
this.type = type;
this.target = target;
}
@Override
public int getCompareToId() {
return getOpcode();
}
@Override
public int internalAcceptCompareTo(
CfInstruction other, CompareToVisitor visitor, CfCompareHelper helper) {
CfIfCmp otherIf = (CfIfCmp) other;
assert kind == otherIf.kind;
assert type == otherIf.type;
return helper.compareLabels(target, otherIf.target, visitor);
}
public Type getKind() {
return kind;
}
public ValueType getType() {
return type;
}
@Override
public CfLabel getTarget() {
return target;
}
public int getOpcode() {
switch (kind) {
case EQ:
return type.isObject() ? Opcodes.IF_ACMPEQ : Opcodes.IF_ICMPEQ;
case GE:
return Opcodes.IF_ICMPGE;
case GT:
return Opcodes.IF_ICMPGT;
case LE:
return Opcodes.IF_ICMPLE;
case LT:
return Opcodes.IF_ICMPLT;
case NE:
return type.isObject() ? Opcodes.IF_ACMPNE : Opcodes.IF_ICMPNE;
default:
throw new Unreachable("Unexpected type " + type);
}
}
@Override
public void print(CfPrinter printer) {
printer.print(this);
}
@Override
public void write(
AppView<?> appView,
ProgramMethod context,
DexItemFactory dexItemFactory,
GraphLens graphLens,
InitClassLens initClassLens,
NamingLens namingLens,
LensCodeRewriterUtils rewriter,
MethodVisitor visitor) {
visitor.visitJumpInsn(getOpcode(), target.getLabel());
}
@Override
public boolean isConditionalJump() {
return true;
}
@Override
public boolean isJump() {
return true;
}
@Override
public void buildIR(IRBuilder builder, CfState state, CfSourceCode code) {
int right = state.pop().register;
int left = state.pop().register;
int trueTargetOffset = code.getLabelOffset(target);
int falseTargetOffset = code.getCurrentInstructionIndex() + 1;
builder.addIf(kind, type, left, right, trueTargetOffset, falseTargetOffset);
}
@Override
public ConstraintWithTarget inliningConstraint(
InliningConstraints inliningConstraints, CfCode code, ProgramMethod context) {
return inliningConstraints.forJumpInstruction();
}
@Override
public void evaluate(
CfFrameVerificationHelper frameBuilder,
DexMethod context,
AppView<?> appView,
DexItemFactory dexItemFactory) {
// ..., value1, value2 →
// ...
DexType type =
this.type.isObject()
? dexItemFactory.objectType
: this.type.toPrimitiveType().toDexType(dexItemFactory);
frameBuilder.popAndDiscardInitialized(type, type);
frameBuilder.checkTarget(target);
}
}