kfr

Fast, modern C++ DSP framework, FFT, Sample Rate Conversion, FIR/IIR/Biquad Filters (SSE, AVX, AVX-512, ARM NEON)
Log | Files | Refs | README

commit efdb74f011a82742840b9f3be67eea24b75ed6fa
parent cc3dd2f413959c6665f23a032bd2d95a278f18f0
Author: [email protected] <[email protected]>
Date:   Tue,  8 Nov 2016 06:41:06 +0300

operators.hpp: horner_even, horner_odd

Diffstat:
Minclude/kfr/base/operators.hpp | 60++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 60 insertions(+), 0 deletions(-)

diff --git a/include/kfr/base/operators.hpp b/include/kfr/base/operators.hpp @@ -492,6 +492,34 @@ constexpr CMT_INLINE common_type<T1, T2, T3, Ts...> horner(const T1& x, const T2 { return fmadd(horner(x, c1, values...), x, c0); } + +template <typename T1, typename T2> +constexpr CMT_INLINE common_type<T1, T2> horner_even(const T1&, const T2& c0) +{ + return c0; +} + +template <typename T1, typename T2, typename T3, typename... Ts> +constexpr CMT_INLINE common_type<T1, T2, T3, Ts...> horner_even(const T1& x, const T2& c0, const T3& c2, + const Ts&... values) +{ + const T1 x2 = x * x; + return fmadd(horner(x2, c2, values...), x2, c0); +} + +template <typename T1, typename T2> +constexpr CMT_INLINE common_type<T1, T2> horner_odd(const T1& x, const T2& c1) +{ + return c1 * x; +} + +template <typename T1, typename T2, typename T3, typename... Ts> +constexpr CMT_INLINE common_type<T1, T2, T3, Ts...> horner_odd(const T1& x, const T2& c1, const T3& c3, + const Ts&... values) +{ + const T1 x2 = x * x; + return fmadd(horner(x2, c3, values...), x2, c1) * x; +} } /// @brief Calculate polynomial using Horner's method @@ -510,6 +538,38 @@ CMT_INLINE internal::expression_function<fn::horner, E...> horner(E&&... x) return { fn::horner(), std::forward<E>(x)... }; } +/// @brief Calculate polynomial using Horner's method (even powers) +/// +/// ``horner_even(x, 1, 2, 3)`` is equivalent to \(3x^4 + 2x^2 + 1\) +template <typename T1, typename... Ts, KFR_ENABLE_IF(is_numeric_args<T1, Ts...>::value)> +constexpr CMT_INLINE common_type<T1, Ts...> horner_even(const T1& x, const Ts&... c) +{ + return internal::horner_even(x, c...); +} +KFR_FN(horner_even) + +template <typename... E, KFR_ENABLE_IF(is_input_expressions<E...>::value)> +CMT_INLINE internal::expression_function<fn::horner_even, E...> horner_even(E&&... x) +{ + return { fn::horner_even(), std::forward<E>(x)... }; +} + +/// @brief Calculate polynomial using Horner's method (odd powers) +/// +/// ``horner_odd(x, 1, 2, 3)`` is equivalent to \(3x^5 + 2x^3 + 1x\) +template <typename T1, typename... Ts, KFR_ENABLE_IF(is_numeric_args<T1, Ts...>::value)> +constexpr CMT_INLINE common_type<T1, Ts...> horner_odd(const T1& x, const Ts&... c) +{ + return internal::horner_odd(x, c...); +} +KFR_FN(horner_odd) + +template <typename... E, KFR_ENABLE_IF(is_input_expressions<E...>::value)> +CMT_INLINE internal::expression_function<fn::horner_odd, E...> horner_odd(E&&... x) +{ + return { fn::horner_odd(), std::forward<E>(x)... }; +} + /// @brief Calculate Multiplicative Inverse of `x` /// Returns `1/x` template <typename T>