Merge "Increase usages of 2Addr instructions"
diff --git a/src/main/java/com/android/tools/r8/ir/regalloc/LinearScanRegisterAllocator.java b/src/main/java/com/android/tools/r8/ir/regalloc/LinearScanRegisterAllocator.java
index d6a64a4..52653a7 100644
--- a/src/main/java/com/android/tools/r8/ir/regalloc/LinearScanRegisterAllocator.java
+++ b/src/main/java/com/android/tools/r8/ir/regalloc/LinearScanRegisterAllocator.java
@@ -8,6 +8,7 @@
 import com.android.tools.r8.graph.DebugLocalInfo;
 import com.android.tools.r8.ir.code.Add;
 import com.android.tools.r8.ir.code.And;
+import com.android.tools.r8.ir.code.ArithmeticBinop;
 import com.android.tools.r8.ir.code.BasicBlock;
 import com.android.tools.r8.ir.code.DebugLocalsChange;
 import com.android.tools.r8.ir.code.IRCode;
@@ -811,6 +812,9 @@
     // Go through each unhandled live interval and find a register for it.
     while (!unhandled.isEmpty()) {
       LiveIntervals unhandledInterval = unhandled.poll();
+
+      setHintToPromote2AddrInstruction(unhandledInterval);
+
       // If this interval value is the src of an argument move. Fix the registers for the
       // consecutive arguments now and add hints to the move sources. This looks forward
       // and propagate hints backwards to avoid many moves in connection with ranged invokes.
@@ -862,6 +866,31 @@
     return true;
   }
 
+  /*
+   * This method tries to promote arithmetic binary instruction to use the 2Addr form.
+   * To achieve this goal the output interval of the binary instruction is set with an hint
+   * that is the left interval or the right interval if possible when intervals do not overlap.
+   */
+  private void setHintToPromote2AddrInstruction(LiveIntervals unhandledInterval) {
+    if (unhandledInterval.getHint() == null &&
+        unhandledInterval.getValue().definition instanceof ArithmeticBinop) {
+      ArithmeticBinop binOp = unhandledInterval.getValue().definition.asArithmeticBinop();
+      Value left = binOp.leftValue();
+      assert left != null;
+      if (left.getLiveIntervals() != null &&
+          !left.getLiveIntervals().overlaps(unhandledInterval)) {
+        unhandledInterval.setHint(left.getLiveIntervals());
+      } else {
+        Value right = binOp.rightValue();
+        assert right != null;
+        if (binOp.isCommutative() && right.getLiveIntervals() != null &&
+            !right.getLiveIntervals().overlaps(unhandledInterval)) {
+          unhandledInterval.setHint(right.getLiveIntervals());
+        }
+      }
+    }
+  }
+
   /**
    * Perform look-ahead and allocate registers for linked argument chains that have the argument
    * interval as an argument move source.
diff --git a/src/test/java/com/android/tools/r8/smali/OutlineTest.java b/src/test/java/com/android/tools/r8/smali/OutlineTest.java
index d0e3ade..c64e39c 100644
--- a/src/test/java/com/android/tools/r8/smali/OutlineTest.java
+++ b/src/test/java/com/android/tools/r8/smali/OutlineTest.java
@@ -12,6 +12,7 @@
 import com.android.tools.r8.code.ConstWide;
 import com.android.tools.r8.code.ConstWideHigh16;
 import com.android.tools.r8.code.DivInt;
+import com.android.tools.r8.code.DivInt2Addr;
 import com.android.tools.r8.code.InvokeStatic;
 import com.android.tools.r8.code.InvokeVirtual;
 import com.android.tools.r8.code.MoveResult;
@@ -1176,7 +1177,7 @@
     assertTrue(code.instructions[0] instanceof DivInt);
     assertTrue(code.instructions[1] instanceof InvokeStatic);
     assertTrue(code.instructions[2] instanceof MoveResult);
-    assertTrue(code.instructions[3] instanceof DivInt);
+    assertTrue(code.instructions[3] instanceof DivInt2Addr);
     assertTrue(code.instructions[4] instanceof Return);
     assertTrue(code.instructions[5] instanceof Const4);
     assertTrue(code.instructions[6] instanceof Return);