/* * Copyright (C) 2018 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. */ package other; /** * Tests for dot product idiom vectorization: byte case. */ public class TestByte { public static final int ARRAY_SIZE = 1024; /// CHECK-START: int other.TestByte.testDotProdSimple(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdSimple(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: /// CHECK-START-ARM64: int other.TestByte.testDotProdSimple(byte[], byte[]) disassembly (after) /// CHECK: VecDotProd /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' /// CHECK: sdot z{{\d+}}.s, z{{\d+}}.b, z{{\d+}}.b /// CHECK-ELIF: hasIsaFeature("dotprod") /// CHECK-NEXT: sdot v{{\d+}}.4s, v{{\d+}}.16b, v{{\d+}}.16b /// CHECK-ELSE: /// CHECK-NOT: sdot /// CHECK-NOT: udot /// CHECK-FI: public static final int testDotProdSimple(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = a[i] * b[i]; s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdComplex(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdComplex(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecReplicateScalar [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: public static final int testDotProdComplex(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = ((byte)(a[i] + 1)) * ((byte)(b[i] + 1)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsigned(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdSimpleUnsigned(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: /// CHECK-START-ARM64: int other.TestByte.testDotProdSimpleUnsigned(byte[], byte[]) disassembly (after) /// CHECK: VecDotProd /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' /// CHECK: udot z{{\d+}}.s, z{{\d+}}.b, z{{\d+}}.b /// CHECK-ELIF: hasIsaFeature("dotprod") /// CHECK-NEXT: udot v{{\d+}}.4s, v{{\d+}}.16b, v{{\d+}}.16b /// CHECK-ELSE: /// CHECK-NOT: sdot /// CHECK-NOT: udot /// CHECK-FI: public static final int testDotProdSimpleUnsigned(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = (a[i] & 0xff) * (b[i] & 0xff); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdComplexUnsigned(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdComplexUnsigned(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecReplicateScalar [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: public static final int testDotProdComplexUnsigned(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = (((a[i] & 0xff) + 1) & 0xff) * (((b[i] & 0xff) + 1) & 0xff); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdComplexUnsignedCastToSigned(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdComplexUnsignedCastToSigned(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecReplicateScalar [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Int8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: public static final int testDotProdComplexUnsignedCastToSigned(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = ((byte)((a[i] & 0xff) + 1)) * ((byte)((b[i] & 0xff) + 1)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdComplexSignedCastToUnsigned(byte[], byte[]) loop_optimization (before) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> ArrayGet [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> TypeConversion [<>] loop:<> outer_loop:none /// CHECK-DAG: <> Mul [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none /// CHECK-START-ARM64: int other.TestByte.testDotProdComplexSignedCastToUnsigned(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: <> IntConstant 0 loop:none /// CHECK-DAG: <> IntConstant 1 loop:none /// CHECK-IF: hasIsaFeature("sve") and os.environ.get('ART_FORCE_TRY_PREDICATED_SIMD') == 'true' // /// CHECK-DAG: <> VecReplicateScalar [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> VecSetScalars [<>,{{j\d+}}] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecPredWhile [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,{{i\d+}}] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>,{{j\d+}}] loop:none /// CHECK-DAG: VecExtractScalar [<>,{{j\d+}}] loop:none // /// CHECK-ELSE: // /// CHECK-DAG: <> IntConstant 16 loop:none /// CHECK-DAG: <> VecReplicateScalar [<>] loop:none /// CHECK-DAG: <> VecSetScalars [<>] loop:none /// CHECK-DAG: <> Phi [<>,{{i\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> Phi [<>,{{d\d+}}] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecLoad [{{l\d+}},<>] loop:<> outer_loop:none /// CHECK-DAG: <> VecAdd [<>,<>] loop:<> outer_loop:none /// CHECK-DAG: VecDotProd [<>,<>,<>] type:Uint8 loop:<> outer_loop:none /// CHECK-DAG: Add [<>,<>] loop:<> outer_loop:none // /// CHECK-DAG: <> VecReduce [<>] loop:none /// CHECK-DAG: VecExtractScalar [<>] loop:none // /// CHECK-FI: public static final int testDotProdComplexSignedCastToUnsigned(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = ((a[i] + 1) & 0xff) * ((b[i] + 1) & 0xff); s += temp; } return s - 1; } /// CHECK-START-ARM64: int other.TestByte.testDotProdSignedWidening(byte[], byte[]) loop_optimization (after) /// CHECK-DAG: VecDotProd type:Int8 public static final int testDotProdSignedWidening(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = ((short)(a[i])) * ((short)(b[i])); s += temp; } return s - 1; } /// CHECK-START-ARM64: int other.TestByte.testDotProdParamSigned(int, byte[]) loop_optimization (after) /// CHECK-DAG: VecDotProd type:Int8 public static final int testDotProdParamSigned(int x, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = (byte)(x) * b[i]; s += temp; } return s - 1; } /// CHECK-START-ARM64: int other.TestByte.testDotProdParamUnsigned(int, byte[]) loop_optimization (after) /// CHECK-DAG: VecDotProd type:Uint8 public static final int testDotProdParamUnsigned(int x, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = (x & 0xff) * (b[i] & 0xff); s += temp; } return s - 1; } // No DOTPROD cases. /// CHECK-START: int other.TestByte.testDotProdIntParam(int, byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdIntParam(int x, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = b[i] * (x); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSignedToChar(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSignedToChar(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = ((char)(a[i])) * ((char)(b[i])); s += temp; } return s - 1; } // Cases when result of Mul is type-converted are not supported. /// CHECK-START: int other.TestByte.testDotProdSimpleCastToSignedByte(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleCastToSignedByte(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { byte temp = (byte)(a[i] * b[i]); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleCastToUnsignedByte(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleCastToUnsignedByte(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { s += (a[i] * b[i]) & 0xff; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsignedCastToSignedByte(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleUnsignedCastToSignedByte(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { byte temp = (byte)((a[i] & 0xff) * (b[i] & 0xff)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsignedCastToUnsignedByte(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleUnsignedCastToUnsignedByte(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { s += ((a[i] & 0xff) * (b[i] & 0xff)) & 0xff; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleCastToShort(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleCastToShort(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { short temp = (short)(a[i] * b[i]); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleCastToChar(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleCastToChar(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { char temp = (char)(a[i] * b[i]); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsignedCastToShort(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleUnsignedCastToShort(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { short temp = (short)((a[i] & 0xff) * (b[i] & 0xff)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsignedCastToChar(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleUnsignedCastToChar(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { char temp = (char)((a[i] & 0xff) * (b[i] & 0xff)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdSimpleUnsignedCastToLong(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdSimpleUnsignedCastToLong(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { long temp = (long)((a[i] & 0xff) * (b[i] & 0xff)); s += temp; } return s - 1; } /// CHECK-START: int other.TestByte.testDotProdUnsignedSigned(byte[], byte[]) loop_optimization (after) /// CHECK-NOT: VecDotProd public static final int testDotProdUnsignedSigned(byte[] a, byte[] b) { int s = 1; for (int i = 0; i < b.length; i++) { int temp = (a[i] & 0xff) * b[i]; s += temp; } return s - 1; } private static void expectEquals(int expected, int result) { if (expected != result) { throw new Error("Expected: " + expected + ", found: " + result); } } private static void testDotProd(byte[] b1, byte[] b2, int[] results) { expectEquals(results[0], testDotProdSimple(b1, b2)); expectEquals(results[1], testDotProdComplex(b1, b2)); expectEquals(results[2], testDotProdSimpleUnsigned(b1, b2)); expectEquals(results[3], testDotProdComplexUnsigned(b1, b2)); expectEquals(results[4], testDotProdComplexUnsignedCastToSigned(b1, b2)); expectEquals(results[5], testDotProdComplexSignedCastToUnsigned(b1, b2)); expectEquals(results[6], testDotProdSignedWidening(b1, b2)); expectEquals(results[7], testDotProdParamSigned(-128, b2)); expectEquals(results[8], testDotProdParamUnsigned(-128, b2)); expectEquals(results[9], testDotProdIntParam(-128, b2)); expectEquals(results[10], testDotProdSignedToChar(b1, b2)); expectEquals(results[11], testDotProdSimpleCastToSignedByte(b1, b2)); expectEquals(results[12], testDotProdSimpleCastToUnsignedByte(b1, b2)); expectEquals(results[13], testDotProdSimpleUnsignedCastToSignedByte(b1, b2)); expectEquals(results[14], testDotProdSimpleUnsignedCastToUnsignedByte(b1, b2)); expectEquals(results[15], testDotProdSimpleCastToShort(b1, b2)); expectEquals(results[16], testDotProdSimpleCastToChar(b1, b2)); expectEquals(results[17], testDotProdSimpleUnsignedCastToShort(b1, b2)); expectEquals(results[18], testDotProdSimpleUnsignedCastToChar(b1, b2)); expectEquals(results[19], testDotProdSimpleUnsignedCastToLong(b1, b2)); expectEquals(results[20], testDotProdUnsignedSigned(b1, b2)); } public static void run() { byte[] b1_1 = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 127, 127, 127, 127 }; byte[] b2_1 = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 127, 127, 127, 127 }; int[] results_1 = { 64516, 65548, 64516, 65548, 65548, 65548, 64516, -65024, 65024, -65024, 64516, 4, 4, 4, 4, 64516, 64516, 64516, 64516, 64516, 64516 }; testDotProd(b1_1, b2_1, results_1); byte[] b1_2 = { 127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 127, 127, 127, 127 }; byte[] b2_2 = { 127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 127, 127, 127, 127 }; int[] results_2 = { 80645, 81931, 80645, 81931, 81931, 81931, 80645, -81280, 81280, -81280, 80645, 5, 5, 5, 5, 80645, 80645, 80645, 80645, 80645, 80645 }; testDotProd(b1_2, b2_2, results_2); byte[] b1_3 = { -128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -128, -128, -128, -128 }; byte[] b2_3 = { 127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 127, 127, 127, 127 }; int[] results_3 = { -81280, 81291, 81280, 82571, 81291, 82571, -81280, -81280, 81280, -81280, 41534080, -640, 640, -640, 640, -81280, 246400, 81280, 81280, 81280, 81280 }; testDotProd(b1_3, b2_3, results_3); byte[] b1_4 = { -128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -128, -128, -128, -128 }; byte[] b2_4 = { -128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -128, -128, -128, -128 }; int[] results_4 = { 81920, 80656, 81920, 83216, 80656, 83216, 81920, 81920, 81920, 81920, -83804160, 0, 0, 0, 0, 81920, 81920, 81920, 81920, 81920, -81920 }; testDotProd(b1_4, b2_4, results_4); } public static void main(String[] args) { run(); } }