MATH-483 Add FastMath special test case code
TODO - enable fail when bugs have been fixed git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1060903 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
6ad3d5edf1
commit
3719af9ea2
|
@ -16,6 +16,9 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.commons.math.util;
|
package org.apache.commons.math.util;
|
||||||
|
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
import org.apache.commons.math.dfp.Dfp;
|
import org.apache.commons.math.dfp.Dfp;
|
||||||
import org.apache.commons.math.dfp.DfpField;
|
import org.apache.commons.math.dfp.DfpField;
|
||||||
import org.apache.commons.math.dfp.DfpMath;
|
import org.apache.commons.math.dfp.DfpMath;
|
||||||
|
@ -31,6 +34,23 @@ public class FastMathTest {
|
||||||
private static final double MAX_ERROR_ULP = 0.51;
|
private static final double MAX_ERROR_ULP = 0.51;
|
||||||
private static final int NUMBER_OF_TRIALS = 1000;
|
private static final int NUMBER_OF_TRIALS = 1000;
|
||||||
|
|
||||||
|
// Values which often need special handling
|
||||||
|
private static final double [] DOUBLE_SPECIAL_VALUES = {
|
||||||
|
-0.0, +0.0, // 1,2
|
||||||
|
Double.NaN, // 3
|
||||||
|
Double.NEGATIVE_INFINITY, Double.POSITIVE_INFINITY, // 4,5
|
||||||
|
Double.MIN_VALUE, Double.MAX_VALUE, // 6,7
|
||||||
|
-Double.MIN_VALUE, -Double.MAX_VALUE, // 8,9
|
||||||
|
};
|
||||||
|
|
||||||
|
private static final float [] FLOAT_SPECIAL_VALUES = {
|
||||||
|
-0.0f, +0.0f, // 1,2
|
||||||
|
Float.NaN, // 3
|
||||||
|
Float.NEGATIVE_INFINITY, Float.POSITIVE_INFINITY, // 4,5
|
||||||
|
Float.MIN_VALUE, Float.MAX_VALUE, // 6,7
|
||||||
|
-Float.MIN_VALUE, -Float.MAX_VALUE, // 8,9
|
||||||
|
};
|
||||||
|
|
||||||
private DfpField field;
|
private DfpField field;
|
||||||
private RandomGenerator generator;
|
private RandomGenerator generator;
|
||||||
|
|
||||||
|
@ -950,6 +970,86 @@ public class FastMathTest {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void reportError(String message) {
|
||||||
|
final boolean fatal = false; // TODO set true once all bugs have been fixed
|
||||||
|
if (fatal) {
|
||||||
|
Assert.fail(message);
|
||||||
|
} else {
|
||||||
|
System.out.println(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static abstract class SpecialComparer {
|
||||||
|
abstract void compareSpecials(Method mathMethod, Method fastMethod) throws Exception;
|
||||||
|
void check(Method mathMethod, float f, Object expected, Object actual, int entry){
|
||||||
|
if (!expected.equals(actual)){
|
||||||
|
reportError(mathMethod.getName()+"(float "+f+") expected "+expected+" actual "+actual+ " entry "+entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void check(Method mathMethod, double d, Object expected, Object actual, int entry){
|
||||||
|
if (!expected.equals(actual)){
|
||||||
|
reportError(mathMethod.getName()+"(float "+d+") expected "+expected+" actual "+actual+ " entry "+entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class CompareFloatSpecials extends SpecialComparer {
|
||||||
|
@Override
|
||||||
|
public void compareSpecials(Method mathMethod, Method fastMethod) throws Exception {
|
||||||
|
int entry = 0;
|
||||||
|
for(float f : FLOAT_SPECIAL_VALUES) {
|
||||||
|
entry++;
|
||||||
|
Object expected = mathMethod.invoke(mathMethod, new Object[]{f});
|
||||||
|
Object actual = fastMethod.invoke(mathMethod, new Object[]{f});
|
||||||
|
check(mathMethod, f, expected, actual, entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class CompareDoubleSpecials extends SpecialComparer {
|
||||||
|
@Override
|
||||||
|
public void compareSpecials(Method mathMethod, Method fastMethod) throws Exception {
|
||||||
|
int entry = 0;
|
||||||
|
for(double d : DOUBLE_SPECIAL_VALUES) {
|
||||||
|
entry++;
|
||||||
|
Object expected = mathMethod.invoke(mathMethod, new Object[]{d});
|
||||||
|
Object actual = fastMethod.invoke(mathMethod, new Object[]{d});
|
||||||
|
check(mathMethod, d, expected, actual, entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void testSpecialCases(Class<?> type, SpecialComparer comparer) throws Exception {
|
||||||
|
Class<?> param[] = new Class<?> [] {type};
|
||||||
|
Method math[] = StrictMath.class.getDeclaredMethods();
|
||||||
|
for(Method mathMethod : math) {
|
||||||
|
Type ret = mathMethod.getGenericReturnType();
|
||||||
|
if (ret.equals(type)){
|
||||||
|
Type []params = mathMethod.getGenericParameterTypes();
|
||||||
|
if (params.length ==1 && params[0].equals(type)) {
|
||||||
|
Method fastMethod = null;
|
||||||
|
String name = mathMethod.getName();
|
||||||
|
try {
|
||||||
|
fastMethod = FastMath.class.getDeclaredMethod(name, param);
|
||||||
|
comparer.compareSpecials(mathMethod, fastMethod);
|
||||||
|
} catch (NoSuchMethodException e) {
|
||||||
|
System.out.println("Cannot find FastMath method corresponding to: "+mathMethod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFloatSpecialCases() throws Exception {
|
||||||
|
testSpecialCases(float.class, new CompareFloatSpecials());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDoubleSpecialCases() throws Exception {
|
||||||
|
testSpecialCases(double.class, new CompareDoubleSpecials());
|
||||||
|
}
|
||||||
|
|
||||||
@Ignore
|
@Ignore
|
||||||
@Test
|
@Test
|
||||||
public void testPerformance() {
|
public void testPerformance() {
|
||||||
|
@ -1157,6 +1257,7 @@ public class FastMathTest {
|
||||||
x += FastMath.expm1(-i / 100000.0);
|
x += FastMath.expm1(-i / 100000.0);
|
||||||
time = System.currentTimeMillis() - time;
|
time = System.currentTimeMillis() - time;
|
||||||
System.out.println("FastMath.expm1 " + time + "\t" + x);
|
System.out.println("FastMath.expm1 " + time + "\t" + x);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue