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