Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorCampbell Barton <ideasman42@gmail.com>2020-11-13 08:58:14 +0300
committerCampbell Barton <ideasman42@gmail.com>2020-11-13 09:05:46 +0300
commitccf8df66fea3650b3d43ed079f9eec4cf35e99f6 (patch)
treeb0fef6285945c04849aa8a0af6f3ab032c255181 /source/blender/blenlib
parent40b2ce5ea73b4b7235435aa2e2ef7dc7742084a0 (diff)
BLI_math: add floor_power_of_10, ceil_power_of_10
Add utility functions to get the floor/ceiling of a float value to the next power of 10.
Diffstat (limited to 'source/blender/blenlib')
-rw-r--r--source/blender/blenlib/BLI_math_base.h3
-rw-r--r--source/blender/blenlib/intern/math_base.c36
-rw-r--r--source/blender/blenlib/tests/BLI_math_base_test.cc18
3 files changed, 57 insertions, 0 deletions
diff --git a/source/blender/blenlib/BLI_math_base.h b/source/blender/blenlib/BLI_math_base.h
index d2bb60717ca..0ec65999006 100644
--- a/source/blender/blenlib/BLI_math_base.h
+++ b/source/blender/blenlib/BLI_math_base.h
@@ -219,6 +219,9 @@ MINLINE unsigned int round_db_to_uint_clamp(double a);
int pow_i(int base, int exp);
double double_round(double x, int ndigits);
+float floor_power_of_10(float f);
+float ceil_power_of_10(float f);
+
#ifdef BLI_MATH_GCC_WARN_PRAGMA
# pragma GCC diagnostic pop
#endif
diff --git a/source/blender/blenlib/intern/math_base.c b/source/blender/blenlib/intern/math_base.c
index 30267661ae7..2bf06371740 100644
--- a/source/blender/blenlib/intern/math_base.c
+++ b/source/blender/blenlib/intern/math_base.c
@@ -79,3 +79,39 @@ double double_round(double x, int ndigits)
/* if computation resulted in overflow, raise OverflowError */
return z;
}
+
+/**
+ * Floor to the nearest power of 10, e.g.:
+ * - 15.0 -> 10.0
+ * - 0.015 -> 0.01
+ * - 1.0 -> 1.0
+ *
+ * \param f: Value to floor, must be over 0.0.
+ * \note If we wanted to support signed values we could if this becomes necessary.
+ */
+float floor_power_of_10(float f)
+{
+ BLI_assert(!(f < 0.0f));
+ if (f != 0.0f) {
+ return 1.0f / (powf(10.0f, ceilf(log10f(1.0f / f))));
+ }
+ return 0.0f;
+}
+
+/**
+ * Ceiling to the nearest power of 10, e.g.:
+ * - 15.0 -> 100.0
+ * - 0.015 -> 0.1
+ * - 1.0 -> 1.0
+ *
+ * \param f: Value to ceiling, must be over 0.0.
+ * \note If we wanted to support signed values we could if this becomes necessary.
+ */
+float ceil_power_of_10(float f)
+{
+ BLI_assert(!(f < 0.0f));
+ if (f != 0.0f) {
+ return 1.0f / (powf(10.0f, floorf(log10f(1.0f / f))));
+ }
+ return 0.0f;
+}
diff --git a/source/blender/blenlib/tests/BLI_math_base_test.cc b/source/blender/blenlib/tests/BLI_math_base_test.cc
index dc20c75576d..d006a2eb59a 100644
--- a/source/blender/blenlib/tests/BLI_math_base_test.cc
+++ b/source/blender/blenlib/tests/BLI_math_base_test.cc
@@ -113,3 +113,21 @@ TEST(math_base, Log2CeilU)
EXPECT_EQ(log2_ceil_u(9), 4);
EXPECT_EQ(log2_ceil_u(123456), 17);
}
+
+TEST(math_base, CeilPowerOf10)
+{
+ EXPECT_EQ(ceil_power_of_10(0), 0);
+ EXPECT_EQ(ceil_power_of_10(1), 1);
+ EXPECT_EQ(ceil_power_of_10(1e-6f), 1e-6f);
+ EXPECT_NEAR(ceil_power_of_10(100.1f), 1000.0f, 1e-4f);
+ EXPECT_NEAR(ceil_power_of_10(99.9f), 100.0f, 1e-4f);
+}
+
+TEST(math_base, FloorPowerOf10)
+{
+ EXPECT_EQ(floor_power_of_10(0), 0);
+ EXPECT_EQ(floor_power_of_10(1), 1);
+ EXPECT_EQ(floor_power_of_10(1e-6f), 1e-6f);
+ EXPECT_NEAR(floor_power_of_10(100.1f), 100.0f, 1e-4f);
+ EXPECT_NEAR(floor_power_of_10(99.9f), 10.0f, 1e-4f);
+}