/* * 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. */ /** * Tests for zero vectorization. */ public class Main { /// CHECK-START: void Main.staticallyAligned(int[]) loop_optimization (before) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyAligned(int[]) loop_optimization (after) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-NOT: ArrayGet /// CHECK-NOT: ArraySet static void staticallyAligned(int[] a) { // Starts at offset 12 (hidden) + 1 * 4 relative to base alignment. // So no peeling, aligned vector, no cleanup. for (int i = 1; i < 9; i++) { a[i] += 1; } } /// CHECK-START: void Main.staticallyAlignedN(int[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> NullCheck loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyAlignedN(int[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> NullCheck loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none static void staticallyAlignedN(int[] a) { // Starts at offset 12 (hidden) + 1 * 4 relative to base alignment. // So no peeling, aligned vector, cleanup. for (int i = 1; i < a.length; i++) { a[i] += 1; } } /// CHECK-START: void Main.staticallyMisaligned(int[]) loop_optimization (before) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyMisaligned(int[]) loop_optimization (after) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-NOT: ArrayGet /// CHECK-NOT: ArraySet static void staticallyMisaligned(int[] a) { // Starts at offset 12 (hidden) + 0 * 4 relative to base alignment. // Yes, Art runtime misaligns the most common access pattern :-( // Static peeling to the rescue, aligned vector, no cleanup. for (int i = 0; i < 9; i++) { a[i] += 1; } } /// CHECK-START: void Main.staticallyMisalignedN(int[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> NullCheck loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyMisalignedN(int[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> IntConstant 2 loop:none /// CHECK-DAG: <> NullCheck loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none static void staticallyMisalignedN(int[] a) { // Starts at offset 12 (hidden) + 0 * 4 relative to base alignment. // Yes, Art runtime misaligns the most common access pattern :-( // Static peeling to the rescue, aligned vector, cleanup. for (int i = 0; i < a.length; i++) { a[i] += 1; } } /// CHECK-START: void Main.staticallyUnknownAligned(int[], int) loop_optimization (before) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 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 [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyUnknownAligned(int[], int) loop_optimization (after) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// 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: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none static void staticallyUnknownAligned(int[] a, int off) { // Starts at an unknown offset due to parameter off. // Dynamic peeling to the rescue, aligned vector, cleanup. for (int i = 0; i < 9; i++) { a[off + i] += 1; } } /// CHECK-START: void Main.staticallyUnknownAlignedN(int[], int, int) loop_optimization (before) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 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 [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none // /// CHECK-START-ARM: void Main.staticallyUnknownAlignedN(int[], int, int) loop_optimization (after) /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> ParameterValue loop:none /// CHECK-DAG: <> IntConstant 0 loop:none /// 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: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecStore [<>,<>,<>] alignment:ALIGN(8,0) loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: ArraySet [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none static void staticallyUnknownAlignedN(int[] a, int off, int n) { // Starts at an unknown offset due to parameter off. // Dynamic peeling to the rescue, aligned vector, cleanup. for (int i = 0; i < n; i++) { a[off + i] += 1; } } // // Test drivers. // private static void test1() { int[] a = new int[9]; staticallyAligned(a); for (int i = 0; i < a.length; i++) { int e = i > 0 ? 1 : 0; expectEquals(e, a[i]); } } private static void test2() { for (int n = 0; n <= 71; n++) { int[] a = new int[n]; staticallyAlignedN(a); for (int i = 0; i < a.length; i++) { int e = i > 0 ? 1 : 0; expectEquals(e, a[i]); } } } private static void test3() { int[] a = new int[9]; staticallyMisaligned(a); for (int i = 0; i < a.length; i++) { expectEquals(1, a[i]); } } private static void test4() { for (int n = 0; n <= 71; n++) { int[] a = new int[n]; staticallyMisalignedN(a); for (int i = 0; i < a.length; i++) { expectEquals(1, a[i]); } } } private static void test5() { for (int off = 0; off <= 8; off++) { int[] a = new int[17]; staticallyUnknownAligned(a, off); for (int i = 0; i < a.length; i++) { int e = (off <= i && i < off + 9) ? 1 : 0; expectEquals(e, a[i]); } } } private static void test6() { for (int off = 0; off <= 8; off++) { for (int n = 0; n <= 9; n++) { int[] a = new int[17]; staticallyUnknownAlignedN(a, off, n); for (int i = 0; i < a.length; i++) { int e = (off <= i && i < off + n) ? 1 : 0; expectEquals(e, a[i]); } } } } public static void main(String[] args) { test1(); test2(); test4(); test5(); test6(); System.out.println("passed"); } private static void expectEquals(int expected, int result) { if (expected != result) { throw new Error("Expected: " + expected + ", found: " + result); } } }