2015-11-23 15:02:19 -08:00
|
|
|
/*
|
2022-11-23 13:29:20 +00:00
|
|
|
* Copyright (c) 2015, 2022, Oracle and/or its affiliates. All rights reserved.
|
2015-11-23 15:02:19 -08:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import sun.java2d.marlin.FloatMath;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @summary Check for correct implementation of FloatMath.ceil/floor
|
|
|
|
* @run main CeilAndFloorTests
|
2016-01-09 14:04:32 +01:00
|
|
|
* @modules java.desktop/sun.java2d.marlin
|
2015-11-23 15:02:19 -08:00
|
|
|
*/
|
|
|
|
public class CeilAndFloorTests {
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
public static String toHexString(double f) {
|
|
|
|
if (!Double.isNaN(f)) {
|
|
|
|
return Double.toHexString(f);
|
|
|
|
} else {
|
|
|
|
return "NaN(0x" + Long.toHexString(Double.doubleToRawLongBits(f)) + ")";
|
|
|
|
}
|
2015-11-23 15:02:19 -08:00
|
|
|
}
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
public static int test(String testName, double input,
|
|
|
|
double result, double expected) {
|
|
|
|
if (Double.compare(expected, result) != 0) {
|
2015-11-23 15:02:19 -08:00
|
|
|
System.err.println("Failure for " + testName + ":\n" +
|
|
|
|
"\tFor input " + input + "\t(" + toHexString(input) + ")\n" +
|
|
|
|
"\texpected " + expected + "\t(" + toHexString(expected) + ")\n" +
|
|
|
|
"\tgot " + result + "\t(" + toHexString(result) + ").");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
public static int test_skip_0(String testName, double input,
|
|
|
|
double result, double expected)
|
2015-11-23 15:02:19 -08:00
|
|
|
{
|
|
|
|
// floor_int does not distinguish +0f and -0f
|
|
|
|
// but it is not critical for Marlin
|
2022-11-23 13:29:20 +00:00
|
|
|
if (Double.compare(expected, result) != 0 && (expected != 0.0))
|
2015-11-23 15:02:19 -08:00
|
|
|
{
|
|
|
|
System.err.println("Failure for " + testName + ":\n" +
|
|
|
|
"\tFor input " + input + "\t(" + toHexString(input) + ")\n" +
|
|
|
|
"\texpected " + expected + "\t(" + toHexString(expected) + ")\n" +
|
|
|
|
"\tgot " + result + "\t(" + toHexString(result) + ").");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
private static int testCeilCase(double input, double expected) {
|
2015-11-23 15:02:19 -08:00
|
|
|
int failures = 0;
|
|
|
|
// int result:
|
|
|
|
failures += test("FloatMath.ceil_int", input, FloatMath.ceil_int(input), (int)expected);
|
|
|
|
return failures;
|
|
|
|
}
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
private static int testFloorCase(double input, double expected) {
|
2015-11-23 15:02:19 -08:00
|
|
|
int failures = 0;
|
|
|
|
// ignore difference between +0f and -0f:
|
|
|
|
failures += test_skip_0("FloatMath.floor_int", input, FloatMath.floor_int(input), (int)expected);
|
|
|
|
return failures;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int nearIntegerTests() {
|
|
|
|
int failures = 0;
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
double [] fixedPoints = {
|
|
|
|
-0.0,
|
|
|
|
0.0,
|
|
|
|
-1.0,
|
|
|
|
1.0,
|
|
|
|
-0x1.0p52,
|
|
|
|
0x1.0p52,
|
2015-11-23 15:02:19 -08:00
|
|
|
-Float.MAX_VALUE,
|
|
|
|
Float.MAX_VALUE,
|
|
|
|
Float.NEGATIVE_INFINITY,
|
|
|
|
Float.POSITIVE_INFINITY,
|
|
|
|
Float.NaN,
|
|
|
|
};
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
for(double fixedPoint : fixedPoints) {
|
2015-11-23 15:02:19 -08:00
|
|
|
failures += testCeilCase(fixedPoint, fixedPoint);
|
|
|
|
failures += testFloorCase(fixedPoint, fixedPoint);
|
|
|
|
}
|
|
|
|
|
|
|
|
for(int i = Float.MIN_EXPONENT; i <= Float.MAX_EXPONENT; i++) {
|
2022-11-23 13:29:20 +00:00
|
|
|
double powerOfTwo = Math.scalb(1.0f, i);
|
|
|
|
double neighborDown = Math.nextDown(powerOfTwo);
|
|
|
|
double neighborUp = Math.nextUp(powerOfTwo);
|
2015-11-23 15:02:19 -08:00
|
|
|
|
|
|
|
if (i < 0) {
|
2022-11-23 13:29:20 +00:00
|
|
|
failures += testCeilCase( powerOfTwo, 1.0);
|
|
|
|
failures += testCeilCase(-powerOfTwo, -0.0);
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
failures += testFloorCase( powerOfTwo, 0.0);
|
|
|
|
failures += testFloorCase(-powerOfTwo, -1.0);
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
failures += testCeilCase( neighborDown, 1.0);
|
|
|
|
failures += testCeilCase(-neighborDown, -0.0);
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
failures += testFloorCase( neighborUp, 0.0);
|
|
|
|
failures += testFloorCase(-neighborUp, -1.0);
|
2015-11-23 15:02:19 -08:00
|
|
|
} else {
|
|
|
|
failures += testCeilCase(powerOfTwo, powerOfTwo);
|
|
|
|
failures += testFloorCase(powerOfTwo, powerOfTwo);
|
|
|
|
|
|
|
|
if (neighborDown==Math.rint(neighborDown)) {
|
|
|
|
failures += testCeilCase( neighborDown, neighborDown);
|
|
|
|
failures += testCeilCase(-neighborDown, -neighborDown);
|
|
|
|
|
|
|
|
failures += testFloorCase( neighborDown, neighborDown);
|
|
|
|
failures += testFloorCase(-neighborDown,-neighborDown);
|
|
|
|
} else {
|
|
|
|
failures += testCeilCase( neighborDown, powerOfTwo);
|
|
|
|
failures += testFloorCase(-neighborDown, -powerOfTwo);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (neighborUp==Math.rint(neighborUp)) {
|
|
|
|
failures += testCeilCase(neighborUp, neighborUp);
|
|
|
|
failures += testCeilCase(-neighborUp, -neighborUp);
|
|
|
|
|
|
|
|
failures += testFloorCase(neighborUp, neighborUp);
|
|
|
|
failures += testFloorCase(-neighborUp, -neighborUp);
|
|
|
|
} else {
|
|
|
|
failures += testFloorCase(neighborUp, powerOfTwo);
|
|
|
|
failures += testCeilCase(-neighborUp, -powerOfTwo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(int i = -(0x10000); i <= 0x10000; i++) {
|
2022-11-23 13:29:20 +00:00
|
|
|
double f = (double) i;
|
|
|
|
double neighborDown = Math.nextDown(f);
|
|
|
|
double neighborUp = Math.nextUp(f);
|
2015-11-23 15:02:19 -08:00
|
|
|
|
|
|
|
failures += testCeilCase( f, f);
|
|
|
|
failures += testCeilCase(-f, -f);
|
|
|
|
|
|
|
|
failures += testFloorCase( f, f);
|
|
|
|
failures += testFloorCase(-f, -f);
|
|
|
|
|
|
|
|
if (Math.abs(f) > 1.0) {
|
|
|
|
failures += testCeilCase( neighborDown, f);
|
|
|
|
failures += testCeilCase(-neighborDown, -f+1);
|
|
|
|
|
|
|
|
failures += testFloorCase( neighborUp, f);
|
|
|
|
failures += testFloorCase(-neighborUp, -f-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return failures;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static int roundingTests() {
|
|
|
|
int failures = 0;
|
2022-11-23 13:29:20 +00:00
|
|
|
double [][] testCases = {
|
|
|
|
{ Float.MIN_VALUE, 1.0},
|
|
|
|
{-Float.MIN_VALUE, -0.0},
|
|
|
|
{ Math.nextDown(Float.MIN_NORMAL), 1.0},
|
|
|
|
{-Math.nextDown(Float.MIN_NORMAL), -0.0},
|
|
|
|
{ Float.MIN_NORMAL, 1.0},
|
|
|
|
{-Float.MIN_NORMAL, -0.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 0.1, 1.0},
|
|
|
|
{-0.1, -0.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 0.5, 1.0},
|
|
|
|
{-0.5, -0.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 1.5, 2.0},
|
|
|
|
{-1.5, -1.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 2.5, 3.0},
|
|
|
|
{-2.5, -2.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 12.3456789, 13.0},
|
|
|
|
{-12.3456789, -12.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ Math.nextDown(1.0), 1.0},
|
|
|
|
{ Math.nextDown(-1.0), -1.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ Math.nextUp(1.0), 2.0},
|
|
|
|
{ Math.nextUp(-1.0), -0.0},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ 0x1.0p22, 0x1.0p22},
|
|
|
|
{-0x1.0p22, -0x1.0p22},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ Math.nextDown(0x1.0p22), 0x1.0p22},
|
|
|
|
{-Math.nextUp(0x1.0p22), -0x1.0p22},
|
2015-11-23 15:02:19 -08:00
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
{ Math.nextUp(0x1.0p22), 0x1.0p22 + 1.0},
|
|
|
|
{-Math.nextDown(0x1.0p22), -0x1.0p22 + 1.0}
|
2015-11-23 15:02:19 -08:00
|
|
|
};
|
|
|
|
|
2022-11-23 13:29:20 +00:00
|
|
|
for(double[] testCase : testCases) {
|
2015-11-23 15:02:19 -08:00
|
|
|
failures += testCeilCase(testCase[0], testCase[1]);
|
|
|
|
failures += testFloorCase(-testCase[0], -testCase[1]);
|
|
|
|
}
|
|
|
|
return failures;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String... args) {
|
|
|
|
int failures = 0;
|
|
|
|
|
|
|
|
System.out.println("nearIntegerTests");
|
|
|
|
failures += nearIntegerTests();
|
|
|
|
|
|
|
|
System.out.println("roundingTests");
|
|
|
|
failures += roundingTests();
|
|
|
|
|
|
|
|
if (failures > 0) {
|
|
|
|
System.err.println("Testing {FloatMath}.ceil/floor incurred "
|
|
|
|
+ failures + " failures.");
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|