|  | // Copyright (c) 2018, 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.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.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.conversion.CfSourceCode; | 
|  | import com.android.tools.r8.ir.conversion.CfState; | 
|  | import com.android.tools.r8.ir.conversion.CfState.Slot; | 
|  | 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 it.unimi.dsi.fastutil.ints.IntArrayList; | 
|  | import java.util.List; | 
|  | import org.objectweb.asm.Label; | 
|  | import org.objectweb.asm.MethodVisitor; | 
|  | import org.objectweb.asm.Opcodes; | 
|  |  | 
|  | public class CfSwitch extends CfInstruction { | 
|  |  | 
|  | public enum Kind { LOOKUP, TABLE } | 
|  |  | 
|  | private final Kind kind; | 
|  | private final CfLabel defaultTarget; | 
|  | private final int[] keys; | 
|  | private final List<CfLabel> targets; | 
|  |  | 
|  | public CfSwitch(Kind kind, CfLabel defaultTarget, int[] keys, List<CfLabel> targets) { | 
|  | this.kind = kind; | 
|  | this.defaultTarget = defaultTarget; | 
|  | this.keys = keys; | 
|  | this.targets = targets; | 
|  | assert kind != Kind.LOOKUP || keys.length == targets.size(); | 
|  | assert kind != Kind.TABLE || keys.length == 1; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int getCompareToId() { | 
|  | return kind == Kind.LOOKUP ? Opcodes.LOOKUPSWITCH : Opcodes.TABLESWITCH; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int internalAcceptCompareTo( | 
|  | CfInstruction other, CompareToVisitor visitor, CfCompareHelper helper) { | 
|  | assert kind == ((CfSwitch) other).kind; | 
|  | return visitor.visit( | 
|  | this, | 
|  | (CfSwitch) other, | 
|  | spec -> | 
|  | spec.withCustomItem(CfSwitch::getDefaultTarget, helper.labelAcceptor()) | 
|  | .withIntArray(i -> i.keys) | 
|  | .withCustomItemCollection(CfSwitch::getSwitchTargets, helper.labelAcceptor())); | 
|  | } | 
|  |  | 
|  | public Kind getKind() { | 
|  | return kind; | 
|  | } | 
|  |  | 
|  | public CfLabel getDefaultTarget() { | 
|  | return defaultTarget; | 
|  | } | 
|  |  | 
|  | public List<Integer> getKeys() { | 
|  | return new IntArrayList(keys); | 
|  | } | 
|  |  | 
|  | public List<CfLabel> getSwitchTargets() { | 
|  | return targets; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public CfSwitch asSwitch() { | 
|  | return this; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isSwitch() { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isJump() { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void write( | 
|  | AppView<?> appView, | 
|  | ProgramMethod context, | 
|  | DexItemFactory dexItemFactory, | 
|  | GraphLens graphLens, | 
|  | InitClassLens initClassLens, | 
|  | NamingLens namingLens, | 
|  | LensCodeRewriterUtils rewriter, | 
|  | MethodVisitor visitor) { | 
|  | Label[] labels = new Label[targets.size()]; | 
|  | for (int i = 0; i < targets.size(); i++) { | 
|  | labels[i] = targets.get(i).getLabel(); | 
|  | } | 
|  | switch (kind) { | 
|  | case LOOKUP: | 
|  | visitor.visitLookupSwitchInsn(defaultTarget.getLabel(), keys, labels); | 
|  | break; | 
|  | case TABLE: { | 
|  | int min = keys[0]; | 
|  | int max = min + targets.size() - 1; | 
|  | visitor.visitTableSwitchInsn(min, max, defaultTarget.getLabel(), labels); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void print(CfPrinter printer) { | 
|  | printer.print(this); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void buildIR(IRBuilder builder, CfState state, CfSourceCode code) { | 
|  | int[] labelOffsets = new int[targets.size()]; | 
|  | for (int i = 0; i < targets.size(); i++) { | 
|  | labelOffsets[i] = code.getLabelOffset(targets.get(i)); | 
|  | } | 
|  | Slot value = state.pop(); | 
|  | builder.addSwitch(value.register, keys, code.getLabelOffset(defaultTarget), labelOffsets); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public ConstraintWithTarget inliningConstraint( | 
|  | InliningConstraints inliningConstraints, CfCode code, ProgramMethod context) { | 
|  | return inliningConstraints.forJumpInstruction(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void evaluate( | 
|  | CfFrameVerificationHelper frameBuilder, | 
|  | DexType context, | 
|  | DexType returnType, | 
|  | DexItemFactory factory, | 
|  | InitClassLens initClassLens) { | 
|  | // ..., index/key → | 
|  | // ... | 
|  | frameBuilder.popInitialized(factory.intType); | 
|  | frameBuilder.checkTarget(defaultTarget); | 
|  | for (CfLabel target : targets) { | 
|  | frameBuilder.checkTarget(target); | 
|  | } | 
|  | frameBuilder.setNoFrame(); | 
|  | } | 
|  | } |