blob: dc0b13343c6b8d642c7aaf3441e832ecfd6b46f4 [file] [log] [blame]
// Copyright (c) 2016, 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.code;
import com.android.tools.r8.cf.LoadStoreHelper;
import com.android.tools.r8.cf.code.CfArrayLength;
import com.android.tools.r8.dex.Constants;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.ir.analysis.AbstractError;
import com.android.tools.r8.ir.analysis.type.TypeElement;
import com.android.tools.r8.ir.conversion.CfBuilder;
import com.android.tools.r8.ir.conversion.DexBuilder;
import com.android.tools.r8.ir.optimize.Inliner.ConstraintWithTarget;
import com.android.tools.r8.ir.optimize.InliningConstraints;
import com.android.tools.r8.ir.regalloc.RegisterAllocator;
public class ArrayLength extends Instruction {
public ArrayLength(Value dest, Value array) {
super(dest, array);
}
@Override
public int opcode() {
return Opcodes.ARRAY_LENGTH;
}
@Override
public <T> T accept(InstructionVisitor<T> visitor) {
return visitor.visit(this);
}
public Value dest() {
return outValue;
}
public Value array() {
return inValues.get(0);
}
@Override
public boolean isArrayLength() {
return true;
}
@Override
public ArrayLength asArrayLength() {
return this;
}
@Override
public void buildDex(DexBuilder builder) {
int dest = builder.allocatedRegister(dest(), getNumber());
int array = builder.allocatedRegister(array(), getNumber());
builder.add(this, new com.android.tools.r8.code.ArrayLength(dest, array));
}
@Override
public int maxInValueRegister() {
return Constants.U4BIT_MAX;
}
@Override
public int maxOutValueRegister() {
return Constants.U4BIT_MAX;
}
@Override
public boolean instructionTypeCanThrow() {
return true;
}
@Override
public AbstractError instructionInstanceCanThrow(AppView<?> appView, DexType context) {
if (array().type.isNullable()) {
return AbstractError.specific(appView.dexItemFactory().npeType);
}
return AbstractError.bottom();
}
@Override
public boolean instructionMayHaveSideEffects(
AppView<?> appView, DexType context, SideEffectAssumption assumption) {
return instructionInstanceCanThrow(appView, context).isThrowing();
}
@Override
public boolean canBeDeadCode(AppView<?> appView, IRCode code) {
return !instructionMayHaveSideEffects(appView, code.method.method.holder);
}
@Override
public boolean identicalAfterRegisterAllocation(Instruction other, RegisterAllocator allocator) {
if (super.identicalAfterRegisterAllocation(other, allocator)) {
// The array length instruction doesn't carry the element type. The art verifier doesn't
// allow an array length instruction into which arrays of two different base types can
// flow. Therefore, as a safe approximation we only consider array length instructions
// equal when they have the same inflowing SSA value.
// TODO(ager): We could perform conservative type propagation earlier in the pipeline and
// add a member type to array length instructions.
return array() == other.asArrayLength().array();
}
return false;
}
@Override
public boolean identicalNonValueNonPositionParts(Instruction other) {
return other.isArrayLength();
}
@Override
public ConstraintWithTarget inliningConstraint(
InliningConstraints inliningConstraints, DexType invocationContext) {
return inliningConstraints.forArrayLength();
}
@Override
public void insertLoadAndStores(InstructionListIterator it, LoadStoreHelper helper) {
helper.loadInValues(this, it);
helper.storeOutValue(this, it);
}
@Override
public void buildCf(CfBuilder builder) {
builder.add(new CfArrayLength());
}
@Override
public TypeElement evaluate(AppView<?> appView) {
return TypeElement.getInt();
}
@Override
public boolean hasInvariantOutType() {
return true;
}
@Override
public boolean throwsNpeIfValueIsNull(Value value, AppView<?> appView, DexType context) {
return array() == value;
}
@Override
public boolean throwsOnNullInput() {
return true;
}
@Override
public Value getNonNullInput() {
return array();
}
@Override
public boolean instructionMayTriggerMethodInvocation(AppView<?> appView, DexType context) {
return false;
}
}