| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393 |
- #include "config.h"
- #include <mmintrin.h>
- #include <xmmintrin.h>
- #include <algorithm>
- #include <array>
- #include <cstddef>
- #include <cstdint>
- #include <limits>
- #include <variant>
- #include "alnumeric.h"
- #include "alspan.h"
- #include "core/bsinc_defs.h"
- #include "core/bufferline.h"
- #include "core/cubic_defs.h"
- #include "core/mixer/hrtfdefs.h"
- #include "core/resampler_limits.h"
- #include "defs.h"
- #include "hrtfbase.h"
- #include "opthelpers.h"
- struct CTag;
- struct SSETag;
- struct CubicTag;
- struct BSincTag;
- struct FastBSincTag;
- #if defined(__GNUC__) && !defined(__clang__) && !defined(__SSE__)
- #pragma GCC target("sse")
- #endif
- namespace {
- constexpr uint BSincPhaseDiffBits{MixerFracBits - BSincPhaseBits};
- constexpr uint BSincPhaseDiffOne{1 << BSincPhaseDiffBits};
- constexpr uint BSincPhaseDiffMask{BSincPhaseDiffOne - 1u};
- constexpr uint CubicPhaseDiffBits{MixerFracBits - CubicPhaseBits};
- constexpr uint CubicPhaseDiffOne{1 << CubicPhaseDiffBits};
- constexpr uint CubicPhaseDiffMask{CubicPhaseDiffOne - 1u};
- force_inline __m128 vmadd(const __m128 x, const __m128 y, const __m128 z) noexcept
- { return _mm_add_ps(x, _mm_mul_ps(y, z)); }
- inline void ApplyCoeffs(const al::span<float2> Values, const size_t IrSize,
- const ConstHrirSpan Coeffs, const float left, const float right)
- {
- ASSUME(IrSize >= MinIrLength);
- ASSUME(IrSize <= HrirLength);
- const auto lrlr = _mm_setr_ps(left, right, left, right);
- /* Round up the IR size to a multiple of 2 for SIMD (2 IRs for 2 channels
- * is 4 floats), to avoid cutting the last sample for odd IR counts. The
- * underlying HRIR is a fixed-size multiple of 2, any extra samples are
- * either 0 (silence) or more IR samples that get applied for "free".
- */
- const auto count4 = size_t{(IrSize+1) >> 1};
- /* This isn't technically correct to test alignment, but it's true for
- * systems that support SSE, which is the only one that needs to know the
- * alignment of Values (which alternates between 8- and 16-byte aligned).
- */
- if(!(reinterpret_cast<uintptr_t>(Values.data())&15))
- {
- const auto vals4 = al::span{reinterpret_cast<__m128*>(Values[0].data()), count4};
- const auto coeffs4 = al::span{reinterpret_cast<const __m128*>(Coeffs[0].data()), count4};
- std::transform(vals4.cbegin(), vals4.cend(), coeffs4.cbegin(), vals4.begin(),
- [lrlr](const __m128 &val, const __m128 &coeff) -> __m128
- { return vmadd(val, coeff, lrlr); });
- }
- else
- {
- auto coeffs = _mm_load_ps(Coeffs[0].data());
- auto vals = _mm_loadl_pi(_mm_setzero_ps(), reinterpret_cast<__m64*>(Values[0].data()));
- auto imp0 = _mm_mul_ps(lrlr, coeffs);
- vals = _mm_add_ps(imp0, vals);
- _mm_storel_pi(reinterpret_cast<__m64*>(Values[0].data()), vals);
- size_t td{count4 - 1};
- size_t i{1};
- do {
- coeffs = _mm_load_ps(Coeffs[i+1].data());
- vals = _mm_load_ps(Values[i].data());
- const auto imp1 = _mm_mul_ps(lrlr, coeffs);
- imp0 = _mm_shuffle_ps(imp0, imp1, _MM_SHUFFLE(1, 0, 3, 2));
- vals = _mm_add_ps(imp0, vals);
- _mm_store_ps(Values[i].data(), vals);
- imp0 = imp1;
- i += 2;
- } while(--td);
- vals = _mm_loadl_pi(vals, reinterpret_cast<__m64*>(Values[i].data()));
- imp0 = _mm_movehl_ps(imp0, imp0);
- vals = _mm_add_ps(imp0, vals);
- _mm_storel_pi(reinterpret_cast<__m64*>(Values[i].data()), vals);
- }
- }
- force_inline void MixLine(const al::span<const float> InSamples, const al::span<float> dst,
- float &CurrentGain, const float TargetGain, const float delta, const size_t fade_len,
- const size_t realign_len, size_t Counter)
- {
- const auto step = float{(TargetGain-CurrentGain) * delta};
- size_t pos{0};
- if(std::abs(step) > std::numeric_limits<float>::epsilon())
- {
- const auto gain = CurrentGain;
- auto step_count = 0.0f;
- /* Mix with applying gain steps in aligned multiples of 4. */
- if(const size_t todo{fade_len >> 2})
- {
- const auto four4 = _mm_set1_ps(4.0f);
- const auto step4 = _mm_set1_ps(step);
- const auto gain4 = _mm_set1_ps(gain);
- auto step_count4 = _mm_setr_ps(0.0f, 1.0f, 2.0f, 3.0f);
- const auto in4 = al::span{reinterpret_cast<const __m128*>(InSamples.data()),
- InSamples.size()/4}.first(todo);
- const auto out4 = al::span{reinterpret_cast<__m128*>(dst.data()), dst.size()/4};
- std::transform(in4.begin(), in4.end(), out4.begin(), out4.begin(),
- [gain4,step4,four4,&step_count4](const __m128 val4, __m128 dry4) -> __m128
- {
- /* dry += val * (gain + step*step_count) */
- dry4 = vmadd(dry4, val4, vmadd(gain4, step4, step_count4));
- step_count4 = _mm_add_ps(step_count4, four4);
- return dry4;
- });
- pos += in4.size()*4;
- /* NOTE: step_count4 now represents the next four counts after the
- * last four mixed samples, so the lowest element represents the
- * next step count to apply.
- */
- step_count = _mm_cvtss_f32(step_count4);
- }
- /* Mix with applying left over gain steps that aren't aligned multiples of 4. */
- if(const size_t leftover{fade_len&3})
- {
- const auto in = InSamples.subspan(pos, leftover);
- const auto out = dst.subspan(pos);
- std::transform(in.begin(), in.end(), out.begin(), out.begin(),
- [gain,step,&step_count](const float val, float dry) noexcept -> float
- {
- dry += val * (gain + step*step_count);
- step_count += 1.0f;
- return dry;
- });
- pos += leftover;
- }
- if(pos < Counter)
- {
- CurrentGain = gain + step*step_count;
- return;
- }
- /* Mix until pos is aligned with 4 or the mix is done. */
- if(const size_t leftover{realign_len&3})
- {
- const auto in = InSamples.subspan(pos, leftover);
- const auto out = dst.subspan(pos);
- std::transform(in.begin(), in.end(), out.begin(), out.begin(),
- [TargetGain](const float val, const float dry) noexcept -> float
- { return dry + val*TargetGain; });
- pos += leftover;
- }
- }
- CurrentGain = TargetGain;
- if(!(std::abs(TargetGain) > GainSilenceThreshold))
- return;
- if(size_t todo{(InSamples.size()-pos) >> 2})
- {
- const auto in4 = al::span{reinterpret_cast<const __m128*>(InSamples.data()),
- InSamples.size()/4}.last(todo);
- const auto out = dst.subspan(pos);
- const auto out4 = al::span{reinterpret_cast<__m128*>(out.data()), out.size()/4};
- const auto gain4 = _mm_set1_ps(TargetGain);
- std::transform(in4.begin(), in4.end(), out4.begin(), out4.begin(),
- [gain4](const __m128 val4, const __m128 dry4) -> __m128
- { return vmadd(dry4, val4, gain4); });
- pos += in4.size()*4;
- }
- if(const size_t leftover{(InSamples.size()-pos)&3})
- {
- const auto in = InSamples.last(leftover);
- const auto out = dst.subspan(pos);
- std::transform(in.begin(), in.end(), out.begin(), out.begin(),
- [TargetGain](const float val, const float dry) noexcept -> float
- { return dry + val*TargetGain; });
- }
- }
- } // namespace
- template<>
- void Resample_<CubicTag,SSETag>(const InterpState *state, const al::span<const float> src,
- uint frac, const uint increment, const al::span<float> dst)
- {
- ASSUME(frac < MixerFracOne);
- const auto filter = std::get<CubicState>(*state).filter;
- size_t pos{MaxResamplerEdge-1};
- std::generate(dst.begin(), dst.end(), [&pos,&frac,src,increment,filter]() -> float
- {
- const uint pi{frac >> CubicPhaseDiffBits}; ASSUME(pi < CubicPhaseCount);
- const float pf{static_cast<float>(frac&CubicPhaseDiffMask) * (1.0f/CubicPhaseDiffOne)};
- const __m128 pf4{_mm_set1_ps(pf)};
- /* Apply the phase interpolated filter. */
- /* f = fil + pf*phd */
- const __m128 f4 = vmadd(_mm_load_ps(filter[pi].mCoeffs.data()), pf4,
- _mm_load_ps(filter[pi].mDeltas.data()));
- /* r = f*src */
- __m128 r4{_mm_mul_ps(f4, _mm_loadu_ps(&src[pos]))};
- r4 = _mm_add_ps(r4, _mm_shuffle_ps(r4, r4, _MM_SHUFFLE(0, 1, 2, 3)));
- r4 = _mm_add_ps(r4, _mm_movehl_ps(r4, r4));
- const float output{_mm_cvtss_f32(r4)};
- frac += increment;
- pos += frac>>MixerFracBits;
- frac &= MixerFracMask;
- return output;
- });
- }
- template<>
- void Resample_<BSincTag,SSETag>(const InterpState *state, const al::span<const float> src,
- uint frac, const uint increment, const al::span<float> dst)
- {
- const auto &bsinc = std::get<BsincState>(*state);
- const auto sf4 = _mm_set1_ps(bsinc.sf);
- const auto m = size_t{bsinc.m};
- ASSUME(m > 0);
- ASSUME(m <= MaxResamplerPadding);
- ASSUME(frac < MixerFracOne);
- const auto filter = bsinc.filter.first(4_uz*BSincPhaseCount*m);
- ASSUME(bsinc.l <= MaxResamplerEdge);
- auto pos = size_t{MaxResamplerEdge-bsinc.l};
- std::generate(dst.begin(), dst.end(), [&pos,&frac,src,increment,sf4,m,filter]() -> float
- {
- // Calculate the phase index and factor.
- const size_t pi{frac >> BSincPhaseDiffBits}; ASSUME(pi < BSincPhaseCount);
- const float pf{static_cast<float>(frac&BSincPhaseDiffMask) * (1.0f/BSincPhaseDiffOne)};
- // Apply the scale and phase interpolated filter.
- auto r4 = _mm_setzero_ps();
- {
- const auto pf4 = _mm_set1_ps(pf);
- const auto fil = filter.subspan(2_uz*pi*m);
- const auto phd = fil.subspan(m);
- const auto scd = fil.subspan(2_uz*BSincPhaseCount*m);
- const auto spd = scd.subspan(m);
- auto td = size_t{m >> 2};
- auto j = size_t{0};
- do {
- /* f = ((fil + sf*scd) + pf*(phd + sf*spd)) */
- const __m128 f4 = vmadd(
- vmadd(_mm_load_ps(&fil[j]), sf4, _mm_load_ps(&scd[j])),
- pf4, vmadd(_mm_load_ps(&phd[j]), sf4, _mm_load_ps(&spd[j])));
- /* r += f*src */
- r4 = vmadd(r4, f4, _mm_loadu_ps(&src[pos+j]));
- j += 4;
- } while(--td);
- }
- r4 = _mm_add_ps(r4, _mm_shuffle_ps(r4, r4, _MM_SHUFFLE(0, 1, 2, 3)));
- r4 = _mm_add_ps(r4, _mm_movehl_ps(r4, r4));
- const auto output = _mm_cvtss_f32(r4);
- frac += increment;
- pos += frac>>MixerFracBits;
- frac &= MixerFracMask;
- return output;
- });
- }
- template<>
- void Resample_<FastBSincTag,SSETag>(const InterpState *state, const al::span<const float> src,
- uint frac, const uint increment, const al::span<float> dst)
- {
- const auto &bsinc = std::get<BsincState>(*state);
- const auto m = size_t{bsinc.m};
- ASSUME(m > 0);
- ASSUME(m <= MaxResamplerPadding);
- ASSUME(frac < MixerFracOne);
- const auto filter = bsinc.filter.first(2_uz*m*BSincPhaseCount);
- ASSUME(bsinc.l <= MaxResamplerEdge);
- size_t pos{MaxResamplerEdge-bsinc.l};
- std::generate(dst.begin(), dst.end(), [&pos,&frac,src,increment,filter,m]() -> float
- {
- // Calculate the phase index and factor.
- const size_t pi{frac >> BSincPhaseDiffBits}; ASSUME(pi < BSincPhaseCount);
- const float pf{static_cast<float>(frac&BSincPhaseDiffMask) * (1.0f/BSincPhaseDiffOne)};
- // Apply the phase interpolated filter.
- auto r4 = _mm_setzero_ps();
- {
- const auto pf4 = _mm_set1_ps(pf);
- const auto fil = filter.subspan(2_uz*m*pi);
- const auto phd = fil.subspan(m);
- auto td = size_t{m >> 2};
- auto j = size_t{0};
- do {
- /* f = fil + pf*phd */
- const auto f4 = vmadd(_mm_load_ps(&fil[j]), pf4, _mm_load_ps(&phd[j]));
- /* r += f*src */
- r4 = vmadd(r4, f4, _mm_loadu_ps(&src[pos+j]));
- j += 4;
- } while(--td);
- }
- r4 = _mm_add_ps(r4, _mm_shuffle_ps(r4, r4, _MM_SHUFFLE(0, 1, 2, 3)));
- r4 = _mm_add_ps(r4, _mm_movehl_ps(r4, r4));
- const auto output = _mm_cvtss_f32(r4);
- frac += increment;
- pos += frac>>MixerFracBits;
- frac &= MixerFracMask;
- return output;
- });
- }
- template<>
- void MixHrtf_<SSETag>(const al::span<const float> InSamples, const al::span<float2> AccumSamples,
- const uint IrSize, const MixHrtfFilter *hrtfparams, const size_t SamplesToDo)
- { MixHrtfBase<ApplyCoeffs>(InSamples, AccumSamples, IrSize, hrtfparams, SamplesToDo); }
- template<>
- void MixHrtfBlend_<SSETag>(const al::span<const float> InSamples,
- const al::span<float2> AccumSamples, const uint IrSize, const HrtfFilter *oldparams,
- const MixHrtfFilter *newparams, const size_t SamplesToDo)
- {
- MixHrtfBlendBase<ApplyCoeffs>(InSamples, AccumSamples, IrSize, oldparams, newparams,
- SamplesToDo);
- }
- template<>
- void MixDirectHrtf_<SSETag>(const FloatBufferSpan LeftOut, const FloatBufferSpan RightOut,
- const al::span<const FloatBufferLine> InSamples, const al::span<float2> AccumSamples,
- const al::span<float,BufferLineSize> TempBuf, const al::span<HrtfChannelState> ChanState,
- const size_t IrSize, const size_t SamplesToDo)
- {
- MixDirectHrtfBase<ApplyCoeffs>(LeftOut, RightOut, InSamples, AccumSamples, TempBuf, ChanState,
- IrSize, SamplesToDo);
- }
- template<>
- void Mix_<SSETag>(const al::span<const float> InSamples, const al::span<FloatBufferLine> OutBuffer,
- const al::span<float> CurrentGains, const al::span<const float> TargetGains,
- const size_t Counter, const size_t OutPos)
- {
- if((OutPos&3) != 0) UNLIKELY
- return Mix_<CTag>(InSamples, OutBuffer, CurrentGains, TargetGains, Counter, OutPos);
- const float delta{(Counter > 0) ? 1.0f / static_cast<float>(Counter) : 0.0f};
- const auto fade_len = std::min(Counter, InSamples.size());
- const auto realign_len = std::min((fade_len+3_uz) & ~3_uz, InSamples.size()) - fade_len;
- auto curgains = CurrentGains.begin();
- auto targetgains = TargetGains.cbegin();
- for(FloatBufferLine &output : OutBuffer)
- MixLine(InSamples, al::span{output}.subspan(OutPos), *curgains++, *targetgains++, delta,
- fade_len, realign_len, Counter);
- }
- template<>
- void Mix_<SSETag>(const al::span<const float> InSamples, const al::span<float> OutBuffer,
- float &CurrentGain, const float TargetGain, const size_t Counter)
- {
- if((reinterpret_cast<uintptr_t>(OutBuffer.data())&15) != 0) UNLIKELY
- return Mix_<CTag>(InSamples, OutBuffer, CurrentGain, TargetGain, Counter);
- const float delta{(Counter > 0) ? 1.0f / static_cast<float>(Counter) : 0.0f};
- const auto fade_len = std::min(Counter, InSamples.size());
- const auto realign_len = std::min((fade_len+3_uz) & ~3_uz, InSamples.size()) - fade_len;
- MixLine(InSamples, OutBuffer, CurrentGain, TargetGain, delta, fade_len, realign_len, Counter);
- }
|