|  | // Copyright (c) 2019, 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 backport; | 
|  |  | 
|  | import com.android.tools.r8.TestBase; | 
|  | import com.android.tools.r8.TestParameters; | 
|  | import com.android.tools.r8.TestRuntime.CfVm; | 
|  | import com.android.tools.r8.desugar.backports.AbstractBackportTest; | 
|  | import com.android.tools.r8.utils.AndroidApiLevel; | 
|  | import java.math.BigInteger; | 
|  | import org.junit.runner.RunWith; | 
|  | import org.junit.runners.Parameterized; | 
|  | import org.junit.runners.Parameterized.Parameters; | 
|  |  | 
|  | @RunWith(Parameterized.class) | 
|  | public final class MathBackportJava9Test extends AbstractBackportTest { | 
|  | @Parameters(name = "{0}") | 
|  | public static Iterable<?> data() { | 
|  | return TestBase.getTestParameters() | 
|  | .withDexRuntimes() | 
|  | .withCfRuntimesStartingFromIncluding(CfVm.JDK9) | 
|  | .withAllApiLevelsAlsoForCf() | 
|  | .build(); | 
|  | } | 
|  |  | 
|  | public MathBackportJava9Test(TestParameters parameters) { | 
|  | super(parameters, Math.class, MathBackportJava9Main.class); | 
|  |  | 
|  | // Math.floorDiv, Math.floorMod, Math.multiplyExact, Math.multiplyFull and Math.multiplyHigh | 
|  | // added in API 31. | 
|  | registerTarget(AndroidApiLevel.S, 27); | 
|  | } | 
|  |  | 
|  | public static class MathBackportJava9Main { | 
|  |  | 
|  | public static void main(String[] args) { | 
|  | testMultiplyExactLongInt(); | 
|  | testMultiplyFull(); | 
|  | testMultiplyHigh(); | 
|  | testFloorDivLongInt(); | 
|  | testFloorModLongInt(); | 
|  | } | 
|  |  | 
|  | public static void testMultiplyExactLongInt() { | 
|  | assertEquals(8L, Math.multiplyExact(2L, 4)); | 
|  | assertEquals(Long.MAX_VALUE, Math.multiplyExact(Long.MAX_VALUE, 1)); | 
|  | assertEquals(Long.MIN_VALUE, Math.multiplyExact(Long.MIN_VALUE / 2L, 2)); | 
|  | try { | 
|  | throw new AssertionError(Math.multiplyExact(Long.MAX_VALUE, 2)); | 
|  | } catch (ArithmeticException expected) { | 
|  | } | 
|  | try { | 
|  | throw new AssertionError(Math.multiplyExact(Long.MIN_VALUE, 2)); | 
|  | } catch (ArithmeticException expected) { | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void testMultiplyFull() { | 
|  | assertEquals(8L, Math.multiplyFull(2, 4)); | 
|  | assertEquals(4611686014132420609L, Math.multiplyFull(Integer.MAX_VALUE, Integer.MAX_VALUE)); | 
|  | assertEquals(-4611686016279904256L, Math.multiplyFull(Integer.MAX_VALUE, Integer.MIN_VALUE)); | 
|  | assertEquals(4611686018427387904L, Math.multiplyFull(Integer.MIN_VALUE, Integer.MIN_VALUE)); | 
|  | } | 
|  |  | 
|  | public static void testMultiplyHigh() { | 
|  | long[] interestingValues = { | 
|  | Long.MIN_VALUE, | 
|  | Long.MAX_VALUE, | 
|  | Integer.MIN_VALUE, | 
|  | Integer.MAX_VALUE, | 
|  | Short.MIN_VALUE, | 
|  | Short.MAX_VALUE, | 
|  | Byte.MIN_VALUE, | 
|  | Byte.MAX_VALUE, | 
|  | 0L, | 
|  | -1L, | 
|  | 1L, | 
|  | -42L, | 
|  | 42L | 
|  | }; | 
|  | for (long x : interestingValues) { | 
|  | for (long y : interestingValues) { | 
|  | long expected = | 
|  | BigInteger.valueOf(x).multiply(BigInteger.valueOf(y)).shiftRight(64).longValue(); | 
|  | assertEquals(expected, Math.multiplyHigh(x, y)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void testFloorDivLongInt() { | 
|  | assertEquals(1L, Math.floorDiv(4L, 4)); | 
|  | assertEquals(1L, Math.floorDiv(-4L, -4)); | 
|  | assertEquals(-1L, Math.floorDiv(-4L, 4)); | 
|  | assertEquals(-1L, Math.floorDiv(4L, -4)); | 
|  |  | 
|  | assertEquals(1L, Math.floorDiv(4L, 3)); | 
|  | assertEquals(1L, Math.floorDiv(-4L, -3)); | 
|  | assertEquals(-2L, Math.floorDiv(-4L, 3)); | 
|  | assertEquals(-2L, Math.floorDiv(4L, -3)); | 
|  |  | 
|  | // Spec edge case: result is actually MAX_VALUE+1 which becomes MIN_VALUE. | 
|  | assertEquals(Long.MIN_VALUE, Math.floorDiv(Long.MIN_VALUE, -1)); | 
|  | } | 
|  |  | 
|  | public static void testFloorModLongInt() { | 
|  | assertEquals(0, Math.floorMod(4L, 4)); | 
|  | assertEquals(0, Math.floorMod(-4L, -4)); | 
|  | assertEquals(0, Math.floorMod(-4L, 4)); | 
|  | assertEquals(0, Math.floorMod(4L, -4)); | 
|  |  | 
|  | assertEquals(1, Math.floorMod(4L, 3)); | 
|  | assertEquals(-1, Math.floorMod(-4L, -3)); | 
|  | assertEquals(2, Math.floorMod(-4L, 3)); | 
|  | assertEquals(-2, Math.floorMod(4L, -3)); | 
|  | } | 
|  |  | 
|  | private static void assertEquals(int expected, int actual) { | 
|  | if (expected != actual) { | 
|  | throw new AssertionError("Expected <" + expected + "> but was <" + actual + '>'); | 
|  | } | 
|  | } | 
|  |  | 
|  | private static void assertEquals(long expected, long actual) { | 
|  | if (expected != actual) { | 
|  | throw new AssertionError("Expected <" + expected + "> but was <" + actual + '>'); | 
|  | } | 
|  | } | 
|  | } | 
|  | } |