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 17451d4038fda5ad6a720003554db67ee56a9b2e
parent 57dcd901c1db7e733c845b08984103218503212c
Author: [email protected] <[email protected]>
Date:   Wed,  7 Sep 2016 15:00:13 +0300

New function: truncate(Expr, size)

Diffstat:
Mexamples/sample_rate_conversion.cpp | 8++++----
Minclude/kfr/base/basic_expressions.hpp | 6++++++
Minclude/kfr/base/univector.hpp | 12++++++++++++
Minclude/kfr/dft/conv.hpp | 2+-
Minclude/kfr/dsp/oscillators.hpp | 6+++---
Minclude/kfr/dsp/sample_rate_conversion.hpp | 8++++----
Mtests/dft_test.cpp | 10+++++-----
7 files changed, 35 insertions(+), 17 deletions(-)

diff --git a/examples/sample_rate_conversion.cpp b/examples/sample_rate_conversion.cpp @@ -29,7 +29,7 @@ int main(int argc, char** argv) const size_t destsize = r(resampled.data(), swept_sine); - univector<i32> i32data = clamp(resampled.slice(0, destsize) * i32max, -i32max, +i32max); + univector<i32> i32data = clamp(resampled.truncate(destsize) * i32max, -i32max, +i32max); univector2d<i32> data = { i32data }; auto wr = sequential_file_writer("audio_high_quality.wav"); @@ -44,7 +44,7 @@ int main(int argc, char** argv) const size_t destsize = r(resampled.data(), swept_sine); - univector<i32> i32data = clamp(resampled.slice(0, destsize) * i32max, -i32max, +i32max); + univector<i32> i32data = clamp(resampled.truncate(destsize) * i32max, -i32max, +i32max); univector2d<i32> data = { i32data }; auto wr = sequential_file_writer("audio_normal_quality.wav"); @@ -59,7 +59,7 @@ int main(int argc, char** argv) const size_t destsize = r(resampled.data(), swept_sine); - univector<i32> i32data = clamp(resampled.slice(0, destsize) * i32max, -i32max, +i32max); + univector<i32> i32data = clamp(resampled.truncate(destsize) * i32max, -i32max, +i32max); univector2d<i32> data = { i32data }; auto wr = sequential_file_writer("audio_low_quality.wav"); @@ -74,7 +74,7 @@ int main(int argc, char** argv) const size_t destsize = r(resampled.data(), swept_sine); - univector<i32> i32data = clamp(resampled.slice(0, destsize) * i32max, -i32max, +i32max); + univector<i32> i32data = clamp(resampled.truncate(destsize) * i32max, -i32max, +i32max); univector2d<i32> data = { i32data }; auto wr = sequential_file_writer("audio_draft_quality.wav"); diff --git a/include/kfr/base/basic_expressions.hpp b/include/kfr/base/basic_expressions.hpp @@ -328,6 +328,12 @@ CMT_INLINE internal::expression_slice<E1> slice(E1&& e1, size_t start, size_t si return internal::expression_slice<E1>(std::forward<E1>(e1), start, size); } +template <typename E1> +CMT_INLINE internal::expression_slice<E1> truncate(E1&& e1, size_t size) +{ + return internal::expression_slice<E1>(std::forward<E1>(e1), 0, size); +} + template <typename T1, typename T2, bool precise = false, typename TF = ftype<common_type<T1, T2>>> CMT_INLINE internal::expression_linspace<TF, precise> linspace(T1 start, T2 stop, size_t size, bool endpoint = false) diff --git a/include/kfr/base/univector.hpp b/include/kfr/base/univector.hpp @@ -75,6 +75,18 @@ struct univector_base : input_expression, output_expression const size_t this_size = derived_cast<Class>(this)->size(); return array_ref<const T>(data + start, std::min(size, this_size - start)); } + univector<T, 0> truncate(size_t size = max_size_t) + { + T* data = derived_cast<Class>(this)->data(); + const size_t this_size = derived_cast<Class>(this)->size(); + return array_ref<T>(data, std::min(size, this_size)); + } + univector<const T, 0> truncate(size_t size = max_size_t) const + { + const T* data = derived_cast<Class>(this)->data(); + const size_t this_size = derived_cast<Class>(this)->size(); + return array_ref<const T>(data, std::min(size, this_size)); + } array_ref<T> ref() { diff --git a/include/kfr/dft/conv.hpp b/include/kfr/dft/conv.hpp @@ -55,7 +55,7 @@ KFR_INTRIN univector<T> convolve(const univector<T, Tag1>& src1, const univector plan.execute(src2padded, src2padded, temp); src1padded = src1padded * src2padded; plan.execute(src1padded, src1padded, temp, true); - return slice(real(src1padded), 0, src1.size() + src2.size() - 1) / T(size); + return truncate(real(src1padded), src1.size() + src2.size() - 1) / T(size); } } #pragma clang diagnostic pop diff --git a/include/kfr/dsp/oscillators.hpp b/include/kfr/dsp/oscillators.hpp @@ -34,15 +34,15 @@ namespace kfr template <typename T = fbase> auto jaehne(identity<T> magn, size_t size) { - return slice(magn * sin(c_pi<T, 1, 2> * sqr(linspace(T(0), T(size), size, false)) / size), 0, size); + return truncate(magn * sin(c_pi<T, 1, 2> * sqr(linspace(T(0), T(size), size, false)) / size), size); } template <typename T = fbase> auto swept(identity<T> magn, size_t size) { - return slice( + return truncate( magn * sin(c_pi<T, 1, 4> * sqr(sqr(linspace(T(0), T(size), size, false)) / sqr(T(size))) * T(size)), - 0, size); + size); } template <typename T = fbase> diff --git a/include/kfr/dsp/sample_rate_conversion.hpp b/include/kfr/dsp/sample_rate_conversion.hpp @@ -134,8 +134,8 @@ struct sample_rate_converter } else { - delay.slice(0, size_t(depth - srcsize)) = delay.slice(size_t(srcsize)); - delay.slice(size_t(depth - srcsize)) = zeros(); + delay.truncate(size_t(depth - srcsize)) = delay.slice(size_t(srcsize)); + delay.slice(size_t(depth - srcsize)) = zeros(); } input_position += srcsize; @@ -183,8 +183,8 @@ struct sample_rate_converter } else { - delay.slice(0, size_t(depth - srcsize)) = delay.slice(size_t(srcsize)); - delay.slice(size_t(depth - srcsize)) = src; + delay.truncate(size_t(depth - srcsize)) = delay.slice(size_t(srcsize)); + delay.slice(size_t(depth - srcsize)) = src; } input_position += srcsize; diff --git a/tests/dft_test.cpp b/tests/dft_test.cpp @@ -47,7 +47,7 @@ TEST(fft_accuracy) { univector<complex<float_type>> in = - slice(gen_random_range<float_type>(gen, -1.0, +1.0), 0, size * 2); + truncate(gen_random_range<float_type>(gen, -1.0, +1.0), size); univector<complex<float_type>> out = in; univector<complex<float_type>> refout = out; const dft_plan<float_type> dft(size); @@ -65,17 +65,17 @@ TEST(fft_accuracy) if (size >= 16) { univector<float_type> in = - slice(gen_random_range<float_type>(gen, -1.0, +1.0), 0, size); + truncate(gen_random_range<float_type>(gen, -1.0, +1.0), size); - univector<complex<float_type>> out = slice(scalar(qnan), 0, size); - univector<complex<float_type>> refout = slice(scalar(qnan), 0, size); + univector<complex<float_type>> out = truncate(scalar(qnan), size); + univector<complex<float_type>> refout = truncate(scalar(qnan), size); const dft_plan_real<float_type> dft(size); univector<u8> temp(dft.temp_size); reference_fft(refout.data(), in.data(), size); dft.execute(out, in, temp); const float_type rms_diff_r = - rms(cabs(refout.slice(0, size / 2 + 1) - out.slice(0, size / 2 + 1))); + rms(cabs(refout.truncate(size / 2 + 1) - out.truncate(size / 2 + 1))); const double ops = log2size * 200; const double epsilon = std::numeric_limits<float_type>::epsilon(); CHECK(rms_diff_r < epsilon * ops);