/* * Copyright (c) 2022, 2024, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * @test * @bug 8205592 8339252 8341260 * @summary Verify {double, float, float16}Value methods work * @modules jdk.incubator.vector * @library /test/lib * @key randomness * @build jdk.test.lib.RandomFactory * @run main DoubleFloatValueTests */ import jdk.test.lib.RandomFactory; import java.math.BigDecimal; import java.util.Random; import jdk.incubator.vector.Float16; public class DoubleFloatValueTests { private static final BigDecimal HALF = BigDecimal.valueOf(5, 1); private static final BigDecimal EPS = BigDecimal.valueOf(1, 10_000); private static BigDecimal nextHalfUp(double v) { BigDecimal bv = new BigDecimal(v); BigDecimal ulp = new BigDecimal(Math.ulp(v)); return bv.add(ulp.multiply(HALF)); } private static BigDecimal nextHalfDown(double v) { BigDecimal bv = new BigDecimal(v); BigDecimal ulp = new BigDecimal(v - Math.nextDown(v)); return bv.subtract(ulp.multiply(HALF)); } private static BigDecimal nextHalfUp(float v) { BigDecimal bv = new BigDecimal(v); BigDecimal ulp = new BigDecimal(Math.ulp(v)); return bv.add(ulp.multiply(HALF)); } private static BigDecimal nextHalfDown(float v) { BigDecimal bv = new BigDecimal(v); BigDecimal ulp = new BigDecimal(v - Math.nextDown(v)); return bv.subtract(ulp.multiply(HALF)); } private static BigDecimal nextHalfUp(Float16 v) { BigDecimal bv = new BigDecimal(v.doubleValue()); BigDecimal ulp = new BigDecimal(Float16.ulp(v).doubleValue()); return bv.add(ulp.multiply(HALF)); } private static BigDecimal nextHalfDown(Float16 v) { BigDecimal bv = new BigDecimal(v.doubleValue()); BigDecimal ulp = new BigDecimal(v.doubleValue() - Float16.nextDown(v).doubleValue()); return bv.subtract(ulp.multiply(HALF)); } private static String toDecHexString(double v) { return v + " (" + Double.toHexString(v) + ")"; } private static String toDecHexString(float v) { return v + " (" + Float.toHexString(v) + ")"; } private static String toDecHexString(Float16 v) { return v + " (" + Float16.toHexString(v) + ")"; } private static void checkDouble(BigDecimal bd, double exp) { double res = bd.doubleValue(); if (exp != res ) { String message = "Bad conversion: got " + toDecHexString(res) + ", expected " + toDecHexString(exp); throw new RuntimeException(message); } } private static void checkFloat(BigDecimal bv, float exp) { float res = bv.floatValue(); if (exp != res ) { String message = "Bad conversion: got " + toDecHexString(res) + ", expected " + toDecHexString(exp); throw new RuntimeException(message); } } private static void checkFloat16(BigDecimal bv, Float16 exp) { Float16 res = Float16.valueOf(bv); // bv.float16Value(); if (exp.floatValue() != res.floatValue()) { String message = "Bad conversion: got " + toDecHexString(res) + ", expected " + toDecHexString(exp); throw new RuntimeException(message); } } private static boolean isOdd(int n) { return (n & 0x1) != 0; } private static void testDoubleValueNearMinValue() { for (int n = 0; n < 100; ++n) { BigDecimal b = nextHalfUp(n * Double.MIN_VALUE); checkDouble(b, ((n + 1) / 2 * 2) * Double.MIN_VALUE); checkDouble(b.subtract(EPS), n * Double.MIN_VALUE); checkDouble(b.add(EPS), (n + 1) * Double.MIN_VALUE); } } private static void testFloatValueNearMinValue() { for (int n = 0; n < 100; ++n) { BigDecimal b = nextHalfUp(n * Float.MIN_VALUE); checkFloat(b, ((n + 1) / 2 * 2) * Float.MIN_VALUE); checkFloat(b.subtract(EPS), n * Float.MIN_VALUE); checkFloat(b.add(EPS), (n + 1) * Float.MIN_VALUE); } } private static void testFloat16ValueNearMinValue() { for (int n = 0; n < 100; ++n) { BigDecimal b = nextHalfUp(Float16.multiply(Float16.valueOf(n), Float16.MIN_VALUE)); checkFloat16(b, Float16.multiply(Float16.valueOf((n + 1) / 2 * 2), Float16.MIN_VALUE)); checkFloat16(b.subtract(EPS), Float16.multiply(Float16.valueOf(n), Float16.MIN_VALUE)); checkFloat16(b.add(EPS), Float16.multiply(Float16.valueOf(n + 1), Float16.MIN_VALUE)); } } private static void testDoubleValueNearMinNormal() { double v = Double.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkDouble(bv, isOdd(n) ? Math.nextDown(v) : v); checkDouble(bv.subtract(EPS), Math.nextDown(v)); checkDouble(bv.add(EPS), v); v = Math.nextDown(v); } v = Double.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfUp(v); checkDouble(bv, isOdd(n) ? Math.nextUp(v) : v); checkDouble(bv.subtract(EPS), v); checkDouble(bv.add(EPS), Math.nextUp(v)); v = Math.nextUp(v); } } private static void testFloatValueNearMinNormal() { float v = Float.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkFloat(bv, isOdd(n) ? Math.nextDown(v) : v); checkFloat(bv.subtract(EPS), Math.nextDown(v)); checkFloat(bv.add(EPS), v); v = Math.nextDown(v); } v = Float.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfUp(v); checkFloat(bv, isOdd(n) ? Math.nextUp(v) : v); checkFloat(bv.subtract(EPS), v); checkFloat(bv.add(EPS), Math.nextUp(v)); v = Math.nextUp(v); } } private static void testFloat16ValueNearMinNormal() { Float16 v = Float16.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkFloat16(bv, isOdd(n) ? Float16.nextDown(v) : v); checkFloat16(bv.subtract(EPS), Float16.nextDown(v)); checkFloat16(bv.add(EPS), v); v = Float16.nextDown(v); } v = Float16.MIN_NORMAL; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfUp(v); checkFloat16(bv, isOdd(n) ? Float16.nextUp(v) : v); checkFloat16(bv.subtract(EPS), v); checkFloat16(bv.add(EPS), Float16.nextUp(v)); v = Float16.nextUp(v); } } private static void testDoubleValueNearMaxValue() { double v = Double.MAX_VALUE; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkDouble(bv, isOdd(n) ? v : Math.nextDown(v)); checkDouble(bv.subtract(EPS), Math.nextDown(v)); checkDouble(bv.add(EPS), v); v = Math.nextDown(v); } BigDecimal bv = nextHalfUp(Double.MAX_VALUE); checkDouble(bv, Double.POSITIVE_INFINITY); checkDouble(bv.subtract(EPS), Double.MAX_VALUE); checkDouble(bv.add(EPS), Double.POSITIVE_INFINITY); } private static void testFloatValueNearMaxValue() { float v = Float.MAX_VALUE; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkFloat(bv, isOdd(n) ? v : Math.nextDown(v)); checkFloat(bv.subtract(EPS), Math.nextDown(v)); checkFloat(bv.add(EPS), v); v = Math.nextDown(v); } BigDecimal bv = nextHalfUp(Float.MAX_VALUE); checkFloat(bv, Float.POSITIVE_INFINITY); checkFloat(bv.subtract(EPS), Float.MAX_VALUE); checkFloat(bv.add(EPS), Float.POSITIVE_INFINITY); } private static void testFloat16ValueNearMaxValue() { Float16 v = Float16.MAX_VALUE; for (int n = 0; n < 100; ++n) { BigDecimal bv = nextHalfDown(v); checkFloat16(bv, isOdd(n) ? v : Float16.nextDown(v)); checkFloat16(bv.subtract(EPS), Float16.nextDown(v)); checkFloat16(bv.add(EPS), v); v = Float16.nextDown(v); } BigDecimal bv = nextHalfUp(Float16.MAX_VALUE); checkFloat16(bv, Float16.POSITIVE_INFINITY); checkFloat16(bv.subtract(EPS), Float16.MAX_VALUE); checkFloat16(bv.add(EPS), Float16.POSITIVE_INFINITY); } private static void testDoubleValueRandom() { Random r = RandomFactory.getRandom(); for (int i = 0; i < 10_000; ++i) { double v = r.nextDouble(-Double.MAX_VALUE, Double.MAX_VALUE); checkDouble(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { double v = r.nextDouble(-1e9, 1e9); checkDouble(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { double v = r.nextDouble(-1e6, 1e6); checkDouble(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { double v = r.nextDouble(-1e-6, 1e-6); checkDouble(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { double v = r.nextDouble(-1e-9, 1e-9); checkDouble(new BigDecimal(v), v); } } private static void testFloatValueRandom() { Random r = RandomFactory.getRandom(); for (int i = 0; i < 10_000; ++i) { float v = r.nextFloat(-Float.MAX_VALUE, Float.MAX_VALUE); checkFloat(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { float v = r.nextFloat(-1e9f, 1e9f); checkFloat(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { float v = r.nextFloat(-1e6f, 1e6f); checkFloat(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { float v = r.nextFloat(-1e-6f, 1e-6f); checkFloat(new BigDecimal(v), v); } for (int i = 0; i < 10_000; ++i) { float v = r.nextFloat(-1e-9f, 1e-9f); checkFloat(new BigDecimal(v), v); } } private static void testFloat16ValueRandom() { Random r = RandomFactory.getRandom(); for (int i = 0; i < 10_000; ++i) { Float16 v = Float16.valueOf(r.nextFloat(-Float16.MAX_VALUE.floatValue(), Float16.MAX_VALUE.floatValue())); checkFloat16(new BigDecimal(v.floatValue()), v); } for (int i = 0; i < 10_000; ++i) { Float16 v = Float16.valueOf(r.nextFloat(-1e4f, 1e4f)); checkFloat16(new BigDecimal(v.floatValue()), v); } for (int i = 0; i < 10_000; ++i) { Float16 v = Float16.valueOf(r.nextFloat(-1e3f, 1e3f)); checkFloat16(new BigDecimal(v.floatValue()), v); } for (int i = 0; i < 10_000; ++i) { Float16 v = Float16.valueOf(r.nextFloat(-1e-3f, 1e-3f)); checkFloat16(new BigDecimal(v.floatValue()), v); } for (int i = 0; i < 10_000; ++i) { Float16 v = Float16.valueOf(r.nextFloat(-1e-4f, 1e-4f)); checkFloat16(new BigDecimal(v.floatValue()), v); } } private static void testDoubleValueExtremes() { checkDouble(BigDecimal.valueOf(1, 1000), 0.0); checkDouble(BigDecimal.valueOf(-1, 1000), -0.0); checkDouble(BigDecimal.valueOf(1, -1000), Double.POSITIVE_INFINITY); checkDouble(BigDecimal.valueOf(-1, -1000), Double.NEGATIVE_INFINITY); } private static void testFloatValueExtremes() { checkFloat(BigDecimal.valueOf(1, 1000), 0.0f); checkFloat(BigDecimal.valueOf(-1, 1000), -0.0f); checkFloat(BigDecimal.valueOf(1, -1000), Float.POSITIVE_INFINITY); checkFloat(BigDecimal.valueOf(-1, -1000), Float.NEGATIVE_INFINITY); } private static void testFloat16ValueExtremes() { checkFloat16(BigDecimal.valueOf(1, 1000), Float16.valueOf(0.0f)); checkFloat16(BigDecimal.valueOf(-1, 1000), Float16.valueOf(-0.0f)); checkFloat16(BigDecimal.valueOf(1, -1000), Float16.POSITIVE_INFINITY); checkFloat16(BigDecimal.valueOf(-1, -1000), Float16.NEGATIVE_INFINITY); } public static void main(String[] args) { testDoubleValueNearMinValue(); testDoubleValueNearMinNormal(); testDoubleValueNearMaxValue(); testDoubleValueRandom(); testDoubleValueExtremes(); testFloatValueNearMinValue(); testFloatValueNearMinNormal(); testFloatValueNearMaxValue(); testFloatValueRandom(); testFloatValueExtremes(); testFloat16ValueNearMinValue(); testFloat16ValueNearMinNormal(); testFloat16ValueNearMaxValue(); testFloat16ValueRandom(); testFloat16ValueExtremes(); } }