| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179 |
- #include "config.h"
- #include "bformatdec.h"
- #include <algorithm>
- #include <array>
- #include <cmath>
- #include <functional>
- #include <utility>
- #include "alnumbers.h"
- #include "bufferline.h"
- #include "filters/splitter.h"
- #include "flexarray.h"
- #include "front_stablizer.h"
- #include "mixer.h"
- #include "opthelpers.h"
- namespace {
- template<typename... Ts>
- struct overloaded : Ts... { using Ts::operator()...; };
- template<typename... Ts>
- overloaded(Ts...) -> overloaded<Ts...>;
- } // namespace
- BFormatDec::BFormatDec(const size_t inchans, const al::span<const ChannelDec> coeffs,
- const al::span<const ChannelDec> coeffslf, const float xover_f0norm,
- std::unique_ptr<FrontStablizer> stablizer)
- : mStablizer{std::move(stablizer)}
- {
- if(coeffslf.empty())
- {
- auto &decoder = mChannelDec.emplace<std::vector<ChannelDecoderSingle>>(inchans);
- for(size_t j{0};j < decoder.size();++j)
- {
- std::transform(coeffs.cbegin(), coeffs.cend(), decoder[j].mGains.begin(),
- [j](const ChannelDec &incoeffs) { return incoeffs[j]; });
- }
- }
- else
- {
- auto &decoder = mChannelDec.emplace<std::vector<ChannelDecoderDual>>(inchans);
- decoder[0].mXOver.init(xover_f0norm);
- for(size_t j{1};j < decoder.size();++j)
- decoder[j].mXOver = decoder[0].mXOver;
- for(size_t j{0};j < decoder.size();++j)
- {
- std::transform(coeffs.cbegin(), coeffs.cend(), decoder[j].mGains[sHFBand].begin(),
- [j](const ChannelDec &incoeffs) { return incoeffs[j]; });
- std::transform(coeffslf.cbegin(), coeffslf.cend(), decoder[j].mGains[sLFBand].begin(),
- [j](const ChannelDec &incoeffs) { return incoeffs[j]; });
- }
- }
- }
- void BFormatDec::process(const al::span<FloatBufferLine> OutBuffer,
- const al::span<const FloatBufferLine> InSamples, const size_t SamplesToDo)
- {
- ASSUME(SamplesToDo > 0);
- auto decode_dualband = [=](std::vector<ChannelDecoderDual> &decoder)
- {
- auto input = InSamples.cbegin();
- const auto hfSamples = al::span<float>{mSamples[sHFBand]}.first(SamplesToDo);
- const auto lfSamples = al::span<float>{mSamples[sLFBand]}.first(SamplesToDo);
- for(auto &chandec : decoder)
- {
- chandec.mXOver.process(al::span{*input++}.first(SamplesToDo), hfSamples, lfSamples);
- MixSamples(hfSamples, OutBuffer, chandec.mGains[sHFBand], chandec.mGains[sHFBand],0,0);
- MixSamples(lfSamples, OutBuffer, chandec.mGains[sLFBand], chandec.mGains[sLFBand],0,0);
- }
- };
- auto decode_singleband = [=](std::vector<ChannelDecoderSingle> &decoder)
- {
- auto input = InSamples.cbegin();
- for(auto &chandec : decoder)
- {
- MixSamples(al::span{*input++}.first(SamplesToDo), OutBuffer, chandec.mGains,
- chandec.mGains, 0, 0);
- }
- };
- std::visit(overloaded{decode_dualband, decode_singleband}, mChannelDec);
- }
- void BFormatDec::processStablize(const al::span<FloatBufferLine> OutBuffer,
- const al::span<const FloatBufferLine> InSamples, const size_t lidx, const size_t ridx,
- const size_t cidx, const size_t SamplesToDo)
- {
- ASSUME(SamplesToDo > 0);
- /* Move the existing direct L/R signal out so it doesn't get processed by
- * the stablizer.
- */
- const auto leftout = al::span<float>{OutBuffer[lidx]}.first(SamplesToDo);
- const auto rightout = al::span<float>{OutBuffer[ridx]}.first(SamplesToDo);
- const auto mid = al::span{mStablizer->MidDirect}.first(SamplesToDo);
- const auto side = al::span{mStablizer->Side}.first(SamplesToDo);
- std::transform(leftout.cbegin(), leftout.cend(), rightout.cbegin(), mid.begin(),std::plus{});
- std::transform(leftout.cbegin(), leftout.cend(), rightout.cbegin(), side.begin(),std::minus{});
- std::fill_n(leftout.begin(), leftout.size(), 0.0f);
- std::fill_n(rightout.begin(), rightout.size(), 0.0f);
- /* Decode the B-Format input to OutBuffer. */
- process(OutBuffer, InSamples, SamplesToDo);
- /* Include the decoded side signal with the direct side signal. */
- for(size_t i{0};i < SamplesToDo;++i)
- side[i] += leftout[i] - rightout[i];
- /* Get the decoded mid signal and band-split it. */
- const auto tmpsamples = al::span{mStablizer->Temp}.first(SamplesToDo);
- std::transform(leftout.cbegin(), leftout.cend(), rightout.cbegin(), tmpsamples.begin(),
- std::plus{});
- mStablizer->MidFilter.process(tmpsamples, mStablizer->MidHF, mStablizer->MidLF);
- /* Apply an all-pass to all channels to match the band-splitter's phase
- * shift. This is to keep the phase synchronized between the existing
- * signal and the split mid signal.
- */
- const size_t NumChannels{OutBuffer.size()};
- for(size_t i{0u};i < NumChannels;i++)
- {
- /* Skip the left and right channels, which are going to get overwritten,
- * and substitute the direct mid signal and direct+decoded side signal.
- */
- if(i == lidx)
- mStablizer->ChannelFilters[i].processAllPass(mid);
- else if(i == ridx)
- mStablizer->ChannelFilters[i].processAllPass(side);
- else
- mStablizer->ChannelFilters[i].processAllPass({OutBuffer[i].data(), SamplesToDo});
- }
- /* This pans the separate low- and high-frequency signals between being on
- * the center channel and the left+right channels. The low-frequency signal
- * is panned 1/3rd toward center and the high-frequency signal is panned
- * 1/4th toward center. These values can be tweaked.
- */
- const float cos_lf{std::cos(1.0f/3.0f * (al::numbers::pi_v<float>*0.5f))};
- const float cos_hf{std::cos(1.0f/4.0f * (al::numbers::pi_v<float>*0.5f))};
- const float sin_lf{std::sin(1.0f/3.0f * (al::numbers::pi_v<float>*0.5f))};
- const float sin_hf{std::sin(1.0f/4.0f * (al::numbers::pi_v<float>*0.5f))};
- const auto centerout = al::span<float>{OutBuffer[cidx]}.first(SamplesToDo);
- for(size_t i{0};i < SamplesToDo;i++)
- {
- /* Add the direct mid signal to the processed mid signal so it can be
- * properly combined with the direct+decoded side signal.
- */
- const float m{mStablizer->MidLF[i]*cos_lf + mStablizer->MidHF[i]*cos_hf + mid[i]};
- const float c{mStablizer->MidLF[i]*sin_lf + mStablizer->MidHF[i]*sin_hf};
- const float s{side[i]};
- /* The generated center channel signal adds to the existing signal,
- * while the modified left and right channels replace.
- */
- leftout[i] = (m + s) * 0.5f;
- rightout[i] = (m - s) * 0.5f;
- centerout[i] += c * 0.5f;
- }
- }
- std::unique_ptr<BFormatDec> BFormatDec::Create(const size_t inchans,
- const al::span<const ChannelDec> coeffs, const al::span<const ChannelDec> coeffslf,
- const float xover_f0norm, std::unique_ptr<FrontStablizer> stablizer)
- {
- return std::make_unique<BFormatDec>(inchans, coeffs, coeffslf, xover_f0norm,
- std::move(stablizer));
- }
|