xref: /aosp_15_r20/art/test/057-math-intrinsics/src/CopySignTest.java (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2024 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker // Note that $opt$ is a marker for the optimizing compiler to test
18*795d594fSAndroid Build Coastguard Worker // it does compile the method, and that $noinline$ is a marker to
19*795d594fSAndroid Build Coastguard Worker // test that it does not inline it.
20*795d594fSAndroid Build Coastguard Worker 
21*795d594fSAndroid Build Coastguard Worker public class CopySignTest {
22*795d594fSAndroid Build Coastguard Worker 
main()23*795d594fSAndroid Build Coastguard Worker   public static void main() {
24*795d594fSAndroid Build Coastguard Worker     copySignFloat();
25*795d594fSAndroid Build Coastguard Worker     copySignDouble();
26*795d594fSAndroid Build Coastguard Worker   }
27*795d594fSAndroid Build Coastguard Worker 
$opt$noinline$copySignFloat(float a, float b)28*795d594fSAndroid Build Coastguard Worker   private static float $opt$noinline$copySignFloat(float a, float b) {
29*795d594fSAndroid Build Coastguard Worker     return Math.copySign(a, b);
30*795d594fSAndroid Build Coastguard Worker   }
31*795d594fSAndroid Build Coastguard Worker 
copySignFloat()32*795d594fSAndroid Build Coastguard Worker   private static void copySignFloat() {
33*795d594fSAndroid Build Coastguard Worker     float testCases [][] = {
34*795d594fSAndroid Build Coastguard Worker       { +0.0f,
35*795d594fSAndroid Build Coastguard Worker          Float.MIN_VALUE,
36*795d594fSAndroid Build Coastguard Worker          0x0.fffffcP-126f,
37*795d594fSAndroid Build Coastguard Worker          0x0.fffffeP-126f,
38*795d594fSAndroid Build Coastguard Worker          Float.MIN_NORMAL,
39*795d594fSAndroid Build Coastguard Worker          1.0f,
40*795d594fSAndroid Build Coastguard Worker          3.0f,
41*795d594fSAndroid Build Coastguard Worker          0x1.fffffcP+127f,
42*795d594fSAndroid Build Coastguard Worker          Float.MAX_VALUE,
43*795d594fSAndroid Build Coastguard Worker          Float.POSITIVE_INFINITY,
44*795d594fSAndroid Build Coastguard Worker          Float.NaN
45*795d594fSAndroid Build Coastguard Worker       },
46*795d594fSAndroid Build Coastguard Worker       {  Float.NEGATIVE_INFINITY,
47*795d594fSAndroid Build Coastguard Worker         -Float.MAX_VALUE,
48*795d594fSAndroid Build Coastguard Worker         -3.0f,
49*795d594fSAndroid Build Coastguard Worker         -1.0f,
50*795d594fSAndroid Build Coastguard Worker         -Float.MIN_NORMAL,
51*795d594fSAndroid Build Coastguard Worker         -0x0.fffffcP-126f,
52*795d594fSAndroid Build Coastguard Worker         -0x0.fffffeP-126f,
53*795d594fSAndroid Build Coastguard Worker         -Float.MIN_VALUE,
54*795d594fSAndroid Build Coastguard Worker         -0.0f,
55*795d594fSAndroid Build Coastguard Worker          Float.intBitsToFloat(0xFFC00000),  // "negative" NaN
56*795d594fSAndroid Build Coastguard Worker         -0x1.fffffcP+127f
57*795d594fSAndroid Build Coastguard Worker       }
58*795d594fSAndroid Build Coastguard Worker     };
59*795d594fSAndroid Build Coastguard Worker 
60*795d594fSAndroid Build Coastguard Worker     for (int i = 0; i < 2; i++) {
61*795d594fSAndroid Build Coastguard Worker       for (int j = 0; j < 2; j++) {
62*795d594fSAndroid Build Coastguard Worker         for (int m = 0; m < testCases[i].length; m++) {
63*795d594fSAndroid Build Coastguard Worker           for (int n = 0; n < testCases[j].length; n++) {
64*795d594fSAndroid Build Coastguard Worker             float expected = (j == 0 ? 1.0f : -1.0f) * Math.abs(testCases[i][m]);
65*795d594fSAndroid Build Coastguard Worker 
66*795d594fSAndroid Build Coastguard Worker             float calculated = Math.copySign(testCases[i][m], testCases[j][n]);
67*795d594fSAndroid Build Coastguard Worker             if (Float.isNaN(testCases[i][m])) {
68*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, Float.NaN);
69*795d594fSAndroid Build Coastguard Worker             } else if (Float.isNaN(testCases[j][n])) {
70*795d594fSAndroid Build Coastguard Worker               assertEquals(Math.abs(calculated), Math.abs(testCases[i][m]));
71*795d594fSAndroid Build Coastguard Worker             } else {
72*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, expected);
73*795d594fSAndroid Build Coastguard Worker             }
74*795d594fSAndroid Build Coastguard Worker 
75*795d594fSAndroid Build Coastguard Worker             calculated = $opt$noinline$copySignFloat(testCases[i][m], testCases[j][n]);
76*795d594fSAndroid Build Coastguard Worker             if (Float.isNaN(testCases[i][m])) {
77*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, Float.NaN);
78*795d594fSAndroid Build Coastguard Worker             } else if (Float.isNaN(testCases[j][n])) {
79*795d594fSAndroid Build Coastguard Worker               assertEquals(Math.abs(calculated), Math.abs(testCases[i][m]));
80*795d594fSAndroid Build Coastguard Worker             } else {
81*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, expected);
82*795d594fSAndroid Build Coastguard Worker             }
83*795d594fSAndroid Build Coastguard Worker           }
84*795d594fSAndroid Build Coastguard Worker         }
85*795d594fSAndroid Build Coastguard Worker       }
86*795d594fSAndroid Build Coastguard Worker     }
87*795d594fSAndroid Build Coastguard Worker   }
88*795d594fSAndroid Build Coastguard Worker 
$opt$noinline$copySignDouble(double a, double b)89*795d594fSAndroid Build Coastguard Worker   private static double $opt$noinline$copySignDouble(double a, double b) {
90*795d594fSAndroid Build Coastguard Worker     return Math.copySign(a, b);
91*795d594fSAndroid Build Coastguard Worker   }
92*795d594fSAndroid Build Coastguard Worker 
copySignDouble()93*795d594fSAndroid Build Coastguard Worker   private static void copySignDouble() {
94*795d594fSAndroid Build Coastguard Worker     double testCases [][] = {
95*795d594fSAndroid Build Coastguard Worker       { +0.0d,
96*795d594fSAndroid Build Coastguard Worker          Double.MIN_VALUE,
97*795d594fSAndroid Build Coastguard Worker          0x0.ffffffffffffeP-1022,
98*795d594fSAndroid Build Coastguard Worker          0x0.fffffffffffffP-1022,
99*795d594fSAndroid Build Coastguard Worker          Double.MIN_NORMAL,
100*795d594fSAndroid Build Coastguard Worker          1.0d,
101*795d594fSAndroid Build Coastguard Worker          3.0d,
102*795d594fSAndroid Build Coastguard Worker          0x1.ffffffffffffeP+1023,
103*795d594fSAndroid Build Coastguard Worker          Double.MAX_VALUE,
104*795d594fSAndroid Build Coastguard Worker          Double.POSITIVE_INFINITY,
105*795d594fSAndroid Build Coastguard Worker          Double.NaN
106*795d594fSAndroid Build Coastguard Worker       },
107*795d594fSAndroid Build Coastguard Worker       {  Double.NEGATIVE_INFINITY,
108*795d594fSAndroid Build Coastguard Worker         -Double.MAX_VALUE,
109*795d594fSAndroid Build Coastguard Worker         -3.0d,
110*795d594fSAndroid Build Coastguard Worker         -1.0d,
111*795d594fSAndroid Build Coastguard Worker         -Double.MIN_NORMAL,
112*795d594fSAndroid Build Coastguard Worker         -0x0.ffffffffffffeP-1022,
113*795d594fSAndroid Build Coastguard Worker         -0x0.fffffffffffffP-1022,
114*795d594fSAndroid Build Coastguard Worker         -Double.MIN_VALUE,
115*795d594fSAndroid Build Coastguard Worker         -0.0d,
116*795d594fSAndroid Build Coastguard Worker          Double.longBitsToDouble(0xfff8000000000000L),  // "negative" NaN
117*795d594fSAndroid Build Coastguard Worker         -0x1.ffffffffffffeP+1023
118*795d594fSAndroid Build Coastguard Worker       }
119*795d594fSAndroid Build Coastguard Worker     };
120*795d594fSAndroid Build Coastguard Worker 
121*795d594fSAndroid Build Coastguard Worker     for (int i = 0; i < 2; i++) {
122*795d594fSAndroid Build Coastguard Worker       for (int j = 0; j < 2; j++) {
123*795d594fSAndroid Build Coastguard Worker         for (int m = 0; m < testCases[i].length; m++) {
124*795d594fSAndroid Build Coastguard Worker           for (int n = 0; n < testCases[j].length; n++) {
125*795d594fSAndroid Build Coastguard Worker             double expected = (j == 0 ? 1.0f : -1.0f) * Math.abs(testCases[i][m]);
126*795d594fSAndroid Build Coastguard Worker 
127*795d594fSAndroid Build Coastguard Worker             double calculated = Math.copySign(testCases[i][m], testCases[j][n]);
128*795d594fSAndroid Build Coastguard Worker             if (Double.isNaN(testCases[i][m])) {
129*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, Double.NaN);
130*795d594fSAndroid Build Coastguard Worker             } else if (Double.isNaN(testCases[j][n])) {
131*795d594fSAndroid Build Coastguard Worker               assertEquals(Math.abs(calculated), Math.abs(testCases[i][m]));
132*795d594fSAndroid Build Coastguard Worker             } else {
133*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, expected);
134*795d594fSAndroid Build Coastguard Worker             }
135*795d594fSAndroid Build Coastguard Worker 
136*795d594fSAndroid Build Coastguard Worker             calculated = $opt$noinline$copySignDouble(testCases[i][m], testCases[j][n]);
137*795d594fSAndroid Build Coastguard Worker             if (Double.isNaN(testCases[i][m])) {
138*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, Double.NaN);
139*795d594fSAndroid Build Coastguard Worker             } else if (Double.isNaN(testCases[j][n])) {
140*795d594fSAndroid Build Coastguard Worker               assertEquals(Math.abs(calculated), Math.abs(testCases[i][m]));
141*795d594fSAndroid Build Coastguard Worker             } else {
142*795d594fSAndroid Build Coastguard Worker               assertEquals(calculated, expected);
143*795d594fSAndroid Build Coastguard Worker             }
144*795d594fSAndroid Build Coastguard Worker           }
145*795d594fSAndroid Build Coastguard Worker         }
146*795d594fSAndroid Build Coastguard Worker       }
147*795d594fSAndroid Build Coastguard Worker     }
148*795d594fSAndroid Build Coastguard Worker   }
149*795d594fSAndroid Build Coastguard Worker 
assertEquals(float calculated, float expected)150*795d594fSAndroid Build Coastguard Worker   private static void assertEquals(float calculated, float expected) {
151*795d594fSAndroid Build Coastguard Worker     if (0 != Float.compare(calculated, expected)) {
152*795d594fSAndroid Build Coastguard Worker       throw new Error("Expected: " + expected + ", found: " + calculated);
153*795d594fSAndroid Build Coastguard Worker     }
154*795d594fSAndroid Build Coastguard Worker   }
155*795d594fSAndroid Build Coastguard Worker 
assertEquals(double calculated, double expected)156*795d594fSAndroid Build Coastguard Worker   private static void assertEquals(double calculated, double expected) {
157*795d594fSAndroid Build Coastguard Worker     if (0 != Double.compare(calculated, expected)) {
158*795d594fSAndroid Build Coastguard Worker       throw new Error("Expected: " + expected + ", found: " + calculated);
159*795d594fSAndroid Build Coastguard Worker     }
160*795d594fSAndroid Build Coastguard Worker   }
161*795d594fSAndroid Build Coastguard Worker 
162*795d594fSAndroid Build Coastguard Worker }
163