1 /* 2 * Copyright (C) 2017 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 /** 18 * Tests for SAD (sum of absolute differences). 19 */ 20 public class Main { 21 22 // TODO: lower precision still coming, b/64091002 23 24 // TODO: consider unsigned SAD too, b/64091002 25 sadChar2Char(char[] s1, char[] s2)26 private static char sadChar2Char(char[] s1, char[] s2) { 27 int min_length = Math.min(s1.length, s2.length); 28 char sad = 0; 29 for (int i = 0; i < min_length; i++) { 30 sad += Math.abs(s1[i] - s2[i]); 31 } 32 return sad; 33 } 34 sadChar2CharAlt(char[] s1, char[] s2)35 private static char sadChar2CharAlt(char[] s1, char[] s2) { 36 int min_length = Math.min(s1.length, s2.length); 37 char sad = 0; 38 for (int i = 0; i < min_length; i++) { 39 char s = s1[i]; 40 char p = s2[i]; 41 sad += s >= p ? s - p : p - s; 42 } 43 return sad; 44 } 45 sadChar2CharAlt2(char[] s1, char[] s2)46 private static char sadChar2CharAlt2(char[] s1, char[] s2) { 47 int min_length = Math.min(s1.length, s2.length); 48 char sad = 0; 49 for (int i = 0; i < min_length; i++) { 50 char s = s1[i]; 51 char p = s2[i]; 52 int x = s - p; 53 if (x < 0) x = -x; 54 sad += x; 55 } 56 return sad; 57 } 58 59 /// CHECK-START: int Main.sadChar2Int(char[], char[]) loop_optimization (before) 60 /// CHECK-DAG: <<Cons0:i\d+>> IntConstant 0 loop:none 61 /// CHECK-DAG: <<Cons1:i\d+>> IntConstant 1 loop:none 62 /// CHECK-DAG: <<Phi1:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop:B\d+>> outer_loop:none 63 /// CHECK-DAG: <<Phi2:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop>> outer_loop:none 64 /// CHECK-DAG: <<Get1:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 65 /// CHECK-DAG: <<Get2:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 66 /// CHECK-DAG: <<Sub:i\d+>> Sub [<<Get1>>,<<Get2>>] loop:<<Loop>> outer_loop:none 67 /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect [<<Sub>>] intrinsic:MathAbsInt loop:<<Loop>> outer_loop:none 68 /// CHECK-DAG: Add [<<Phi2>>,<<Intrin>>] loop:<<Loop>> outer_loop:none 69 /// CHECK-DAG: Add [<<Phi1>>,<<Cons1>>] loop:<<Loop>> outer_loop:none 70 // 71 /// CHECK-START: int Main.sadChar2Int(char[], char[]) loop_optimization (after) 72 /// CHECK-NOT: VecSADAccumulate sadChar2Int(char[] s1, char[] s2)73 private static int sadChar2Int(char[] s1, char[] s2) { 74 int min_length = Math.min(s1.length, s2.length); 75 int sad = 0; 76 for (int i = 0; i < min_length; i++) { 77 sad += Math.abs(s1[i] - s2[i]); 78 } 79 return sad; 80 } 81 82 /// CHECK-START: int Main.sadChar2IntAlt(char[], char[]) loop_optimization (before) 83 /// CHECK-DAG: <<Cons0:i\d+>> IntConstant 0 loop:none 84 /// CHECK-DAG: <<Cons1:i\d+>> IntConstant 1 loop:none 85 /// CHECK-DAG: <<Phi1:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop:B\d+>> outer_loop:none 86 /// CHECK-DAG: <<Phi2:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop>> outer_loop:none 87 /// CHECK-DAG: <<Get1:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 88 /// CHECK-DAG: <<Get2:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 89 /// CHECK-DAG: <<Sub:i\d+>> Sub [<<Get2>>,<<Get1>>] loop:<<Loop>> outer_loop:none 90 /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect [<<Sub>>] intrinsic:MathAbsInt loop:<<Loop>> outer_loop:none 91 /// CHECK-DAG: Add [<<Phi2>>,<<Intrin>>] loop:<<Loop>> outer_loop:none 92 /// CHECK-DAG: Add [<<Phi1>>,<<Cons1>>] loop:<<Loop>> outer_loop:none 93 // 94 /// CHECK-START: int Main.sadChar2IntAlt(char[], char[]) loop_optimization (after) 95 /// CHECK-NOT: VecSADAccumulate sadChar2IntAlt(char[] s1, char[] s2)96 private static int sadChar2IntAlt(char[] s1, char[] s2) { 97 int min_length = Math.min(s1.length, s2.length); 98 int sad = 0; 99 for (int i = 0; i < min_length; i++) { 100 char s = s1[i]; 101 char p = s2[i]; 102 sad += s >= p ? s - p : p - s; 103 } 104 return sad; 105 } 106 107 /// CHECK-START: int Main.sadChar2IntAlt2(char[], char[]) loop_optimization (before) 108 /// CHECK-DAG: <<Cons0:i\d+>> IntConstant 0 loop:none 109 /// CHECK-DAG: <<Cons1:i\d+>> IntConstant 1 loop:none 110 /// CHECK-DAG: <<Phi1:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop:B\d+>> outer_loop:none 111 /// CHECK-DAG: <<Phi2:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop>> outer_loop:none 112 /// CHECK-DAG: <<Get1:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 113 /// CHECK-DAG: <<Get2:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 114 /// CHECK-DAG: <<Sub:i\d+>> Sub [<<Get1>>,<<Get2>>] loop:<<Loop>> outer_loop:none 115 /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect [<<Sub>>] intrinsic:MathAbsInt loop:<<Loop>> outer_loop:none 116 /// CHECK-DAG: Add [<<Phi2>>,<<Intrin>>] loop:<<Loop>> outer_loop:none 117 /// CHECK-DAG: Add [<<Phi1>>,<<Cons1>>] loop:<<Loop>> outer_loop:none 118 // 119 /// CHECK-START: int Main.sadChar2IntAlt2(char[], char[]) loop_optimization (after) 120 /// CHECK-NOT: VecSADAccumulate sadChar2IntAlt2(char[] s1, char[] s2)121 private static int sadChar2IntAlt2(char[] s1, char[] s2) { 122 int min_length = Math.min(s1.length, s2.length); 123 int sad = 0; 124 for (int i = 0; i < min_length; i++) { 125 char s = s1[i]; 126 char p = s2[i]; 127 int x = s - p; 128 if (x < 0) x = -x; 129 sad += x; 130 } 131 return sad; 132 } 133 134 /// CHECK-START: long Main.sadChar2Long(char[], char[]) loop_optimization (before) 135 /// CHECK-DAG: <<Cons0:i\d+>> IntConstant 0 loop:none 136 /// CHECK-DAG: <<Cons1:i\d+>> IntConstant 1 loop:none 137 /// CHECK-DAG: <<ConsL:j\d+>> LongConstant 0 loop:none 138 /// CHECK-DAG: <<Phi1:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop:B\d+>> outer_loop:none 139 /// CHECK-DAG: <<Phi2:j\d+>> Phi [<<ConsL>>,{{j\d+}}] loop:<<Loop>> outer_loop:none 140 /// CHECK-DAG: <<Get1:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 141 /// CHECK-DAG: <<Get2:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 142 /// CHECK-DAG: <<Cnv1:j\d+>> TypeConversion [<<Get1>>] loop:<<Loop>> outer_loop:none 143 /// CHECK-DAG: <<Cnv2:j\d+>> TypeConversion [<<Get2>>] loop:<<Loop>> outer_loop:none 144 /// CHECK-DAG: <<Sub:j\d+>> Sub [<<Cnv1>>,<<Cnv2>>] loop:<<Loop>> outer_loop:none 145 /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect [<<Sub>>] intrinsic:MathAbsLong loop:<<Loop>> outer_loop:none 146 /// CHECK-DAG: Add [<<Phi2>>,<<Intrin>>] loop:<<Loop>> outer_loop:none 147 /// CHECK-DAG: Add [<<Phi1>>,<<Cons1>>] loop:<<Loop>> outer_loop:none 148 // 149 /// CHECK-START: long Main.sadChar2Long(char[], char[]) loop_optimization (after) 150 /// CHECK-NOT: VecSADAccumulate sadChar2Long(char[] s1, char[] s2)151 private static long sadChar2Long(char[] s1, char[] s2) { 152 int min_length = Math.min(s1.length, s2.length); 153 long sad = 0; 154 for (int i = 0; i < min_length; i++) { 155 long x = s1[i]; 156 long y = s2[i]; 157 sad += Math.abs(x - y); 158 } 159 return sad; 160 } 161 162 /// CHECK-START: long Main.sadChar2LongAt1(char[], char[]) loop_optimization (before) 163 /// CHECK-DAG: <<Cons0:i\d+>> IntConstant 0 loop:none 164 /// CHECK-DAG: <<Cons1:i\d+>> IntConstant 1 loop:none 165 /// CHECK-DAG: <<ConsL:j\d+>> LongConstant 1 loop:none 166 /// CHECK-DAG: <<Phi1:i\d+>> Phi [<<Cons0>>,{{i\d+}}] loop:<<Loop:B\d+>> outer_loop:none 167 /// CHECK-DAG: <<Phi2:j\d+>> Phi [<<ConsL>>,{{j\d+}}] loop:<<Loop>> outer_loop:none 168 /// CHECK-DAG: <<Get1:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 169 /// CHECK-DAG: <<Get2:c\d+>> ArrayGet [{{l\d+}},<<Phi1>>] loop:<<Loop>> outer_loop:none 170 /// CHECK-DAG: <<Cnv1:j\d+>> TypeConversion [<<Get1>>] loop:<<Loop>> outer_loop:none 171 /// CHECK-DAG: <<Cnv2:j\d+>> TypeConversion [<<Get2>>] loop:<<Loop>> outer_loop:none 172 /// CHECK-DAG: <<Sub:j\d+>> Sub [<<Cnv1>>,<<Cnv2>>] loop:<<Loop>> outer_loop:none 173 /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect [<<Sub>>] intrinsic:MathAbsLong loop:<<Loop>> outer_loop:none 174 /// CHECK-DAG: Add [<<Phi2>>,<<Intrin>>] loop:<<Loop>> outer_loop:none 175 /// CHECK-DAG: Add [<<Phi1>>,<<Cons1>>] loop:<<Loop>> outer_loop:none 176 // 177 /// CHECK-START: long Main.sadChar2LongAt1(char[], char[]) loop_optimization (after) 178 /// CHECK-NOT: VecSADAccumulate sadChar2LongAt1(char[] s1, char[] s2)179 private static long sadChar2LongAt1(char[] s1, char[] s2) { 180 int min_length = Math.min(s1.length, s2.length); 181 long sad = 1; // starts at 1 182 for (int i = 0; i < min_length; i++) { 183 long x = s1[i]; 184 long y = s2[i]; 185 sad += Math.abs(x - y); 186 } 187 return sad; 188 } 189 main(String[] args)190 public static void main(String[] args) { 191 // Cross-test the two most extreme values individually. 192 char[] s1 = { 0, 0x8000, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; 193 char[] s2 = { 0, 0x7fff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; 194 expectEquals(1, sadChar2Char(s1, s2)); 195 expectEquals(1, sadChar2Char(s2, s1)); 196 expectEquals(1, sadChar2CharAlt(s1, s2)); 197 expectEquals(1, sadChar2CharAlt(s2, s1)); 198 expectEquals(1, sadChar2CharAlt2(s1, s2)); 199 expectEquals(1, sadChar2CharAlt2(s2, s1)); 200 expectEquals(1, sadChar2Int(s1, s2)); 201 expectEquals(1, sadChar2Int(s2, s1)); 202 expectEquals(1, sadChar2IntAlt(s1, s2)); 203 expectEquals(1, sadChar2IntAlt(s2, s1)); 204 expectEquals(1, sadChar2IntAlt2(s1, s2)); 205 expectEquals(1, sadChar2IntAlt2(s2, s1)); 206 expectEquals(1L, sadChar2Long(s1, s2)); 207 expectEquals(1L, sadChar2Long(s2, s1)); 208 expectEquals(2L, sadChar2LongAt1(s1, s2)); 209 expectEquals(2L, sadChar2LongAt1(s2, s1)); 210 211 // Use cross-values to test all cases. 212 char[] interesting = { 213 (char) 0x0000, 214 (char) 0x0001, 215 (char) 0x0002, 216 (char) 0x1234, 217 (char) 0x8000, 218 (char) 0x8001, 219 (char) 0x7fff, 220 (char) 0xffff 221 }; 222 int n = interesting.length; 223 int m = n * n + 1; 224 s1 = new char[m]; 225 s2 = new char[m]; 226 int k = 0; 227 for (int i = 0; i < n; i++) { 228 for (int j = 0; j < n; j++) { 229 s1[k] = interesting[i]; 230 s2[k] = interesting[j]; 231 k++; 232 } 233 } 234 s1[k] = 10; 235 s2[k] = 2; 236 expectEquals(56196, sadChar2Char(s1, s2)); 237 expectEquals(56196, sadChar2CharAlt(s1, s2)); 238 expectEquals(56196, sadChar2CharAlt2(s1, s2)); 239 expectEquals(1497988, sadChar2Int(s1, s2)); 240 expectEquals(1497988, sadChar2IntAlt(s1, s2)); 241 expectEquals(1497988, sadChar2IntAlt2(s1, s2)); 242 expectEquals(1497988L, sadChar2Long(s1, s2)); 243 expectEquals(1497989L, sadChar2LongAt1(s1, s2)); 244 245 System.out.println("passed"); 246 } 247 expectEquals(int expected, int result)248 private static void expectEquals(int expected, int result) { 249 if (expected != result) { 250 throw new Error("Expected: " + expected + ", found: " + result); 251 } 252 } 253 expectEquals(long expected, long result)254 private static void expectEquals(long expected, long result) { 255 if (expected != result) { 256 throw new Error("Expected: " + expected + ", found: " + result); 257 } 258 } 259 } 260