| // 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.code; |
| |
| import com.android.tools.r8.dex.Constants; |
| import com.android.tools.r8.dex.IndexedItemCollection; |
| import com.android.tools.r8.graph.DexMethod; |
| import com.android.tools.r8.graph.DexProto; |
| import com.android.tools.r8.graph.GraphLens; |
| import com.android.tools.r8.graph.GraphLens.MethodLookupResult; |
| import com.android.tools.r8.graph.IndexedDexItem; |
| import com.android.tools.r8.graph.ObjectToOffsetMapping; |
| import com.android.tools.r8.graph.ProgramMethod; |
| import com.android.tools.r8.ir.code.Invoke.Type; |
| import com.android.tools.r8.ir.conversion.LensCodeRewriterUtils; |
| import com.android.tools.r8.naming.ClassNameMapper; |
| import com.android.tools.r8.utils.structural.CompareToVisitor; |
| import com.android.tools.r8.utils.structural.StructuralSpecification; |
| import java.nio.ShortBuffer; |
| import java.util.function.BiPredicate; |
| |
| /** Format4rcc for instructions of size 4, with a range of registers and 2 constant pool index. */ |
| public abstract class Format4rcc extends Base4Format { |
| |
| public final short AA; |
| public final char CCCC; |
| public DexMethod BBBB; |
| public DexProto HHHH; |
| |
| private static void specify(StructuralSpecification<Format4rcc, ?> spec) { |
| spec.withInt(i -> i.AA).withInt(i -> i.CCCC).withItem(i -> i.BBBB).withItem(i -> i.HHHH); |
| } |
| |
| // AA | op | [meth]@BBBB | CCCC | [proto]@HHHH |
| Format4rcc(int high, BytecodeStream stream, DexMethod[] methodMap, DexProto[] protoMap) { |
| super(stream); |
| this.AA = (short) high; |
| this.BBBB = methodMap[read16BitValue(stream)]; |
| this.CCCC = read16BitValue(stream); |
| this.HHHH = protoMap[read16BitValue(stream)]; |
| } |
| |
| Format4rcc(int firstArgumentRegister, int argumentCount, DexMethod method, DexProto proto) { |
| assert 0 <= firstArgumentRegister && firstArgumentRegister <= Constants.U16BIT_MAX; |
| assert 0 <= argumentCount && argumentCount <= Constants.U8BIT_MAX; |
| this.CCCC = (char) firstArgumentRegister; |
| this.AA = (short) argumentCount; |
| BBBB = method; |
| HHHH = proto; |
| } |
| |
| @Override |
| public void write( |
| ShortBuffer dest, |
| ProgramMethod context, |
| GraphLens graphLens, |
| ObjectToOffsetMapping mapping, |
| LensCodeRewriterUtils rewriter) { |
| MethodLookupResult lookup = |
| graphLens.lookupMethod(getMethod(), context.getReference(), Type.POLYMORPHIC); |
| assert lookup.getType() == Type.POLYMORPHIC; |
| writeFirst(AA, dest); |
| write16BitReference(lookup.getReference(), dest, mapping); |
| write16BitValue(CCCC, dest); |
| |
| DexProto rewrittenProto = rewriter.rewriteProto(getProto()); |
| write16BitReference(rewrittenProto, dest, mapping); |
| } |
| |
| @Override |
| public final int hashCode() { |
| return ((CCCC << 24) | (HHHH.hashCode() << 12) | (BBBB.hashCode() << 4) | AA) |
| ^ getClass().hashCode(); |
| } |
| |
| @Override |
| final int internalAcceptCompareTo(Instruction other, CompareToVisitor visitor) { |
| return visitor.visit(this, (Format4rcc) other, Format4rcc::specify); |
| } |
| |
| @Override |
| public String toString(ClassNameMapper naming) { |
| StringBuilder builder = new StringBuilder(); |
| appendRegisterRange(builder); |
| builder.append(" "); |
| if (naming == null) { |
| builder.append(BBBB.toSmaliString()); |
| } else { |
| builder.append(naming.originalNameOf(BBBB)); |
| } |
| if (naming == null) { |
| builder.append(HHHH.toSmaliString()); |
| } else { |
| builder.append(naming.originalNameOf(HHHH)); |
| } |
| return formatString(builder.toString()); |
| } |
| |
| @Override |
| public String toSmaliString(ClassNameMapper naming) { |
| StringBuilder builder = new StringBuilder(); |
| appendRegisterRange(builder); |
| builder.append(", "); |
| // TODO(sgjesse): Add support for smali name mapping. |
| builder.append(BBBB.toSmaliString()); |
| builder.append(", "); |
| builder.append(HHHH.toSmaliString()); |
| return formatSmaliString(builder.toString()); |
| } |
| |
| @Override |
| public void collectIndexedItems( |
| IndexedItemCollection indexedItems, |
| ProgramMethod context, |
| GraphLens graphLens, |
| LensCodeRewriterUtils rewriter) { |
| MethodLookupResult lookup = |
| graphLens.lookupMethod(getMethod(), context.getReference(), Type.POLYMORPHIC); |
| assert lookup.getType() == Type.POLYMORPHIC; |
| lookup.getReference().collectIndexedItems(indexedItems); |
| |
| DexProto rewrittenProto = rewriter.rewriteProto(getProto()); |
| rewrittenProto.collectIndexedItems(indexedItems); |
| } |
| |
| @Override |
| public boolean equals(Instruction other, BiPredicate<IndexedDexItem, IndexedDexItem> equality) { |
| if (other == null || (this.getClass() != other.getClass())) { |
| return false; |
| } |
| Format4rcc o = (Format4rcc) other; |
| return o.AA == AA |
| && o.CCCC == CCCC |
| && equality.test(BBBB, o.BBBB) |
| && equality.test(HHHH, o.HHHH); |
| } |
| |
| private void appendRegisterRange(StringBuilder builder) { |
| int firstRegister = CCCC; |
| builder.append("{ "); |
| builder.append("v").append(firstRegister); |
| if (AA != 1) { |
| builder.append(" .. v").append(firstRegister + AA - 1); |
| } |
| builder.append(" }"); |
| } |
| |
| @Override |
| public DexMethod getMethod() { |
| return BBBB; |
| } |
| |
| @Override |
| public DexProto getProto() { |
| return HHHH; |
| } |
| } |