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 aa4b716e9b46f91f6cbb2bcd7fa72fbc1a2e3db7
parent 0d8fc456f716bc1f0420dcc8f27274fa87e1ccad
Author: [email protected] <[email protected]>
Date:   Fri, 12 Aug 2016 09:36:21 +0300

Version 1.0.0.

Diffstat:
Minclude/kfr/base/basic_expressions.hpp | 2+-
Minclude/kfr/base/kfr.h | 10+++++-----
Minclude/kfr/base/operators.hpp | 7+++++++
Minclude/kfr/dsp/dcremove.hpp | 2+-
Minclude/kfr/dsp/fir.hpp | 2+-
Minclude/kfr/dsp/mixdown.hpp | 3+++
Mtests/CMakeLists.txt | 20+++++---------------
Dtests/basic_vector_test.cpp | 104-------------------------------------------------------------------------------
Dtests/conv_test.cpp | 34----------------------------------
Mtests/dft_test.cpp | 10++++++++++
Atests/dsp_test.cpp | 75+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Dtests/fracdelay_test.cpp | 41-----------------------------------------
Mtests/intrinsic_test.cpp | 79+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Mtests/vec_test.cpp | 169+++++++++++++++++++++++++++++++++++++++----------------------------------------
14 files changed, 270 insertions(+), 288 deletions(-)

