| // 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.dex.code.DexArrayLength; |
| import com.android.tools.r8.graph.AppView; |
| import com.android.tools.r8.graph.ProgramMethod; |
| 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.conversion.MethodConversionOptions; |
| 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; |
| import com.android.tools.r8.lightir.LirBuilder; |
| |
| 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 DexArrayLength(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 boolean instructionInstanceCanThrow(AppView<?> appView, ProgramMethod context) { |
| return array().type.isNullable(); |
| } |
| |
| @Override |
| public boolean instructionMayHaveSideEffects( |
| AppView<?> appView, ProgramMethod context, SideEffectAssumption assumption) { |
| return instructionInstanceCanThrow(appView, context); |
| } |
| |
| @Override |
| public boolean identicalAfterRegisterAllocation( |
| Instruction other, RegisterAllocator allocator, MethodConversionOptions conversionOptions) { |
| if (super.identicalAfterRegisterAllocation(other, allocator, conversionOptions)) { |
| // 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, ProgramMethod context) { |
| 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(), this); |
| } |
| |
| @Override |
| public TypeElement evaluate(AppView<?> appView) { |
| return TypeElement.getInt(); |
| } |
| |
| @Override |
| public boolean hasInvariantOutType() { |
| return true; |
| } |
| |
| @Override |
| public boolean throwsNpeIfValueIsNull(Value value, AppView<?> appView, ProgramMethod context) { |
| return array() == value; |
| } |
| |
| @Override |
| public boolean throwsOnNullInput() { |
| return true; |
| } |
| |
| @Override |
| public Value getNonNullInput() { |
| return array(); |
| } |
| |
| @Override |
| public boolean instructionMayTriggerMethodInvocation(AppView<?> appView, ProgramMethod context) { |
| return false; |
| } |
| |
| @Override |
| public void buildLir(LirBuilder<Value, BasicBlock> builder) { |
| builder.addArrayLength(array()); |
| } |
| } |