Blender  V2.93
BLI_math_base_test.cc
Go to the documentation of this file.
1 /* Apache License, Version 2.0 */
2 
3 #include "testing/testing.h"
4 
5 #include "BLI_math.h"
6 
7 /* In tests below, when we are using -1.0f as max_diff value, we actually turn the function into a
8  * pure-ULP one. */
9 
10 /* Put this here, since we cannot use BLI_assert() in inline math files it seems... */
11 TEST(math_base, CompareFFRelativeValid)
12 {
13  EXPECT_TRUE(sizeof(float) == sizeof(int));
14 }
15 
16 TEST(math_base, CompareFFRelativeNormal)
17 {
18  float f1 = 1.99999988f; /* *(float *)&(*(int *)&f2 - 1) */
19  float f2 = 2.00000000f;
20  float f3 = 2.00000048f; /* *(float *)&(*(int *)&f2 + 2) */
21  float f4 = 2.10000000f; /* *(float *)&(*(int *)&f2 + 419430) */
22 
23  const float max_diff = FLT_EPSILON * 0.1f;
24 
25  EXPECT_TRUE(compare_ff_relative(f1, f2, max_diff, 1));
26  EXPECT_TRUE(compare_ff_relative(f2, f1, max_diff, 1));
27 
28  EXPECT_TRUE(compare_ff_relative(f3, f2, max_diff, 2));
29  EXPECT_TRUE(compare_ff_relative(f2, f3, max_diff, 2));
30 
31  EXPECT_FALSE(compare_ff_relative(f3, f2, max_diff, 1));
32  EXPECT_FALSE(compare_ff_relative(f2, f3, max_diff, 1));
33 
34  EXPECT_FALSE(compare_ff_relative(f3, f2, -1.0f, 1));
35  EXPECT_FALSE(compare_ff_relative(f2, f3, -1.0f, 1));
36 
37  EXPECT_TRUE(compare_ff_relative(f3, f2, -1.0f, 2));
38  EXPECT_TRUE(compare_ff_relative(f2, f3, -1.0f, 2));
39 
40  EXPECT_FALSE(compare_ff_relative(f4, f2, max_diff, 64));
41  EXPECT_FALSE(compare_ff_relative(f2, f4, max_diff, 64));
42 
43  EXPECT_TRUE(compare_ff_relative(f1, f3, max_diff, 64));
44  EXPECT_TRUE(compare_ff_relative(f3, f1, max_diff, 64));
45 }
46 
47 TEST(math_base, CompareFFRelativeZero)
48 {
49  float f0 = 0.0f;
50  float f1 = 4.2038954e-045f; /* *(float *)&(*(int *)&f0 + 3) */
51 
52  float fn0 = -0.0f;
53  float fn1 = -2.8025969e-045f; /* *(float *)&(*(int *)&fn0 - 2) */
54 
55  const float max_diff = FLT_EPSILON * 0.1f;
56 
57  EXPECT_TRUE(compare_ff_relative(f0, f1, -1.0f, 3));
58  EXPECT_TRUE(compare_ff_relative(f1, f0, -1.0f, 3));
59 
60  EXPECT_FALSE(compare_ff_relative(f0, f1, -1.0f, 1));
61  EXPECT_FALSE(compare_ff_relative(f1, f0, -1.0f, 1));
62 
63  EXPECT_TRUE(compare_ff_relative(fn0, fn1, -1.0f, 8));
64  EXPECT_TRUE(compare_ff_relative(fn1, fn0, -1.0f, 8));
65 
66  EXPECT_TRUE(compare_ff_relative(f0, f1, max_diff, 1));
67  EXPECT_TRUE(compare_ff_relative(f1, f0, max_diff, 1));
68 
69  EXPECT_TRUE(compare_ff_relative(fn0, f0, max_diff, 1));
70  EXPECT_TRUE(compare_ff_relative(f0, fn0, max_diff, 1));
71 
72  EXPECT_TRUE(compare_ff_relative(f0, fn1, max_diff, 1));
73  EXPECT_TRUE(compare_ff_relative(fn1, f0, max_diff, 1));
74 
75  /* Note: in theory, this should return false, since 0.0f and -0.0f have 0x80000000 diff,
76  * but overflow in subtraction seems to break something here
77  * (abs(*(int *)&fn0 - *(int *)&f0) == 0x80000000 == fn0), probably because int32 cannot
78  * hold this abs value. this is yet another illustration of why one shall never use (near-)zero
79  * floats in pure-ULP comparison. */
80  // EXPECT_FALSE(compare_ff_relative(fn0, f0, -1.0f, 1024));
81  // EXPECT_FALSE(compare_ff_relative(f0, fn0, -1.0f, 1024));
82 
83  EXPECT_FALSE(compare_ff_relative(fn0, f1, -1.0f, 1024));
84  EXPECT_FALSE(compare_ff_relative(f1, fn0, -1.0f, 1024));
85 }
86 
87 TEST(math_base, Log2FloorU)
88 {
89  EXPECT_EQ(log2_floor_u(0), 0);
90  EXPECT_EQ(log2_floor_u(1), 0);
91  EXPECT_EQ(log2_floor_u(2), 1);
92  EXPECT_EQ(log2_floor_u(3), 1);
93  EXPECT_EQ(log2_floor_u(4), 2);
94  EXPECT_EQ(log2_floor_u(5), 2);
95  EXPECT_EQ(log2_floor_u(6), 2);
96  EXPECT_EQ(log2_floor_u(7), 2);
97  EXPECT_EQ(log2_floor_u(8), 3);
98  EXPECT_EQ(log2_floor_u(9), 3);
99  EXPECT_EQ(log2_floor_u(123456), 16);
100 }
101 
102 TEST(math_base, Log2CeilU)
103 {
104  EXPECT_EQ(log2_ceil_u(0), 0);
105  EXPECT_EQ(log2_ceil_u(1), 0);
106  EXPECT_EQ(log2_ceil_u(2), 1);
107  EXPECT_EQ(log2_ceil_u(3), 2);
108  EXPECT_EQ(log2_ceil_u(4), 2);
109  EXPECT_EQ(log2_ceil_u(5), 3);
110  EXPECT_EQ(log2_ceil_u(6), 3);
111  EXPECT_EQ(log2_ceil_u(7), 3);
112  EXPECT_EQ(log2_ceil_u(8), 3);
113  EXPECT_EQ(log2_ceil_u(9), 4);
114  EXPECT_EQ(log2_ceil_u(123456), 17);
115 }
116 
117 TEST(math_base, CeilPowerOf10)
118 {
121  EXPECT_EQ(ceil_power_of_10(1e-6f), 1e-6f);
122  EXPECT_NEAR(ceil_power_of_10(100.1f), 1000.0f, 1e-4f);
123  EXPECT_NEAR(ceil_power_of_10(99.9f), 100.0f, 1e-4f);
124 }
125 
126 TEST(math_base, FloorPowerOf10)
127 {
130  EXPECT_EQ(floor_power_of_10(1e-6f), 1e-6f);
131  EXPECT_NEAR(floor_power_of_10(100.1f), 100.0f, 1e-4f);
132  EXPECT_NEAR(floor_power_of_10(99.9f), 10.0f, 1e-4f);
133 }
EXPECT_EQ(BLI_expr_pylike_eval(expr, nullptr, 0, &result), EXPR_PYLIKE_INVALID)
MINLINE unsigned int log2_ceil_u(unsigned int x)
float floor_power_of_10(float f)
Definition: math_base.c:91
MINLINE int compare_ff_relative(float a, float b, const float max_diff, const int max_ulps)
MINLINE unsigned int log2_floor_u(unsigned int x)
float ceil_power_of_10(float f)
Definition: math_base.c:109
TEST(math_base, CompareFFRelativeValid)