| /* |
| * Copyright (C) 2012 Intel Corporation |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| |
| #include <gtest/gtest.h> |
| #include "sky/engine/wtf/MathExtras.h" |
| |
| namespace { |
| |
| TEST(MathExtrasTest, Lrint) |
| { |
| EXPECT_EQ(-8, lrint(-7.5)); |
| EXPECT_EQ(-8, lrint(-8.5)); |
| EXPECT_EQ(0, lrint(-0.5)); |
| EXPECT_EQ(0, lrint(0.5)); |
| EXPECT_EQ(0, lrint(-0.5)); |
| EXPECT_EQ(1, lrint(1.3)); |
| EXPECT_EQ(2, lrint(1.7)); |
| EXPECT_EQ(0, lrint(0)); |
| EXPECT_EQ(0, lrint(-0)); |
| if (sizeof(long int) == 8) { |
| // Largest double number with 0.5 precision and one halfway rounding case below. |
| EXPECT_EQ(pow(2.0, 52), lrint(pow(2.0, 52) - 0.5)); |
| EXPECT_EQ(pow(2.0, 52) - 2, lrint(pow(2.0, 52) - 1.5)); |
| // Smallest double number with 0.5 precision and one halfway rounding case above. |
| EXPECT_EQ(-pow(2.0, 52), lrint(-pow(2.0, 52) + 0.5)); |
| EXPECT_EQ(-pow(2.0, 52) + 2, lrint(-pow(2.0, 52) + 1.5)); |
| } |
| } |
| |
| TEST(MathExtrasTest, clampToIntLong) |
| { |
| if (sizeof(long) == sizeof(int)) |
| return; |
| |
| long maxInt = std::numeric_limits<int>::max(); |
| long minInt = std::numeric_limits<int>::min(); |
| long overflowInt = maxInt + 1; |
| long underflowInt = minInt - 1; |
| |
| EXPECT_GT(overflowInt, maxInt); |
| EXPECT_LT(underflowInt, minInt); |
| |
| EXPECT_EQ(maxInt, clampTo<int>(maxInt)); |
| EXPECT_EQ(minInt, clampTo<int>(minInt)); |
| |
| EXPECT_EQ(maxInt, clampTo<int>(overflowInt)); |
| EXPECT_EQ(minInt, clampTo<int>(underflowInt)); |
| } |
| |
| TEST(MathExtrasTest, clampToIntLongLong) |
| { |
| long long maxInt = std::numeric_limits<int>::max(); |
| long long minInt = std::numeric_limits<int>::min(); |
| long long overflowInt = maxInt + 1; |
| long long underflowInt = minInt - 1; |
| |
| EXPECT_GT(overflowInt, maxInt); |
| EXPECT_LT(underflowInt, minInt); |
| |
| EXPECT_EQ(maxInt, clampTo<int>(maxInt)); |
| EXPECT_EQ(minInt, clampTo<int>(minInt)); |
| |
| EXPECT_EQ(maxInt, clampTo<int>(overflowInt)); |
| EXPECT_EQ(minInt, clampTo<int>(underflowInt)); |
| } |
| |
| TEST(MathExtrasTest, clampToIntegerFloat) |
| { |
| // This test is inaccurate as floats will round the min / max integer |
| // due to the narrow mantissa. However it will properly checks within |
| // (close to the extreme) and outside the integer range. |
| float maxInt = std::numeric_limits<int>::max(); |
| float minInt = std::numeric_limits<int>::min(); |
| float overflowInt = maxInt * 1.1; |
| float underflowInt = minInt * 1.1; |
| |
| EXPECT_GT(overflowInt, maxInt); |
| EXPECT_LT(underflowInt, minInt); |
| |
| // If maxInt == 2^31 - 1 (ie on I32 architecture), the closest float used to represent it is 2^31. |
| EXPECT_NEAR(clampToInteger(maxInt), maxInt, 1); |
| EXPECT_EQ(minInt, clampToInteger(minInt)); |
| |
| EXPECT_NEAR(clampToInteger(overflowInt), maxInt, 1); |
| EXPECT_EQ(minInt, clampToInteger(underflowInt)); |
| } |
| |
| TEST(MathExtrasTest, clampToIntegerDouble) |
| { |
| double maxInt = std::numeric_limits<int>::max(); |
| double minInt = std::numeric_limits<int>::min(); |
| double overflowInt = maxInt + 1; |
| double underflowInt = minInt - 1; |
| |
| EXPECT_GT(overflowInt, maxInt); |
| EXPECT_LT(underflowInt, minInt); |
| |
| EXPECT_EQ(maxInt, clampToInteger(maxInt)); |
| EXPECT_EQ(minInt, clampToInteger(minInt)); |
| |
| EXPECT_EQ(clampToInteger(overflowInt), maxInt); |
| EXPECT_EQ(clampToInteger(underflowInt), minInt); |
| } |
| |
| TEST(MathExtrasTest, clampToFloat) |
| { |
| double maxFloat = std::numeric_limits<float>::max(); |
| double minFloat = -maxFloat; |
| double overflowFloat = maxFloat * 1.1; |
| double underflowFloat = minFloat * 1.1; |
| |
| EXPECT_GT(overflowFloat, maxFloat); |
| EXPECT_LT(underflowFloat, minFloat); |
| |
| EXPECT_EQ(maxFloat, clampToFloat(maxFloat)); |
| EXPECT_EQ(minFloat, clampToFloat(minFloat)); |
| |
| EXPECT_EQ(maxFloat, clampToFloat(overflowFloat)); |
| EXPECT_EQ(minFloat, clampToFloat(underflowFloat)); |
| |
| EXPECT_EQ(maxFloat, clampToFloat(std::numeric_limits<float>::infinity())); |
| EXPECT_EQ(minFloat, clampToFloat(-std::numeric_limits<float>::infinity())); |
| } |
| |
| TEST(MathExtrasTest, clampToUnsignedLong) |
| { |
| if (sizeof(unsigned long) == sizeof(unsigned)) |
| return; |
| |
| unsigned long maxUnsigned = std::numeric_limits<unsigned>::max(); |
| unsigned long overflowUnsigned = maxUnsigned + 1; |
| |
| EXPECT_GT(overflowUnsigned, maxUnsigned); |
| |
| EXPECT_EQ(maxUnsigned, clampTo<unsigned>(maxUnsigned)); |
| |
| EXPECT_EQ(maxUnsigned, clampTo<unsigned>(overflowUnsigned)); |
| EXPECT_EQ(0u, clampTo<unsigned>(-1)); |
| } |
| |
| TEST(MathExtrasTest, clampToUnsignedLongLong) |
| { |
| unsigned long long maxUnsigned = std::numeric_limits<unsigned>::max(); |
| unsigned long long overflowUnsigned = maxUnsigned + 1; |
| |
| EXPECT_GT(overflowUnsigned, maxUnsigned); |
| |
| EXPECT_EQ(maxUnsigned, clampTo<unsigned>(maxUnsigned)); |
| |
| EXPECT_EQ(maxUnsigned, clampTo<unsigned>(overflowUnsigned)); |
| EXPECT_EQ(0u, clampTo<unsigned>(-1)); |
| } |
| |
| // Make sure that various +-inf cases are handled properly (they aren't |
| // by default on VS). |
| TEST(MathExtrasTest, infinityMath) |
| { |
| double posInf = std::numeric_limits<double>::infinity(); |
| double negInf = -std::numeric_limits<double>::infinity(); |
| double nan = std::numeric_limits<double>::quiet_NaN(); |
| |
| EXPECT_EQ(M_PI_4, atan2(posInf, posInf)); |
| EXPECT_EQ(3.0 * M_PI_4, atan2(posInf, negInf)); |
| EXPECT_EQ(-M_PI_4, atan2(negInf, posInf)); |
| EXPECT_EQ(-3.0 * M_PI_4, atan2(negInf, negInf)); |
| |
| EXPECT_EQ(0.0, fmod(0.0, posInf)); |
| EXPECT_EQ(7.0, fmod(7.0, posInf)); |
| EXPECT_EQ(-7.0, fmod(-7.0, posInf)); |
| EXPECT_EQ(0.0, fmod(0.0, negInf)); |
| EXPECT_EQ(7.0, fmod(7.0, negInf)); |
| EXPECT_EQ(-7.0, fmod(-7.0, negInf)); |
| |
| EXPECT_EQ(1.0, pow(5.0, 0.0)); |
| EXPECT_EQ(1.0, pow(-5.0, 0.0)); |
| EXPECT_EQ(1.0, pow(nan, 0.0)); |
| } |
| |
| } // namespace |