/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import java.lang.reflect.Array; import java.lang.reflect.Method; /** * Tests for SIMD related optimizations. */ public class Main { /// CHECK-START: void Main.unroll(float[], float[]) loop_optimization (before) /// CHECK-DAG: <> FloatConstant 2.5 loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [{{l\d+}},<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM64: void Main.unroll(float[], float[]) loop_optimization (after) /// CHECK-DAG: <> FloatConstant 2.5 loop:none /// CHECK-DAG: <> IntConstant 4 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-NOT: VecReplicateScalar /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecMul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecMul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none private static void unroll(float[] x, float[] y) { for (int i = 0; i < 100; i++) { x[i] = y[i] * 2.5f; } } /// CHECK-START-ARM64: void Main.stencil(int[], int[], int) loop_optimization (after) /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none private static void stencil(int[] a, int[] b, int n) { for (int i = 1; i < n - 1; i++) { a[i] = b[i - 1] + b[i] + b[i + 1]; } } /// CHECK-START: void Main.stencilAddInt(int[], int[], int) loop_optimization (before) /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant -1 loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-START-{X86_64,ARM64}: void Main.stencilAddInt(int[], int[], int) loop_optimization (after) /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none private static void stencilAddInt(int[] a, int[] b, int n) { int minus1 = $inline$constMinus1(); for (int i = 1; i < n + minus1; i++) { a[i] = b[i + minus1] + b[i] + b[i + 1]; } } private static int $inline$constMinus1() { return -1; } /// CHECK-START: void Main.stencilSubInt(int[], int[], int) loop_optimization (before) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Sub [<>,<>] loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> Sub [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-START-{X86_64,ARM64}: void Main.stencilSubInt(int[], int[], int) loop_optimization (after) /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> Phi loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none private static void stencilSubInt(int[] a, int[] b, int n) { int plus1 = $inline$constPlus1(); for (int i = 1; i < n - plus1; i++) { a[i] = b[i - plus1] + b[i] + b[i + 1]; } } private static int $inline$constPlus1() { return 1; } /// CHECK-START: long Main.longInductionReduction(long[]) loop_optimization (before) /// CHECK-DAG: <> LongConstant 0 loop:none /// CHECK-DAG: <> LongConstant 1 loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM64: long Main.longInductionReduction(long[]) loop_optimization (after) /// CHECK-DAG: <> LongConstant 0 loop:none /// CHECK-DAG: <> LongConstant 1 loop:none /// CHECK-DAG: <> LongConstant 2 loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{j\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none static long longInductionReduction(long[] y) { long x = 1; for (long i = 0; i < 10; i++) { x += y[0]; } return x; } /// CHECK-START: void Main.intVectorLongInvariant(int[], long[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM64: void Main.intVectorLongInvariant(int[], long[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 4 loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none static void intVectorLongInvariant(int[] x, long[] y) { for (int i = 0; i < 100; i++) { x[i] = (int) y[0]; } } /// CHECK-START: void Main.longCanBeDoneWithInt(int[], int[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> LongConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM64: void Main.longCanBeDoneWithInt(int[], int[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 4 loop:none /// CHECK-DAG: <> LongConstant 1 loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none static void longCanBeDoneWithInt(int[] x, int[] y) { for (int i = 0; i < 100; i++) { x[i] = (int) (y[i] + 1L); } } static void testUnroll() { float[] x = new float[100]; float[] y = new float[100]; for (int i = 0; i < 100; i++) { x[i] = 0.0f; y[i] = 2.0f; } unroll(x, y); for (int i = 0; i < 100; i++) { expectEquals(5.0f, x[i]); expectEquals(2.0f, y[i]); } } static void testStencil1() { int[] a = new int[100]; int[] b = new int[100]; for (int i = 0; i < 100; i++) { a[i] = 0; b[i] = i; } stencil(a, b, 100); for (int i = 1; i < 99; i++) { int e = i + i + i; expectEquals(e, a[i]); expectEquals(i, b[i]); } } static void testStencil2() { int[] a = new int[100]; int[] b = new int[100]; for (int i = 0; i < 100; i++) { a[i] = 0; b[i] = i; } stencilSubInt(a, b, 100); for (int i = 1; i < 99; i++) { int e = i + i + i; expectEquals(e, a[i]); expectEquals(i, b[i]); } } static void testStencil3() { int[] a = new int[100]; int[] b = new int[100]; for (int i = 0; i < 100; i++) { a[i] = 0; b[i] = i; } stencilAddInt(a, b, 100); for (int i = 1; i < 99; i++) { int e = i + i + i; expectEquals(e, a[i]); expectEquals(i, b[i]); } } static void testTypes() { int[] a = new int[100]; int[] b = new int[100]; long[] l = { 3 }; expectEquals(31, longInductionReduction(l)); intVectorLongInvariant(a, l); for (int i = 0; i < 100; i++) { expectEquals(3, a[i]); } longCanBeDoneWithInt(b, a); for (int i = 0; i < 100; i++) { expectEquals(4, b[i]); } } public static void main(String[] args) { testUnroll(); testStencil1(); testStencil2(); testStencil3(); testTypes(); System.out.println("passed"); } private static void expectEquals(int expected, int result) { if (expected != result) { throw new Error("Expected: " + expected + ", found: " + result); } } private static void expectEquals(long expected, long result) { if (expected != result) { throw new Error("Expected: " + expected + ", found: " + result); } } private static void expectEquals(float expected, float result) { if (expected != result) { throw new Error("Expected: " + expected + ", found: " + result); } } }