Revert "[LIR] Strategy for indirect phi references."
This reverts commit 47e11df0f45dde13b7e5cc6bd836e8577036307e.
Reason for revert: Test failures
Change-Id: If7529c7260ba4181c66803b5dd945bc41c03e888
diff --git a/src/main/java/com/android/tools/r8/ir/conversion/IRConverter.java b/src/main/java/com/android/tools/r8/ir/conversion/IRConverter.java
index ffda264..e0d7bc0 100644
--- a/src/main/java/com/android/tools/r8/ir/conversion/IRConverter.java
+++ b/src/main/java/com/android/tools/r8/ir/conversion/IRConverter.java
@@ -65,8 +65,7 @@
import com.android.tools.r8.lightir.Lir2IRConverter;
import com.android.tools.r8.lightir.LirCode;
import com.android.tools.r8.lightir.LirStrategy;
-import com.android.tools.r8.lightir.LirStrategy.ExternalPhisStrategy;
-import com.android.tools.r8.lightir.PhiInInstructionsStrategy;
+import com.android.tools.r8.lightir.LirStrategy.PhiInInstructionsStrategy;
import com.android.tools.r8.naming.IdentifierNameStringMarker;
import com.android.tools.r8.optimize.argumentpropagation.ArgumentPropagatorIROptimizer;
import com.android.tools.r8.position.MethodPosition;
@@ -1084,18 +1083,12 @@
OptimizationFeedback feedback,
BytecodeMetadataProvider bytecodeMetadataProvider,
Timing timing) {
- IRCode round1 = doRoundtripWithStrategy(code, new ExternalPhisStrategy(), "indirect phis");
- IRCode round2 = doRoundtripWithStrategy(round1, new PhiInInstructionsStrategy(), "inline phis");
- return round2;
- }
-
- private <EV, S extends LirStrategy<Value, EV>> IRCode doRoundtripWithStrategy(
- IRCode code, S strategy, String name) {
- timing.begin("IR->LIR (" + name + ")");
- LirCode<EV> lirCode =
+ LirStrategy<Value, Integer> strategy = new PhiInInstructionsStrategy();
+ timing.begin("IR->LIR");
+ LirCode<Integer> lirCode =
IR2LirConverter.translate(code, strategy.getEncodingStrategy(), appView.dexItemFactory());
timing.end();
- timing.begin("LIR->IR (" + name + ")");
+ timing.begin("LIR->IR");
IRCode irCode =
Lir2IRConverter.translate(
code.context(), lirCode, strategy.getDecodingStrategy(lirCode), appView);
diff --git a/src/main/java/com/android/tools/r8/lightir/ByteUtils.java b/src/main/java/com/android/tools/r8/lightir/ByteUtils.java
index 5d34483..f719f11 100644
--- a/src/main/java/com/android/tools/r8/lightir/ByteUtils.java
+++ b/src/main/java/com/android/tools/r8/lightir/ByteUtils.java
@@ -51,15 +51,6 @@
return (value >= 0) && (value <= 0xFFFF);
}
- private static int truncateToU2(int value) {
- return value & 0xFFFF;
- }
-
- public static int ensureU2(int value) {
- assert isU2(value);
- return truncateToU2(value);
- }
-
public static int unsetBitAtIndex(int value, int index) {
return value & ~(1 << (index - 1));
}
diff --git a/src/main/java/com/android/tools/r8/lightir/IR2LirConverter.java b/src/main/java/com/android/tools/r8/lightir/IR2LirConverter.java
index ee53ceb..21af715 100644
--- a/src/main/java/com/android/tools/r8/lightir/IR2LirConverter.java
+++ b/src/main/java/com/android/tools/r8/lightir/IR2LirConverter.java
@@ -48,11 +48,6 @@
strategy.defineBlock(block, blockIndex);
}
- private boolean recordPhi(Phi phi, int valueIndex) {
- recordValue(phi, valueIndex);
- return strategy.isPhiInlineInstruction();
- }
-
private void recordValue(Value value, int valueIndex) {
EV encodedValue = strategy.defineValue(value, valueIndex);
if (value.hasLocalInfo()) {
@@ -74,8 +69,16 @@
BasicBlockIterator blockIt = irCode.listIterator();
while (blockIt.hasNext()) {
BasicBlock block = blockIt.next();
- if (strategy.isPhiInlineInstruction()) {
- currentValueIndex += computePhis(block);
+ if (block.hasPhis()) {
+ // The block order of the predecessors may change, since the LIR does not encode the
+ // direct links, the block order is used to determine predecessor order.
+ int[] permutation = computePermutation(block.getPredecessors(), strategy::getBlockIndex);
+ Value[] operands = new Value[block.getPredecessors().size()];
+ for (Phi phi : block.getPhis()) {
+ permuteOperands(phi.getOperands(), permutation, operands);
+ builder.addPhi(phi.getType(), Arrays.asList(operands));
+ currentValueIndex++;
+ }
}
if (block.hasCatchHandlers()) {
CatchHandlers<BasicBlock> handlers = block.getCatchHandlers();
@@ -109,25 +112,6 @@
}
assert builder.verifyCurrentValueIndex(currentValueIndex);
}
- if (!strategy.isPhiInlineInstruction()) {
- irCode.listIterator().forEachRemaining(this::computePhis);
- }
- }
-
- private int computePhis(BasicBlock block) {
- int valuesOffset = 0;
- if (block.hasPhis()) {
- // The block order of the predecessors may change, since the LIR does not encode the
- // direct links, the block order is used to determine predecessor order.
- int[] permutation = computePermutation(block.getPredecessors(), strategy::getBlockIndex);
- Value[] operands = new Value[block.getPredecessors().size()];
- for (Phi phi : block.getPhis()) {
- permuteOperands(phi.getOperands(), permutation, operands);
- builder.addPhi(phi.getType(), Arrays.asList(operands));
- valuesOffset++;
- }
- }
- return valuesOffset;
}
private void computeBlockAndValueTables() {
@@ -136,10 +120,9 @@
for (BasicBlock block : irCode.blocks) {
recordBlock(block, instructionIndex);
for (Phi phi : block.getPhis()) {
- if (recordPhi(phi, valueIndex)) {
- valueIndex++;
- instructionIndex++;
- }
+ recordValue(phi, valueIndex);
+ valueIndex++;
+ instructionIndex++;
}
for (Instruction instruction : block.getInstructions()) {
if (instruction.hasOutValue()) {
diff --git a/src/main/java/com/android/tools/r8/lightir/Lir2IRConverter.java b/src/main/java/com/android/tools/r8/lightir/Lir2IRConverter.java
index 21e634b..cc64c4e 100644
--- a/src/main/java/com/android/tools/r8/lightir/Lir2IRConverter.java
+++ b/src/main/java/com/android/tools/r8/lightir/Lir2IRConverter.java
@@ -4,6 +4,7 @@
package com.android.tools.r8.lightir;
import com.android.tools.r8.graph.AppView;
+import com.android.tools.r8.graph.DebugLocalInfo;
import com.android.tools.r8.graph.DexField;
import com.android.tools.r8.graph.DexMethod;
import com.android.tools.r8.graph.DexString;
@@ -57,7 +58,6 @@
AppView<?> appView) {
Parser<EV> parser = new Parser<>(lirCode, method.getReference(), appView, strategy);
parser.parseArguments(method);
- parser.ensureDebugInfo();
lirCode.forEach(view -> view.accept(parser));
return parser.getIRCode(method);
}
@@ -157,21 +157,6 @@
advanceNextPositionEntry();
}
- public void ensureDebugInfo() {
- if (code.getDebugLocalInfoTable() == null) {
- return;
- }
- code.getDebugLocalInfoTable()
- .forEachLocalDefinition(
- (encodedValue, localInfo) -> {
- Value value = getValue(encodedValue);
- if (!value.hasLocalInfo()) {
- value.setLocalInfo(localInfo);
- }
- assert value.getLocalInfo() == localInfo;
- });
- }
-
public IRCode getIRCode(ProgramMethod method) {
LinkedList<BasicBlock> blockList = new LinkedList<>();
IntList blockIndices = new IntArrayList(blocks.keySet());
@@ -207,7 +192,7 @@
}
public Value getValue(EV encodedValue) {
- return strategy.getValue(encodedValue, code.getStrategyInfo());
+ return strategy.getValue(encodedValue);
}
public List<Value> getValues(List<EV> indices) {
@@ -228,34 +213,20 @@
public Value getOutValueForNextInstruction(TypeElement type) {
int valueIndex = toInstructionIndexInIR(peekNextInstructionIndex());
- return strategy.getValueDefinitionForInstructionIndex(
- valueIndex, type, code::getDebugLocalInfo);
+ DebugLocalInfo localInfo = code.getDebugLocalInfo(valueIndex);
+ return strategy.getValueDefinitionForInstructionIndex(valueIndex, type, localInfo);
}
public Phi getPhiForNextInstructionAndAdvanceState(TypeElement type) {
- int instructionIndex = peekNextInstructionIndex();
- int valueIndex = toInstructionIndexInIR(instructionIndex);
- Phi phi =
- strategy.getPhiDefinitionForInstructionIndex(
- valueIndex,
- blockIndex -> getBasicBlockOrEnsureCurrentBlock(blockIndex, instructionIndex),
- type,
- code::getDebugLocalInfo,
- code.getStrategyInfo());
- ensureCurrentPosition();
- ++nextInstructionIndex;
- return phi;
- }
-
- private BasicBlock getBasicBlockOrEnsureCurrentBlock(int index, int currentInstructionIndex) {
- // If the index is at current or past it ensure the block.
- if (index >= currentInstructionIndex) {
- ensureCurrentBlock();
- return currentBlock;
- }
- // Otherwise we assume the index is an exact block index for an existing block.
- assert blocks.containsKey(index);
- return getBasicBlock(index);
+ int valueIndex = toInstructionIndexInIR(peekNextInstructionIndex());
+ DebugLocalInfo localInfo = code.getDebugLocalInfo(valueIndex);
+ // TODO(b/225838009): The phi constructor implicitly adds to the block, so we need to ensure
+ // the block. However, we must grab the index above. Find a way to clean this up so it is
+ // uniform with instructions.
+ advanceInstructionState();
+ // Creating the phi implicitly adds it to currentBlock.
+ return strategy.getPhiDefinitionForInstructionIndex(
+ valueIndex, currentBlock, type, localInfo);
}
private void advanceInstructionState() {
@@ -289,9 +260,8 @@
// Arguments are not included in the "instructions" so this does not call "addInstruction"
// which would otherwise advance the state.
TypeElement typeElement = type.toTypeElement(appView);
- Value dest =
- strategy.getValueDefinitionForInstructionIndex(
- index, typeElement, code::getDebugLocalInfo);
+ DebugLocalInfo localInfo = code.getDebugLocalInfo(index);
+ Value dest = strategy.getValueDefinitionForInstructionIndex(index, typeElement, localInfo);
Argument argument = new Argument(dest, index, type.isBooleanType());
assert currentBlock != null;
assert currentPosition.isSyntheticPosition();
diff --git a/src/main/java/com/android/tools/r8/lightir/LirBuilder.java b/src/main/java/com/android/tools/r8/lightir/LirBuilder.java
index 62b30fd..0f78a4e 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirBuilder.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirBuilder.java
@@ -441,6 +441,6 @@
instructionCount,
new TryCatchTable(tryCatchRanges),
debugTable,
- strategy.getStrategyInfo());
+ strategy.getSsaValueStrategy());
}
}
diff --git a/src/main/java/com/android/tools/r8/lightir/LirCode.java b/src/main/java/com/android/tools/r8/lightir/LirCode.java
index 3f84cdf..f62d9f6 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirCode.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirCode.java
@@ -12,7 +12,6 @@
import com.android.tools.r8.ir.code.Position;
import it.unimi.dsi.fastutil.ints.Int2ReferenceMap;
import java.util.Map;
-import java.util.function.BiConsumer;
public class LirCode<EV> implements Iterable<LirInstructionView> {
@@ -49,13 +48,9 @@
this.valueToLocalMap = valueToLocalMap;
this.instructionToEndUseMap = instructionToEndUseMap;
}
-
- public void forEachLocalDefinition(BiConsumer<EV, DebugLocalInfo> fn) {
- valueToLocalMap.forEach(fn);
- }
}
- private final LirStrategyInfo<EV> strategyInfo;
+ private final LirSsaValueStrategy<EV> ssaValueStrategy;
private final IRMetadata metadata;
@@ -94,7 +89,7 @@
int instructionCount,
TryCatchTable tryCatchTable,
DebugLocalInfoTable<EV> debugLocalInfoTable,
- LirStrategyInfo<EV> strategyInfo) {
+ LirSsaValueStrategy<EV> ssaValueStrategy) {
this.metadata = metadata;
this.constants = constants;
this.positionTable = positions;
@@ -103,17 +98,11 @@
this.instructionCount = instructionCount;
this.tryCatchTable = tryCatchTable;
this.debugLocalInfoTable = debugLocalInfoTable;
- this.strategyInfo = strategyInfo;
+ this.ssaValueStrategy = ssaValueStrategy;
}
public EV decodeValueIndex(int encodedValueIndex, int currentValueIndex) {
- return strategyInfo
- .getReferenceStrategy()
- .decodeValueIndex(encodedValueIndex, currentValueIndex);
- }
-
- public LirStrategyInfo<EV> getStrategyInfo() {
- return strategyInfo;
+ return ssaValueStrategy.decodeValueIndex(encodedValueIndex, currentValueIndex);
}
public int getArgumentCount() {
@@ -148,7 +137,7 @@
return debugLocalInfoTable;
}
- public DebugLocalInfo getDebugLocalInfo(EV valueIndex) {
+ public DebugLocalInfo getDebugLocalInfo(int valueIndex) {
return debugLocalInfoTable == null ? null : debugLocalInfoTable.valueToLocalMap.get(valueIndex);
}
diff --git a/src/main/java/com/android/tools/r8/lightir/LirDecodingStrategy.java b/src/main/java/com/android/tools/r8/lightir/LirDecodingStrategy.java
index 0b0a51d..1e25e3c 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirDecodingStrategy.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirDecodingStrategy.java
@@ -7,21 +7,15 @@
import com.android.tools.r8.ir.analysis.type.TypeElement;
import com.android.tools.r8.ir.code.BasicBlock;
import com.android.tools.r8.ir.code.Phi;
-import java.util.function.Function;
-import java.util.function.IntFunction;
/** Abstraction for how to decode SSA values (and basic blocks) when reading LIR. */
public abstract class LirDecodingStrategy<V, EV> {
- public abstract V getValue(EV encodedValue, LirStrategyInfo<EV> strategyInfo);
+ public abstract V getValue(EV encodedValue);
public abstract V getValueDefinitionForInstructionIndex(
- int instructionIndex, TypeElement type, Function<EV, DebugLocalInfo> getLocalInfo);
+ int instructionIndex, TypeElement type, DebugLocalInfo localInfo);
public abstract Phi getPhiDefinitionForInstructionIndex(
- int valueIndex,
- IntFunction<BasicBlock> getBlock,
- TypeElement type,
- Function<EV, DebugLocalInfo> getLocalInfo,
- LirStrategyInfo<EV> strategyInfo);
+ int instructionIndex, BasicBlock block, TypeElement type, DebugLocalInfo localInfo);
}
diff --git a/src/main/java/com/android/tools/r8/lightir/LirEncodingStrategy.java b/src/main/java/com/android/tools/r8/lightir/LirEncodingStrategy.java
index eb814d1..8bbba90 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirEncodingStrategy.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirEncodingStrategy.java
@@ -14,17 +14,13 @@
public abstract EV defineValue(V value, int index);
- public abstract boolean isPhiInlineInstruction();
-
public abstract boolean verifyValueIndex(V value, int expectedIndex);
public abstract EV getEncodedValue(V value);
public int getEncodedValueIndexForReference(EV encodedValue, int referencingValueIndex) {
- return getStrategyInfo()
- .getReferenceStrategy()
- .encodeValueIndex(encodedValue, referencingValueIndex);
+ return getSsaValueStrategy().encodeValueIndex(encodedValue, referencingValueIndex);
}
- public abstract LirStrategyInfo<EV> getStrategyInfo();
+ public abstract LirSsaValueStrategy<EV> getSsaValueStrategy();
}
diff --git a/src/main/java/com/android/tools/r8/lightir/LirPrinter.java b/src/main/java/com/android/tools/r8/lightir/LirPrinter.java
index f4a840b..20dfc30 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirPrinter.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirPrinter.java
@@ -53,7 +53,7 @@
}
private String fmtValueIndex(EV valueIndex) {
- return valueIndex.toString();
+ return "v" + valueIndex;
}
private String fmtInsnIndex(int instructionIndex) {
diff --git a/src/main/java/com/android/tools/r8/lightir/LirStrategy.java b/src/main/java/com/android/tools/r8/lightir/LirStrategy.java
index ba0fa92..70cda39 100644
--- a/src/main/java/com/android/tools/r8/lightir/LirStrategy.java
+++ b/src/main/java/com/android/tools/r8/lightir/LirStrategy.java
@@ -3,8 +3,6 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.lightir;
-import com.android.tools.r8.errors.Unimplemented;
-import com.android.tools.r8.errors.Unreachable;
import com.android.tools.r8.graph.DebugLocalInfo;
import com.android.tools.r8.ir.analysis.type.TypeElement;
import com.android.tools.r8.ir.code.BasicBlock;
@@ -13,11 +11,6 @@
import com.android.tools.r8.ir.code.Value;
import it.unimi.dsi.fastutil.objects.Reference2IntMap;
import it.unimi.dsi.fastutil.objects.Reference2IntOpenHashMap;
-import java.util.ArrayList;
-import java.util.IdentityHashMap;
-import java.util.Map;
-import java.util.function.Function;
-import java.util.function.IntFunction;
/**
* Abstraction for encoding and decoding LIR values.
@@ -33,322 +26,114 @@
public abstract LirDecodingStrategy<V, EV> getDecodingStrategy(LirCode<EV> code);
- /**
- * Encoding of a value with a phi-bit.
- *
- * <p>Due to the generic signature the encoding is boxed so this just adds some convenient
- * predicates and formatting since it is boxed anyway.
- *
- * <p>JVM code attribute has length u2 (16-bit / max 65536). Thus, the number of basic blocks and
- * phi count is also bounded by the same value. The encoding here is taken to be
- *
- * <ul>
- * <li>1-bit for value/phi bit (sign bit / most significant bit),
- * <li>15-bit phi index (the following most significant bits).
- * <li>16-bit block index (the least significant bits).
- * </ul>
- *
- * <p>TODO(b/225838009): Fix this encoding to support pathological block counts above 32k.
- */
- public static class PhiOrValue {
- private final int value;
-
- public static PhiOrValue forPhi(int blockIndex, int phiIndex) {
- int sign = Integer.MIN_VALUE;
- int block = ByteUtils.ensureU2(blockIndex) << 16;
- int phi = ByteUtils.ensureU2(phiIndex);
- int raw = sign | block | phi;
- assert raw < 0;
- if (block >= (1 << 15)) {
- // TODO(b/225838009): Support 16-bit values.
- throw new Unimplemented("No support for more than 15-bit block index.");
- }
- return new PhiOrValue(raw);
- }
-
- public static PhiOrValue forNonPhi(int index) {
- assert index >= 0;
- return new PhiOrValue(index);
- }
-
- private PhiOrValue(int value) {
- this.value = value;
- }
-
- public boolean isPhi() {
- return value < 0;
- }
-
- public boolean isNonPhi() {
- return !isPhi();
- }
-
- public int getRawValue() {
- return value;
- }
-
- public int getDecodedValue() {
- assert isNonPhi();
- return value;
- }
-
- public int getBlockIndex() {
- assert isPhi();
- return (value & ~Integer.MIN_VALUE) >> 16;
- }
-
- public int getPhiIndex() {
- assert isPhi();
- return value & 0xFFFF;
- }
+ // Strategy that implements the encoding of phi values as instructions in the LIR instruction
+ // stream.
+ public static class PhiInInstructionsStrategy extends LirStrategy<Value, Integer> {
@Override
- public String toString() {
- if (isPhi()) {
- return "phi(" + getBlockIndex() + "," + getPhiIndex() + ")";
- }
- return "v" + getDecodedValue();
- }
-
- @Override
- public int hashCode() {
- return value;
- }
-
- @Override
- public boolean equals(Object obj) {
- if (obj == null) {
- return false;
- }
- if (obj == this) {
- return true;
- }
- return obj instanceof PhiOrValue && (value == ((PhiOrValue) obj).value);
- }
- }
-
- public static class ExternalPhisStrategy extends LirStrategy<Value, PhiOrValue> {
-
- @Override
- public LirEncodingStrategy<Value, PhiOrValue> getEncodingStrategy() {
+ public LirEncodingStrategy<Value, Integer> getEncodingStrategy() {
return new EncodingStrategy();
}
@Override
- public LirDecodingStrategy<Value, PhiOrValue> getDecodingStrategy(LirCode<PhiOrValue> code) {
+ public LirDecodingStrategy<Value, Integer> getDecodingStrategy(LirCode<Integer> code) {
return new DecodingStrategy(code);
}
-
- private static class StrategyInfo extends LirStrategyInfo<PhiOrValue> {
- private static final StrategyInfo EMPTY = new StrategyInfo(new int[0]);
-
- private final int[] phiTable;
-
- public StrategyInfo(int[] phiTable) {
- this.phiTable = phiTable;
- }
-
- @Override
- public LirSsaValueStrategy<PhiOrValue> getReferenceStrategy() {
- return ReferenceStrategy.INSTANCE;
- }
- }
-
- private static class EncodingStrategy extends LirEncodingStrategy<Value, PhiOrValue> {
- private final Map<Value, PhiOrValue> values = new IdentityHashMap<>();
- private final Reference2IntMap<BasicBlock> blocks = new Reference2IntOpenHashMap<>();
- private final ArrayList<Integer> phiTable = new ArrayList<>();
-
- @Override
- public boolean isPhiInlineInstruction() {
- return false;
- }
-
- @Override
- public void defineBlock(BasicBlock block, int index) {
- assert !blocks.containsKey(block);
- blocks.put(block, index);
- if (block.getPhis().isEmpty()) {
- return;
- }
- int i = 0;
- for (Phi phi : block.getPhis()) {
- values.put(phi, PhiOrValue.forPhi(index, i++));
- }
- // Amend the phi table with the index of the basic block and the number of its phis.
- phiTable.add(index);
- phiTable.add(i);
- }
-
- @Override
- public PhiOrValue defineValue(Value value, int index) {
- if (value.isPhi()) {
- // Phis are defined as part of blocks.
- PhiOrValue encodedValue = values.get(value);
- assert encodedValue != null;
- return encodedValue;
- }
- PhiOrValue encodedValue = PhiOrValue.forNonPhi(index);
- values.put(value, encodedValue);
- return encodedValue;
- }
-
- @Override
- public boolean verifyValueIndex(Value value, int expectedIndex) {
- PhiOrValue encodedValue = values.get(value);
- assert encodedValue.isNonPhi();
- assert expectedIndex == encodedValue.getDecodedValue();
- return true;
- }
-
- @Override
- public PhiOrValue getEncodedValue(Value value) {
- return values.get(value);
- }
-
- @Override
- public int getBlockIndex(BasicBlock block) {
- assert blocks.containsKey(block);
- return blocks.getInt(block);
- }
-
- @Override
- public LirStrategyInfo<PhiOrValue> getStrategyInfo() {
- if (phiTable.isEmpty()) {
- return StrategyInfo.EMPTY;
- }
- int[] array = new int[phiTable.size()];
- for (int i = 0; i < phiTable.size(); i++) {
- array[i] = phiTable.get(i);
- }
- return new StrategyInfo(array);
- }
- }
-
- private static class DecodingStrategy extends LirDecodingStrategy<Value, PhiOrValue> {
-
- private final Value[] values;
- private final int firstPhiValueIndex;
-
- DecodingStrategy(LirCode<PhiOrValue> code) {
- values = new Value[code.getArgumentCount() + code.getInstructionCount()];
- int phiValueIndex = -1;
- for (LirInstructionView view : code) {
- if (view.getOpcode() == LirOpcodes.PHI) {
- phiValueIndex = code.getArgumentCount() + view.getInstructionIndex();
- break;
- }
- }
- this.firstPhiValueIndex = phiValueIndex;
- }
-
- private int decode(PhiOrValue encodedValue, LirStrategyInfo<PhiOrValue> strategyInfo) {
- if (encodedValue.isNonPhi()) {
- return encodedValue.getDecodedValue();
- }
- StrategyInfo info = (StrategyInfo) strategyInfo;
- int phiBlock = encodedValue.getBlockIndex();
- int phiIndex = encodedValue.getPhiIndex();
- assert firstPhiValueIndex != -1;
- int index = firstPhiValueIndex;
- for (int i = 0; i < info.phiTable.length; i++) {
- int blockIndex = info.phiTable[i];
- if (blockIndex == phiBlock) {
- return index + phiIndex;
- }
- index += info.phiTable[++i];
- }
- throw new Unreachable("Unexpectedly fell off the end of the phi table");
- }
-
- @Override
- public Value getValue(PhiOrValue encodedValue, LirStrategyInfo<PhiOrValue> strategyInfo) {
- int index = decode(encodedValue, strategyInfo);
- Value value = values[index];
- if (value == null) {
- value = new Value(index, TypeElement.getBottom(), null);
- values[index] = value;
- }
- return value;
- }
-
- @Override
- public Value getValueDefinitionForInstructionIndex(
- int index, TypeElement type, Function<PhiOrValue, DebugLocalInfo> getLocalInfo) {
- PhiOrValue encodedValue = new PhiOrValue(index);
- assert encodedValue.isNonPhi();
- DebugLocalInfo localInfo = getLocalInfo.apply(encodedValue);
- Value value = values[index];
- if (value == null) {
- value = new Value(index, type, localInfo);
- values[index] = value;
- } else {
- value.setType(type);
- if (localInfo != null && !value.hasLocalInfo()) {
- value.setLocalInfo(localInfo);
- }
- assert localInfo == value.getLocalInfo();
- }
- return value;
- }
-
- @Override
- public Phi getPhiDefinitionForInstructionIndex(
- int valueIndex,
- IntFunction<BasicBlock> getBlock,
- TypeElement type,
- Function<PhiOrValue, DebugLocalInfo> getLocalInfo,
- LirStrategyInfo<PhiOrValue> strategyInfo) {
- PhiOrValue encodedValue = getEncodedPhiForAbsoluteValueIndex(valueIndex, strategyInfo);
- BasicBlock block = getBlock.apply(encodedValue.getBlockIndex());
- DebugLocalInfo localInfo = getLocalInfo.apply(encodedValue);
- Phi phi = new Phi(valueIndex, block, type, localInfo, RegisterReadType.NORMAL);
- Value value = values[valueIndex];
- if (value != null) {
- // A fake ssa value has already been created, replace the users by the actual phi.
- // TODO(b/225838009): We could consider encoding the value phi-bit in the value index
- // and avoid the overhead of replacing users at phi-definition time.
- assert !value.isPhi();
- value.replaceUsers(phi);
- }
- values[valueIndex] = phi;
- return phi;
- }
-
- private PhiOrValue getEncodedPhiForAbsoluteValueIndex(
- int phiValueIndex, LirStrategyInfo<PhiOrValue> strategyInfo) {
- StrategyInfo info = (StrategyInfo) strategyInfo;
- int currentPhiValueIndex = firstPhiValueIndex;
- for (int i = 0; i < info.phiTable.length; i += 2) {
- assert currentPhiValueIndex <= phiValueIndex;
- int blockIndex = info.phiTable[i];
- int phiCount = info.phiTable[i + 1];
- assert phiCount > 0;
- if (phiValueIndex < currentPhiValueIndex + phiCount) {
- int phiOffsetInBlock = phiValueIndex - currentPhiValueIndex;
- return PhiOrValue.forPhi(blockIndex, phiOffsetInBlock);
- }
- currentPhiValueIndex += phiCount;
- }
- throw new Unreachable("Unexpected fall off the end of the phi table");
- }
- }
-
- // TODO(b/225838009): Consider still encoding the local value refs as small relative indexes.
- private static class ReferenceStrategy extends LirSsaValueStrategy<PhiOrValue> {
-
- private static final ReferenceStrategy INSTANCE = new ReferenceStrategy();
-
- @Override
- public int encodeValueIndex(PhiOrValue value, int currentValueIndex) {
- return value.getRawValue();
- }
-
- @Override
- public PhiOrValue decodeValueIndex(int encodedValueIndex, int currentValueIndex) {
- return new PhiOrValue(encodedValueIndex);
- }
- }
}
+ private static class EncodingStrategy extends LirEncodingStrategy<Value, Integer> {
+
+ // EV == Integer and its definition is equal to its shifted instruction index.
+ // The conversion for EV to its int-valued reference is determined by the 'valueStrategy'.
+
+ private final LirSsaValueStrategy<Integer> valueStrategy = LirSsaValueStrategy.get();
+ private final Reference2IntMap<Value> values = new Reference2IntOpenHashMap<>();
+ private final Reference2IntMap<BasicBlock> blocks = new Reference2IntOpenHashMap<>();
+
+ @Override
+ public void defineBlock(BasicBlock block, int index) {
+ assert !blocks.containsKey(block);
+ blocks.put(block, index);
+ }
+
+ @Override
+ public Integer defineValue(Value value, int index) {
+ values.put(value, index);
+ return index;
+ }
+
+ @Override
+ public boolean verifyValueIndex(Value value, int expectedIndex) {
+ assert expectedIndex == values.getInt(value);
+ return true;
+ }
+
+ @Override
+ public Integer getEncodedValue(Value value) {
+ return values.getInt(value);
+ }
+
+ @Override
+ public int getBlockIndex(BasicBlock block) {
+ assert blocks.containsKey(block);
+ return blocks.getInt(block);
+ }
+
+ @Override
+ public LirSsaValueStrategy<Integer> getSsaValueStrategy() {
+ return valueStrategy;
+ }
+ }
+
+ private static class DecodingStrategy extends LirDecodingStrategy<Value, Integer> {
+
+ private final Value[] values;
+
+ DecodingStrategy(LirCode<Integer> code) {
+ values = new Value[code.getArgumentCount() + code.getInstructionCount()];
+ }
+
+ @Override
+ public Value getValue(Integer encodedValue) {
+ int index = encodedValue;
+ Value value = values[index];
+ if (value == null) {
+ value = new Value(index, TypeElement.getBottom(), null);
+ values[index] = value;
+ }
+ return value;
+ }
+
+ @Override
+ public Value getValueDefinitionForInstructionIndex(
+ int index, TypeElement type, DebugLocalInfo localInfo) {
+ Value value = values[index];
+ if (value == null) {
+ value = new Value(index, type, localInfo);
+ values[index] = value;
+ } else {
+ value.setType(type);
+ if (localInfo != null) {
+ value.setLocalInfo(localInfo);
+ }
+ }
+ return value;
+ }
+
+ @Override
+ public Phi getPhiDefinitionForInstructionIndex(
+ int index, BasicBlock block, TypeElement type, DebugLocalInfo localInfo) {
+ Phi phi = new Phi(index, block, type, localInfo, RegisterReadType.NORMAL);
+ Value value = values[index];
+ if (value != null) {
+ // A fake ssa value has already been created, replace the users by the actual phi.
+ // TODO(b/225838009): We could consider encoding the value type as a bit in the value index
+ // and avoid the overhead of replacing users at phi-definition time.
+ assert !value.isPhi();
+ value.replaceUsers(phi);
+ }
+ values[index] = phi;
+ return phi;
+ }
+ }
}
diff --git a/src/main/java/com/android/tools/r8/lightir/LirStrategyInfo.java b/src/main/java/com/android/tools/r8/lightir/LirStrategyInfo.java
deleted file mode 100644
index 088786c..0000000
--- a/src/main/java/com/android/tools/r8/lightir/LirStrategyInfo.java
+++ /dev/null
@@ -1,9 +0,0 @@
-// Copyright (c) 2023, 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.lightir;
-
-public abstract class LirStrategyInfo<EV> {
-
- public abstract LirSsaValueStrategy<EV> getReferenceStrategy();
-}
diff --git a/src/main/java/com/android/tools/r8/lightir/PhiInInstructionsStrategy.java b/src/main/java/com/android/tools/r8/lightir/PhiInInstructionsStrategy.java
deleted file mode 100644
index 0b0d537..0000000
--- a/src/main/java/com/android/tools/r8/lightir/PhiInInstructionsStrategy.java
+++ /dev/null
@@ -1,145 +0,0 @@
-// Copyright (c) 2023, 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.lightir;
-
-import com.android.tools.r8.graph.DebugLocalInfo;
-import com.android.tools.r8.ir.analysis.type.TypeElement;
-import com.android.tools.r8.ir.code.BasicBlock;
-import com.android.tools.r8.ir.code.Phi;
-import com.android.tools.r8.ir.code.Phi.RegisterReadType;
-import com.android.tools.r8.ir.code.Value;
-import it.unimi.dsi.fastutil.objects.Reference2IntMap;
-import it.unimi.dsi.fastutil.objects.Reference2IntOpenHashMap;
-import java.util.function.Function;
-import java.util.function.IntFunction;
-
-/** Strategy encoding phi values as instructions in the LIR instruction stream. */
-public class PhiInInstructionsStrategy extends LirStrategy<Value, Integer> {
-
- @Override
- public LirEncodingStrategy<Value, Integer> getEncodingStrategy() {
- return new EncodingStrategy();
- }
-
- @Override
- public LirDecodingStrategy<Value, Integer> getDecodingStrategy(LirCode<Integer> code) {
- return new DecodingStrategy(code);
- }
-
- private static class EncodingStrategy extends LirEncodingStrategy<Value, Integer> {
-
- // EV == Integer and its definition is equal to its shifted instruction index.
- // The conversion for EV to its int-valued reference is determined by the 'valueStrategy'.
-
- private final LirSsaValueStrategy<Integer> referenceStrategy = LirSsaValueStrategy.get();
- private final Reference2IntMap<Value> values = new Reference2IntOpenHashMap<>();
- private final Reference2IntMap<BasicBlock> blocks = new Reference2IntOpenHashMap<>();
-
- @Override
- public boolean isPhiInlineInstruction() {
- return true;
- }
-
- @Override
- public void defineBlock(BasicBlock block, int index) {
- assert !blocks.containsKey(block);
- blocks.put(block, index);
- }
-
- @Override
- public Integer defineValue(Value value, int index) {
- values.put(value, index);
- return index;
- }
-
- @Override
- public boolean verifyValueIndex(Value value, int expectedIndex) {
- assert expectedIndex == values.getInt(value);
- return true;
- }
-
- @Override
- public Integer getEncodedValue(Value value) {
- return values.getInt(value);
- }
-
- @Override
- public int getBlockIndex(BasicBlock block) {
- assert blocks.containsKey(block);
- return blocks.getInt(block);
- }
-
- @Override
- public LirStrategyInfo<Integer> getStrategyInfo() {
- return new LirStrategyInfo<Integer>() {
- @Override
- public LirSsaValueStrategy<Integer> getReferenceStrategy() {
- return referenceStrategy;
- }
- };
- }
- }
-
- private static class DecodingStrategy extends LirDecodingStrategy<Value, Integer> {
-
- private final Value[] values;
-
- DecodingStrategy(LirCode<Integer> code) {
- values = new Value[code.getArgumentCount() + code.getInstructionCount()];
- }
-
- @Override
- public Value getValue(Integer encodedValue, LirStrategyInfo<Integer> strategyInfo) {
- int index = encodedValue;
- Value value = values[index];
- if (value == null) {
- value = new Value(index, TypeElement.getBottom(), null);
- values[index] = value;
- }
- return value;
- }
-
- @Override
- public Value getValueDefinitionForInstructionIndex(
- int index, TypeElement type, Function<Integer, DebugLocalInfo> getLocalInfo) {
- DebugLocalInfo localInfo = getLocalInfo.apply(index);
- Value value = values[index];
- if (value == null) {
- value = new Value(index, type, localInfo);
- values[index] = value;
- } else {
- value.setType(type);
- if (localInfo != null) {
- if (!value.hasLocalInfo()) {
- value.setLocalInfo(localInfo);
- }
- assert localInfo == value.getLocalInfo();
- }
- }
- return value;
- }
-
- @Override
- public Phi getPhiDefinitionForInstructionIndex(
- int valueIndex,
- IntFunction<BasicBlock> getBlock,
- TypeElement type,
- Function<Integer, DebugLocalInfo> getLocalInfo,
- LirStrategyInfo<Integer> strategyInfo) {
- BasicBlock block = getBlock.apply(valueIndex);
- DebugLocalInfo localInfo = getLocalInfo.apply(valueIndex);
- Phi phi = new Phi(valueIndex, block, type, localInfo, RegisterReadType.NORMAL);
- Value value = values[valueIndex];
- if (value != null) {
- // A fake ssa value has already been created, replace the users by the actual phi.
- // TODO(b/225838009): We could consider encoding the value type as a bit in the value index
- // and avoid the overhead of replacing users at phi-definition time.
- assert !value.isPhi();
- value.replaceUsers(phi);
- }
- values[valueIndex] = phi;
- return phi;
- }
- }
-}
diff --git a/src/test/java/com/android/tools/r8/lightir/LirBasicCallbackTest.java b/src/test/java/com/android/tools/r8/lightir/LirBasicCallbackTest.java
index abd980d..062470c 100644
--- a/src/test/java/com/android/tools/r8/lightir/LirBasicCallbackTest.java
+++ b/src/test/java/com/android/tools/r8/lightir/LirBasicCallbackTest.java
@@ -37,11 +37,6 @@
private static class ThrowingStrategy extends LirEncodingStrategy<Value, Integer> {
@Override
- public boolean isPhiInlineInstruction() {
- return false;
- }
-
- @Override
public void defineBlock(BasicBlock block, int index) {
throw new Unreachable();
}
@@ -67,7 +62,7 @@
}
@Override
- public LirStrategyInfo<Integer> getStrategyInfo() {
+ public LirSsaValueStrategy<Integer> getSsaValueStrategy() {
return null;
}
}