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   /// CHECK-START: int Main.sad1(int, int) instruction_simplifier$after_inlining (before)
23   /// CHECK-DAG: <<Select:i\d+>> Select
24   /// CHECK-DAG:                 Return [<<Select>>]
25   //
26   /// CHECK-START: int Main.sad1(int, int) instruction_simplifier$after_inlining (after)
27   /// CHECK-DAG: <<Select:i\d+>> Select
28   /// CHECK-DAG:                 Return [<<Select>>]
29   //
30   /// CHECK-START: int Main.sad1(int, int) instruction_simplifier$after_inlining (after)
31   /// CHECK-NOT: InvokeStaticOrDirect intrinsic:MathAbsInt
32   //
33   // NOTE: for direct 32-bit operands, this is not an ABS.
sad1(int x, int y)34   static int sad1(int x, int y) {
35     return x >= y ? x - y : y - x;
36   }
37 
38   /// CHECK-START: int Main.sad2(int, int) instruction_simplifier$after_inlining (before)
39   /// CHECK-DAG: <<Select:i\d+>> Select
40   /// CHECK-DAG:                 Return [<<Select>>]
41   //
42   /// CHECK-START: int Main.sad2(int, int) instruction_simplifier$after_inlining (after)
43   /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect intrinsic:MathAbsInt
44   /// CHECK-DAG:                 Return [<<Intrin>>]
sad2(int x, int y)45   static int sad2(int x, int y) {
46     int diff = x - y;
47     if (diff < 0) diff = -diff;
48     return diff;
49   }
50 
51   /// CHECK-START: int Main.sad3(int, int) instruction_simplifier$after_inlining (before)
52   /// CHECK-DAG: <<Select:i\d+>> Select
53   /// CHECK-DAG:                 Return [<<Select>>]
54   //
55   /// CHECK-START: int Main.sad3(int, int) instruction_simplifier$after_inlining (after)
56   /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect intrinsic:MathAbsInt
57   /// CHECK-DAG:                 Return [<<Intrin>>]
sad3(int x, int y)58   static int sad3(int x, int y) {
59     int diff = x - y;
60     return diff >= 0 ? diff : -diff;
61   }
62 
63   /// CHECK-START: int Main.sad3Alt(int, int) instruction_simplifier$after_inlining (before)
64   /// CHECK-DAG: <<Select:i\d+>> Select
65   /// CHECK-DAG:                 Return [<<Select>>]
66   //
67   /// CHECK-START: int Main.sad3Alt(int, int) instruction_simplifier$after_inlining (after)
68   /// CHECK-DAG: <<Intrin:i\d+>> InvokeStaticOrDirect intrinsic:MathAbsInt
69   /// CHECK-DAG:                 Return [<<Intrin>>]
sad3Alt(int x, int y)70   static int sad3Alt(int x, int y) {
71     int diff = x - y;
72     return 0 <= diff ? diff : -diff;
73   }
74 
75   /// CHECK-START: long Main.sadL1(int, int) instruction_simplifier$after_inlining (before)
76   /// CHECK-DAG: <<Select:j\d+>> Select
77   /// CHECK-DAG:                 Return [<<Select>>]
78   //
79   /// CHECK-START: long Main.sadL1(int, int) instruction_simplifier$after_inlining (after)
80   /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect intrinsic:MathAbsLong
81   /// CHECK-DAG:                 Return [<<Intrin>>]
sadL1(int x, int y)82   static long sadL1(int x, int y) {
83     long xl = x;
84     long yl = y;
85     return xl >= yl ? xl - yl : yl - xl;
86   }
87 
88   /// CHECK-START: long Main.sadL2(int, int) instruction_simplifier$after_inlining (before)
89   /// CHECK-DAG: <<Select:j\d+>> Select
90   /// CHECK-DAG:                 Return [<<Select>>]
91   //
92   /// CHECK-START: long Main.sadL2(int, int) instruction_simplifier$after_inlining (after)
93   /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect intrinsic:MathAbsLong
94   /// CHECK-DAG:                 Return [<<Intrin>>]
sadL2(int x, int y)95   static long sadL2(int x, int y) {
96     long diff = x - y;
97     if (diff < 0L) diff = -diff;
98     return diff;
99   }
100 
101   /// CHECK-START: long Main.sadL3(int, int) instruction_simplifier$after_inlining (before)
102   /// CHECK-DAG: <<Select:j\d+>> Select
103   /// CHECK-DAG:                 Return [<<Select>>]
104   //
105   /// CHECK-START: long Main.sadL3(int, int) instruction_simplifier$after_inlining (after)
106   /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect intrinsic:MathAbsLong
107   /// CHECK-DAG:                 Return [<<Intrin>>]
sadL3(int x, int y)108   static long sadL3(int x, int y) {
109     long diff = x - y;
110     return diff >= 0L ? diff : -diff;
111   }
112 
113   /// CHECK-START: long Main.sadL3Alt(int, int) instruction_simplifier$after_inlining (before)
114   /// CHECK-DAG: <<Select:j\d+>> Select
115   /// CHECK-DAG:                 Return [<<Select>>]
116   //
117   /// CHECK-START: long Main.sadL3Alt(int, int) instruction_simplifier$after_inlining (after)
118   /// CHECK-DAG: <<Intrin:j\d+>> InvokeStaticOrDirect intrinsic:MathAbsLong
119   /// CHECK-DAG:                 Return [<<Intrin>>]
sadL3Alt(int x, int y)120   static long sadL3Alt(int x, int y) {
121     long diff = x - y;
122     return 0L <= diff ? diff : -diff;
123   }
124 
main(String[] args)125   public static void main(String[] args) {
126     // Use cross-values for the interesting values.
127     int[] interesting = {
128       0x00000000, 0x00000001, 0x00007fff, 0x00008000, 0x00008001, 0x0000ffff,
129       0x00010000, 0x00010001, 0x00017fff, 0x00018000, 0x00018001, 0x0001ffff,
130       0x7fff0000, 0x7fff0001, 0x7fff7fff, 0x7fff8000, 0x7fff8001, 0x7fffffff,
131       0x80000000, 0x80000001, 0x80007fff, 0x80008000, 0x80008001, 0x8000ffff,
132       0x80010000, 0x80010001, 0x80017fff, 0x80018000, 0x80018001, 0x8001ffff,
133       0xffff0000, 0xffff0001, 0xffff7fff, 0xffff8000, 0xffff8001, 0xffffffff
134     };
135     for (int i = 0; i < interesting.length; i++) {
136       for (int j = 0; j < interesting.length; j++) {
137         int x = interesting[i];
138         int y = interesting[j];
139         int e1 = x >= y ? x - y : y - x;  // still select
140         expectEquals(e1, sad1(x, y));
141         int e2 = Math.abs(x - y);  // pure abs
142         expectEquals(e2, sad2(x, y));
143         expectEquals(e2, sad3(x, y));
144         expectEquals(e2, sad3Alt(x, y));
145         long eL1 = Math.abs(((long)x) - ((long)y));  // now, different, but abs
146         expectEquals(eL1, sadL1(x, y));
147         long eL2 = Math.abs((long)(x - y));  // also, different, but abs
148         expectEquals(eL2, sadL2(x, y));
149         expectEquals(eL2, sadL3(x, y));
150         expectEquals(eL2, sadL3Alt(x, y));
151       }
152     }
153     System.out.println("passed");
154   }
155 
expectEquals(int expected, int result)156   private static void expectEquals(int expected, int result) {
157     if (expected != result) {
158       throw new Error("Expected: " + expected + ", found: " + result);
159     }
160   }
161 
expectEquals(long expected, long result)162   private static void expectEquals(long expected, long result) {
163     if (expected != result) {
164       throw new Error("Expected: " + expected + ", found: " + result);
165     }
166   }
167 }
168