1*35238bceSAndroid Build Coastguard Worker /*------------------------------------------------------------------------- 2*35238bceSAndroid Build Coastguard Worker * drawElements Base Portability Library 3*35238bceSAndroid Build Coastguard Worker * ------------------------------------- 4*35238bceSAndroid Build Coastguard Worker * 5*35238bceSAndroid Build Coastguard Worker * Copyright 2015 The Android Open Source Project 6*35238bceSAndroid Build Coastguard Worker * 7*35238bceSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 8*35238bceSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 9*35238bceSAndroid Build Coastguard Worker * You may obtain a copy of the License at 10*35238bceSAndroid Build Coastguard Worker * 11*35238bceSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 12*35238bceSAndroid Build Coastguard Worker * 13*35238bceSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 14*35238bceSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 15*35238bceSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 16*35238bceSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 17*35238bceSAndroid Build Coastguard Worker * limitations under the License. 18*35238bceSAndroid Build Coastguard Worker * 19*35238bceSAndroid Build Coastguard Worker *//*! 20*35238bceSAndroid Build Coastguard Worker * \file 21*35238bceSAndroid Build Coastguard Worker * \brief Testing of deMath functions. 22*35238bceSAndroid Build Coastguard Worker *//*--------------------------------------------------------------------*/ 23*35238bceSAndroid Build Coastguard Worker 24*35238bceSAndroid Build Coastguard Worker #include "deMath.h" 25*35238bceSAndroid Build Coastguard Worker #include "deRandom.h" 26*35238bceSAndroid Build Coastguard Worker 27*35238bceSAndroid Build Coastguard Worker DE_BEGIN_EXTERN_C 28*35238bceSAndroid Build Coastguard Worker conversionToFloatLosesPrecision(int32_t x)29*35238bceSAndroid Build Coastguard Workerstatic bool conversionToFloatLosesPrecision(int32_t x) 30*35238bceSAndroid Build Coastguard Worker { 31*35238bceSAndroid Build Coastguard Worker if (x == -0x7FFFFFFF - 1) 32*35238bceSAndroid Build Coastguard Worker return false; 33*35238bceSAndroid Build Coastguard Worker else if (x < 0) 34*35238bceSAndroid Build Coastguard Worker return conversionToFloatLosesPrecision(-x); 35*35238bceSAndroid Build Coastguard Worker else if (x == 0) 36*35238bceSAndroid Build Coastguard Worker return false; 37*35238bceSAndroid Build Coastguard Worker else if (((uint32_t)x & 0x1) == 0) 38*35238bceSAndroid Build Coastguard Worker return conversionToFloatLosesPrecision(x >> 1); /* remove trailing zeros */ 39*35238bceSAndroid Build Coastguard Worker else 40*35238bceSAndroid Build Coastguard Worker return x > ((1 << 24) - 1); /* remaining part does not fit in the mantissa? */ 41*35238bceSAndroid Build Coastguard Worker } 42*35238bceSAndroid Build Coastguard Worker testSingleInt32ToFloat(int32_t x)43*35238bceSAndroid Build Coastguard Workerstatic void testSingleInt32ToFloat(int32_t x) 44*35238bceSAndroid Build Coastguard Worker { 45*35238bceSAndroid Build Coastguard Worker /* roundTowardsToNegInf(x) <= round(x) <= roundTowardsPosInf(x). */ 46*35238bceSAndroid Build Coastguard Worker /* \note: Need to use inequalities since round(x) returns arbitrary precision floats. */ 47*35238bceSAndroid Build Coastguard Worker DE_TEST_ASSERT(deInt32ToFloatRoundToNegInf(x) <= deInt32ToFloat(x)); 48*35238bceSAndroid Build Coastguard Worker DE_TEST_ASSERT(deInt32ToFloat(x) <= deInt32ToFloatRoundToPosInf(x)); 49*35238bceSAndroid Build Coastguard Worker 50*35238bceSAndroid Build Coastguard Worker /* if precision is lost, floor(x) < ceil(x). Else floor(x) == ceil(x) */ 51*35238bceSAndroid Build Coastguard Worker if (conversionToFloatLosesPrecision(x)) 52*35238bceSAndroid Build Coastguard Worker DE_TEST_ASSERT(deInt32ToFloatRoundToNegInf(x) < deInt32ToFloatRoundToPosInf(x)); 53*35238bceSAndroid Build Coastguard Worker else 54*35238bceSAndroid Build Coastguard Worker DE_TEST_ASSERT(deInt32ToFloatRoundToNegInf(x) == deInt32ToFloatRoundToPosInf(x)); 55*35238bceSAndroid Build Coastguard Worker 56*35238bceSAndroid Build Coastguard Worker /* max one ulp from each other */ 57*35238bceSAndroid Build Coastguard Worker if (deInt32ToFloatRoundToNegInf(x) < deInt32ToFloatRoundToPosInf(x)) 58*35238bceSAndroid Build Coastguard Worker { 59*35238bceSAndroid Build Coastguard Worker union 60*35238bceSAndroid Build Coastguard Worker { 61*35238bceSAndroid Build Coastguard Worker float f; 62*35238bceSAndroid Build Coastguard Worker int32_t u; 63*35238bceSAndroid Build Coastguard Worker } v0, v1; 64*35238bceSAndroid Build Coastguard Worker 65*35238bceSAndroid Build Coastguard Worker v0.f = deInt32ToFloatRoundToNegInf(x); 66*35238bceSAndroid Build Coastguard Worker v1.f = deInt32ToFloatRoundToPosInf(x); 67*35238bceSAndroid Build Coastguard Worker 68*35238bceSAndroid Build Coastguard Worker DE_TEST_ASSERT(v0.u + 1 == v1.u || v0.u == v1.u + 1); 69*35238bceSAndroid Build Coastguard Worker } 70*35238bceSAndroid Build Coastguard Worker } 71*35238bceSAndroid Build Coastguard Worker testInt32ToFloat(void)72*35238bceSAndroid Build Coastguard Workerstatic void testInt32ToFloat(void) 73*35238bceSAndroid Build Coastguard Worker { 74*35238bceSAndroid Build Coastguard Worker const int numIterations = 2500000; 75*35238bceSAndroid Build Coastguard Worker 76*35238bceSAndroid Build Coastguard Worker int sign; 77*35238bceSAndroid Build Coastguard Worker int numBits; 78*35238bceSAndroid Build Coastguard Worker int delta; 79*35238bceSAndroid Build Coastguard Worker int ndx; 80*35238bceSAndroid Build Coastguard Worker deRandom rnd; 81*35238bceSAndroid Build Coastguard Worker 82*35238bceSAndroid Build Coastguard Worker deRandom_init(&rnd, 0xdeadbeefu - 1); 83*35238bceSAndroid Build Coastguard Worker 84*35238bceSAndroid Build Coastguard Worker for (sign = -1; sign < 1; ++sign) 85*35238bceSAndroid Build Coastguard Worker for (numBits = 0; numBits < 32; ++numBits) 86*35238bceSAndroid Build Coastguard Worker for (delta = -2; delta < 3; ++delta) 87*35238bceSAndroid Build Coastguard Worker { 88*35238bceSAndroid Build Coastguard Worker const int64_t x = (int64_t)(sign == -1 ? (-1) : (+1)) * (1LL << (int64_t)numBits) + (int64_t)delta; 89*35238bceSAndroid Build Coastguard Worker 90*35238bceSAndroid Build Coastguard Worker /* would overflow */ 91*35238bceSAndroid Build Coastguard Worker if (x > 0x7FFFFFFF || x < -0x7FFFFFFF - 1) 92*35238bceSAndroid Build Coastguard Worker continue; 93*35238bceSAndroid Build Coastguard Worker 94*35238bceSAndroid Build Coastguard Worker testSingleInt32ToFloat((int32_t)x); 95*35238bceSAndroid Build Coastguard Worker } 96*35238bceSAndroid Build Coastguard Worker 97*35238bceSAndroid Build Coastguard Worker for (ndx = 0; ndx < numIterations; ++ndx) 98*35238bceSAndroid Build Coastguard Worker testSingleInt32ToFloat((int32_t)deRandom_getUint32(&rnd)); 99*35238bceSAndroid Build Coastguard Worker } 100*35238bceSAndroid Build Coastguard Worker deMath_selfTest(void)101*35238bceSAndroid Build Coastguard Workervoid deMath_selfTest(void) 102*35238bceSAndroid Build Coastguard Worker { 103*35238bceSAndroid Build Coastguard Worker /* Test Int32ToFloat*(). */ 104*35238bceSAndroid Build Coastguard Worker testInt32ToFloat(); 105*35238bceSAndroid Build Coastguard Worker } 106*35238bceSAndroid Build Coastguard Worker 107*35238bceSAndroid Build Coastguard Worker DE_END_EXTERN_C 108