|
- /*
- * Carla math utils
- * Copyright (C) 2011-2022 Filipe Coelho <falktx@falktx.com>
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of
- * the License, or any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * For a full copy of the GNU General Public License see the doc/GPL.txt file.
- */
-
- #ifndef CARLA_MATH_UTILS_HPP_INCLUDED
- #define CARLA_MATH_UTILS_HPP_INCLUDED
-
- #include "CarlaUtils.hpp"
-
- #include <cmath>
- #include <limits>
-
- // --------------------------------------------------------------------------------------------------------------------
- // math functions (base)
-
- /*
- * Return the lower of 2 values, with 'min' as the minimum possible value (ie, constrain).
- */
- template<typename T>
- static inline
- const T& carla_minConstrained(const T& v1, const T& v2, const T& min) noexcept
- {
- return (v1 <= min || v2 <= min) ? min : (v1 < v2 ? v1 : v2);
- }
-
- /*
- * Return the lower positive of 2 values.
- * If one of the values is zero, returns zero.
- * If one value is negative but the other positive, returns the positive.
- * Returned value is guaranteed to be >= 0.
- */
- template<typename T>
- static inline
- T carla_minPositive(const T& v1, const T& v2) noexcept
- {
- if (v1 == 0 || v2 == 0)
- return 0;
- if (v1 < 0)
- return (v2 > 0) ? v2 : 0;
- if (v2 < 0)
- return (v1 > 0) ? v1 : 0;
- return (v1 < v2) ? v1 : v2;
- }
-
- /*
- * Return the higher of 2 values, with 'max' as the maximum possible value (ie, limit).
- */
- template<typename T>
- static inline
- const T& carla_maxLimited(const T& v1, const T& v2, const T& max) noexcept
- {
- return (v1 >= max || v2 >= max) ? max : (v1 > v2 ? v1 : v2);
- }
-
- /*
- * Return the higher negative of 2 values.
- * If one of the values is zero, returns zero.
- * If one value is positive but the other negative, returns the negative.
- * Returned value is guaranteed to be <= 0.
- */
- template<typename T>
- static inline
- T carla_maxNegative(const T& v1, const T& v2) noexcept
- {
- if (v1 == 0 || v2 == 0)
- return 0;
- if (v1 > 0)
- return (v2 < 0) ? v2 : 0;
- if (v2 > 0)
- return (v1 < 0) ? v1 : 0;
- return (v1 > v2) ? v1 : v2;
- }
-
- /*
- * Return a value between 'min' and 'max'.
- */
- template<typename T>
- static inline
- const T& carla_fixedValue(const T& min, const T& max, const T& value) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(max > min, min);
-
- if (value <= min)
- return min;
- if (value >= max)
- return max;
- return value;
- }
-
- /*
- * Get next power of 2.
- */
- static inline
- uint32_t carla_nextPowerOf2(uint32_t size) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(size > 0, 0);
-
- // http://graphics.stanford.edu/~seander/bithacks.html#RoundUpPowerOf2
- --size;
- size |= size >> 1;
- size |= size >> 2;
- size |= size >> 4;
- size |= size >> 8;
- size |= size >> 16;
- return ++size;
- }
-
- /*
- * Safely check if 2 numbers are equal.
- */
- template<typename T>
- static inline
- bool carla_isEqual(const T& v1, const T& v2)
- {
- return std::abs(v1-v2) < std::numeric_limits<T>::epsilon();
- }
-
- /*
- * Safely check if 2 numbers are not equal.
- */
- template<typename T>
- static inline
- bool carla_isNotEqual(const T& v1, const T& v2)
- {
- return std::abs(v1-v2) >= std::numeric_limits<T>::epsilon();
- }
-
- /*
- * Safely check if a number is zero.
- */
- template<typename T>
- static inline
- bool carla_isZero(const T& value)
- {
- return std::abs(value) < std::numeric_limits<T>::epsilon();
- }
-
- /*
- * Safely check if a number is not zero.
- */
- template<typename T>
- static inline
- bool carla_isNotZero(const T& value)
- {
- return std::abs(value) >= std::numeric_limits<T>::epsilon();
- }
-
- // --------------------------------------------------------------------------------------------------------------------
- // math functions (extended)
-
- /*
- * Add float array values to another float array.
- */
- static inline
- void carla_addFloats(float dest[], const float src[], const std::size_t count) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(dest != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(src != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(count > 0,);
-
- for (std::size_t i=0; i<count; ++i)
- {
- if (!std::isfinite(dest[i]))
- __builtin_unreachable();
- if (!std::isfinite(src[i]))
- __builtin_unreachable();
- *dest++ += *src++;
- }
- }
-
- /*
- * Copy float array values to another float array.
- */
- static inline
- void carla_copyFloats(float dest[], const float src[], const std::size_t count) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(dest != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(src != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(count > 0,);
-
- std::memcpy(dest, src, count*sizeof(float));
- }
-
- /*
- * Fill a float array with a single float value.
- */
- static inline
- void carla_fillFloatsWithSingleValue(float data[], const float& value, const std::size_t count) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(data != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(count > 0,);
-
- if (carla_isZero(value))
- {
- std::memset(data, 0, count*sizeof(float));
- }
- else
- {
- for (std::size_t i=0; i<count; ++i)
- {
- if (!std::isfinite(data[i]))
- __builtin_unreachable();
- *data++ = value;
- }
- }
- }
-
- /*
- * Clear a float array.
- */
- static inline
- void carla_zeroFloats(float floats[], const std::size_t count) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(floats != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(count > 0,);
-
- std::memset(floats, 0, count*sizeof(float));
- }
-
- // --------------------------------------------------------------------------------------------------------------------
-
- /*
- * Fill an array with a fixed value, floating-point version.
- */
- template <>
- inline
- void carla_fill<float>(float data[], const float& value, const std::size_t count) noexcept
- {
- carla_fillFloatsWithSingleValue(data, value, count);
- }
-
- /*
- * Find the highest absolute and normalized value within a float array.
- */
- static inline
- float carla_findMaxNormalizedFloat(const float floats[], const std::size_t count)
- {
- CARLA_SAFE_ASSERT_RETURN(floats != nullptr, 0.0f);
- CARLA_SAFE_ASSERT_RETURN(count > 0, 0.0f);
-
- static constexpr const float kEmptyFloats[8192] = {};
-
- if (count <= 8192 && std::memcmp(floats, kEmptyFloats, count) == 0)
- return 0.0f;
-
- float tmp, maxf2 = std::abs(floats[0]);
-
- for (std::size_t i=1; i<count; ++i)
- {
- if (!std::isfinite(floats[i]))
- __builtin_unreachable();
-
- tmp = std::abs(*floats++);
-
- if (tmp > maxf2)
- maxf2 = tmp;
- }
-
- if (maxf2 > 1.0f)
- maxf2 = 1.0f;
-
- return maxf2;
- }
-
- /*
- * Multiply an array with a fixed value, float-specific version.
- */
- static inline
- void carla_multiply(float data[], const float& multiplier, const std::size_t count) noexcept
- {
- CARLA_SAFE_ASSERT_RETURN(data != nullptr,);
- CARLA_SAFE_ASSERT_RETURN(count > 0,);
-
- if (carla_isZero(multiplier))
- {
- std::memset(data, 0, count*sizeof(float));
- }
- else
- {
- for (std::size_t i=0; i<count; ++i)
- {
- if (!std::isfinite(data[i]))
- __builtin_unreachable();
- *data++ *= multiplier;
- }
- }
- }
-
- // --------------------------------------------------------------------------------------------------------------------
- // Missing functions in old OSX versions.
-
- #if ! defined(DISTRHO_UTILS_HPP_INCLUDED) && ! defined(CARLA_PROPER_CPP11_SUPPORT)
- namespace std {
- inline float fmin(float __x, float __y)
- { return __builtin_fminf(__x, __y); }
- inline float fmax(float __x, float __y)
- { return __builtin_fmaxf(__x, __y); }
- inline float rint(float __x)
- { return __builtin_rintf(__x); }
- inline float round(float __x)
- { return __builtin_roundf(__x); }
- }
- #endif
-
- // --------------------------------------------------------------------------------------------------------------------
-
- #endif // CARLA_MATH_UTILS_HPP_INCLUDED
|