diff --git a/include/kfr/base/basic_expressions.hpp b/include/kfr/base/basic_expressions.hpp @@ -70,7 +70,7 @@ CMT_INLINE auto sequence(T x, Ts... rest) { const T seq[] = { x, static_cast<T>(rest)... }; constexpr size_t N = arraysize(seq); - return lambda([=](size_t index) { return seq[index % N]; }); + return typed<T>(lambda([=](size_t index) { return seq[index % N]; })); } CMT_INLINE auto zeros() { diff --git a/include/kfr/base/kfr.h b/include/kfr/base/kfr.h @@ -12,11 +12,11 @@ using ::cid::arraysize; } #endif -#define KFR_VERSION_STRING "0.9.1" -#define KFR_VERSION_MAJOR 0 -#define KFR_VERSION_MINOR 9 -#define KFR_VERSION_BUILD 1 -#define KFR_VERSION 901 +#define KFR_VERSION_STRING "1.0.0" +#define KFR_VERSION_MAJOR 1 +#define KFR_VERSION_MINOR 0 +#define KFR_VERSION_BUILD 0 +#define KFR_VERSION 10000 #ifdef __cplusplus namespace kfr diff --git a/include/kfr/base/operators.hpp b/include/kfr/base/operators.hpp @@ -696,6 +696,13 @@ struct expression_pack : expression<E...>, output_expression output(index, x, csizeseq<count>); } + template <typename Input, KFR_ENABLE_IF(is_input_expression<Input>::value)> + CMT_INLINE expression_pack& operator=(Input&& input) + { + process<value_type>(*this, std::forward<Input>(input), size()); + return *this; + } + private: template <typename U, size_t N, size_t... indices> void output(size_t index, const vec<vec<U, count>, N>& x, csizes_t<indices...>) diff --git a/include/kfr/dsp/dcremove.hpp b/include/kfr/dsp/dcremove.hpp @@ -28,7 +28,7 @@ namespace kfr { -template <typename T, typename E1> +template <typename E1, typename T = flt_type<value_type_of<E1>>> CMT_INLINE internal::expression_biquads<1, T, internal::arg<E1>> dcremove(E1&& e1, double cutoff = 0.00025) { const biquad_params<T> bqs[1] = { biquad_highpass(cutoff, 0.5) }; diff --git a/include/kfr/dsp/fir.hpp b/include/kfr/dsp/fir.hpp @@ -107,7 +107,7 @@ template <typename T, size_t TapCount, typename E1> CMT_INLINE internal::expression_short_fir<TapCount, T, E1> short_fir(E1&& e1, const univector<T, TapCount>& taps) { - static_assert(TapCount >= 1 && TapCount < 16, "Use short_fir only for small FIR filters"); + static_assert(TapCount >= 1 && TapCount <= 32, "Use short_fir only for small FIR filters"); return internal::expression_short_fir<TapCount, T, E1>(std::forward<E1>(e1), taps.ref()); } } diff --git a/include/kfr/dsp/mixdown.hpp b/include/kfr/dsp/mixdown.hpp @@ -51,6 +51,9 @@ struct stereo_matrix }; } +constexpr f64x2x2 matrix_sum_diff() { return { f64x2{ 1, 1 }, f64x2{ 1, -1 } }; } +constexpr f64x2x2 matrix_halfsum_halfdiff() { return { f64x2{ 0.5, 0.5 }, f64x2{ 0.5, -0.5 } }; } + template <typename Left, typename Right, typename Result = internal::expression_function< internal::stereo_matrix, internal::expression_pack<internal::arg<Left>, internal::arg<Right>>>> diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt @@ -44,20 +44,14 @@ set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/tests/cmake/") find_package(MPFR) -set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/tests/cmake/") - -find_package(MPFR) - -add_executable(basic_vector_test basic_vector_test.cpp ${KFR_SRC}) add_executable(intrinsic_test intrinsic_test.cpp ${KFR_SRC}) add_executable(dft_test dft_test.cpp ${KFR_SRC}) -add_executable(conv_test conv_test.cpp ${KFR_SRC}) if (MPFR_FOUND) include_directories(${MPFR_INCLUDE_DIR}) add_executable(transcendental_test transcendental_test.cpp ${KFR_SRC}) target_link_libraries(transcendental_test ${MPFR_LIBRARIES}) endif () -add_executable(fracdelay_test fracdelay_test.cpp ${KFR_SRC}) +add_executable(dsp_test dsp_test.cpp ${KFR_SRC}) add_executable(empty_test empty_test.cpp ${KFR_SRC}) add_executable(complex_test complex_test.cpp ${KFR_SRC}) add_executable(vec_test vec_test.cpp ${KFR_SRC}) @@ -72,22 +66,18 @@ endif () if (NOT IOS) enable_testing() - add_test(NAME basic_vector_test - COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/basic_vector_test) + add_test(NAME vec_test + COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/vec_test) add_test(NAME intrinsic_test COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/intrinsic_test) - add_test(NAME fracdelay_test - COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/fracdelay_test) - add_test(NAME conv_test - COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/conv_test) + add_test(NAME dsp_test + COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/dsp_test) if (MPFR_FOUND) add_test(NAME transcendental_test COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/transcendental_test) endif () add_test(NAME complex_test COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/complex_test) - add_test(NAME vec_test - COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/vec_test) add_test(NAME stat_test COMMAND ${EMULATOR} ${PROJECT_BINARY_DIR}/tests/stat_test) diff --git a/tests/basic_vector_test.cpp b/tests/basic_vector_test.cpp @@ -1,104 +0,0 @@ -/** - * KFR (http://kfrlib.com) - * Copyright (C) 2016 D Levin - * See LICENSE.txt for details - */ - -#include <kfr/io/tostring.hpp> - -#include <kfr/cometa/string.hpp> -#include <kfr/math.hpp> -#include <kfr/version.hpp> - -#include "testo/testo.hpp" - -using namespace kfr; - -TEST(test) -{ - // How to make a vector: - - // * Use constructor - const vec<double, 4> first{ 1, 2.5, -infinity, 3.1415926 }; - CHECK(first == vec<double, 4>{ 1, 2.5, -infinity, 3.1415926 }); - - // * Use make_vector function - const auto second = make_vector(-1, +1); - CHECK(second == vec<int, 2>{ -1, 1 }); - - // * Convert from vector of other type: - const vec<int, 4> int_vector{ 10, 20, 30, 40 }; - const vec<double, 4> double_vector = cast<double>(int_vector); - CHECK(double_vector == vec<double, 4>{ 10, 20, 30, 40 }); - - // * Concat two vectors: - const vec<int, 1> left_part{ 1 }; - const vec<int, 1> right_part{ 2 }; - const vec<int, 2> pair{ left_part, right_part }; - CHECK(pair == vec<int, 2>{ 1, 2 }); - - // * Same, but using make_vector and concat: - const vec<int, 2> pair2 = concat(make_vector(10), make_vector(20)); - CHECK(pair2 == vec<int, 2>{ 10, 20 }); - - // * Repeat vector multiple times: - const vec<short, 8> repeated = repeat<4>(make_vector<short>(0, -1)); - CHECK(repeated == vec<short, 8>{ 0, -1, 0, -1, 0, -1, 0, -1 }); - - // * Use enumerate to generate sequence of numbers: - const vec<int, 8> eight = enumerate<int, 8>(); - CHECK(eight == vec<int, 8>{ 0, 1, 2, 3, 4, 5, 6, 7 }); - - // * Vectors can be of any length... - const vec<int, 1> one{ 42 }; - const vec<int, 2> two = concat(one, make_vector(42)); - CHECK(two == vec<int, 2>{ 42, 42 }); - - const vec<u8, 256> very_long_vector = repeat<64>(make_vector<u8>(1, 2, 4, 8)); - CHECK(slice<0, 17>(very_long_vector) == - vec<unsigned char, 17>{ 1, 2, 4, 8, 1, 2, 4, 8, 1, 2, 4, 8, 1, 2, 4, 8, 1 }); - - // * ...really any: - using big_vector = vec<i16, 107>; - big_vector v107 = enumerate<i16, 107>(); - CHECK(hadd(v107) == static_cast<short>(5671)); - - using color = vec<u8, 3>; - const color green = cast<u8>(make_vector(0.0, 1.0, 0.0) * 255); - CHECK(green == vec<unsigned char, 3>{ 0, 255, 0 }); - - // Vectors support all standard operators: - const auto op1 = make_vector(0, 1, 10, 100); - const auto op2 = make_vector(20, 2, -2, 200); - const auto result = op1 * op2 - 4; - CHECK(result == vec<int, 4>{ -4, -2, -24, 19996 }); - - // * Transform vector: - const vec<int, 8> numbers1 = enumerate<int, 8>(); - const vec<int, 8> numbers2 = enumerate<int, 8>() + 100; - CHECK(odd(numbers1) == vec<int, 4>{ 1, 3, 5, 7 }); - CHECK(even(numbers2) == vec<int, 4>{ 100, 102, 104, 106 }); - - // * The following command pairs are equivalent: - CHECK(permute(numbers1, elements<0, 2, 1, 3, 4, 6, 5, 7>) == vec<int, 8>{ 0, 2, 1, 3, 4, 6, 5, 7 }); - CHECK(permute(numbers1, elements<0, 2, 1, 3>) == vec<int, 8>{ 0, 2, 1, 3, 4, 6, 5, 7 }); - - CHECK(shuffle(numbers1, numbers2, elements<0, 8, 2, 10, 4, 12, 6, 14>) == - vec<int, 8>{ 0, 100, 2, 102, 4, 104, 6, 106 }); - CHECK(shuffle(numbers1, numbers2, elements<0, 8>) == vec<int, 8>{ 0, 100, 2, 102, 4, 104, 6, 106 }); - - CHECK(blend(numbers1, numbers2, elements<0, 1, 1, 0, 1, 1, 0, 1>) == - vec<int, 8>{ 0, 101, 102, 3, 104, 105, 6, 107 }); - CHECK(blend(numbers1, numbers2, elements<0, 1, 1>) == vec<int, 8>{ 0, 101, 102, 3, 104, 105, 6, 107 }); - - // * Transpose matrix: - const auto sixteen = enumerate<float, 16>(); - CHECK(transpose<4>(sixteen) == vec<float, 16>{ 0, 4, 8, 12, 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15 }); -} - -int main(int /*argc*/, char** /*argv*/) -{ - println(library_version()); - - return testo::run_all("", true); -} diff --git a/tests/conv_test.cpp b/tests/conv_test.cpp @@ -1,34 +0,0 @@ -/** - * KFR (http://kfrlib.com) - * Copyright (C) 2016 D Levin - * See LICENSE.txt for details - */ - -// library_version() -#include <kfr/io/tostring.hpp> -#include <kfr/version.hpp> - -#include <kfr/base/reduce.hpp> - -#include <tuple> - -#include "testo/testo.hpp" -#include <kfr/dft/conv.hpp> - -using namespace kfr; - -TEST(test_convolve) -{ - univector<fbase, 5> a({ 1, 2, 3, 4, 5 }); - univector<fbase, 5> b({ 0.25, 0.5, 1.0, 0.5, 0.25 }); - univector<fbase> c = convolve(a, b); - CHECK(c.size() == 9); - CHECK(rms(c - univector<fbase>({ 0.25, 1., 2.75, 5., 7.5, 8.5, 7.75, 3.5, 1.25 })) < 0.0001); -} - -int main(int argc, char** argv) -{ - println(library_version()); - - return testo::run_all("", true); -} diff --git a/tests/dft_test.cpp b/tests/dft_test.cpp @@ -15,6 +15,7 @@ #include <kfr/base/reduce.hpp> #include <kfr/cometa/string.hpp> #include <kfr/dft/fft.hpp> +#include <kfr/dft/conv.hpp> #include <kfr/dft/reference_dft.hpp> #include <kfr/io/tostring.hpp> #include <kfr/math.hpp> @@ -28,6 +29,15 @@ constexpr ctypes_t<float, double> float_types{}; constexpr ctypes_t<float> float_types{}; #endif +TEST(test_convolve) +{ + univector<fbase, 5> a({ 1, 2, 3, 4, 5 }); + univector<fbase, 5> b({ 0.25, 0.5, 1.0, 0.5, 0.25 }); + univector<fbase> c = convolve(a, b); + CHECK(c.size() == 9); + CHECK(rms(c - univector<fbase>({ 0.25, 1., 2.75, 5., 7.5, 8.5, 7.75, 3.5, 1.25 })) < 0.0001); +} + TEST(fft_accuracy) { testo::active_test()->show_progress = true; diff --git a/tests/dsp_test.cpp b/tests/dsp_test.cpp @@ -0,0 +1,75 @@ +/** + * KFR (http://kfrlib.com) + * Copyright (C) 2016 D Levin + * See LICENSE.txt for details + */ + +#include <kfr/io/tostring.hpp> + +#include "testo/testo.hpp" +#include <kfr/dsp.hpp> +#include <kfr/math.hpp> + +using namespace kfr; + +TEST(delay) +{ + const univector<float, 33> v1 = counter() + 100; + const univector<float, 33> v2 = delay(v1); + CHECK(v2[0] == 0); + CHECK(v2[1] == 100); + CHECK(v2[2] == 101); + CHECK(v2[19] == 118); + + const univector<float, 33> v3 = delay(v1, csize<3>); + CHECK(v3[0] == 0); + CHECK(v3[1] == 0); + CHECK(v3[2] == 0); + CHECK(v3[3] == 100); + CHECK(v3[4] == 101); + CHECK(v3[19] == 116); +} + +TEST(fracdelay) +{ + univector<double, 5> a({ 1, 2, 3, 4, 5 }); + univector<double, 5> b = fracdelay(a, 0.5); + CHECK(rms(b - univector<double>({ 0.5, 1.5, 2.5, 3.5, 4.5 })) < c_epsilon<double> * 5); + + b = fracdelay(a, 0.1); + CHECK(rms(b - univector<double>({ 0.9, 1.9, 2.9, 3.9, 4.9 })) < c_epsilon<double> * 5); + + b = fracdelay(a, 0.0); + CHECK(rms(b - univector<double>({ 1, 2, 3, 4, 5 })) < c_epsilon<double> * 5); + + b = fracdelay(a, 1.0); + CHECK(rms(b - univector<double>({ 0, 1, 2, 3, 4 })) < c_epsilon<double> * 5); +} + +TEST(mixdown) +{ + univector<double, 20> ch1 = counter(); + univector<double, 20> ch2 = counter() * 2 + 100; + univector<double, 20> mix = mixdown(ch1, ch2); + CHECK(mix[0] == 100); + CHECK(mix[1] == 103); + CHECK(mix[19] == 157); +} + +TEST(mixdown_stereo) +{ + const univector<double, 21> left = counter(); + const univector<double, 21> right = counter() * 2 + 100; + univector<double, 21> mid; + univector<double, 21> side; + pack(mid, side) = mixdown_stereo(left, right, matrix_sum_diff()); + + CHECK(mid[0] == 100); + CHECK(side[0] == -100); + CHECK(mid[1] == 103); + CHECK(side[1] == -101); + CHECK(mid[20] == 160); + CHECK(side[20] == -120); +} + +int main(int argc, char** argv) { return testo::run_all("", true); } diff --git a/tests/fracdelay_test.cpp b/tests/fracdelay_test.cpp @@ -1,41 +0,0 @@ -/** - * KFR (http://kfrlib.com) - * Copyright (C) 2016 D Levin - * See LICENSE.txt for details - */ - -// library_version() -#include <kfr/io/tostring.hpp> -#include <kfr/version.hpp> - -#include <kfr/base/reduce.hpp> - -#include <tuple> - -#include "testo/testo.hpp" -#include <kfr/dsp/fracdelay.hpp> - -using namespace kfr; - -TEST(test_fracdelay) -{ - univector<double, 5> a({ 1, 2, 3, 4, 5 }); - univector<double, 5> b = fracdelay(a, 0.5); - CHECK(rms(b - univector<double>({ 0.5, 1.5, 2.5, 3.5, 4.5 })) < c_epsilon<double> * 5); - - b = fracdelay(a, 0.1); - CHECK(rms(b - univector<double>({ 0.9, 1.9, 2.9, 3.9, 4.9 })) < c_epsilon<double> * 5); - - b = fracdelay(a, 0.0); - CHECK(rms(b - univector<double>({ 1, 2, 3, 4, 5 })) < c_epsilon<double> * 5); - - b = fracdelay(a, 1.0); - CHECK(rms(b - univector<double>({ 0, 1, 2, 3, 4 })) < c_epsilon<double> * 5); -} - -int main(int argc, char** argv) -{ - println(library_version()); - - return testo::run_all("", true); -} diff --git a/tests/intrinsic_test.cpp b/tests/intrinsic_test.cpp @@ -8,6 +8,7 @@ #include "testo/testo.hpp" #include <kfr/math.hpp> +#include <kfr/dsp.hpp> using namespace kfr; @@ -210,6 +211,15 @@ TEST(intrin_round) TEST(intrin_min_max) { + testo::assert_is_same<decltype(min(1, 2)), int>(); + testo::assert_is_same<decltype(min(1, 2u)), unsigned int>(); + testo::assert_is_same<decltype(min(1, 2)), int>(); + testo::assert_is_same<decltype(min(pack(1), 2u)), u32x1>(); + testo::assert_is_same<decltype(min(2u, pack(1))), u32x1>(); + testo::assert_is_same<decltype(min(pack(1), pack(2u))), u32x1>(); + testo::assert_is_same<decltype(min(pack(1, 2, 3), pack(1.0, 2.0, 3.0))), f64x3>(); + testo::assert_is_same<decltype(min(pack(1.0, 2.0, 3.0), pack(1, 2, 3))), f64x3>(); + CHECK(min(1, 2) == 1); CHECK(min(1, 2u) == 1u); CHECK(min(pack(1), 2) == pack(1)); @@ -316,4 +326,73 @@ TEST(intrin_any_all) }); } +TEST(intrin_math) +{ + testo::assert_is_same<decltype(pack(11) * pack(0.5)), f64x1>(); + testo::assert_is_same<decltype(pack(11) * 0.5), f64x1>(); + testo::assert_is_same<decltype(kfr::sin(2)), fbase>(); + testo::assert_is_same<decltype(kfr::sin(pack(2))), vec<fbase, 1>>(); + testo::assert_is_same<decltype(kfr::sindeg(2)), fbase>(); + testo::assert_is_same<decltype(kfr::sindeg(pack(2))), vec<fbase, 1>>(); + + CHECK(pack(11) * pack(0.5) == 5.5); + CHECK(pack(11) * 0.5 == 5.5); + CHECK(kfr::sin(2) == fbase(0.90929742682568169539601986591174)); + CHECK(kfr::sin(pack(2)) == pack(fbase(0.90929742682568169539601986591174))); + CHECK(kfr::sindeg(2) == fbase(0.03489949670250097164599518162533)); + CHECK(kfr::sindeg(pack(2)) == pack(fbase(0.03489949670250097164599518162533))); + CHECK(kfr::cos(2) == fbase(-0.41614683654714238699756822950076)); + CHECK(kfr::cos(pack(2)) == pack(fbase(-0.41614683654714238699756822950076))); + CHECK(kfr::cosdeg(2) == fbase(0.99939082701909573000624344004393)); + CHECK(kfr::cosdeg(pack(2)) == pack(fbase(0.99939082701909573000624344004393))); + + CHECK(kfr::log(2) == fbase(0.6931471805599453)); + CHECK(kfr::log(pack(2)) == pack(fbase(0.6931471805599453))); + CHECK(kfr::log2(2) == fbase(1.0)); + CHECK(kfr::log2(pack(2)) == pack(fbase(1.0))); + CHECK(kfr::log10(2) == fbase(0.30102999566398119521373889472449)); + CHECK(kfr::log10(pack(2)) == pack(fbase(0.30102999566398119521373889472449))); + + CHECK(kfr::exp(2) == fbase(7.3890560989306502)); + CHECK(kfr::exp(pack(2)) == pack(fbase(7.3890560989306502))); + CHECK(kfr::exp2(2) == fbase(4.0)); + CHECK(kfr::exp2(pack(2)) == pack(fbase(4.0))); + + CHECK(kfr::logn(2, 10) == fbase(0.30102999566398119521373889472449)); + CHECK(kfr::logn(pack(2), pack(10)) == pack(fbase(0.30102999566398119521373889472449))); + + CHECK(kfr::pow(2, fbase(0.9)) == fbase(1.8660659830736148319626865322999)); + CHECK(kfr::pow(pack(2), pack(fbase(0.9))) == pack(fbase(1.8660659830736148319626865322999))); + + CHECK(kfr::root(fbase(1.5), 2) == fbase(1.2247448713915890490986420373529)); + CHECK(kfr::root(pack(fbase(1.5)), pack(2)) == pack(fbase(1.2247448713915890490986420373529))); + + testo::epsilon<float>() *= 10.0; + testo::epsilon<double>() *= 10.0; + + CHECK(kfr::sinh(2) == fbase(3.6268604078470187676682139828013)); + CHECK(kfr::sinh(pack(2)) == pack(fbase(3.6268604078470187676682139828013))); + CHECK(kfr::cosh(2) == fbase(3.7621956910836314595622134777737)); + CHECK(kfr::cosh(pack(2)) == pack(fbase(3.7621956910836314595622134777737))); + + CHECK(kfr::tanh(2) == fbase(0.96402758007581688394641372410092)); + CHECK(kfr::tanh(pack(2)) == pack(fbase(0.96402758007581688394641372410092))); + CHECK(kfr::coth(2) == fbase(1.0373147207275480958778097647678)); + CHECK(kfr::coth(pack(2)) == pack(fbase(1.0373147207275480958778097647678))); + + testo::epsilon<float>() *= 10.0; + testo::epsilon<double>() *= 10.0; + + CHECK(kfr::tan(2) == fbase(-2.1850398632615189916433061023137)); + CHECK(kfr::tan(pack(2)) == pack(fbase(-2.1850398632615189916433061023137))); + CHECK(kfr::tandeg(2) == fbase(0.03492076949174773050040262577373)); + CHECK(kfr::tandeg(pack(2)) == pack(fbase(0.03492076949174773050040262577373))); + + testo::epsilon<float>() *= 10.0; + testo::epsilon<double>() *= 10.0; + + CHECK(kfr::note_to_hertz(60) == fbase(261.6255653005986346778499935233)); + CHECK(kfr::note_to_hertz(pack(60)) == pack(fbase(261.6255653005986346778499935233))); +} + int main(int argc, char** argv) { return testo::run_all("", false); } diff --git a/tests/vec_test.cpp b/tests/vec_test.cpp @@ -12,6 +12,88 @@ using namespace kfr; +TEST(test_basic) +{ + // How to make a vector: + + // * Use constructor + const vec<double, 4> first{ 1, 2.5, -infinity, 3.1415926 }; + CHECK(first == vec<double, 4>{ 1, 2.5, -infinity, 3.1415926 }); + + // * Use make_vector function + const auto second = make_vector(-1, +1); + CHECK(second == vec<int, 2>{ -1, 1 }); + + // * Convert from vector of other type: + const vec<int, 4> int_vector{ 10, 20, 30, 40 }; + const vec<double, 4> double_vector = cast<double>(int_vector); + CHECK(double_vector == vec<double, 4>{ 10, 20, 30, 40 }); + + // * Concat two vectors: + const vec<int, 1> left_part{ 1 }; + const vec<int, 1> right_part{ 2 }; + const vec<int, 2> pair{ left_part, right_part }; + CHECK(pair == vec<int, 2>{ 1, 2 }); + + // * Same, but using make_vector and concat: + const vec<int, 2> pair2 = concat(make_vector(10), make_vector(20)); + CHECK(pair2 == vec<int, 2>{ 10, 20 }); + + // * Repeat vector multiple times: + const vec<short, 8> repeated = repeat<4>(make_vector<short>(0, -1)); + CHECK(repeated == vec<short, 8>{ 0, -1, 0, -1, 0, -1, 0, -1 }); + + // * Use enumerate to generate sequence of numbers: + const vec<int, 8> eight = enumerate<int, 8>(); + CHECK(eight == vec<int, 8>{ 0, 1, 2, 3, 4, 5, 6, 7 }); + + // * Vectors can be of any length... + const vec<int, 1> one{ 42 }; + const vec<int, 2> two = concat(one, make_vector(42)); + CHECK(two == vec<int, 2>{ 42, 42 }); + + const vec<u8, 256> very_long_vector = repeat<64>(make_vector<u8>(1, 2, 4, 8)); + CHECK(slice<0, 17>(very_long_vector) == + vec<unsigned char, 17>{ 1, 2, 4, 8, 1, 2, 4, 8, 1, 2, 4, 8, 1, 2, 4, 8, 1 }); + + // * ...really any: + using big_vector = vec<i16, 107>; + big_vector v107 = enumerate<i16, 107>(); + CHECK(hadd(v107) == static_cast<short>(5671)); + + using color = vec<u8, 3>; + const color green = cast<u8>(make_vector(0.0, 1.0, 0.0) * 255); + CHECK(green == vec<unsigned char, 3>{ 0, 255, 0 }); + + // Vectors support all standard operators: + const auto op1 = make_vector(0, 1, 10, 100); + const auto op2 = make_vector(20, 2, -2, 200); + const auto result = op1 * op2 - 4; + CHECK(result == vec<int, 4>{ -4, -2, -24, 19996 }); + + // * Transform vector: + const vec<int, 8> numbers1 = enumerate<int, 8>(); + const vec<int, 8> numbers2 = enumerate<int, 8>() + 100; + CHECK(odd(numbers1) == vec<int, 4>{ 1, 3, 5, 7 }); + CHECK(even(numbers2) == vec<int, 4>{ 100, 102, 104, 106 }); + + // * The following command pairs are equivalent: + CHECK(permute(numbers1, elements<0, 2, 1, 3, 4, 6, 5, 7>) == vec<int, 8>{ 0, 2, 1, 3, 4, 6, 5, 7 }); + CHECK(permute(numbers1, elements<0, 2, 1, 3>) == vec<int, 8>{ 0, 2, 1, 3, 4, 6, 5, 7 }); + + CHECK(shuffle(numbers1, numbers2, elements<0, 8, 2, 10, 4, 12, 6, 14>) == + vec<int, 8>{ 0, 100, 2, 102, 4, 104, 6, 106 }); + CHECK(shuffle(numbers1, numbers2, elements<0, 8>) == vec<int, 8>{ 0, 100, 2, 102, 4, 104, 6, 106 }); + + CHECK(blend(numbers1, numbers2, elements<0, 1, 1, 0, 1, 1, 0, 1>) == + vec<int, 8>{ 0, 101, 102, 3, 104, 105, 6, 107 }); + CHECK(blend(numbers1, numbers2, elements<0, 1, 1>) == vec<int, 8>{ 0, 101, 102, 3, 104, 105, 6, 107 }); + + // * Transpose matrix: + const auto sixteen = enumerate<float, 16>(); + CHECK(transpose<4>(sixteen) == vec<float, 16>{ 0, 4, 8, 12, 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15 }); +} + TEST(vec_concat) { CHECK(concat(vec<f32, 1>{ 1 }, vec<f32, 2>{ 2, 3 }, vec<f32, 1>{ 4 }, vec<f32, 3>{ 5, 6, 7 }) // @@ -115,92 +197,6 @@ TEST(vec_low_high) CHECK(high(vec<u8, 2>(1, 2)) == vec<u8, 1>(2)); } -TEST(vec_conv) -{ - testo::assert_is_same<common_type<i32, f32>, f32>(); - testo::assert_is_same<common_type<f32, f64>, f64>(); - testo::assert_is_same<common_type<i32, f32x4>, f32x4>(); - testo::assert_is_same<common_type<f32x4, f64>, f64x4>(); - testo::assert_is_same<common_type<f64, f32x4>, f64x4>(); - testo::assert_is_same<common_type<f32x4, f64x4>, f64x4>(); - - testo::assert_is_same<decltype(min(1, 2)), int>(); - testo::assert_is_same<decltype(min(1, 2u)), unsigned int>(); - testo::assert_is_same<decltype(min(1, 2)), int>(); - testo::assert_is_same<decltype(min(pack(1), 2u)), u32x1>(); - testo::assert_is_same<decltype(min(2u, pack(1))), u32x1>(); - testo::assert_is_same<decltype(min(pack(1), pack(2u))), u32x1>(); - testo::assert_is_same<decltype(min(pack(1, 2, 3), pack(1.0, 2.0, 3.0))), f64x3>(); - testo::assert_is_same<decltype(min(pack(1.0, 2.0, 3.0), pack(1, 2, 3))), f64x3>(); - - testo::assert_is_same<decltype(pack(11) * pack(0.5)), f64x1>(); - testo::assert_is_same<decltype(pack(11) * 0.5), f64x1>(); - - testo::assert_is_same<decltype(kfr::sin(2)), fbase>(); - testo::assert_is_same<decltype(kfr::sin(pack(2))), vec<fbase, 1>>(); - testo::assert_is_same<decltype(kfr::sindeg(2)), fbase>(); - testo::assert_is_same<decltype(kfr::sindeg(pack(2))), vec<fbase, 1>>(); - - CHECK(pack(11) * pack(0.5) == 5.5); - CHECK(pack(11) * 0.5 == 5.5); - CHECK(kfr::sin(2) == fbase(0.90929742682568169539601986591174)); - CHECK(kfr::sin(pack(2)) == pack(fbase(0.90929742682568169539601986591174))); - CHECK(kfr::sindeg(2) == fbase(0.03489949670250097164599518162533)); - CHECK(kfr::sindeg(pack(2)) == pack(fbase(0.03489949670250097164599518162533))); - CHECK(kfr::cos(2) == fbase(-0.41614683654714238699756822950076)); - CHECK(kfr::cos(pack(2)) == pack(fbase(-0.41614683654714238699756822950076))); - CHECK(kfr::cosdeg(2) == fbase(0.99939082701909573000624344004393)); - CHECK(kfr::cosdeg(pack(2)) == pack(fbase(0.99939082701909573000624344004393))); - - CHECK(kfr::log(2) == fbase(0.6931471805599453)); - CHECK(kfr::log(pack(2)) == pack(fbase(0.6931471805599453))); - CHECK(kfr::log2(2) == fbase(1.0)); - CHECK(kfr::log2(pack(2)) == pack(fbase(1.0))); - CHECK(kfr::log10(2) == fbase(0.30102999566398119521373889472449)); - CHECK(kfr::log10(pack(2)) == pack(fbase(0.30102999566398119521373889472449))); - - CHECK(kfr::exp(2) == fbase(7.3890560989306502)); - CHECK(kfr::exp(pack(2)) == pack(fbase(7.3890560989306502))); - CHECK(kfr::exp2(2) == fbase(4.0)); - CHECK(kfr::exp2(pack(2)) == pack(fbase(4.0))); - - CHECK(kfr::logn(2, 10) == fbase(0.30102999566398119521373889472449)); - CHECK(kfr::logn(pack(2), pack(10)) == pack(fbase(0.30102999566398119521373889472449))); - - CHECK(kfr::pow(2, fbase(0.9)) == fbase(1.8660659830736148319626865322999)); - CHECK(kfr::pow(pack(2), pack(fbase(0.9))) == pack(fbase(1.8660659830736148319626865322999))); - - CHECK(kfr::root(fbase(1.5), 2) == fbase(1.2247448713915890490986420373529)); - CHECK(kfr::root(pack(fbase(1.5)), pack(2)) == pack(fbase(1.2247448713915890490986420373529))); - - testo::epsilon<float>() *= 10.0; - testo::epsilon<double>() *= 10.0; - - CHECK(kfr::sinh(2) == fbase(3.6268604078470187676682139828013)); - CHECK(kfr::sinh(pack(2)) == pack(fbase(3.6268604078470187676682139828013))); - CHECK(kfr::cosh(2) == fbase(3.7621956910836314595622134777737)); - CHECK(kfr::cosh(pack(2)) == pack(fbase(3.7621956910836314595622134777737))); - - CHECK(kfr::tanh(2) == fbase(0.96402758007581688394641372410092)); - CHECK(kfr::tanh(pack(2)) == pack(fbase(0.96402758007581688394641372410092))); - CHECK(kfr::coth(2) == fbase(1.0373147207275480958778097647678)); - CHECK(kfr::coth(pack(2)) == pack(fbase(1.0373147207275480958778097647678))); - - testo::epsilon<float>() *= 10.0; - testo::epsilon<double>() *= 10.0; - - CHECK(kfr::tan(2) == fbase(-2.1850398632615189916433061023137)); - CHECK(kfr::tan(pack(2)) == pack(fbase(-2.1850398632615189916433061023137))); - CHECK(kfr::tandeg(2) == fbase(0.03492076949174773050040262577373)); - CHECK(kfr::tandeg(pack(2)) == pack(fbase(0.03492076949174773050040262577373))); - - testo::epsilon<float>() *= 10.0; - testo::epsilon<double>() *= 10.0; - - CHECK(kfr::note_to_hertz(60) == fbase(261.6255653005986346778499935233)); - CHECK(kfr::note_to_hertz(pack(60)) == pack(fbase(261.6255653005986346778499935233))); -} - TEST(vec_matrix) { using i32x2x2 = vec<vec<int, 2>, 2>; @@ -293,6 +289,7 @@ TEST(test_delay) CHECK(v3[3] == 100); CHECK(v3[4] == 101); CHECK(v3[19] == 116); + } int main(int argc, char** argv) { return testo::run_all("", true); }