1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2016 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 /** 18*795d594fSAndroid Build Coastguard Worker * A few tests of Math.abs for floating-point data. 19*795d594fSAndroid Build Coastguard Worker * 20*795d594fSAndroid Build Coastguard Worker * Note, as a "quality of implementation", rather than pure "spec compliance", 21*795d594fSAndroid Build Coastguard Worker * we require that Math.abs() clears the sign bit (but changes nothing else) 22*795d594fSAndroid Build Coastguard Worker * for all numbers, including NaN (signaling NaN may become quiet though). 23*795d594fSAndroid Build Coastguard Worker */ 24*795d594fSAndroid Build Coastguard Worker public class TestFpAbs { 25*795d594fSAndroid Build Coastguard Worker 26*795d594fSAndroid Build Coastguard Worker private final static boolean isDalvik = 27*795d594fSAndroid Build Coastguard Worker System.getProperty("java.vm.name").equals("Dalvik"); 28*795d594fSAndroid Build Coastguard Worker 29*795d594fSAndroid Build Coastguard Worker private static final int SPQUIET = 1 << 22; 30*795d594fSAndroid Build Coastguard Worker private static final long DPQUIET = 1L << 51; 31*795d594fSAndroid Build Coastguard Worker 32*795d594fSAndroid Build Coastguard Worker /// CHECK-START: float TestFpAbs.$opt$noinline$absSP(float) builder (after) 33*795d594fSAndroid Build Coastguard Worker /// CHECK-DAG: <<Result:f\d+>> Abs 34*795d594fSAndroid Build Coastguard Worker /// CHECK-DAG: Return [<<Result>>] $opt$noinline$absSP(float f)35*795d594fSAndroid Build Coastguard Worker private static float $opt$noinline$absSP(float f) { 36*795d594fSAndroid Build Coastguard Worker return Math.abs(f); 37*795d594fSAndroid Build Coastguard Worker } 38*795d594fSAndroid Build Coastguard Worker 39*795d594fSAndroid Build Coastguard Worker /// CHECK-START: double TestFpAbs.$opt$noinline$absDP(double) builder (after) 40*795d594fSAndroid Build Coastguard Worker /// CHECK-DAG: <<Result:d\d+>> Abs 41*795d594fSAndroid Build Coastguard Worker /// CHECK-DAG: Return [<<Result>>] $opt$noinline$absDP(double d)42*795d594fSAndroid Build Coastguard Worker private static double $opt$noinline$absDP(double d) { 43*795d594fSAndroid Build Coastguard Worker return Math.abs(d); 44*795d594fSAndroid Build Coastguard Worker } 45*795d594fSAndroid Build Coastguard Worker main()46*795d594fSAndroid Build Coastguard Worker public static void main() { 47*795d594fSAndroid Build Coastguard Worker // A few obvious numbers. 48*795d594fSAndroid Build Coastguard Worker for (float f = -100.0f; f < 0.0f; f += 0.5f) { 49*795d594fSAndroid Build Coastguard Worker expectEqualsSP(-f, $opt$noinline$absSP(f)); 50*795d594fSAndroid Build Coastguard Worker } 51*795d594fSAndroid Build Coastguard Worker for (float f = 0.0f; f <= 100.0f; f += 0.5f) { 52*795d594fSAndroid Build Coastguard Worker expectEqualsSP(f, $opt$noinline$absSP(f)); 53*795d594fSAndroid Build Coastguard Worker } 54*795d594fSAndroid Build Coastguard Worker for (float f = -1.5f; f <= -1.499f; f = Math.nextAfter(f, Float.POSITIVE_INFINITY)) { 55*795d594fSAndroid Build Coastguard Worker expectEqualsSP(-f, $opt$noinline$absSP(f)); 56*795d594fSAndroid Build Coastguard Worker } 57*795d594fSAndroid Build Coastguard Worker for (float f = 1.499f; f <= 1.5f; f = Math.nextAfter(f, Float.POSITIVE_INFINITY)) { 58*795d594fSAndroid Build Coastguard Worker expectEqualsSP(f, $opt$noinline$absSP(f)); 59*795d594fSAndroid Build Coastguard Worker } 60*795d594fSAndroid Build Coastguard Worker 61*795d594fSAndroid Build Coastguard Worker // Zero 62*795d594fSAndroid Build Coastguard Worker expectEquals32(0, Float.floatToRawIntBits($opt$noinline$absSP(+0.0f))); 63*795d594fSAndroid Build Coastguard Worker expectEquals32(0, Float.floatToRawIntBits($opt$noinline$absSP(-0.0f))); 64*795d594fSAndroid Build Coastguard Worker 65*795d594fSAndroid Build Coastguard Worker // Inf. 66*795d594fSAndroid Build Coastguard Worker expectEqualsSP(Float.POSITIVE_INFINITY, $opt$noinline$absSP(Float.NEGATIVE_INFINITY)); 67*795d594fSAndroid Build Coastguard Worker expectEqualsSP(Float.POSITIVE_INFINITY, $opt$noinline$absSP(Float.POSITIVE_INFINITY)); 68*795d594fSAndroid Build Coastguard Worker 69*795d594fSAndroid Build Coastguard Worker // A few NaN numbers. 70*795d594fSAndroid Build Coastguard Worker int[] spnans = { 71*795d594fSAndroid Build Coastguard Worker 0x7f800001, // signaling 72*795d594fSAndroid Build Coastguard Worker 0x7fa00000, 73*795d594fSAndroid Build Coastguard Worker 0x7fbfffff, 74*795d594fSAndroid Build Coastguard Worker 0x7fc00000, // quiet 75*795d594fSAndroid Build Coastguard Worker 0x7fc00001, 76*795d594fSAndroid Build Coastguard Worker 0x7fffffff, 77*795d594fSAndroid Build Coastguard Worker 0xff800001, // signaling 78*795d594fSAndroid Build Coastguard Worker 0xffa00000, 79*795d594fSAndroid Build Coastguard Worker 0xffbfffff, 80*795d594fSAndroid Build Coastguard Worker 0xffc00000, // quiet 81*795d594fSAndroid Build Coastguard Worker 0xffffffff 82*795d594fSAndroid Build Coastguard Worker }; 83*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < spnans.length; i++) { 84*795d594fSAndroid Build Coastguard Worker float f = Float.intBitsToFloat(spnans[i]); 85*795d594fSAndroid Build Coastguard Worker expectEqualsNaN32( 86*795d594fSAndroid Build Coastguard Worker spnans[i] & Integer.MAX_VALUE, 87*795d594fSAndroid Build Coastguard Worker Float.floatToRawIntBits($opt$noinline$absSP(f))); 88*795d594fSAndroid Build Coastguard Worker } 89*795d594fSAndroid Build Coastguard Worker 90*795d594fSAndroid Build Coastguard Worker // A few obvious numbers. 91*795d594fSAndroid Build Coastguard Worker for (double d = -100.0; d < 0.0; d += 0.5) { 92*795d594fSAndroid Build Coastguard Worker expectEqualsDP(-d, $opt$noinline$absDP(d)); 93*795d594fSAndroid Build Coastguard Worker } 94*795d594fSAndroid Build Coastguard Worker for (double d = 0.0; d <= 100.0; d += 0.5) { 95*795d594fSAndroid Build Coastguard Worker expectEqualsDP(d, $opt$noinline$absDP(d)); 96*795d594fSAndroid Build Coastguard Worker } 97*795d594fSAndroid Build Coastguard Worker for (double d = -1.5d; d <= -1.49999999999d; d = Math.nextAfter(d, Double.POSITIVE_INFINITY)) { 98*795d594fSAndroid Build Coastguard Worker expectEqualsDP(-d, $opt$noinline$absDP(d)); 99*795d594fSAndroid Build Coastguard Worker } 100*795d594fSAndroid Build Coastguard Worker for (double d = 1.49999999999d; d <= 1.5; d = Math.nextAfter(d, Double.POSITIVE_INFINITY)) { 101*795d594fSAndroid Build Coastguard Worker expectEqualsDP(d, $opt$noinline$absDP(d)); 102*795d594fSAndroid Build Coastguard Worker } 103*795d594fSAndroid Build Coastguard Worker 104*795d594fSAndroid Build Coastguard Worker // Zero 105*795d594fSAndroid Build Coastguard Worker expectEquals64(0L, Double.doubleToRawLongBits($opt$noinline$absDP(+0.0f))); 106*795d594fSAndroid Build Coastguard Worker expectEquals64(0L, Double.doubleToRawLongBits($opt$noinline$absDP(-0.0f))); 107*795d594fSAndroid Build Coastguard Worker 108*795d594fSAndroid Build Coastguard Worker // Inf. 109*795d594fSAndroid Build Coastguard Worker expectEqualsDP(Double.POSITIVE_INFINITY, $opt$noinline$absDP(Double.NEGATIVE_INFINITY)); 110*795d594fSAndroid Build Coastguard Worker expectEqualsDP(Double.POSITIVE_INFINITY, $opt$noinline$absDP(Double.POSITIVE_INFINITY)); 111*795d594fSAndroid Build Coastguard Worker 112*795d594fSAndroid Build Coastguard Worker // A few NaN numbers. 113*795d594fSAndroid Build Coastguard Worker long[] dpnans = { 114*795d594fSAndroid Build Coastguard Worker 0x7ff0000000000001L, 115*795d594fSAndroid Build Coastguard Worker 0x7ff4000000000000L, 116*795d594fSAndroid Build Coastguard Worker 0x7ff8000000000000L, 117*795d594fSAndroid Build Coastguard Worker 0x7fffffffffffffffL, 118*795d594fSAndroid Build Coastguard Worker 0xfff0000000000001L, 119*795d594fSAndroid Build Coastguard Worker 0xfff4000000000000L, 120*795d594fSAndroid Build Coastguard Worker 0xfff8000000000000L, 121*795d594fSAndroid Build Coastguard Worker 0xffffffffffffffffL 122*795d594fSAndroid Build Coastguard Worker }; 123*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < dpnans.length; i++) { 124*795d594fSAndroid Build Coastguard Worker double d = Double.longBitsToDouble(dpnans[i]); 125*795d594fSAndroid Build Coastguard Worker expectEqualsNaN64( 126*795d594fSAndroid Build Coastguard Worker dpnans[i] & Long.MAX_VALUE, 127*795d594fSAndroid Build Coastguard Worker Double.doubleToRawLongBits($opt$noinline$absDP(d))); 128*795d594fSAndroid Build Coastguard Worker } 129*795d594fSAndroid Build Coastguard Worker 130*795d594fSAndroid Build Coastguard Worker System.out.println("TestFpAbs passed"); 131*795d594fSAndroid Build Coastguard Worker } 132*795d594fSAndroid Build Coastguard Worker expectEquals32(int expected, int result)133*795d594fSAndroid Build Coastguard Worker private static void expectEquals32(int expected, int result) { 134*795d594fSAndroid Build Coastguard Worker if (expected != result) { 135*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: 0x" + Integer.toHexString(expected) 136*795d594fSAndroid Build Coastguard Worker + ", found: 0x" + Integer.toHexString(result)); 137*795d594fSAndroid Build Coastguard Worker } 138*795d594fSAndroid Build Coastguard Worker } 139*795d594fSAndroid Build Coastguard Worker 140*795d594fSAndroid Build Coastguard Worker // We allow that an expected NaN result has become quiet. expectEqualsNaN32(int expected, int result)141*795d594fSAndroid Build Coastguard Worker private static void expectEqualsNaN32(int expected, int result) { 142*795d594fSAndroid Build Coastguard Worker if (expected != result && (expected | SPQUIET) != result) { 143*795d594fSAndroid Build Coastguard Worker if (!isDalvik) { 144*795d594fSAndroid Build Coastguard Worker // If not on ART, relax the expected value more towards just 145*795d594fSAndroid Build Coastguard Worker // "spec compliance" and allow sign bit to remain set for NaN. 146*795d594fSAndroid Build Coastguard Worker if (expected == (result & Integer.MAX_VALUE)) { 147*795d594fSAndroid Build Coastguard Worker return; 148*795d594fSAndroid Build Coastguard Worker } 149*795d594fSAndroid Build Coastguard Worker } 150*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: 0x" + Integer.toHexString(expected) 151*795d594fSAndroid Build Coastguard Worker + ", found: 0x" + Integer.toHexString(result)); 152*795d594fSAndroid Build Coastguard Worker } 153*795d594fSAndroid Build Coastguard Worker } 154*795d594fSAndroid Build Coastguard Worker expectEquals64(long expected, long result)155*795d594fSAndroid Build Coastguard Worker private static void expectEquals64(long expected, long result) { 156*795d594fSAndroid Build Coastguard Worker if (expected != result) { 157*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: 0x" + Long.toHexString(expected) 158*795d594fSAndroid Build Coastguard Worker + ", found: 0x" + Long.toHexString(result)); 159*795d594fSAndroid Build Coastguard Worker } 160*795d594fSAndroid Build Coastguard Worker } 161*795d594fSAndroid Build Coastguard Worker 162*795d594fSAndroid Build Coastguard Worker // We allow that an expected NaN result has become quiet. expectEqualsNaN64(long expected, long result)163*795d594fSAndroid Build Coastguard Worker private static void expectEqualsNaN64(long expected, long result) { 164*795d594fSAndroid Build Coastguard Worker if (expected != result && (expected | DPQUIET) != result) { 165*795d594fSAndroid Build Coastguard Worker if (!isDalvik) { 166*795d594fSAndroid Build Coastguard Worker // If not on ART, relax the expected value more towards just 167*795d594fSAndroid Build Coastguard Worker // "spec compliance" and allow sign bit to remain set for NaN. 168*795d594fSAndroid Build Coastguard Worker if (expected == (result & Long.MAX_VALUE)) { 169*795d594fSAndroid Build Coastguard Worker return; 170*795d594fSAndroid Build Coastguard Worker } 171*795d594fSAndroid Build Coastguard Worker } 172*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: 0x" + Long.toHexString(expected) 173*795d594fSAndroid Build Coastguard Worker + ", found: 0x" + Long.toHexString(result)); 174*795d594fSAndroid Build Coastguard Worker } 175*795d594fSAndroid Build Coastguard Worker } 176*795d594fSAndroid Build Coastguard Worker expectEqualsSP(float expected, float result)177*795d594fSAndroid Build Coastguard Worker private static void expectEqualsSP(float expected, float result) { 178*795d594fSAndroid Build Coastguard Worker if (expected != result) { 179*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: " + expected + ", found: " + result); 180*795d594fSAndroid Build Coastguard Worker } 181*795d594fSAndroid Build Coastguard Worker } 182*795d594fSAndroid Build Coastguard Worker expectEqualsDP(double expected, double result)183*795d594fSAndroid Build Coastguard Worker private static void expectEqualsDP(double expected, double result) { 184*795d594fSAndroid Build Coastguard Worker if (expected != result) { 185*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: " + expected + ", found: " + result); 186*795d594fSAndroid Build Coastguard Worker } 187*795d594fSAndroid Build Coastguard Worker } 188*795d594fSAndroid Build Coastguard Worker } 189