blob: 4da5b404a4c29a47eaff4e34582495d7bb69db06 [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.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;
}
}