2007-11-13 18:02:18 -08:00
|
|
|
/**
|
|
|
|
* OpenAL cross platform audio library
|
|
|
|
* Copyright (C) 1999-2007 by authors.
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
2014-08-18 14:11:03 +02:00
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2007-11-13 18:02:18 -08:00
|
|
|
* Or go to http://www.gnu.org/copyleft/lgpl.html
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "alu.h"
|
2009-01-25 19:20:47 -08:00
|
|
|
|
2019-07-28 11:28:36 -07:00
|
|
|
#include <algorithm>
|
|
|
|
#include <array>
|
|
|
|
#include <atomic>
|
2019-12-01 17:57:31 -08:00
|
|
|
#include <cassert>
|
2019-07-28 11:28:36 -07:00
|
|
|
#include <chrono>
|
|
|
|
#include <climits>
|
|
|
|
#include <cstdarg>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <functional>
|
|
|
|
#include <iterator>
|
2018-12-22 16:01:14 -08:00
|
|
|
#include <limits>
|
2019-07-28 11:28:36 -07:00
|
|
|
#include <memory>
|
|
|
|
#include <new>
|
2021-04-26 07:56:00 -07:00
|
|
|
#include <stdint.h>
|
2019-07-28 11:28:36 -07:00
|
|
|
#include <utility>
|
2018-11-21 05:06:31 -08:00
|
|
|
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "almalloc.h"
|
2022-01-27 02:59:07 -08:00
|
|
|
#include "alnumbers.h"
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "alnumeric.h"
|
|
|
|
#include "alspan.h"
|
2019-09-16 15:10:36 -07:00
|
|
|
#include "alstring.h"
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "atomic.h"
|
2020-12-12 10:38:24 -08:00
|
|
|
#include "core/ambidefs.h"
|
2021-04-25 14:29:21 -07:00
|
|
|
#include "core/async_event.h"
|
2021-04-25 18:08:08 -07:00
|
|
|
#include "core/bformatdec.h"
|
2020-12-04 13:13:52 -08:00
|
|
|
#include "core/bs2b.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "core/bsinc_defs.h"
|
2020-12-04 07:35:40 -08:00
|
|
|
#include "core/bsinc_tables.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "core/bufferline.h"
|
2021-04-27 08:26:42 -07:00
|
|
|
#include "core/buffer_storage.h"
|
2021-04-27 08:04:09 -07:00
|
|
|
#include "core/context.h"
|
2020-12-31 16:47:12 -08:00
|
|
|
#include "core/cpu_caps.h"
|
2020-11-27 19:18:17 -08:00
|
|
|
#include "core/devformat.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "core/device.h"
|
2021-12-17 17:13:59 -08:00
|
|
|
#include "core/effects/base.h"
|
|
|
|
#include "core/effectslot.h"
|
2020-12-04 09:42:13 -08:00
|
|
|
#include "core/filters/biquad.h"
|
|
|
|
#include "core/filters/nfc.h"
|
2020-12-31 16:47:12 -08:00
|
|
|
#include "core/fpu_ctrl.h"
|
2021-04-22 10:26:20 -07:00
|
|
|
#include "core/hrtf.h"
|
2020-12-03 23:37:50 -08:00
|
|
|
#include "core/mastering.h"
|
2021-04-25 11:36:37 -07:00
|
|
|
#include "core/mixer.h"
|
2020-12-12 14:58:09 -08:00
|
|
|
#include "core/mixer/defs.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "core/mixer/hrtfdefs.h"
|
|
|
|
#include "core/resampler_limits.h"
|
2020-12-04 13:13:52 -08:00
|
|
|
#include "core/uhjfilter.h"
|
2021-04-27 08:26:42 -07:00
|
|
|
#include "core/voice.h"
|
2021-04-27 08:04:09 -07:00
|
|
|
#include "core/voice_change.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "intrusive_ptr.h"
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "opthelpers.h"
|
2018-02-03 01:07:06 -08:00
|
|
|
#include "ringbuffer.h"
|
2019-08-12 03:59:52 -07:00
|
|
|
#include "strutils.h"
|
2019-07-28 11:28:36 -07:00
|
|
|
#include "threads.h"
|
|
|
|
#include "vecmat.h"
|
2021-04-26 07:56:00 -07:00
|
|
|
#include "vector.h"
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2020-04-03 03:52:23 -07:00
|
|
|
struct CTag;
|
|
|
|
#ifdef HAVE_SSE
|
|
|
|
struct SSETag;
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE2
|
|
|
|
struct SSE2Tag;
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE4_1
|
|
|
|
struct SSE4Tag;
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_NEON
|
|
|
|
struct NEONTag;
|
|
|
|
#endif
|
|
|
|
struct PointTag;
|
|
|
|
struct LerpTag;
|
|
|
|
struct CubicTag;
|
|
|
|
struct BSincTag;
|
|
|
|
struct FastBSincTag;
|
|
|
|
|
2018-11-11 14:56:25 -08:00
|
|
|
|
2020-11-28 03:38:20 -08:00
|
|
|
static_assert(!(MaxResamplerPadding&1), "MaxResamplerPadding is not a multiple of two");
|
2019-09-28 16:41:26 -07:00
|
|
|
|
|
|
|
|
2018-12-08 21:58:44 -08:00
|
|
|
namespace {
|
|
|
|
|
2021-04-26 07:56:00 -07:00
|
|
|
using uint = unsigned int;
|
|
|
|
|
2020-12-03 23:26:07 -08:00
|
|
|
constexpr uint MaxPitch{10};
|
|
|
|
|
|
|
|
static_assert((BufferLineSize-1)/MaxPitch > 0, "MaxPitch is too large for BufferLineSize!");
|
|
|
|
static_assert((INT_MAX>>MixerFracBits)/MaxPitch > BufferLineSize,
|
|
|
|
"MaxPitch and/or BufferLineSize are too large for MixerFracBits!");
|
|
|
|
|
2018-12-26 21:55:39 -08:00
|
|
|
using namespace std::placeholders;
|
|
|
|
|
2020-04-03 08:49:15 -07:00
|
|
|
float InitConeScale()
|
2018-12-08 21:58:44 -08:00
|
|
|
{
|
2020-04-03 08:49:15 -07:00
|
|
|
float ret{1.0f};
|
2019-08-12 03:59:52 -07:00
|
|
|
if(auto optval = al::getenv("__ALSOFT_HALF_ANGLE_CONES"))
|
|
|
|
{
|
2019-09-16 15:10:36 -07:00
|
|
|
if(al::strcasecmp(optval->c_str(), "true") == 0
|
|
|
|
|| strtol(optval->c_str(), nullptr, 0) == 1)
|
2019-08-12 03:59:52 -07:00
|
|
|
ret *= 0.5f;
|
|
|
|
}
|
2018-12-08 21:58:44 -08:00
|
|
|
return ret;
|
|
|
|
}
|
2017-08-07 01:38:26 -07:00
|
|
|
/* Cone scalar */
|
2020-04-03 08:49:15 -07:00
|
|
|
const float ConeScale{InitConeScale()};
|
2017-08-07 01:38:26 -07:00
|
|
|
|
2022-03-10 22:37:02 -08:00
|
|
|
/* Localized scalars for mono sources (initialized in aluInit, after
|
2022-03-10 17:47:42 -08:00
|
|
|
* configuration is loaded).
|
|
|
|
*/
|
2022-03-10 22:37:02 -08:00
|
|
|
float XScale{1.0f};
|
|
|
|
float YScale{1.0f};
|
2022-03-10 17:47:42 -08:00
|
|
|
float ZScale{1.0f};
|
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
|
2017-08-07 01:38:26 -07:00
|
|
|
struct ChanMap {
|
2018-12-24 19:29:01 -08:00
|
|
|
Channel channel;
|
2020-04-03 08:49:15 -07:00
|
|
|
float angle;
|
|
|
|
float elevation;
|
2017-08-07 01:38:26 -07:00
|
|
|
};
|
|
|
|
|
2021-02-06 14:39:30 -08:00
|
|
|
using HrtfDirectMixerFunc = void(*)(const FloatBufferSpan LeftOut, const FloatBufferSpan RightOut,
|
|
|
|
const al::span<const FloatBufferLine> InSamples, float2 *AccumSamples, float *TempBuf,
|
|
|
|
HrtfChannelState *ChanState, const size_t IrSize, const size_t BufferSize);
|
2020-04-16 02:05:07 -07:00
|
|
|
|
2020-03-25 22:24:09 -07:00
|
|
|
HrtfDirectMixerFunc MixDirectHrtf{MixDirectHrtf_<CTag>};
|
2019-10-02 16:53:23 -07:00
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
inline HrtfDirectMixerFunc SelectHrtfMixer(void)
|
2014-11-22 04:20:17 -08:00
|
|
|
{
|
|
|
|
#ifdef HAVE_NEON
|
|
|
|
if((CPUCapFlags&CPU_CAP_NEON))
|
2019-01-23 12:23:05 -08:00
|
|
|
return MixDirectHrtf_<NEONTag>;
|
2014-11-22 04:20:17 -08:00
|
|
|
#endif
|
2017-04-12 22:44:16 -07:00
|
|
|
#ifdef HAVE_SSE
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE))
|
2019-01-23 12:23:05 -08:00
|
|
|
return MixDirectHrtf_<SSETag>;
|
2017-04-12 22:44:16 -07:00
|
|
|
#endif
|
2014-11-22 04:20:17 -08:00
|
|
|
|
2019-01-23 12:23:05 -08:00
|
|
|
return MixDirectHrtf_<CTag>;
|
2014-11-22 04:20:17 -08:00
|
|
|
}
|
|
|
|
|
2019-09-28 03:42:17 -07:00
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
inline void BsincPrepare(const uint increment, BsincState *state, const BSincTable *table)
|
2019-09-28 03:42:17 -07:00
|
|
|
{
|
2020-10-19 07:55:25 -07:00
|
|
|
size_t si{BSincScaleCount - 1};
|
2019-09-28 03:42:17 -07:00
|
|
|
float sf{0.0f};
|
|
|
|
|
2020-10-21 17:16:27 -07:00
|
|
|
if(increment > MixerFracOne)
|
2019-09-28 03:42:17 -07:00
|
|
|
{
|
Avoiding cutting all bsinc resampler output at scale 0
This is mostly for the SampleConverter, used by some capture backends. When
recording at really low rates, like 5512hz, with a device capturing at a higher
rate like 44100hz or 48000hz, it hits the filter's downscaling limit and
produces pure silence.
In such cases, it's better to just accept some aliasing noise so that the app
will still get some recognizable audio. The alternative would be to scale the
desired rate by 2x, 3x, etc until it's above the bsinc limit, then take every
2nd, 3rd, etc sample of the result as if by an extra simpler resampler pass.
2021-02-24 21:44:55 -08:00
|
|
|
sf = MixerFracOne/static_cast<float>(increment) - table->scaleBase;
|
|
|
|
sf = maxf(0.0f, BSincScaleCount*sf*table->scaleRange - 1.0f);
|
2019-09-28 03:42:17 -07:00
|
|
|
si = float2uint(sf);
|
|
|
|
/* The interpolation factor is fit to this diagonally-symmetric curve
|
|
|
|
* to reduce the transition ripple caused by interpolating different
|
|
|
|
* scales of the sinc function.
|
|
|
|
*/
|
|
|
|
sf = 1.0f - std::cos(std::asin(sf - static_cast<float>(si)));
|
|
|
|
}
|
|
|
|
|
|
|
|
state->sf = sf;
|
|
|
|
state->m = table->m[si];
|
|
|
|
state->l = (state->m/2) - 1;
|
|
|
|
state->filter = table->Tab + table->filterOffset[si];
|
|
|
|
}
|
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
inline ResamplerFunc SelectResampler(Resampler resampler, uint increment)
|
2019-09-28 03:42:17 -07:00
|
|
|
{
|
|
|
|
switch(resampler)
|
|
|
|
{
|
|
|
|
case Resampler::Point:
|
|
|
|
return Resample_<PointTag,CTag>;
|
|
|
|
case Resampler::Linear:
|
|
|
|
#ifdef HAVE_NEON
|
|
|
|
if((CPUCapFlags&CPU_CAP_NEON))
|
|
|
|
return Resample_<LerpTag,NEONTag>;
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE4_1
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE4_1))
|
|
|
|
return Resample_<LerpTag,SSE4Tag>;
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE2
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE2))
|
|
|
|
return Resample_<LerpTag,SSE2Tag>;
|
|
|
|
#endif
|
|
|
|
return Resample_<LerpTag,CTag>;
|
|
|
|
case Resampler::Cubic:
|
|
|
|
return Resample_<CubicTag,CTag>;
|
|
|
|
case Resampler::BSinc12:
|
|
|
|
case Resampler::BSinc24:
|
2022-04-24 22:32:59 -07:00
|
|
|
if(increment > MixerFracOne)
|
2019-09-28 03:42:17 -07:00
|
|
|
{
|
|
|
|
#ifdef HAVE_NEON
|
|
|
|
if((CPUCapFlags&CPU_CAP_NEON))
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<BSincTag,NEONTag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE))
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<BSincTag,SSETag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
#endif
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<BSincTag,CTag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
}
|
2022-04-24 22:32:59 -07:00
|
|
|
/* fall-through */
|
|
|
|
case Resampler::FastBSinc12:
|
|
|
|
case Resampler::FastBSinc24:
|
2019-09-28 03:42:17 -07:00
|
|
|
#ifdef HAVE_NEON
|
|
|
|
if((CPUCapFlags&CPU_CAP_NEON))
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<FastBSincTag,NEONTag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SSE
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE))
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<FastBSincTag,SSETag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
#endif
|
2022-04-24 22:32:59 -07:00
|
|
|
return Resample_<FastBSincTag,CTag>;
|
2019-09-28 03:42:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return Resample_<PointTag,CTag>;
|
|
|
|
}
|
|
|
|
|
2019-06-15 23:10:11 -07:00
|
|
|
} // namespace
|
|
|
|
|
2022-03-10 17:47:42 -08:00
|
|
|
void aluInit(CompatFlagBitset flags)
|
2019-06-15 23:10:11 -07:00
|
|
|
{
|
|
|
|
MixDirectHrtf = SelectHrtfMixer();
|
2022-03-10 22:37:02 -08:00
|
|
|
XScale = flags.test(CompatFlags::ReverseX) ? -1.0f : 1.0f;
|
|
|
|
YScale = flags.test(CompatFlags::ReverseY) ? -1.0f : 1.0f;
|
2022-03-10 17:47:42 -08:00
|
|
|
ZScale = flags.test(CompatFlags::ReverseZ) ? -1.0f : 1.0f;
|
2019-06-15 23:10:11 -07:00
|
|
|
}
|
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
ResamplerFunc PrepareResampler(Resampler resampler, uint increment, InterpState *state)
|
2019-09-28 03:42:17 -07:00
|
|
|
{
|
|
|
|
switch(resampler)
|
|
|
|
{
|
|
|
|
case Resampler::Point:
|
|
|
|
case Resampler::Linear:
|
|
|
|
case Resampler::Cubic:
|
|
|
|
break;
|
|
|
|
case Resampler::FastBSinc12:
|
|
|
|
case Resampler::BSinc12:
|
|
|
|
BsincPrepare(increment, &state->bsinc, &bsinc12);
|
|
|
|
break;
|
|
|
|
case Resampler::FastBSinc24:
|
|
|
|
case Resampler::BSinc24:
|
|
|
|
BsincPrepare(increment, &state->bsinc, &bsinc24);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return SelectResampler(resampler, increment);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::ProcessHrtf(const size_t SamplesToDo)
|
2018-02-10 15:50:05 -08:00
|
|
|
{
|
2018-12-26 12:46:01 -08:00
|
|
|
/* HRTF is stereo output only. */
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint lidx{RealOut.ChannelIndex[FrontLeft]};
|
|
|
|
const uint ridx{RealOut.ChannelIndex[FrontRight]};
|
2018-12-26 12:46:01 -08:00
|
|
|
|
2019-08-07 11:43:53 -07:00
|
|
|
MixDirectHrtf(RealOut.Buffer[lidx], RealOut.Buffer[ridx], Dry.Buffer, HrtfAccumData,
|
2020-12-12 13:52:14 -08:00
|
|
|
mHrtfState->mTemp.data(), mHrtfState->mChannels.data(), mHrtfState->mIrSize, SamplesToDo);
|
2018-02-10 15:50:05 -08:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::ProcessAmbiDec(const size_t SamplesToDo)
|
2018-02-10 15:50:05 -08:00
|
|
|
{
|
2019-08-07 11:43:53 -07:00
|
|
|
AmbiDecoder->process(RealOut.Buffer, Dry.Buffer.data(), SamplesToDo);
|
2018-02-10 15:50:05 -08:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::ProcessAmbiDecStablized(const size_t SamplesToDo)
|
2020-06-15 02:05:10 -07:00
|
|
|
{
|
|
|
|
/* Decode with front image stablization. */
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint lidx{RealOut.ChannelIndex[FrontLeft]};
|
|
|
|
const uint ridx{RealOut.ChannelIndex[FrontRight]};
|
|
|
|
const uint cidx{RealOut.ChannelIndex[FrontCenter]};
|
2020-06-15 02:05:10 -07:00
|
|
|
|
|
|
|
AmbiDecoder->processStablize(RealOut.Buffer, Dry.Buffer.data(), lidx, ridx, cidx,
|
|
|
|
SamplesToDo);
|
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::ProcessUhj(const size_t SamplesToDo)
|
2018-02-10 15:50:05 -08:00
|
|
|
{
|
2018-12-26 12:46:01 -08:00
|
|
|
/* UHJ is stereo output only. */
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint lidx{RealOut.ChannelIndex[FrontLeft]};
|
|
|
|
const uint ridx{RealOut.ChannelIndex[FrontRight]};
|
2018-04-22 02:38:09 -07:00
|
|
|
|
|
|
|
/* Encode to stereo-compatible 2-channel UHJ output. */
|
2021-11-25 20:29:50 -08:00
|
|
|
mUhjEncoder->encode(RealOut.Buffer[lidx].data(), RealOut.Buffer[ridx].data(),
|
2022-05-06 04:14:34 -07:00
|
|
|
{{Dry.Buffer[0].data(), Dry.Buffer[1].data(), Dry.Buffer[2].data()}}, SamplesToDo);
|
2018-02-10 15:50:05 -08:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::ProcessBs2b(const size_t SamplesToDo)
|
2018-02-10 15:50:05 -08:00
|
|
|
{
|
2019-06-12 22:51:09 -07:00
|
|
|
/* First, decode the ambisonic mix to the "real" output. */
|
2019-08-07 11:43:53 -07:00
|
|
|
AmbiDecoder->process(RealOut.Buffer, Dry.Buffer.data(), SamplesToDo);
|
2019-06-12 22:51:09 -07:00
|
|
|
|
2018-12-26 12:46:01 -08:00
|
|
|
/* BS2B is stereo output only. */
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint lidx{RealOut.ChannelIndex[FrontLeft]};
|
|
|
|
const uint ridx{RealOut.ChannelIndex[FrontRight]};
|
2018-04-22 02:38:09 -07:00
|
|
|
|
2019-06-12 22:51:09 -07:00
|
|
|
/* Now apply the BS2B binaural/crossfeed filter. */
|
2019-08-07 11:43:53 -07:00
|
|
|
bs2b_cross_feed(Bs2b.get(), RealOut.Buffer[lidx].data(), RealOut.Buffer[ridx].data(),
|
|
|
|
SamplesToDo);
|
2018-02-10 15:50:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
namespace {
|
|
|
|
|
2022-04-24 22:53:55 -07:00
|
|
|
using AmbiRotateMatrix = std::array<std::array<float,MaxAmbiChannels>,MaxAmbiChannels>;
|
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
/* This RNG method was created based on the math found in opusdec. It's quick,
|
|
|
|
* and starting with a seed value of 22222, is suitable for generating
|
|
|
|
* whitenoise.
|
|
|
|
*/
|
2020-12-15 20:48:21 -08:00
|
|
|
inline uint dither_rng(uint *seed) noexcept
|
2018-11-20 02:12:04 -08:00
|
|
|
{
|
|
|
|
*seed = (*seed * 96314165) + 907633515;
|
|
|
|
return *seed;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-21 02:05:55 -08:00
|
|
|
inline auto& GetAmbiScales(AmbiScaling scaletype) noexcept
|
2019-12-02 12:50:18 -08:00
|
|
|
{
|
2021-07-13 06:31:13 -07:00
|
|
|
switch(scaletype)
|
|
|
|
{
|
|
|
|
case AmbiScaling::FuMa: return AmbiScale::FromFuMa();
|
|
|
|
case AmbiScaling::SN3D: return AmbiScale::FromSN3D();
|
|
|
|
case AmbiScaling::UHJ: return AmbiScale::FromUHJ();
|
|
|
|
case AmbiScaling::N3D: break;
|
|
|
|
}
|
2021-01-21 02:05:55 -08:00
|
|
|
return AmbiScale::FromN3D();
|
2019-12-02 12:50:18 -08:00
|
|
|
}
|
|
|
|
|
2021-01-21 02:05:55 -08:00
|
|
|
inline auto& GetAmbiLayout(AmbiLayout layouttype) noexcept
|
2019-12-02 12:50:18 -08:00
|
|
|
{
|
2021-01-21 02:05:55 -08:00
|
|
|
if(layouttype == AmbiLayout::FuMa) return AmbiIndex::FromFuMa();
|
|
|
|
return AmbiIndex::FromACN();
|
2019-12-02 12:50:18 -08:00
|
|
|
}
|
|
|
|
|
2021-01-21 02:05:55 -08:00
|
|
|
inline auto& GetAmbi2DLayout(AmbiLayout layouttype) noexcept
|
2019-12-02 12:50:18 -08:00
|
|
|
{
|
2021-01-21 02:05:55 -08:00
|
|
|
if(layouttype == AmbiLayout::FuMa) return AmbiIndex::FromFuMa2D();
|
|
|
|
return AmbiIndex::FromACN2D();
|
2019-12-02 12:50:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
bool CalcContextParams(ContextBase *ctx)
|
2017-09-27 08:55:42 -07:00
|
|
|
{
|
2020-12-15 17:25:45 -08:00
|
|
|
ContextProps *props{ctx->mParams.ContextUpdate.exchange(nullptr, std::memory_order_acq_rel)};
|
2017-09-27 08:55:42 -07:00
|
|
|
if(!props) return false;
|
|
|
|
|
2021-05-25 14:08:16 -07:00
|
|
|
const alu::Vector pos{props->Position[0], props->Position[1], props->Position[2], 1.0f};
|
|
|
|
ctx->mParams.Position = pos;
|
|
|
|
|
2012-10-12 07:38:29 -07:00
|
|
|
/* AT then UP */
|
2018-12-26 22:27:34 -08:00
|
|
|
alu::Vector N{props->OrientAt[0], props->OrientAt[1], props->OrientAt[2], 0.0f};
|
2018-12-12 04:22:11 -08:00
|
|
|
N.normalize();
|
2018-12-26 22:27:34 -08:00
|
|
|
alu::Vector V{props->OrientUp[0], props->OrientUp[1], props->OrientUp[2], 0.0f};
|
2018-12-12 04:22:11 -08:00
|
|
|
V.normalize();
|
2012-10-12 07:38:29 -07:00
|
|
|
/* Build and normalize right-vector */
|
2020-11-13 23:33:40 -08:00
|
|
|
alu::Vector U{N.cross_product(V)};
|
2018-12-12 04:22:11 -08:00
|
|
|
U.normalize();
|
|
|
|
|
2021-05-25 14:08:16 -07:00
|
|
|
const alu::Matrix rot{
|
2020-11-13 23:33:40 -08:00
|
|
|
U[0], V[0], -N[0], 0.0,
|
|
|
|
U[1], V[1], -N[1], 0.0,
|
|
|
|
U[2], V[2], -N[2], 0.0,
|
|
|
|
0.0, 0.0, 0.0, 1.0};
|
2021-05-25 14:08:16 -07:00
|
|
|
const alu::Vector vel{props->Velocity[0], props->Velocity[1], props->Velocity[2], 0.0};
|
2015-08-24 03:02:58 -07:00
|
|
|
|
2021-05-25 14:08:16 -07:00
|
|
|
ctx->mParams.Matrix = rot;
|
|
|
|
ctx->mParams.Velocity = rot * vel;
|
2014-12-16 06:29:31 -08:00
|
|
|
|
2020-12-15 17:25:45 -08:00
|
|
|
ctx->mParams.Gain = props->Gain * ctx->mGainBoost;
|
|
|
|
ctx->mParams.MetersPerUnit = props->MetersPerUnit;
|
2022-02-08 22:41:44 -08:00
|
|
|
ctx->mParams.AirAbsorptionGainHF = props->AirAbsorptionGainHF;
|
2016-08-23 19:17:17 -07:00
|
|
|
|
2022-02-08 20:43:05 -08:00
|
|
|
ctx->mParams.DopplerFactor = props->DopplerFactor;
|
|
|
|
ctx->mParams.SpeedOfSound = props->SpeedOfSound * props->DopplerVelocity;
|
|
|
|
|
|
|
|
ctx->mParams.SourceDistanceModel = props->SourceDistanceModel;
|
|
|
|
ctx->mParams.mDistanceModel = props->mDistanceModel;
|
|
|
|
|
|
|
|
AtomicReplaceHead(ctx->mFreeContextProps, props);
|
2017-09-27 08:55:42 -07:00
|
|
|
return true;
|
2012-10-12 07:38:29 -07:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
bool CalcEffectSlotParams(EffectSlot *slot, EffectSlot **sorted_slots, ContextBase *context)
|
2016-05-12 18:26:33 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
EffectSlotProps *props{slot->Update.exchange(nullptr, std::memory_order_acq_rel)};
|
2019-08-05 11:37:05 -07:00
|
|
|
if(!props) return false;
|
2016-05-12 18:26:33 -07:00
|
|
|
|
2020-01-18 18:53:58 -08:00
|
|
|
/* If the effect slot target changed, clear the first sorted entry to force
|
|
|
|
* a re-sort.
|
|
|
|
*/
|
2020-11-07 08:36:49 -08:00
|
|
|
if(slot->Target != props->Target)
|
2020-01-18 18:53:58 -08:00
|
|
|
*sorted_slots = nullptr;
|
2020-11-07 08:36:49 -08:00
|
|
|
slot->Gain = props->Gain;
|
|
|
|
slot->AuxSendAuto = props->AuxSendAuto;
|
|
|
|
slot->Target = props->Target;
|
|
|
|
slot->EffectType = props->Type;
|
|
|
|
slot->mEffectProps = props->Props;
|
2020-12-26 09:21:30 -08:00
|
|
|
if(props->Type == EffectSlotType::Reverb || props->Type == EffectSlotType::EAXReverb)
|
2019-08-05 11:37:05 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
slot->RoomRolloff = props->Props.Reverb.RoomRolloffFactor;
|
|
|
|
slot->DecayTime = props->Props.Reverb.DecayTime;
|
|
|
|
slot->DecayLFRatio = props->Props.Reverb.DecayLFRatio;
|
|
|
|
slot->DecayHFRatio = props->Props.Reverb.DecayHFRatio;
|
|
|
|
slot->DecayHFLimit = props->Props.Reverb.DecayHFLimit;
|
|
|
|
slot->AirAbsorptionGainHF = props->Props.Reverb.AirAbsorptionGainHF;
|
2019-08-05 11:37:05 -07:00
|
|
|
}
|
2018-12-12 01:09:04 -08:00
|
|
|
else
|
2016-05-12 18:26:33 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
slot->RoomRolloff = 0.0f;
|
|
|
|
slot->DecayTime = 0.0f;
|
|
|
|
slot->DecayLFRatio = 0.0f;
|
|
|
|
slot->DecayHFRatio = 0.0f;
|
|
|
|
slot->DecayHFLimit = false;
|
|
|
|
slot->AirAbsorptionGainHF = 1.0f;
|
2019-08-05 11:37:05 -07:00
|
|
|
}
|
|
|
|
|
2020-08-22 22:45:26 -07:00
|
|
|
EffectState *state{props->State.release()};
|
2020-11-07 08:36:49 -08:00
|
|
|
EffectState *oldstate{slot->mEffectState};
|
|
|
|
slot->mEffectState = state;
|
2019-08-05 11:37:05 -07:00
|
|
|
|
|
|
|
/* Only release the old state if it won't get deleted, since we can't be
|
|
|
|
* deleting/freeing anything in the mixer.
|
|
|
|
*/
|
|
|
|
if(!oldstate->releaseIfNoDelete())
|
|
|
|
{
|
|
|
|
/* Otherwise, if it would be deleted send it off with a release event. */
|
|
|
|
RingBuffer *ring{context->mAsyncEvents.get()};
|
|
|
|
auto evt_vec = ring->getWriteVector();
|
|
|
|
if LIKELY(evt_vec.first.len > 0)
|
2017-09-27 11:13:18 -07:00
|
|
|
{
|
2021-10-08 11:05:36 -07:00
|
|
|
AsyncEvent *evt{al::construct_at(reinterpret_cast<AsyncEvent*>(evt_vec.first.buf),
|
2021-10-10 05:07:31 -07:00
|
|
|
AsyncEvent::ReleaseEffectState)};
|
2019-08-05 11:37:05 -07:00
|
|
|
evt->u.mEffectState = oldstate;
|
|
|
|
ring->writeAdvance(1);
|
2017-09-27 11:13:18 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-05 11:37:05 -07:00
|
|
|
/* If writing the event failed, the queue was probably full. Store
|
|
|
|
* the old state in the property object where it can eventually be
|
|
|
|
* cleaned up sometime later (not ideal, but better than blocking
|
|
|
|
* or leaking).
|
2018-09-21 02:37:37 -07:00
|
|
|
*/
|
2020-08-22 22:45:26 -07:00
|
|
|
props->State.reset(oldstate);
|
2018-09-21 02:37:37 -07:00
|
|
|
}
|
2017-09-27 11:13:18 -07:00
|
|
|
}
|
2016-05-13 18:28:01 -07:00
|
|
|
|
2019-08-05 11:37:05 -07:00
|
|
|
AtomicReplaceHead(context->mFreeEffectslotProps, props);
|
|
|
|
|
2018-12-24 13:29:36 -08:00
|
|
|
EffectTarget output;
|
2020-11-07 08:36:49 -08:00
|
|
|
if(EffectSlot *target{slot->Target})
|
2019-03-22 19:25:55 -07:00
|
|
|
output = EffectTarget{&target->Wet, nullptr};
|
2018-12-24 13:29:36 -08:00
|
|
|
else
|
|
|
|
{
|
2021-04-24 08:28:13 -07:00
|
|
|
DeviceBase *device{context->mDevice};
|
2019-02-22 22:03:04 -08:00
|
|
|
output = EffectTarget{&device->Dry, &device->RealOut};
|
2018-12-24 13:29:36 -08:00
|
|
|
}
|
2020-11-07 08:36:49 -08:00
|
|
|
state->update(context, slot, &slot->mEffectProps, output);
|
2017-09-27 08:55:42 -07:00
|
|
|
return true;
|
Provide asynchronous property updates for sources
This necessitates a change in how source updates are handled. Rather than just
being able to update sources when a dependent object state is changed (e.g. a
listener gain change), now all source updates must be proactively provided.
Consequently, apps that do not utilize any deferring (AL_SOFT_defer_updates or
alcSuspendContext/alcProcessContext) may utilize more CPU since it'll be
filling out more update containers for the mixer thread to use.
The upside is that there's less blocking between the app's calling thread and
the mixer thread, particularly for vectors and other multi-value properties
(filters and sends). Deferring behavior when used is also improved, since
updates that shouldn't be applied yet are simply not provided. And when they
are provided, the mixer doesn't have to ignore them, meaning the actual
deferring of a context doesn't have to synchrnously force an update -- the
process call will send any pending updates, which the mixer will apply even if
another deferral occurs before the mixer runs, because it'll still be there
waiting on the next mixer invocation.
There is one slight bug introduced by this commit. When a listener change is
made, or changes to multiple sources while updates are being deferred, it is
possible for the mixer to run while the sources are prepping their updates,
causing some of the source updates to be seen before the other. This will be
fixed in short order.
2016-05-14 23:43:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-02 20:23:19 -07:00
|
|
|
/* Scales the given azimuth toward the side (+/- pi/2 radians) for positions in
|
|
|
|
* front.
|
|
|
|
*/
|
|
|
|
inline float ScaleAzimuthFront(float azimuth, float scale)
|
|
|
|
{
|
2020-04-08 06:17:04 -07:00
|
|
|
const float abs_azi{std::fabs(azimuth)};
|
2022-01-27 02:59:07 -08:00
|
|
|
if(!(abs_azi >= al::numbers::pi_v<float>*0.5f))
|
|
|
|
return std::copysign(minf(abs_azi*scale, al::numbers::pi_v<float>*0.5f), azimuth);
|
2019-04-02 20:23:19 -07:00
|
|
|
return azimuth;
|
|
|
|
}
|
2011-06-29 23:18:49 -07:00
|
|
|
|
2020-01-31 11:11:06 -08:00
|
|
|
/* Wraps the given value in radians to stay between [-pi,+pi] */
|
|
|
|
inline float WrapRadians(float r)
|
|
|
|
{
|
2022-02-23 01:29:28 -08:00
|
|
|
static constexpr float Pi{al::numbers::pi_v<float>};
|
|
|
|
static constexpr float Pi2{Pi*2.0f};
|
2020-01-31 11:11:06 -08:00
|
|
|
if(r > Pi) return std::fmod(Pi+r, Pi2) - Pi;
|
|
|
|
if(r < -Pi) return Pi - std::fmod(Pi-r, Pi2);
|
|
|
|
return r;
|
|
|
|
}
|
2019-12-01 17:57:31 -08:00
|
|
|
|
|
|
|
/* Begin ambisonic rotation helpers.
|
|
|
|
*
|
|
|
|
* Rotating first-order B-Format just needs a straight-forward X/Y/Z rotation
|
|
|
|
* matrix. Higher orders, however, are more complicated. The method implemented
|
|
|
|
* here is a recursive algorithm (the rotation for first-order is used to help
|
|
|
|
* generate the second-order rotation, which helps generate the third-order
|
|
|
|
* rotation, etc).
|
|
|
|
*
|
|
|
|
* Adapted from
|
|
|
|
* <https://github.com/polarch/Spherical-Harmonic-Transform/blob/master/getSHrotMtx.m>,
|
|
|
|
* provided under the BSD 3-Clause license.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2015, Archontis Politis
|
|
|
|
* Copyright (c) 2019, Christopher Robinson
|
|
|
|
*
|
|
|
|
* The u, v, and w coefficients used for generating higher-order rotations are
|
|
|
|
* precomputed since they're constant. The second-order coefficients are
|
|
|
|
* followed by the third-order coefficients, etc.
|
|
|
|
*/
|
|
|
|
struct RotatorCoeffs {
|
|
|
|
float u, v, w;
|
|
|
|
|
|
|
|
template<size_t N0, size_t N1>
|
|
|
|
static std::array<RotatorCoeffs,N0+N1> ConcatArrays(const std::array<RotatorCoeffs,N0> &lhs,
|
|
|
|
const std::array<RotatorCoeffs,N1> &rhs)
|
|
|
|
{
|
|
|
|
std::array<RotatorCoeffs,N0+N1> ret;
|
|
|
|
auto iter = std::copy(lhs.cbegin(), lhs.cend(), ret.begin());
|
|
|
|
std::copy(rhs.cbegin(), rhs.cend(), iter);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int l, int num_elems=l*2+1>
|
|
|
|
static std::array<RotatorCoeffs,num_elems*num_elems> GenCoeffs()
|
|
|
|
{
|
|
|
|
std::array<RotatorCoeffs,num_elems*num_elems> ret{};
|
|
|
|
auto coeffs = ret.begin();
|
|
|
|
|
|
|
|
for(int m{-l};m <= l;++m)
|
|
|
|
{
|
|
|
|
for(int n{-l};n <= l;++n)
|
|
|
|
{
|
|
|
|
// compute u,v,w terms of Eq.8.1 (Table I)
|
|
|
|
const bool d{m == 0}; // the delta function d_m0
|
|
|
|
const float denom{static_cast<float>((std::abs(n) == l) ?
|
|
|
|
(2*l) * (2*l - 1) : (l*l - n*n))};
|
|
|
|
|
|
|
|
const int abs_m{std::abs(m)};
|
|
|
|
coeffs->u = std::sqrt(static_cast<float>(l*l - m*m)/denom);
|
|
|
|
coeffs->v = std::sqrt(static_cast<float>(l+abs_m-1) * static_cast<float>(l+abs_m) /
|
|
|
|
denom) * (1.0f+d) * (1.0f - 2.0f*d) * 0.5f;
|
|
|
|
coeffs->w = std::sqrt(static_cast<float>(l-abs_m-1) * static_cast<float>(l-abs_m) /
|
|
|
|
denom) * (1.0f-d) * -0.5f;
|
|
|
|
++coeffs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const auto RotatorCoeffArray = RotatorCoeffs::ConcatArrays(RotatorCoeffs::GenCoeffs<2>(),
|
|
|
|
RotatorCoeffs::GenCoeffs<3>());
|
|
|
|
|
|
|
|
/**
|
2019-12-23 23:10:36 -08:00
|
|
|
* Given the matrix, pre-filled with the (zeroth- and) first-order rotation
|
2019-12-01 17:57:31 -08:00
|
|
|
* coefficients, this fills in the coefficients for the higher orders up to and
|
2019-12-23 23:10:36 -08:00
|
|
|
* including the given order. The matrix is in ACN layout.
|
2019-12-01 17:57:31 -08:00
|
|
|
*/
|
2022-04-24 22:53:55 -07:00
|
|
|
void AmbiRotator(AmbiRotateMatrix &matrix, const int order)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
|
|
|
/* Don't do anything for < 2nd order. */
|
2019-12-23 23:10:36 -08:00
|
|
|
if(order < 2) return;
|
2019-12-01 17:57:31 -08:00
|
|
|
|
|
|
|
auto P = [](const int i, const int l, const int a, const int n, const size_t last_band,
|
2022-04-24 22:53:55 -07:00
|
|
|
const AmbiRotateMatrix &R)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
2020-12-15 20:48:21 -08:00
|
|
|
const float ri1{ R[static_cast<uint>(i+2)][ 1+2]};
|
|
|
|
const float rim1{R[static_cast<uint>(i+2)][-1+2]};
|
|
|
|
const float ri0{ R[static_cast<uint>(i+2)][ 0+2]};
|
2019-12-01 17:57:31 -08:00
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
auto vec = R[static_cast<uint>(a+l-1) + last_band].cbegin() + last_band;
|
2019-12-01 17:57:31 -08:00
|
|
|
if(n == -l)
|
2020-12-15 20:48:21 -08:00
|
|
|
return ri1*vec[0] + rim1*vec[static_cast<uint>(l-1)*size_t{2}];
|
2019-12-01 17:57:31 -08:00
|
|
|
if(n == l)
|
2020-12-15 20:48:21 -08:00
|
|
|
return ri1*vec[static_cast<uint>(l-1)*size_t{2}] - rim1*vec[0];
|
|
|
|
return ri0*vec[static_cast<uint>(n+l-1)];
|
2019-12-01 17:57:31 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
auto U = [P](const int l, const int m, const int n, const size_t last_band,
|
2022-04-24 22:53:55 -07:00
|
|
|
const AmbiRotateMatrix &R)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
|
|
|
return P(0, l, m, n, last_band, R);
|
|
|
|
};
|
|
|
|
auto V = [P](const int l, const int m, const int n, const size_t last_band,
|
2022-04-24 22:53:55 -07:00
|
|
|
const AmbiRotateMatrix &R)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
2022-01-27 02:59:07 -08:00
|
|
|
using namespace al::numbers;
|
2019-12-01 17:57:31 -08:00
|
|
|
if(m > 0)
|
|
|
|
{
|
|
|
|
const bool d{m == 1};
|
|
|
|
const float p0{P( 1, l, m-1, n, last_band, R)};
|
|
|
|
const float p1{P(-1, l, -m+1, n, last_band, R)};
|
2022-01-27 02:59:07 -08:00
|
|
|
return d ? p0*sqrt2_v<float> : (p0 - p1);
|
2019-12-01 17:57:31 -08:00
|
|
|
}
|
|
|
|
const bool d{m == -1};
|
|
|
|
const float p0{P( 1, l, m+1, n, last_band, R)};
|
|
|
|
const float p1{P(-1, l, -m-1, n, last_band, R)};
|
2022-01-27 02:59:07 -08:00
|
|
|
return d ? p1*sqrt2_v<float> : (p0 + p1);
|
2019-12-01 17:57:31 -08:00
|
|
|
};
|
|
|
|
auto W = [P](const int l, const int m, const int n, const size_t last_band,
|
2022-04-24 22:53:55 -07:00
|
|
|
const AmbiRotateMatrix &R)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
|
|
|
assert(m != 0);
|
|
|
|
if(m > 0)
|
|
|
|
{
|
|
|
|
const float p0{P( 1, l, m+1, n, last_band, R)};
|
|
|
|
const float p1{P(-1, l, -m-1, n, last_band, R)};
|
|
|
|
return p0 + p1;
|
|
|
|
}
|
|
|
|
const float p0{P( 1, l, m-1, n, last_band, R)};
|
|
|
|
const float p1{P(-1, l, -m+1, n, last_band, R)};
|
|
|
|
return p0 - p1;
|
|
|
|
};
|
|
|
|
|
|
|
|
// compute rotation matrix of each subsequent band recursively
|
|
|
|
auto coeffs = RotatorCoeffArray.cbegin();
|
|
|
|
size_t band_idx{4}, last_band{1};
|
2019-12-23 23:10:36 -08:00
|
|
|
for(int l{2};l <= order;++l)
|
2019-12-01 17:57:31 -08:00
|
|
|
{
|
|
|
|
size_t y{band_idx};
|
|
|
|
for(int m{-l};m <= l;++m,++y)
|
|
|
|
{
|
|
|
|
size_t x{band_idx};
|
|
|
|
for(int n{-l};n <= l;++n,++x)
|
|
|
|
{
|
|
|
|
float r{0.0f};
|
|
|
|
|
|
|
|
// computes Eq.8.1
|
|
|
|
const float u{coeffs->u};
|
2019-12-23 23:10:36 -08:00
|
|
|
if(u != 0.0f) r += u * U(l, m, n, last_band, matrix);
|
2019-12-01 17:57:31 -08:00
|
|
|
const float v{coeffs->v};
|
2019-12-23 23:10:36 -08:00
|
|
|
if(v != 0.0f) r += v * V(l, m, n, last_band, matrix);
|
2019-12-01 17:57:31 -08:00
|
|
|
const float w{coeffs->w};
|
2019-12-23 23:10:36 -08:00
|
|
|
if(w != 0.0f) r += w * W(l, m, n, last_band, matrix);
|
2019-12-01 17:57:31 -08:00
|
|
|
|
2019-12-23 23:10:36 -08:00
|
|
|
matrix[y][x] = r;
|
2019-12-01 17:57:31 -08:00
|
|
|
++coeffs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
last_band = band_idx;
|
2020-12-15 20:48:21 -08:00
|
|
|
band_idx += static_cast<uint>(l)*size_t{2} + 1;
|
2019-12-01 17:57:31 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* End ambisonic rotation helpers. */
|
|
|
|
|
|
|
|
|
2022-01-27 04:04:41 -08:00
|
|
|
constexpr float Deg2Rad(float x) noexcept
|
|
|
|
{ return static_cast<float>(al::numbers::pi / 180.0 * x); }
|
|
|
|
|
2019-12-21 00:48:58 -08:00
|
|
|
struct GainTriplet { float Base, HF, LF; };
|
|
|
|
|
2020-04-08 06:17:04 -07:00
|
|
|
void CalcPanningAndFilters(Voice *voice, const float xpos, const float ypos, const float zpos,
|
|
|
|
const float Distance, const float Spread, const GainTriplet &DryGain,
|
2020-11-07 08:36:49 -08:00
|
|
|
const al::span<const GainTriplet,MAX_SENDS> WetGain, EffectSlot *(&SendSlots)[MAX_SENDS],
|
2021-04-24 08:28:13 -07:00
|
|
|
const VoiceProps *props, const ContextParams &Context, const DeviceBase *Device)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2022-01-27 04:04:41 -08:00
|
|
|
static constexpr ChanMap MonoMap[1]{
|
2019-04-02 20:23:19 -07:00
|
|
|
{ FrontCenter, 0.0f, 0.0f }
|
|
|
|
}, RearMap[2]{
|
|
|
|
{ BackLeft, Deg2Rad(-150.0f), Deg2Rad(0.0f) },
|
|
|
|
{ BackRight, Deg2Rad( 150.0f), Deg2Rad(0.0f) }
|
|
|
|
}, QuadMap[4]{
|
|
|
|
{ FrontLeft, Deg2Rad( -45.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontRight, Deg2Rad( 45.0f), Deg2Rad(0.0f) },
|
|
|
|
{ BackLeft, Deg2Rad(-135.0f), Deg2Rad(0.0f) },
|
|
|
|
{ BackRight, Deg2Rad( 135.0f), Deg2Rad(0.0f) }
|
|
|
|
}, X51Map[6]{
|
|
|
|
{ FrontLeft, Deg2Rad( -30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontRight, Deg2Rad( 30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontCenter, Deg2Rad( 0.0f), Deg2Rad(0.0f) },
|
|
|
|
{ LFE, 0.0f, 0.0f },
|
|
|
|
{ SideLeft, Deg2Rad(-110.0f), Deg2Rad(0.0f) },
|
|
|
|
{ SideRight, Deg2Rad( 110.0f), Deg2Rad(0.0f) }
|
|
|
|
}, X61Map[7]{
|
|
|
|
{ FrontLeft, Deg2Rad(-30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontRight, Deg2Rad( 30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontCenter, Deg2Rad( 0.0f), Deg2Rad(0.0f) },
|
|
|
|
{ LFE, 0.0f, 0.0f },
|
|
|
|
{ BackCenter, Deg2Rad(180.0f), Deg2Rad(0.0f) },
|
|
|
|
{ SideLeft, Deg2Rad(-90.0f), Deg2Rad(0.0f) },
|
|
|
|
{ SideRight, Deg2Rad( 90.0f), Deg2Rad(0.0f) }
|
|
|
|
}, X71Map[8]{
|
|
|
|
{ FrontLeft, Deg2Rad( -30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontRight, Deg2Rad( 30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontCenter, Deg2Rad( 0.0f), Deg2Rad(0.0f) },
|
|
|
|
{ LFE, 0.0f, 0.0f },
|
|
|
|
{ BackLeft, Deg2Rad(-150.0f), Deg2Rad(0.0f) },
|
|
|
|
{ BackRight, Deg2Rad( 150.0f), Deg2Rad(0.0f) },
|
|
|
|
{ SideLeft, Deg2Rad( -90.0f), Deg2Rad(0.0f) },
|
|
|
|
{ SideRight, Deg2Rad( 90.0f), Deg2Rad(0.0f) }
|
|
|
|
};
|
|
|
|
|
2018-12-12 01:09:04 -08:00
|
|
|
ChanMap StereoMap[2]{
|
2018-12-22 16:01:14 -08:00
|
|
|
{ FrontLeft, Deg2Rad(-30.0f), Deg2Rad(0.0f) },
|
|
|
|
{ FrontRight, Deg2Rad( 30.0f), Deg2Rad(0.0f) }
|
2016-03-25 14:40:44 -07:00
|
|
|
};
|
2010-08-05 01:07:20 -07:00
|
|
|
|
2020-04-08 06:17:04 -07:00
|
|
|
const auto Frequency = static_cast<float>(Device->Frequency);
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint NumSends{Device->NumAuxSends};
|
2019-02-24 19:07:21 -08:00
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
const size_t num_channels{voice->mChans.size()};
|
2020-02-08 18:10:59 -08:00
|
|
|
ASSUME(num_channels > 0);
|
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
for(auto &chandata : voice->mChans)
|
2020-02-08 18:10:59 -08:00
|
|
|
{
|
|
|
|
chandata.mDryParams.Hrtf.Target = HrtfFilter{};
|
|
|
|
chandata.mDryParams.Gains.Target.fill(0.0f);
|
|
|
|
std::for_each(chandata.mWetParams.begin(), chandata.mWetParams.begin()+NumSends,
|
|
|
|
[](SendParams ¶ms) -> void { params.Gains.Target.fill(0.0f); });
|
2020-03-25 21:06:24 -07:00
|
|
|
}
|
2020-02-08 18:10:59 -08:00
|
|
|
|
|
|
|
DirectMode DirectChannels{props->DirectChannels};
|
2018-12-12 01:09:04 -08:00
|
|
|
const ChanMap *chans{nullptr};
|
2019-03-10 16:29:06 -07:00
|
|
|
switch(voice->mFmtChannels)
|
2010-12-09 16:37:23 -08:00
|
|
|
{
|
|
|
|
case FmtMono:
|
2011-12-20 01:17:11 -08:00
|
|
|
chans = MonoMap;
|
2017-05-04 04:35:53 -07:00
|
|
|
/* Mono buffers are never played direct. */
|
2019-12-28 11:33:19 -08:00
|
|
|
DirectChannels = DirectMode::Off;
|
2010-12-09 16:37:23 -08:00
|
|
|
break;
|
2011-05-15 00:18:28 -07:00
|
|
|
|
2012-04-29 04:44:53 -07:00
|
|
|
case FmtStereo:
|
2019-12-28 11:33:19 -08:00
|
|
|
if(DirectChannels == DirectMode::Off)
|
2019-12-17 21:49:58 -08:00
|
|
|
{
|
2020-01-31 11:11:06 -08:00
|
|
|
/* Convert counter-clockwise to clock-wise, and wrap between
|
|
|
|
* [-pi,+pi].
|
|
|
|
*/
|
|
|
|
StereoMap[0].angle = WrapRadians(-props->StereoPan[0]);
|
|
|
|
StereoMap[1].angle = WrapRadians(-props->StereoPan[1]);
|
2019-12-17 21:49:58 -08:00
|
|
|
}
|
2016-02-26 16:09:06 -08:00
|
|
|
chans = StereoMap;
|
2010-12-09 16:37:23 -08:00
|
|
|
break;
|
|
|
|
|
2021-06-24 08:21:25 -07:00
|
|
|
case FmtRear: chans = RearMap; break;
|
|
|
|
case FmtQuad: chans = QuadMap; break;
|
|
|
|
case FmtX51: chans = X51Map; break;
|
|
|
|
case FmtX61: chans = X61Map; break;
|
|
|
|
case FmtX71: chans = X71Map; break;
|
2014-10-31 17:18:45 -07:00
|
|
|
|
|
|
|
case FmtBFormat2D:
|
|
|
|
case FmtBFormat3D:
|
2021-03-31 05:37:56 -07:00
|
|
|
case FmtUHJ2:
|
2021-03-31 10:03:31 -07:00
|
|
|
case FmtUHJ3:
|
2021-03-31 20:46:03 -07:00
|
|
|
case FmtUHJ4:
|
2021-12-11 17:50:24 -08:00
|
|
|
case FmtSuperStereo:
|
2021-03-31 05:37:56 -07:00
|
|
|
DirectChannels = DirectMode::Off;
|
|
|
|
break;
|
2011-05-15 02:12:42 -07:00
|
|
|
}
|
2018-03-25 18:02:07 -07:00
|
|
|
|
2021-12-23 13:43:10 -08:00
|
|
|
voice->mFlags.reset(VoiceHasHrtf).reset(VoiceHasNfc);
|
2021-12-15 02:01:22 -08:00
|
|
|
if(auto *decoder{voice->mDecoder.get()})
|
2022-05-17 02:32:10 -07:00
|
|
|
decoder->mWidthControl = minf(props->EnhWidth, 0.7f);
|
2021-12-15 02:01:22 -08:00
|
|
|
|
2021-12-12 15:53:53 -08:00
|
|
|
if(IsAmbisonic(voice->mFmtChannels))
|
2014-10-31 17:18:45 -07:00
|
|
|
{
|
2021-12-12 15:53:53 -08:00
|
|
|
/* Special handling for B-Format and UHJ sources. */
|
2014-10-31 22:43:13 -07:00
|
|
|
|
2021-12-11 17:50:24 -08:00
|
|
|
if(Device->AvgSpeakerDist > 0.0f && voice->mFmtChannels != FmtUHJ2
|
|
|
|
&& voice->mFmtChannels != FmtSuperStereo)
|
2017-03-10 04:35:32 -08:00
|
|
|
{
|
2020-08-27 08:44:29 -07:00
|
|
|
if(!(Distance > std::numeric_limits<float>::epsilon()))
|
|
|
|
{
|
|
|
|
/* NOTE: The NFCtrlFilters were created with a w0 of 0, which
|
|
|
|
* is what we want for FOA input. The first channel may have
|
|
|
|
* been previously re-adjusted if panned, so reset it.
|
|
|
|
*/
|
|
|
|
voice->mChans[0].mDryParams.NFCtrlFilter.adjust(0.0f);
|
|
|
|
}
|
|
|
|
else
|
2016-01-28 00:02:46 -08:00
|
|
|
{
|
2018-12-22 10:00:06 -08:00
|
|
|
/* Clamp the distance for really close sources, to prevent
|
|
|
|
* excessive bass.
|
2017-05-04 11:09:45 -07:00
|
|
|
*/
|
2020-04-08 06:17:04 -07:00
|
|
|
const float mdist{maxf(Distance, Device->AvgSpeakerDist/4.0f)};
|
2020-10-21 16:39:21 -07:00
|
|
|
const float w0{SpeedOfSoundMetersPerSec / (mdist * Frequency)};
|
2017-05-04 11:09:45 -07:00
|
|
|
|
|
|
|
/* Only need to adjust the first channel of a B-Format source. */
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[0].mDryParams.NFCtrlFilter.adjust(w0);
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
|
2021-12-23 13:43:10 -08:00
|
|
|
voice->mFlags.set(VoiceHasNfc);
|
2020-08-27 08:44:29 -07:00
|
|
|
}
|
2020-04-21 23:58:53 -07:00
|
|
|
|
2020-08-27 08:44:29 -07:00
|
|
|
/* Panning a B-Format sound toward some direction is easy. Just pan the
|
|
|
|
* first (W) channel as a normal mono sound. The angular spread is used
|
|
|
|
* as a directional scalar to blend between full coverage and full
|
|
|
|
* panning.
|
|
|
|
*/
|
|
|
|
const float coverage{!(Distance > std::numeric_limits<float>::epsilon()) ? 1.0f :
|
2022-01-27 02:59:07 -08:00
|
|
|
(al::numbers::inv_pi_v<float>/2.0f * Spread)};
|
2019-03-26 11:04:52 -07:00
|
|
|
|
2020-08-27 08:44:29 -07:00
|
|
|
auto calc_coeffs = [xpos,ypos,zpos](RenderMode mode)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2020-09-01 05:46:19 -07:00
|
|
|
if(mode != RenderMode::Pairwise)
|
2020-08-27 08:44:29 -07:00
|
|
|
return CalcDirectionCoeffs({xpos, ypos, zpos}, 0.0f);
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-08-27 08:44:29 -07:00
|
|
|
/* Clamp Y, in case rounding errors caused it to end up outside
|
|
|
|
* of -1...+1.
|
|
|
|
*/
|
|
|
|
const float ev{std::asin(clampf(ypos, -1.0f, 1.0f))};
|
|
|
|
/* Negate Z for right-handed coords with -Z in front. */
|
|
|
|
const float az{std::atan2(xpos, -zpos)};
|
|
|
|
|
|
|
|
/* A scalar of 1.5 for plain stereo results in +/-60 degrees
|
|
|
|
* being moved to +/-90 degrees for direct right and left
|
|
|
|
* speaker responses.
|
|
|
|
*/
|
|
|
|
return CalcAngleCoeffs(ScaleAzimuthFront(az, 1.5f), ev, 0.0f);
|
|
|
|
};
|
|
|
|
auto coeffs = calc_coeffs(Device->mRenderMode);
|
|
|
|
std::transform(coeffs.begin()+1, coeffs.end(), coeffs.begin()+1,
|
|
|
|
std::bind(std::multiplies<float>{}, _1, 1.0f-coverage));
|
|
|
|
|
|
|
|
/* NOTE: W needs to be scaled according to channel scaling. */
|
2021-01-21 02:05:55 -08:00
|
|
|
auto&& scales = GetAmbiScales(voice->mAmbiScaling);
|
2020-08-27 08:44:29 -07:00
|
|
|
ComputePanGains(&Device->Dry, coeffs.data(), DryGain.Base*scales[0],
|
|
|
|
voice->mChans[0].mDryParams.Gains.Target);
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2020-08-27 08:44:29 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2020-08-27 08:44:29 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base*scales[0],
|
|
|
|
voice->mChans[0].mWetParams[i].Gains.Target);
|
|
|
|
}
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-08-27 08:44:29 -07:00
|
|
|
if(coverage > 0.0f)
|
|
|
|
{
|
2018-12-12 01:09:04 -08:00
|
|
|
/* Local B-Format sources have their XYZ channels rotated according
|
|
|
|
* to the orientation.
|
|
|
|
*/
|
2017-05-04 11:09:45 -07:00
|
|
|
/* AT then UP */
|
2018-12-26 22:27:34 -08:00
|
|
|
alu::Vector N{props->OrientAt[0], props->OrientAt[1], props->OrientAt[2], 0.0f};
|
2018-12-12 04:22:11 -08:00
|
|
|
N.normalize();
|
2018-12-26 22:27:34 -08:00
|
|
|
alu::Vector V{props->OrientUp[0], props->OrientUp[1], props->OrientUp[2], 0.0f};
|
2018-12-12 04:22:11 -08:00
|
|
|
V.normalize();
|
2017-05-04 11:09:45 -07:00
|
|
|
if(!props->HeadRelative)
|
|
|
|
{
|
2020-12-15 17:25:45 -08:00
|
|
|
N = Context.Matrix * N;
|
|
|
|
V = Context.Matrix * V;
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
/* Build and normalize right-vector */
|
2020-11-13 23:33:40 -08:00
|
|
|
alu::Vector U{N.cross_product(V)};
|
2018-12-12 04:22:11 -08:00
|
|
|
U.normalize();
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2019-12-01 17:57:31 -08:00
|
|
|
/* Build a rotation matrix. Manually fill the zeroth- and first-
|
|
|
|
* order elements, then construct the rotation for the higher
|
|
|
|
* orders.
|
2018-03-25 08:24:53 -07:00
|
|
|
*/
|
2022-04-24 22:53:55 -07:00
|
|
|
AmbiRotateMatrix shrot{};
|
2019-12-01 17:57:31 -08:00
|
|
|
shrot[0][0] = 1.0f;
|
|
|
|
shrot[1][1] = U[0]; shrot[1][2] = -V[0]; shrot[1][3] = -N[0];
|
|
|
|
shrot[2][1] = -U[1]; shrot[2][2] = V[1]; shrot[2][3] = N[1];
|
|
|
|
shrot[3][1] = U[2]; shrot[3][2] = -V[2]; shrot[3][3] = -N[2];
|
|
|
|
AmbiRotator(shrot, static_cast<int>(minu(voice->mAmbiOrder, Device->mAmbiOrder)));
|
|
|
|
|
2019-12-17 22:25:00 -08:00
|
|
|
/* Convert the rotation matrix for input ordering and scaling, and
|
|
|
|
* whether input is 2D or 3D.
|
2019-12-01 17:57:31 -08:00
|
|
|
*/
|
2021-12-12 15:53:53 -08:00
|
|
|
const uint8_t *index_map{Is2DAmbisonic(voice->mFmtChannels) ?
|
2019-12-02 12:50:18 -08:00
|
|
|
GetAmbi2DLayout(voice->mAmbiLayout).data() :
|
|
|
|
GetAmbiLayout(voice->mAmbiLayout).data()};
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-12-04 13:53:56 -08:00
|
|
|
static const uint8_t ChansPerOrder[MaxAmbiOrder+1]{1, 3, 5, 7,};
|
|
|
|
static const uint8_t OrderOffset[MaxAmbiOrder+1]{0, 1, 4, 9,};
|
2020-08-27 08:44:29 -07:00
|
|
|
for(size_t c{1};c < num_channels;c++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2019-12-01 17:57:31 -08:00
|
|
|
const size_t acn{index_map[c]};
|
2021-01-21 02:05:55 -08:00
|
|
|
const size_t order{AmbiIndex::OrderFromChannel()[acn]};
|
2019-12-01 17:57:31 -08:00
|
|
|
const size_t tocopy{ChansPerOrder[order]};
|
|
|
|
const size_t offset{OrderOffset[order]};
|
2020-08-27 08:44:29 -07:00
|
|
|
const float scale{scales[acn] * coverage};
|
2019-12-01 17:57:31 -08:00
|
|
|
auto in = shrot.cbegin() + offset;
|
|
|
|
|
2020-12-04 13:53:56 -08:00
|
|
|
coeffs = std::array<float,MaxAmbiChannels>{};
|
2019-12-01 17:57:31 -08:00
|
|
|
for(size_t x{0};x < tocopy;++x)
|
|
|
|
coeffs[offset+x] = in[x][acn] * scale;
|
|
|
|
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Device->Dry, coeffs.data(), DryGain.Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target);
|
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2019-06-03 22:24:26 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
|
|
|
}
|
2016-01-28 00:02:46 -08:00
|
|
|
}
|
2015-10-23 20:16:11 -07:00
|
|
|
}
|
2014-10-31 17:18:45 -07:00
|
|
|
}
|
2021-12-21 08:18:07 -08:00
|
|
|
else if(DirectChannels != DirectMode::Off && !Device->RealOut.RemixMap.empty())
|
2011-06-29 23:18:49 -07:00
|
|
|
{
|
2017-05-07 18:28:43 -07:00
|
|
|
/* Direct source channels always play local. Skip the virtual channels
|
|
|
|
* and write inputs to the matching real outputs.
|
2017-05-04 11:09:45 -07:00
|
|
|
*/
|
2019-07-03 23:26:33 -07:00
|
|
|
voice->mDirect.Buffer = Device->RealOut.Buffer;
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2020-12-15 20:48:21 -08:00
|
|
|
uint idx{GetChannelIdxByName(Device->RealOut, chans[c].channel)};
|
2019-09-12 04:17:21 -07:00
|
|
|
if(idx != INVALID_CHANNEL_INDEX)
|
2019-12-21 00:48:58 -08:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target[idx] = DryGain.Base;
|
2019-12-28 11:33:19 -08:00
|
|
|
else if(DirectChannels == DirectMode::RemixMismatch)
|
2019-12-19 04:38:34 -08:00
|
|
|
{
|
2019-12-21 00:48:58 -08:00
|
|
|
auto match_channel = [chans,c](const InputRemixMap &map) noexcept -> bool
|
2019-12-19 04:38:34 -08:00
|
|
|
{ return chans[c].channel == map.channel; };
|
|
|
|
auto remap = std::find_if(Device->RealOut.RemixMap.cbegin(),
|
|
|
|
Device->RealOut.RemixMap.cend(), match_channel);
|
|
|
|
if(remap != Device->RealOut.RemixMap.cend())
|
2021-12-21 08:18:07 -08:00
|
|
|
{
|
2019-12-19 04:38:34 -08:00
|
|
|
for(const auto &target : remap->targets)
|
|
|
|
{
|
|
|
|
idx = GetChannelIdxByName(Device->RealOut, target.channel);
|
|
|
|
if(idx != INVALID_CHANNEL_INDEX)
|
2019-12-21 00:48:58 -08:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target[idx] = DryGain.Base *
|
|
|
|
target.mix;
|
2019-12-19 04:38:34 -08:00
|
|
|
}
|
2021-12-21 08:18:07 -08:00
|
|
|
}
|
2019-12-19 04:38:34 -08:00
|
|
|
}
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
2016-02-14 01:22:01 -08:00
|
|
|
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Auxiliary sends still use normal channel panning since they mix to
|
|
|
|
* B-Format, which can't channel-match.
|
2017-05-04 04:35:53 -07:00
|
|
|
*/
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2014-11-22 13:10:32 -08:00
|
|
|
{
|
2020-04-21 23:58:53 -07:00
|
|
|
const auto coeffs = CalcAngleCoeffs(chans[c].angle, chans[c].elevation, 0.0f);
|
2016-01-28 00:02:46 -08:00
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2016-01-28 00:02:46 -08:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
2016-01-28 00:02:46 -08:00
|
|
|
}
|
2012-03-12 18:27:25 -07:00
|
|
|
}
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
2020-09-01 05:46:19 -07:00
|
|
|
else if(Device->mRenderMode == RenderMode::Hrtf)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Full HRTF rendering. Skip the virtual channels and render to the
|
|
|
|
* real outputs.
|
2017-05-04 04:35:53 -07:00
|
|
|
*/
|
2019-07-03 23:26:33 -07:00
|
|
|
voice->mDirect.Buffer = Device->RealOut.Buffer;
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2018-12-22 16:01:14 -08:00
|
|
|
if(Distance > std::numeric_limits<float>::epsilon())
|
2014-11-23 10:49:54 -08:00
|
|
|
{
|
2020-04-08 06:17:04 -07:00
|
|
|
const float ev{std::asin(clampf(ypos, -1.0f, 1.0f))};
|
|
|
|
const float az{std::atan2(xpos, -zpos)};
|
2019-03-10 11:33:08 -07:00
|
|
|
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Get the HRIR coefficients and delays just once, for the given
|
|
|
|
* source direction.
|
|
|
|
*/
|
2020-03-01 17:16:09 -08:00
|
|
|
GetHrtfCoeffs(Device->mHrtf.get(), ev, az, Distance, Spread,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[0].mDryParams.Hrtf.Target.Coeffs,
|
|
|
|
voice->mChans[0].mDryParams.Hrtf.Target.Delay);
|
2021-06-24 08:21:25 -07:00
|
|
|
voice->mChans[0].mDryParams.Hrtf.Target.Gain = DryGain.Base;
|
2017-05-04 04:35:53 -07:00
|
|
|
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Remaining channels use the same results as the first. */
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{1};c < num_channels;c++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
|
|
|
/* Skip LFE */
|
2019-06-03 22:24:26 -07:00
|
|
|
if(chans[c].channel == LFE) continue;
|
|
|
|
voice->mChans[c].mDryParams.Hrtf.Target = voice->mChans[0].mDryParams.Hrtf.Target;
|
2016-01-28 00:02:46 -08:00
|
|
|
}
|
|
|
|
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Calculate the directional coefficients once, which apply to all
|
|
|
|
* input channels of the source sends.
|
|
|
|
*/
|
2020-04-21 23:58:53 -07:00
|
|
|
const auto coeffs = CalcDirectionCoeffs({xpos, ypos, zpos}, Spread);
|
2017-05-04 04:35:53 -07:00
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2019-06-03 22:24:26 -07:00
|
|
|
/* Skip LFE */
|
|
|
|
if(chans[c].channel == LFE)
|
|
|
|
continue;
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2019-06-03 22:24:26 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2021-06-24 08:21:25 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
|
|
|
}
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
2015-10-23 20:16:11 -07:00
|
|
|
}
|
2017-05-04 11:09:45 -07:00
|
|
|
else
|
|
|
|
{
|
2017-05-07 18:28:43 -07:00
|
|
|
/* Local sources on HRTF play with each channel panned to its
|
|
|
|
* relative location around the listener, providing "virtual
|
|
|
|
* speaker" responses.
|
|
|
|
*/
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2018-12-12 01:09:04 -08:00
|
|
|
/* Skip LFE */
|
2017-05-04 11:09:45 -07:00
|
|
|
if(chans[c].channel == LFE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Get the HRIR coefficients and delays for this channel
|
|
|
|
* position.
|
|
|
|
*/
|
2020-03-01 17:16:09 -08:00
|
|
|
GetHrtfCoeffs(Device->mHrtf.get(), chans[c].elevation, chans[c].angle,
|
2019-01-28 22:07:03 -08:00
|
|
|
std::numeric_limits<float>::infinity(), Spread,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.Hrtf.Target.Coeffs,
|
|
|
|
voice->mChans[c].mDryParams.Hrtf.Target.Delay);
|
2019-12-21 00:48:58 -08:00
|
|
|
voice->mChans[c].mDryParams.Hrtf.Target.Gain = DryGain.Base;
|
2017-05-04 11:09:45 -07:00
|
|
|
|
|
|
|
/* Normal panning for auxiliary sends. */
|
2020-04-21 23:58:53 -07:00
|
|
|
const auto coeffs = CalcAngleCoeffs(chans[c].angle, chans[c].elevation, Spread);
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-04 04:35:53 -07:00
|
|
|
|
2021-12-23 13:43:10 -08:00
|
|
|
voice->mFlags.set(VoiceHasHrtf);
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Non-HRTF rendering. Use normal panning to the output. */
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2018-12-22 16:01:14 -08:00
|
|
|
if(Distance > std::numeric_limits<float>::epsilon())
|
2011-05-02 02:22:30 -07:00
|
|
|
{
|
2017-05-04 11:09:45 -07:00
|
|
|
/* Calculate NFC filter coefficient if needed. */
|
2017-08-21 00:30:14 -07:00
|
|
|
if(Device->AvgSpeakerDist > 0.0f)
|
2011-04-14 21:03:37 -07:00
|
|
|
{
|
2018-12-22 10:00:06 -08:00
|
|
|
/* Clamp the distance for really close sources, to prevent
|
|
|
|
* excessive bass.
|
2017-05-04 11:09:45 -07:00
|
|
|
*/
|
2020-04-08 06:17:04 -07:00
|
|
|
const float mdist{maxf(Distance, Device->AvgSpeakerDist/4.0f)};
|
2020-10-21 16:39:21 -07:00
|
|
|
const float w0{SpeedOfSoundMetersPerSec / (mdist * Frequency)};
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2018-02-11 21:59:59 -08:00
|
|
|
/* Adjust NFC filters. */
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.NFCtrlFilter.adjust(w0);
|
2018-02-11 21:59:59 -08:00
|
|
|
|
2021-12-23 13:43:10 -08:00
|
|
|
voice->mFlags.set(VoiceHasNfc);
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate the directional coefficients once, which apply to all
|
|
|
|
* input channels.
|
|
|
|
*/
|
2020-04-21 23:58:53 -07:00
|
|
|
auto calc_coeffs = [xpos,ypos,zpos,Spread](RenderMode mode)
|
2019-03-10 11:33:08 -07:00
|
|
|
{
|
2020-09-01 05:46:19 -07:00
|
|
|
if(mode != RenderMode::Pairwise)
|
2020-04-21 23:58:53 -07:00
|
|
|
return CalcDirectionCoeffs({xpos, ypos, zpos}, Spread);
|
2020-04-08 06:17:04 -07:00
|
|
|
const float ev{std::asin(clampf(ypos, -1.0f, 1.0f))};
|
|
|
|
const float az{std::atan2(xpos, -zpos)};
|
2020-04-21 23:58:53 -07:00
|
|
|
return CalcAngleCoeffs(ScaleAzimuthFront(az, 1.5f), ev, Spread);
|
|
|
|
};
|
|
|
|
const auto coeffs = calc_coeffs(Device->mRenderMode);
|
2017-05-04 11:09:45 -07:00
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
|
|
|
/* Special-case LFE */
|
|
|
|
if(chans[c].channel == LFE)
|
2016-01-28 00:02:46 -08:00
|
|
|
{
|
2019-07-04 15:02:12 -07:00
|
|
|
if(Device->Dry.Buffer.data() == Device->RealOut.Buffer.data())
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint idx{GetChannelIdxByName(Device->RealOut, chans[c].channel)};
|
2019-09-12 04:17:21 -07:00
|
|
|
if(idx != INVALID_CHANNEL_INDEX)
|
2019-12-21 00:48:58 -08:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target[idx] = DryGain.Base;
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
continue;
|
2016-01-28 00:02:46 -08:00
|
|
|
}
|
2017-05-04 04:35:53 -07:00
|
|
|
|
2021-06-24 08:21:25 -07:00
|
|
|
ComputePanGains(&Device->Dry, coeffs.data(), DryGain.Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target);
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2019-06-03 22:24:26 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2021-06-24 08:21:25 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
|
|
|
}
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(Device->AvgSpeakerDist > 0.0f)
|
|
|
|
{
|
2020-02-08 22:39:32 -08:00
|
|
|
/* If the source distance is 0, simulate a plane-wave by using
|
|
|
|
* infinite distance, which results in a w0 of 0.
|
2017-05-04 11:09:45 -07:00
|
|
|
*/
|
2022-02-23 01:29:28 -08:00
|
|
|
static constexpr float w0{0.0f};
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.NFCtrlFilter.adjust(w0);
|
2018-02-11 21:59:59 -08:00
|
|
|
|
2021-12-23 13:43:10 -08:00
|
|
|
voice->mFlags.set(VoiceHasNfc);
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{0};c < num_channels;c++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
|
|
|
/* Special-case LFE */
|
|
|
|
if(chans[c].channel == LFE)
|
|
|
|
{
|
2019-07-04 15:02:12 -07:00
|
|
|
if(Device->Dry.Buffer.data() == Device->RealOut.Buffer.data())
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint idx{GetChannelIdxByName(Device->RealOut, chans[c].channel)};
|
2019-09-12 04:17:21 -07:00
|
|
|
if(idx != INVALID_CHANNEL_INDEX)
|
2019-12-21 00:48:58 -08:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target[idx] = DryGain.Base;
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-09-01 05:46:19 -07:00
|
|
|
const auto coeffs = CalcAngleCoeffs((Device->mRenderMode == RenderMode::Pairwise)
|
2020-04-21 23:58:53 -07:00
|
|
|
? ScaleAzimuthFront(chans[c].angle, 3.0f) : chans[c].angle,
|
|
|
|
chans[c].elevation, Spread);
|
2018-08-29 01:45:27 -07:00
|
|
|
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Device->Dry, coeffs.data(), DryGain.Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.Gains.Target);
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2017-05-04 11:09:45 -07:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
if(const EffectSlot *Slot{SendSlots[i]})
|
2020-04-21 23:58:53 -07:00
|
|
|
ComputePanGains(&Slot->Wet, coeffs.data(), WetGain[i].Base,
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].Gains.Target);
|
2017-05-04 11:09:45 -07:00
|
|
|
}
|
2014-10-02 18:05:42 -07:00
|
|
|
}
|
2015-10-23 20:16:11 -07:00
|
|
|
}
|
2014-03-23 16:11:21 -07:00
|
|
|
}
|
2009-12-09 07:02:26 -08:00
|
|
|
|
2013-05-27 19:14:02 -07:00
|
|
|
{
|
2019-12-21 02:02:57 -08:00
|
|
|
const float hfNorm{props->Direct.HFReference / Frequency};
|
|
|
|
const float lfNorm{props->Direct.LFReference / Frequency};
|
2017-05-21 03:38:19 -07:00
|
|
|
|
2019-03-10 16:29:06 -07:00
|
|
|
voice->mDirect.FilterType = AF_None;
|
2019-12-21 02:02:57 -08:00
|
|
|
if(DryGain.HF != 1.0f) voice->mDirect.FilterType |= AF_LowPass;
|
|
|
|
if(DryGain.LF != 1.0f) voice->mDirect.FilterType |= AF_HighPass;
|
|
|
|
|
2019-06-03 22:24:26 -07:00
|
|
|
auto &lowpass = voice->mChans[0].mDryParams.LowPass;
|
|
|
|
auto &highpass = voice->mChans[0].mDryParams.HighPass;
|
2019-12-21 02:02:57 -08:00
|
|
|
lowpass.setParamsFromSlope(BiquadType::HighShelf, hfNorm, DryGain.HF, 1.0f);
|
|
|
|
highpass.setParamsFromSlope(BiquadType::LowShelf, lfNorm, DryGain.LF, 1.0f);
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{1};c < num_channels;c++)
|
2014-05-17 07:17:48 -07:00
|
|
|
{
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mDryParams.LowPass.copyParamsFrom(lowpass);
|
|
|
|
voice->mChans[c].mDryParams.HighPass.copyParamsFrom(highpass);
|
2014-05-17 07:17:48 -07:00
|
|
|
}
|
2013-05-27 19:14:02 -07:00
|
|
|
}
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2009-12-09 07:02:26 -08:00
|
|
|
{
|
2019-12-21 02:02:57 -08:00
|
|
|
const float hfNorm{props->Send[i].HFReference / Frequency};
|
|
|
|
const float lfNorm{props->Send[i].LFReference / Frequency};
|
2017-05-21 03:38:19 -07:00
|
|
|
|
2019-03-10 16:29:06 -07:00
|
|
|
voice->mSend[i].FilterType = AF_None;
|
2019-12-21 02:02:57 -08:00
|
|
|
if(WetGain[i].HF != 1.0f) voice->mSend[i].FilterType |= AF_LowPass;
|
|
|
|
if(WetGain[i].LF != 1.0f) voice->mSend[i].FilterType |= AF_HighPass;
|
2019-06-03 22:24:26 -07:00
|
|
|
|
|
|
|
auto &lowpass = voice->mChans[0].mWetParams[i].LowPass;
|
|
|
|
auto &highpass = voice->mChans[0].mWetParams[i].HighPass;
|
2019-12-21 02:02:57 -08:00
|
|
|
lowpass.setParamsFromSlope(BiquadType::HighShelf, hfNorm, WetGain[i].HF, 1.0f);
|
|
|
|
highpass.setParamsFromSlope(BiquadType::LowShelf, lfNorm, WetGain[i].LF, 1.0f);
|
2020-03-25 21:06:24 -07:00
|
|
|
for(size_t c{1};c < num_channels;c++)
|
2014-05-17 07:17:48 -07:00
|
|
|
{
|
2019-06-03 22:24:26 -07:00
|
|
|
voice->mChans[c].mWetParams[i].LowPass.copyParamsFrom(lowpass);
|
|
|
|
voice->mChans[c].mWetParams[i].HighPass.copyParamsFrom(highpass);
|
2014-05-17 07:17:48 -07:00
|
|
|
}
|
2009-12-09 07:02:26 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void CalcNonAttnSourceParams(Voice *voice, const VoiceProps *props, const ContextBase *context)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2021-04-24 08:28:13 -07:00
|
|
|
const DeviceBase *Device{context->mDevice};
|
2020-11-07 08:36:49 -08:00
|
|
|
EffectSlot *SendSlots[MAX_SENDS];
|
2017-05-04 04:35:53 -07:00
|
|
|
|
2019-07-04 15:02:12 -07:00
|
|
|
voice->mDirect.Buffer = Device->Dry.Buffer;
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < Device->NumAuxSends;i++)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
|
|
|
SendSlots[i] = props->Send[i].Slot;
|
2020-12-26 09:21:30 -08:00
|
|
|
if(!SendSlots[i] || SendSlots[i]->EffectType == EffectSlotType::None)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2019-01-07 12:37:13 +01:00
|
|
|
SendSlots[i] = nullptr;
|
2019-05-29 23:06:24 -07:00
|
|
|
voice->mSend[i].Buffer = {};
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
|
|
|
else
|
2019-07-04 15:02:12 -07:00
|
|
|
voice->mSend[i].Buffer = SendSlots[i]->Wet.Buffer;
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate the stepping value */
|
2020-04-08 06:17:04 -07:00
|
|
|
const auto Pitch = static_cast<float>(voice->mFrequency) /
|
|
|
|
static_cast<float>(Device->Frequency) * props->Pitch;
|
2020-12-03 23:26:07 -08:00
|
|
|
if(Pitch > float{MaxPitch})
|
|
|
|
voice->mStep = MaxPitch<<MixerFracBits;
|
2017-05-04 04:35:53 -07:00
|
|
|
else
|
2020-10-21 17:16:27 -07:00
|
|
|
voice->mStep = maxu(fastf2u(Pitch * MixerFracOne), 1);
|
2019-09-28 03:42:17 -07:00
|
|
|
voice->mResampler = PrepareResampler(props->mResampler, voice->mStep, &voice->mResampleState);
|
2017-05-04 04:35:53 -07:00
|
|
|
|
|
|
|
/* Calculate gains */
|
2019-12-21 00:48:58 -08:00
|
|
|
GainTriplet DryGain;
|
|
|
|
DryGain.Base = minf(clampf(props->Gain, props->MinGain, props->MaxGain) * props->Direct.Gain *
|
2020-12-15 17:25:45 -08:00
|
|
|
context->mParams.Gain, GainMixMax);
|
2019-12-21 00:48:58 -08:00
|
|
|
DryGain.HF = props->Direct.GainHF;
|
|
|
|
DryGain.LF = props->Direct.GainLF;
|
|
|
|
GainTriplet WetGain[MAX_SENDS];
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < Device->NumAuxSends;i++)
|
2017-05-04 04:35:53 -07:00
|
|
|
{
|
2019-12-21 00:48:58 -08:00
|
|
|
WetGain[i].Base = minf(clampf(props->Gain, props->MinGain, props->MaxGain) *
|
2020-12-15 17:25:45 -08:00
|
|
|
props->Send[i].Gain * context->mParams.Gain, GainMixMax);
|
2019-12-21 00:48:58 -08:00
|
|
|
WetGain[i].HF = props->Send[i].GainHF;
|
|
|
|
WetGain[i].LF = props->Send[i].GainLF;
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
|
|
|
|
2019-12-21 00:48:58 -08:00
|
|
|
CalcPanningAndFilters(voice, 0.0f, 0.0f, -1.0f, 0.0f, 0.0f, DryGain, WetGain, SendSlots, props,
|
2020-12-15 17:25:45 -08:00
|
|
|
context->mParams, Device);
|
2017-05-04 04:35:53 -07:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void CalcAttnSourceParams(Voice *voice, const VoiceProps *props, const ContextBase *context)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2021-04-24 08:28:13 -07:00
|
|
|
const DeviceBase *Device{context->mDevice};
|
2020-12-15 20:48:21 -08:00
|
|
|
const uint NumSends{Device->NumAuxSends};
|
2018-12-12 01:09:04 -08:00
|
|
|
|
|
|
|
/* Set mixing buffers and get send parameters. */
|
2019-07-04 15:02:12 -07:00
|
|
|
voice->mDirect.Buffer = Device->Dry.Buffer;
|
2020-11-07 08:36:49 -08:00
|
|
|
EffectSlot *SendSlots[MAX_SENDS];
|
2022-03-02 11:14:59 -08:00
|
|
|
uint UseDryAttnForRoom{0};
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2011-07-03 03:34:40 -07:00
|
|
|
{
|
2017-03-08 03:38:28 -08:00
|
|
|
SendSlots[i] = props->Send[i].Slot;
|
2020-12-26 09:21:30 -08:00
|
|
|
if(!SendSlots[i] || SendSlots[i]->EffectType == EffectSlotType::None)
|
2018-12-12 01:09:04 -08:00
|
|
|
SendSlots[i] = nullptr;
|
2022-03-02 11:14:59 -08:00
|
|
|
else if(!SendSlots[i]->AuxSendAuto)
|
2011-07-05 11:00:52 -07:00
|
|
|
{
|
2022-03-02 11:14:59 -08:00
|
|
|
/* If the slot's auxiliary send auto is off, the data sent to the
|
|
|
|
* effect slot is the same as the dry path, sans filter effects.
|
|
|
|
*/
|
|
|
|
UseDryAttnForRoom |= 1u<<i;
|
|
|
|
}
|
2011-07-05 14:14:20 -07:00
|
|
|
|
2016-01-28 00:02:46 -08:00
|
|
|
if(!SendSlots[i])
|
2019-05-29 23:06:24 -07:00
|
|
|
voice->mSend[i].Buffer = {};
|
2013-10-06 10:11:01 -07:00
|
|
|
else
|
2019-07-04 15:02:12 -07:00
|
|
|
voice->mSend[i].Buffer = SendSlots[i]->Wet.Buffer;
|
2011-07-03 03:34:40 -07:00
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-26 00:59:17 -07:00
|
|
|
/* Transform source to listener space (convert to head relative) */
|
2018-12-12 04:22:11 -08:00
|
|
|
alu::Vector Position{props->Position[0], props->Position[1], props->Position[2], 1.0f};
|
|
|
|
alu::Vector Velocity{props->Velocity[0], props->Velocity[1], props->Velocity[2], 0.0f};
|
|
|
|
alu::Vector Direction{props->Direction[0], props->Direction[1], props->Direction[2], 0.0f};
|
2020-11-12 12:46:15 -08:00
|
|
|
if(!props->HeadRelative)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2012-04-26 00:59:17 -07:00
|
|
|
/* Transform source vectors */
|
2021-05-25 14:08:16 -07:00
|
|
|
Position = context->mParams.Matrix * (Position - context->mParams.Position);
|
2020-12-15 17:25:45 -08:00
|
|
|
Velocity = context->mParams.Matrix * Velocity;
|
|
|
|
Direction = context->mParams.Matrix * Direction;
|
2009-10-19 13:25:40 -07:00
|
|
|
}
|
|
|
|
else
|
2011-10-30 05:49:17 -07:00
|
|
|
{
|
2012-03-09 23:53:45 -08:00
|
|
|
/* Offset the source velocity to be relative of the listener velocity */
|
2020-12-15 17:25:45 -08:00
|
|
|
Velocity += context->mParams.Velocity;
|
2011-10-30 05:49:17 -07:00
|
|
|
}
|
2009-11-22 22:36:20 -08:00
|
|
|
|
2018-12-12 04:22:11 -08:00
|
|
|
const bool directional{Direction.normalize() > 0.0f};
|
2019-03-10 11:33:08 -07:00
|
|
|
alu::Vector ToSource{Position[0], Position[1], Position[2], 0.0f};
|
2021-05-25 14:08:16 -07:00
|
|
|
const float Distance{ToSource.normalize()};
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-26 00:59:17 -07:00
|
|
|
/* Calculate distance attenuation */
|
2019-12-21 00:48:58 -08:00
|
|
|
float ClampedDist{Distance};
|
2022-03-02 11:14:59 -08:00
|
|
|
float DryGainBase{props->Gain};
|
|
|
|
float WetGainBase{props->Gain};
|
2009-10-19 13:25:40 -07:00
|
|
|
|
2020-12-15 17:25:45 -08:00
|
|
|
switch(context->mParams.SourceDistanceModel ? props->mDistanceModel
|
|
|
|
: context->mParams.mDistanceModel)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::InverseClamped:
|
2018-12-12 01:09:04 -08:00
|
|
|
if(props->MaxDistance < props->RefDistance) break;
|
2022-03-30 04:32:40 -07:00
|
|
|
ClampedDist = clampf(ClampedDist, props->RefDistance, props->MaxDistance);
|
2012-04-26 00:59:17 -07:00
|
|
|
/*fall-through*/
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::Inverse:
|
2022-03-30 04:32:40 -07:00
|
|
|
if(props->RefDistance > 0.0f)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2022-04-06 17:41:24 -07:00
|
|
|
float dist{lerpf(props->RefDistance, ClampedDist, props->RolloffFactor)};
|
2022-03-02 11:14:59 -08:00
|
|
|
if(dist > 0.0f) DryGainBase *= props->RefDistance / dist;
|
|
|
|
|
2022-04-06 17:41:24 -07:00
|
|
|
dist = lerpf(props->RefDistance, ClampedDist, props->RoomRolloffFactor);
|
2022-03-02 11:14:59 -08:00
|
|
|
if(dist > 0.0f) WetGainBase *= props->RefDistance / dist;
|
2009-10-19 13:25:40 -07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::LinearClamped:
|
2018-12-12 01:09:04 -08:00
|
|
|
if(props->MaxDistance < props->RefDistance) break;
|
2022-03-30 04:32:40 -07:00
|
|
|
ClampedDist = clampf(ClampedDist, props->RefDistance, props->MaxDistance);
|
2012-04-26 00:59:17 -07:00
|
|
|
/*fall-through*/
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::Linear:
|
2022-03-30 04:32:40 -07:00
|
|
|
if(props->MaxDistance != props->RefDistance)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2022-03-02 11:14:59 -08:00
|
|
|
float attn{(ClampedDist-props->RefDistance) /
|
|
|
|
(props->MaxDistance-props->RefDistance) * props->RolloffFactor};
|
|
|
|
DryGainBase *= maxf(1.0f - attn, 0.0f);
|
|
|
|
|
|
|
|
attn = (ClampedDist-props->RefDistance) /
|
|
|
|
(props->MaxDistance-props->RefDistance) * props->RoomRolloffFactor;
|
|
|
|
WetGainBase *= maxf(1.0f - attn, 0.0f);
|
2009-10-19 13:25:40 -07:00
|
|
|
}
|
|
|
|
break;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::ExponentClamped:
|
2018-12-12 01:09:04 -08:00
|
|
|
if(props->MaxDistance < props->RefDistance) break;
|
2022-03-30 04:32:40 -07:00
|
|
|
ClampedDist = clampf(ClampedDist, props->RefDistance, props->MaxDistance);
|
2012-04-26 00:59:17 -07:00
|
|
|
/*fall-through*/
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::Exponent:
|
2022-03-30 04:32:40 -07:00
|
|
|
if(ClampedDist > 0.0f && props->RefDistance > 0.0f)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2020-02-09 14:25:20 -08:00
|
|
|
const float dist_ratio{ClampedDist/props->RefDistance};
|
2022-03-02 11:14:59 -08:00
|
|
|
DryGainBase *= std::pow(dist_ratio, -props->RolloffFactor);
|
|
|
|
WetGainBase *= std::pow(dist_ratio, -props->RoomRolloffFactor);
|
2009-10-19 13:25:40 -07:00
|
|
|
}
|
|
|
|
break;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2018-11-17 23:02:27 -08:00
|
|
|
case DistanceModel::Disable:
|
2009-10-19 13:25:40 -07:00
|
|
|
break;
|
|
|
|
}
|
2009-04-11 20:27:55 -07:00
|
|
|
|
2011-09-30 17:51:21 -07:00
|
|
|
/* Calculate directional soundcones */
|
2022-03-02 11:14:59 -08:00
|
|
|
float ConeHF{1.0f}, WetConeHF{1.0f};
|
2017-05-20 02:22:11 -07:00
|
|
|
if(directional && props->InnerAngle < 360.0f)
|
2009-10-19 13:25:40 -07:00
|
|
|
{
|
2022-02-23 01:29:28 -08:00
|
|
|
static constexpr float Rad2Deg{static_cast<float>(180.0 / al::numbers::pi)};
|
2022-01-27 02:59:07 -08:00
|
|
|
const float Angle{Rad2Deg*2.0f * std::acos(-Direction.dot_product(ToSource)) * ConeScale};
|
2018-12-12 01:09:04 -08:00
|
|
|
|
2022-03-02 11:14:59 -08:00
|
|
|
float ConeGain{1.0f};
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
if(Angle >= props->OuterAngle)
|
2016-05-16 22:42:41 -07:00
|
|
|
{
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
ConeGain = props->OuterGain;
|
2022-04-06 17:41:24 -07:00
|
|
|
ConeHF = lerpf(1.0f, props->OuterGainHF, props->DryGainHFAuto);
|
2017-05-20 02:22:11 -07:00
|
|
|
}
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
else if(Angle >= props->InnerAngle)
|
2017-05-20 02:22:11 -07:00
|
|
|
{
|
2019-12-21 00:48:58 -08:00
|
|
|
const float scale{(Angle-props->InnerAngle) / (props->OuterAngle-props->InnerAngle)};
|
2022-04-06 17:41:24 -07:00
|
|
|
ConeGain = lerpf(1.0f, props->OuterGain, scale);
|
|
|
|
ConeHF = lerpf(1.0f, props->OuterGainHF, scale * props->DryGainHFAuto);
|
2016-05-16 22:42:41 -07:00
|
|
|
}
|
|
|
|
|
2022-03-02 11:14:59 -08:00
|
|
|
DryGainBase *= ConeGain;
|
2022-04-06 17:41:24 -07:00
|
|
|
WetGainBase *= lerpf(1.0f, ConeGain, props->WetGainAuto);
|
2022-03-02 11:14:59 -08:00
|
|
|
|
2022-04-06 17:41:24 -07:00
|
|
|
WetConeHF = lerpf(1.0f, ConeHF, props->WetGainHFAuto);
|
2011-07-05 11:00:52 -07:00
|
|
|
}
|
2009-12-08 14:18:07 -08:00
|
|
|
|
2012-04-26 00:59:17 -07:00
|
|
|
/* Apply gain and frequency filters */
|
2022-03-02 11:14:59 -08:00
|
|
|
DryGainBase = clampf(DryGainBase, props->MinGain, props->MaxGain) * context->mParams.Gain;
|
|
|
|
WetGainBase = clampf(WetGainBase, props->MinGain, props->MaxGain) * context->mParams.Gain;
|
|
|
|
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
GainTriplet DryGain{};
|
2022-03-02 11:14:59 -08:00
|
|
|
DryGain.Base = minf(DryGainBase * props->Direct.Gain, GainMixMax);
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
DryGain.HF = ConeHF * props->Direct.GainHF;
|
|
|
|
DryGain.LF = props->Direct.GainLF;
|
|
|
|
GainTriplet WetGain[MAX_SENDS]{};
|
2020-12-15 20:48:21 -08:00
|
|
|
for(uint i{0};i < NumSends;i++)
|
2011-07-05 11:00:52 -07:00
|
|
|
{
|
2022-03-02 16:58:26 -08:00
|
|
|
/* If this effect slot's Auxiliary Send Auto is off, then use the dry
|
|
|
|
* path distance and cone attenuation, otherwise use the wet (room)
|
|
|
|
* path distance and cone attenuation. The send filter is used instead
|
|
|
|
* of the direct filter, regardless.
|
2022-03-02 15:34:09 -08:00
|
|
|
*/
|
2022-03-02 16:58:26 -08:00
|
|
|
const bool use_room{!(UseDryAttnForRoom&(1u<<i))};
|
|
|
|
const float gain{use_room ? WetGainBase : DryGainBase};
|
2022-03-02 11:14:59 -08:00
|
|
|
WetGain[i].Base = minf(gain * props->Send[i].Gain, GainMixMax);
|
2022-03-02 16:58:26 -08:00
|
|
|
WetGain[i].HF = (use_room ? WetConeHF : ConeHF) * props->Send[i].GainHF;
|
Rework the initial reverb decay
The idea here is that the initial reverb decay can't become less than the dry
path distance attenuation, as the dry attenuation represents the audio that has
not yet had a chance to start reflecting in the environment. As well, the
reference distance indicates where there is no distance attenuation, with any
initial attenuation set by the environment itself.
So what we do is use the dry path attenuation as the baseline for what's mixed
to the reverb, with the decay rate indicating how much of the remaining room
(non-direct) energy attenuates with distance.
This may be over-complicating it. Other sources hint at a more typical XdB per
doubling of distance, with X varying depending on environment properties (room
size, absorbancy, etc). This could be handled by applying a normal inverse
distance attenuation model, with a rolloff factor generated from the reverb
properties (density, decay rate, etc). Will need more testing and research.
2022-03-01 15:38:04 -08:00
|
|
|
WetGain[i].LF = props->Send[i].GainLF;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2008-01-19 00:49:05 -08:00
|
|
|
|
2018-03-29 15:41:51 -07:00
|
|
|
/* Distance-based air absorption and initial send decay. */
|
2022-03-02 15:34:09 -08:00
|
|
|
if(likely(Distance > props->RefDistance))
|
2018-03-29 15:41:51 -07:00
|
|
|
{
|
2022-03-02 16:58:26 -08:00
|
|
|
const float distance_base{(Distance-props->RefDistance) * props->RolloffFactor};
|
|
|
|
const float absorption{distance_base * context->mParams.MetersPerUnit *
|
2022-03-02 11:14:59 -08:00
|
|
|
props->AirAbsorptionFactor};
|
|
|
|
if(absorption > std::numeric_limits<float>::epsilon())
|
2018-03-29 15:41:51 -07:00
|
|
|
{
|
2022-03-02 11:14:59 -08:00
|
|
|
const float hfattn{std::pow(context->mParams.AirAbsorptionGainHF, absorption)};
|
2022-03-02 16:58:26 -08:00
|
|
|
DryGain.HF *= hfattn;
|
|
|
|
for(uint i{0u};i < NumSends;++i)
|
|
|
|
WetGain[i].HF *= hfattn;
|
2018-03-29 15:41:51 -07:00
|
|
|
}
|
|
|
|
|
2022-03-02 15:34:09 -08:00
|
|
|
/* If the source's Auxiliary Send Filter Gain Auto is off, no extra
|
|
|
|
* adjustment is applied to the send gains.
|
|
|
|
*/
|
|
|
|
for(uint i{props->WetGainAuto ? 0u : NumSends};i < NumSends;++i)
|
2022-03-02 11:14:59 -08:00
|
|
|
{
|
2022-06-16 18:36:04 -07:00
|
|
|
if(!SendSlots[i] || !(SendSlots[i]->DecayTime > 0.0f))
|
2022-03-02 15:34:09 -08:00
|
|
|
continue;
|
2022-03-02 11:14:59 -08:00
|
|
|
|
2022-03-02 15:34:09 -08:00
|
|
|
auto calc_attenuation = [](float distance, float refdist, float rolloff) noexcept
|
|
|
|
{
|
2022-04-06 17:41:24 -07:00
|
|
|
const float dist{lerpf(refdist, distance, rolloff)};
|
2022-03-02 15:34:09 -08:00
|
|
|
if(dist > refdist) return refdist / dist;
|
|
|
|
return 1.0f;
|
|
|
|
};
|
2022-03-02 11:14:59 -08:00
|
|
|
|
2022-03-02 15:34:09 -08:00
|
|
|
/* The reverb effect's room rolloff factor always applies to an
|
|
|
|
* inverse distance rolloff model.
|
|
|
|
*/
|
|
|
|
WetGain[i].Base *= calc_attenuation(Distance, props->RefDistance,
|
|
|
|
SendSlots[i]->RoomRolloff);
|
|
|
|
|
|
|
|
/* If this effect slot's Auxiliary Send Auto is off, don't apply
|
|
|
|
* the automatic initial reverb decay (should the reverb's room
|
|
|
|
* rolloff still apply?).
|
|
|
|
*/
|
|
|
|
if(!SendSlots[i]->AuxSendAuto)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
GainTriplet DecayDistance;
|
|
|
|
/* Calculate the distances to where this effect's decay reaches
|
|
|
|
* -60dB.
|
|
|
|
*/
|
|
|
|
DecayDistance.Base = SendSlots[i]->DecayTime * SpeedOfSoundMetersPerSec;
|
|
|
|
DecayDistance.LF = DecayDistance.Base * SendSlots[i]->DecayLFRatio;
|
|
|
|
DecayDistance.HF = DecayDistance.Base * SendSlots[i]->DecayHFRatio;
|
|
|
|
if(SendSlots[i]->DecayHFLimit)
|
|
|
|
{
|
|
|
|
const float airAbsorption{SendSlots[i]->AirAbsorptionGainHF};
|
|
|
|
if(airAbsorption < 1.0f)
|
|
|
|
{
|
|
|
|
/* Calculate the distance to where this effect's air
|
|
|
|
* absorption reaches -60dB, and limit the effect's HF
|
|
|
|
* decay distance (so it doesn't take any longer to decay
|
|
|
|
* than the air would allow).
|
|
|
|
*/
|
|
|
|
static constexpr float log10_decaygain{-3.0f/*std::log10(ReverbDecayGain)*/};
|
|
|
|
const float absorb_dist{log10_decaygain / std::log10(airAbsorption)};
|
|
|
|
DecayDistance.HF = minf(absorb_dist, DecayDistance.HF);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const float baseAttn = calc_attenuation(Distance, props->RefDistance,
|
2022-03-02 11:14:59 -08:00
|
|
|
props->RolloffFactor);
|
|
|
|
|
2018-03-29 15:41:51 -07:00
|
|
|
/* Apply a decay-time transformation to the wet path, based on the
|
2022-03-02 16:58:26 -08:00
|
|
|
* source distance. The initial decay of the reverb effect is
|
|
|
|
* calculated and applied to the wet path.
|
2018-03-29 15:41:51 -07:00
|
|
|
*/
|
2022-03-02 16:58:26 -08:00
|
|
|
const float fact{distance_base / DecayDistance.Base};
|
|
|
|
const float gain{std::pow(ReverbDecayGain, fact)*(1.0f-baseAttn) + baseAttn};
|
2022-03-02 15:34:09 -08:00
|
|
|
WetGain[i].Base *= gain;
|
2018-03-29 15:41:51 -07:00
|
|
|
|
2022-03-02 15:34:09 -08:00
|
|
|
if(gain > 0.0f)
|
|
|
|
{
|
2022-03-02 16:58:26 -08:00
|
|
|
const float hffact{distance_base / DecayDistance.HF};
|
|
|
|
const float gainhf{std::pow(ReverbDecayGain, hffact)*(1.0f-baseAttn) + baseAttn};
|
2022-03-02 15:34:09 -08:00
|
|
|
WetGain[i].HF *= minf(gainhf/gain, 1.0f);
|
2022-03-02 16:58:26 -08:00
|
|
|
const float lffact{distance_base / DecayDistance.LF};
|
|
|
|
const float gainlf{std::pow(ReverbDecayGain, lffact)*(1.0f-baseAttn) + baseAttn};
|
2022-03-02 15:34:09 -08:00
|
|
|
WetGain[i].LF *= minf(gainlf/gain, 1.0f);
|
2018-03-29 15:41:51 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-17 22:49:34 -07:00
|
|
|
|
|
|
|
/* Initial source pitch */
|
2020-04-08 06:17:04 -07:00
|
|
|
float Pitch{props->Pitch};
|
2017-05-17 22:49:34 -07:00
|
|
|
|
2012-04-26 00:59:17 -07:00
|
|
|
/* Calculate velocity-based doppler effect */
|
2020-12-15 17:25:45 -08:00
|
|
|
float DopplerFactor{props->DopplerFactor * context->mParams.DopplerFactor};
|
2012-03-18 08:20:08 -07:00
|
|
|
if(DopplerFactor > 0.0f)
|
2009-12-01 03:32:04 -08:00
|
|
|
{
|
2020-12-15 17:25:45 -08:00
|
|
|
const alu::Vector &lvelocity = context->mParams.Velocity;
|
2020-11-13 23:33:40 -08:00
|
|
|
float vss{Velocity.dot_product(ToSource) * -DopplerFactor};
|
|
|
|
float vls{lvelocity.dot_product(ToSource) * -DopplerFactor};
|
2017-05-20 03:28:40 -07:00
|
|
|
|
2020-12-15 17:25:45 -08:00
|
|
|
const float SpeedOfSound{context->mParams.SpeedOfSound};
|
2017-05-20 03:28:40 -07:00
|
|
|
if(!(vls < SpeedOfSound))
|
2012-03-18 08:20:08 -07:00
|
|
|
{
|
2017-05-20 03:28:40 -07:00
|
|
|
/* Listener moving away from the source at the speed of sound.
|
|
|
|
* Sound waves can't catch it.
|
|
|
|
*/
|
|
|
|
Pitch = 0.0f;
|
|
|
|
}
|
|
|
|
else if(!(vss < SpeedOfSound))
|
|
|
|
{
|
|
|
|
/* Source moving toward the listener at the speed of sound. Sound
|
|
|
|
* waves bunch up to extreme frequencies.
|
|
|
|
*/
|
2018-12-22 16:01:14 -08:00
|
|
|
Pitch = std::numeric_limits<float>::infinity();
|
2017-05-20 03:28:40 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Source and listener movement is nominal. Calculate the proper
|
|
|
|
* doppler shift.
|
|
|
|
*/
|
|
|
|
Pitch *= (SpeedOfSound-vls) / (SpeedOfSound-vss);
|
2012-03-18 08:20:08 -07:00
|
|
|
}
|
2009-12-01 03:32:04 -08:00
|
|
|
}
|
2010-08-07 00:38:02 -07:00
|
|
|
|
2017-05-17 22:49:34 -07:00
|
|
|
/* Adjust pitch based on the buffer and output frequencies, and calculate
|
|
|
|
* fixed-point stepping value.
|
2016-05-09 14:22:26 -07:00
|
|
|
*/
|
2020-04-08 06:17:04 -07:00
|
|
|
Pitch *= static_cast<float>(voice->mFrequency) / static_cast<float>(Device->Frequency);
|
2020-12-03 23:26:07 -08:00
|
|
|
if(Pitch > float{MaxPitch})
|
|
|
|
voice->mStep = MaxPitch<<MixerFracBits;
|
2016-05-09 14:22:26 -07:00
|
|
|
else
|
2020-10-21 17:16:27 -07:00
|
|
|
voice->mStep = maxu(fastf2u(Pitch * MixerFracOne), 1);
|
2019-09-28 03:42:17 -07:00
|
|
|
voice->mResampler = PrepareResampler(props->mResampler, voice->mStep, &voice->mResampleState);
|
2009-12-01 03:32:04 -08:00
|
|
|
|
2020-04-08 06:17:04 -07:00
|
|
|
float spread{0.0f};
|
2017-05-04 11:09:45 -07:00
|
|
|
if(props->Radius > Distance)
|
2022-01-27 02:59:07 -08:00
|
|
|
spread = al::numbers::pi_v<float>*2.0f - Distance/props->Radius*al::numbers::pi_v<float>;
|
2018-03-29 12:11:37 -07:00
|
|
|
else if(Distance > 0.0f)
|
2018-12-12 01:09:04 -08:00
|
|
|
spread = std::asin(props->Radius/Distance) * 2.0f;
|
2009-12-09 07:21:59 -08:00
|
|
|
|
2022-03-10 22:37:02 -08:00
|
|
|
CalcPanningAndFilters(voice, ToSource[0]*XScale, ToSource[1]*YScale, ToSource[2]*ZScale,
|
2020-12-15 17:25:45 -08:00
|
|
|
Distance*context->mParams.MetersPerUnit, spread, DryGain, WetGain, SendSlots, props,
|
|
|
|
context->mParams, Device);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void CalcSourceParams(Voice *voice, ContextBase *context, bool force)
|
2016-05-16 14:46:06 -07:00
|
|
|
{
|
2020-03-28 17:20:38 -07:00
|
|
|
VoicePropsItem *props{voice->mUpdate.exchange(nullptr, std::memory_order_acq_rel)};
|
2020-03-04 21:15:32 -08:00
|
|
|
if(!props && !force) return;
|
2016-08-24 02:17:55 -07:00
|
|
|
|
|
|
|
if(props)
|
2016-08-23 18:56:01 -07:00
|
|
|
{
|
2019-03-10 16:29:06 -07:00
|
|
|
voice->mProps = *props;
|
2016-08-23 18:56:01 -07:00
|
|
|
|
2019-07-30 09:05:54 -07:00
|
|
|
AtomicReplaceHead(context->mFreeVoiceProps, props);
|
2016-08-23 18:56:01 -07:00
|
|
|
}
|
2016-08-24 02:17:55 -07:00
|
|
|
|
2019-12-29 19:17:29 -08:00
|
|
|
if((voice->mProps.DirectChannels != DirectMode::Off && voice->mFmtChannels != FmtMono
|
2021-12-11 17:50:24 -08:00
|
|
|
&& !IsAmbisonic(voice->mFmtChannels))
|
|
|
|
|| voice->mProps.mSpatializeMode == SpatializeMode::Off
|
2020-05-21 09:10:32 -07:00
|
|
|
|| (voice->mProps.mSpatializeMode==SpatializeMode::Auto && voice->mFmtChannels != FmtMono))
|
2019-03-10 16:29:06 -07:00
|
|
|
CalcNonAttnSourceParams(voice, &voice->mProps, context);
|
2019-12-17 22:36:26 -08:00
|
|
|
else
|
|
|
|
CalcAttnSourceParams(voice, &voice->mProps, context);
|
2016-05-16 14:46:06 -07:00
|
|
|
}
|
|
|
|
|
2009-08-26 19:15:17 -07:00
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void SendSourceStateEvent(ContextBase *context, uint id, VChangeState state)
|
2020-02-20 22:50:37 -08:00
|
|
|
{
|
|
|
|
RingBuffer *ring{context->mAsyncEvents.get()};
|
|
|
|
auto evt_vec = ring->getWriteVector();
|
|
|
|
if(evt_vec.first.len < 1) return;
|
|
|
|
|
2021-10-08 11:05:36 -07:00
|
|
|
AsyncEvent *evt{al::construct_at(reinterpret_cast<AsyncEvent*>(evt_vec.first.buf),
|
2021-10-10 05:07:31 -07:00
|
|
|
AsyncEvent::SourceStateChange)};
|
2020-02-20 22:50:37 -08:00
|
|
|
evt->u.srcstate.id = id;
|
2021-04-25 14:29:21 -07:00
|
|
|
switch(state)
|
|
|
|
{
|
|
|
|
case VChangeState::Reset:
|
|
|
|
evt->u.srcstate.state = AsyncEvent::SrcState::Reset;
|
|
|
|
break;
|
|
|
|
case VChangeState::Stop:
|
|
|
|
evt->u.srcstate.state = AsyncEvent::SrcState::Stop;
|
|
|
|
break;
|
|
|
|
case VChangeState::Play:
|
|
|
|
evt->u.srcstate.state = AsyncEvent::SrcState::Play;
|
|
|
|
break;
|
|
|
|
case VChangeState::Pause:
|
|
|
|
evt->u.srcstate.state = AsyncEvent::SrcState::Pause;
|
|
|
|
break;
|
|
|
|
/* Shouldn't happen. */
|
|
|
|
case VChangeState::Restart:
|
|
|
|
ASSUME(0);
|
|
|
|
}
|
2020-02-20 22:50:37 -08:00
|
|
|
|
|
|
|
ring->writeAdvance(1);
|
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void ProcessVoiceChanges(ContextBase *ctx)
|
2020-02-20 22:50:37 -08:00
|
|
|
{
|
|
|
|
VoiceChange *cur{ctx->mCurrentVoiceChange.load(std::memory_order_acquire)};
|
|
|
|
VoiceChange *next{cur->mNext.load(std::memory_order_acquire)};
|
|
|
|
if(!next) return;
|
|
|
|
|
2020-12-27 01:05:16 -08:00
|
|
|
const uint enabledevt{ctx->mEnabledEvts.load(std::memory_order_acquire)};
|
2020-02-20 22:50:37 -08:00
|
|
|
do {
|
|
|
|
cur = next;
|
|
|
|
|
2020-02-23 21:40:54 -08:00
|
|
|
bool sendevt{false};
|
2020-12-16 01:18:11 -08:00
|
|
|
if(cur->mState == VChangeState::Reset || cur->mState == VChangeState::Stop)
|
2020-02-20 22:50:37 -08:00
|
|
|
{
|
2020-03-28 17:20:38 -07:00
|
|
|
if(Voice *voice{cur->mVoice})
|
2020-02-20 22:50:37 -08:00
|
|
|
{
|
|
|
|
voice->mCurrentBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
voice->mLoopBuffer.store(nullptr, std::memory_order_relaxed);
|
2020-12-16 02:12:18 -08:00
|
|
|
/* A source ID indicates the voice was playing or paused, which
|
|
|
|
* gets a reset/stop event.
|
|
|
|
*/
|
|
|
|
sendevt = voice->mSourceID.exchange(0u, std::memory_order_relaxed) != 0u;
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice::State oldvstate{Voice::Playing};
|
2020-12-16 02:12:18 -08:00
|
|
|
voice->mPlayState.compare_exchange_strong(oldvstate, Voice::Stopping,
|
2020-02-20 22:50:37 -08:00
|
|
|
std::memory_order_relaxed, std::memory_order_acquire);
|
2020-02-26 01:48:59 -08:00
|
|
|
voice->mPendingChange.store(false, std::memory_order_release);
|
2020-02-20 22:50:37 -08:00
|
|
|
}
|
2020-12-16 02:12:18 -08:00
|
|
|
/* Reset state change events are always sent, even if the voice is
|
|
|
|
* already stopped or even if there is no voice.
|
2020-02-23 21:40:54 -08:00
|
|
|
*/
|
2020-12-16 01:18:11 -08:00
|
|
|
sendevt |= (cur->mState == VChangeState::Reset);
|
2020-02-20 22:50:37 -08:00
|
|
|
}
|
2020-12-16 01:18:11 -08:00
|
|
|
else if(cur->mState == VChangeState::Pause)
|
2020-02-21 04:29:32 -08:00
|
|
|
{
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice *voice{cur->mVoice};
|
|
|
|
Voice::State oldvstate{Voice::Playing};
|
|
|
|
sendevt = voice->mPlayState.compare_exchange_strong(oldvstate, Voice::Stopping,
|
2020-02-21 04:29:32 -08:00
|
|
|
std::memory_order_release, std::memory_order_acquire);
|
|
|
|
}
|
2020-12-16 01:18:11 -08:00
|
|
|
else if(cur->mState == VChangeState::Play)
|
2020-02-21 04:29:32 -08:00
|
|
|
{
|
2020-02-23 21:40:54 -08:00
|
|
|
/* NOTE: When playing a voice, sending a source state change event
|
|
|
|
* depends if there's an old voice to stop and if that stop is
|
|
|
|
* successful. If there is no old voice, a playing event is always
|
|
|
|
* sent. If there is an old voice, an event is sent only if the
|
|
|
|
* voice is already stopped.
|
|
|
|
*/
|
2020-03-28 17:20:38 -07:00
|
|
|
if(Voice *oldvoice{cur->mOldVoice})
|
2020-02-23 21:40:54 -08:00
|
|
|
{
|
|
|
|
oldvoice->mCurrentBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
oldvoice->mLoopBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
oldvoice->mSourceID.store(0u, std::memory_order_relaxed);
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice::State oldvstate{Voice::Playing};
|
|
|
|
sendevt = !oldvoice->mPlayState.compare_exchange_strong(oldvstate, Voice::Stopping,
|
|
|
|
std::memory_order_relaxed, std::memory_order_acquire);
|
2020-02-26 01:48:59 -08:00
|
|
|
oldvoice->mPendingChange.store(false, std::memory_order_release);
|
2020-02-23 21:40:54 -08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
sendevt = true;
|
|
|
|
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice *voice{cur->mVoice};
|
|
|
|
voice->mPlayState.store(Voice::Playing, std::memory_order_release);
|
2020-02-21 04:29:32 -08:00
|
|
|
}
|
2020-12-16 01:18:11 -08:00
|
|
|
else if(cur->mState == VChangeState::Restart)
|
2020-02-26 03:46:15 -08:00
|
|
|
{
|
2020-12-16 02:12:18 -08:00
|
|
|
/* Restarting a voice never sends a source change event. */
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice *oldvoice{cur->mOldVoice};
|
2020-02-26 03:46:15 -08:00
|
|
|
oldvoice->mCurrentBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
oldvoice->mLoopBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
/* If there's no sourceID, the old voice finished so don't start
|
|
|
|
* the new one at its new offset.
|
|
|
|
*/
|
|
|
|
if(oldvoice->mSourceID.exchange(0u, std::memory_order_relaxed) != 0u)
|
|
|
|
{
|
|
|
|
/* Otherwise, set the voice to stopping if it's not already (it
|
2020-03-04 10:09:44 -08:00
|
|
|
* might already be, if paused), and play the new voice as
|
2020-02-26 03:46:15 -08:00
|
|
|
* appropriate.
|
|
|
|
*/
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice::State oldvstate{Voice::Playing};
|
|
|
|
oldvoice->mPlayState.compare_exchange_strong(oldvstate, Voice::Stopping,
|
2020-02-26 03:46:15 -08:00
|
|
|
std::memory_order_relaxed, std::memory_order_acquire);
|
|
|
|
|
2020-03-28 17:20:38 -07:00
|
|
|
Voice *voice{cur->mVoice};
|
|
|
|
voice->mPlayState.store((oldvstate == Voice::Playing) ? Voice::Playing
|
|
|
|
: Voice::Stopped, std::memory_order_release);
|
2020-02-26 03:46:15 -08:00
|
|
|
}
|
|
|
|
oldvoice->mPendingChange.store(false, std::memory_order_release);
|
|
|
|
}
|
2021-10-10 05:07:31 -07:00
|
|
|
if(sendevt && (enabledevt&AsyncEvent::SourceStateChange))
|
2020-02-20 22:50:37 -08:00
|
|
|
SendSourceStateEvent(ctx, cur->mSourceID, cur->mState);
|
2020-03-22 11:35:08 -07:00
|
|
|
|
|
|
|
next = cur->mNext.load(std::memory_order_acquire);
|
|
|
|
} while(next);
|
2020-02-20 22:50:37 -08:00
|
|
|
ctx->mCurrentVoiceChange.store(cur, std::memory_order_release);
|
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void ProcessParamUpdates(ContextBase *ctx, const EffectSlotArray &slots,
|
2020-03-28 17:20:38 -07:00
|
|
|
const al::span<Voice*> voices)
|
2015-09-18 00:48:43 -07:00
|
|
|
{
|
2020-02-20 22:50:37 -08:00
|
|
|
ProcessVoiceChanges(ctx);
|
|
|
|
|
2019-08-01 13:28:53 -07:00
|
|
|
IncrementRef(ctx->mUpdateCount);
|
2019-08-04 11:59:14 -07:00
|
|
|
if LIKELY(!ctx->mHoldUpdates.load(std::memory_order_acquire))
|
2016-05-15 17:14:58 -07:00
|
|
|
{
|
2019-08-05 11:37:05 -07:00
|
|
|
bool force{CalcContextParams(ctx)};
|
2020-11-07 08:36:49 -08:00
|
|
|
auto sorted_slots = const_cast<EffectSlot**>(slots.data() + slots.size());
|
|
|
|
for(EffectSlot *slot : slots)
|
2020-01-18 18:53:58 -08:00
|
|
|
force |= CalcEffectSlotParams(slot, sorted_slots, ctx);
|
2016-05-12 18:26:33 -07:00
|
|
|
|
2020-03-28 17:20:38 -07:00
|
|
|
for(Voice *voice : voices)
|
2020-03-04 21:15:32 -08:00
|
|
|
{
|
|
|
|
/* Only update voices that have a source. */
|
|
|
|
if(voice->mSourceID.load(std::memory_order_relaxed) != 0)
|
|
|
|
CalcSourceParams(voice, ctx, force);
|
|
|
|
}
|
2015-09-18 00:48:43 -07:00
|
|
|
}
|
2019-08-01 13:28:53 -07:00
|
|
|
IncrementRef(ctx->mUpdateCount);
|
2015-09-18 00:48:43 -07:00
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void ProcessContexts(DeviceBase *device, const uint SamplesToDo)
|
2018-11-23 13:12:48 -08:00
|
|
|
{
|
2018-12-22 22:14:25 -08:00
|
|
|
ASSUME(SamplesToDo > 0);
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
for(ContextBase *ctx : *device->mContexts.load(std::memory_order_acquire))
|
2020-02-21 22:17:29 -08:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
const EffectSlotArray &auxslots = *ctx->mActiveAuxSlots.load(std::memory_order_acquire);
|
2020-03-28 17:20:38 -07:00
|
|
|
const al::span<Voice*> voices{ctx->getVoicesSpanAcquired()};
|
2018-11-23 13:12:48 -08:00
|
|
|
|
2020-02-21 22:17:29 -08:00
|
|
|
/* Process pending propery updates for objects on the context. */
|
|
|
|
ProcessParamUpdates(ctx, auxslots, voices);
|
2018-11-23 13:12:48 -08:00
|
|
|
|
2020-02-21 22:17:29 -08:00
|
|
|
/* Clear auxiliary effect slot mixing buffers. */
|
2020-11-07 08:36:49 -08:00
|
|
|
for(EffectSlot *slot : auxslots)
|
2018-11-23 13:12:48 -08:00
|
|
|
{
|
2020-11-02 04:24:36 -08:00
|
|
|
for(auto &buffer : slot->Wet.Buffer)
|
2020-05-12 04:46:15 -07:00
|
|
|
buffer.fill(0.0f);
|
2020-02-21 22:17:29 -08:00
|
|
|
}
|
2020-01-08 07:31:01 -08:00
|
|
|
|
2020-02-21 22:17:29 -08:00
|
|
|
/* Process voices that have a playing source. */
|
2020-03-28 17:20:38 -07:00
|
|
|
for(Voice *voice : voices)
|
2020-01-18 18:53:58 -08:00
|
|
|
{
|
2020-03-28 17:20:38 -07:00
|
|
|
const Voice::State vstate{voice->mPlayState.load(std::memory_order_acquire)};
|
|
|
|
if(vstate != Voice::Stopped && vstate != Voice::Pending)
|
2020-03-04 10:09:44 -08:00
|
|
|
voice->mix(vstate, ctx, SamplesToDo);
|
2020-01-18 18:53:58 -08:00
|
|
|
}
|
2018-12-22 22:14:25 -08:00
|
|
|
|
2020-02-21 22:17:29 -08:00
|
|
|
/* Process effects. */
|
|
|
|
if(const size_t num_slots{auxslots.size()})
|
2020-01-14 10:38:24 -08:00
|
|
|
{
|
2020-02-21 22:17:29 -08:00
|
|
|
auto slots = auxslots.data();
|
|
|
|
auto slots_end = slots + num_slots;
|
2020-01-14 10:38:24 -08:00
|
|
|
|
2020-11-05 15:35:09 -08:00
|
|
|
/* Sort the slots into extra storage, so that effect slots come
|
|
|
|
* before their effect slot target (or their targets' target).
|
2020-01-14 10:38:24 -08:00
|
|
|
*/
|
2020-11-07 08:36:49 -08:00
|
|
|
const al::span<EffectSlot*> sorted_slots{const_cast<EffectSlot**>(slots_end),
|
2020-11-05 15:35:09 -08:00
|
|
|
num_slots};
|
|
|
|
/* Skip sorting if it has already been done. */
|
|
|
|
if(!sorted_slots[0])
|
2020-02-21 22:17:29 -08:00
|
|
|
{
|
2020-11-05 15:35:09 -08:00
|
|
|
/* First, copy the slots to the sorted list, then partition the
|
|
|
|
* sorted list so that all slots without a target slot go to
|
|
|
|
* the end.
|
2020-02-21 22:17:29 -08:00
|
|
|
*/
|
2020-11-05 15:35:09 -08:00
|
|
|
std::copy(slots, slots_end, sorted_slots.begin());
|
|
|
|
auto split_point = std::partition(sorted_slots.begin(), sorted_slots.end(),
|
2020-11-07 08:36:49 -08:00
|
|
|
[](const EffectSlot *slot) noexcept -> bool
|
|
|
|
{ return slot->Target != nullptr; });
|
2020-11-05 15:35:09 -08:00
|
|
|
/* There must be at least one slot without a slot target. */
|
|
|
|
assert(split_point != sorted_slots.end());
|
|
|
|
|
|
|
|
/* Simple case: no more than 1 slot has a target slot. Either
|
|
|
|
* all slots go right to the output, or the remaining one must
|
|
|
|
* target an already-partitioned slot.
|
|
|
|
*/
|
|
|
|
if(split_point - sorted_slots.begin() > 1)
|
|
|
|
{
|
|
|
|
/* At least two slots target other slots. Starting from the
|
|
|
|
* back of the sorted list, continue partitioning the front
|
|
|
|
* of the list given each target until all targets are
|
|
|
|
* accounted for. This ensures all slots without a target
|
|
|
|
* go last, all slots directly targeting those last slots
|
|
|
|
* go second-to-last, all slots directly targeting those
|
|
|
|
* second-last slots go third-to-last, etc.
|
|
|
|
*/
|
|
|
|
auto next_target = sorted_slots.end();
|
|
|
|
do {
|
|
|
|
/* This shouldn't happen, but if there's unsorted slots
|
|
|
|
* left that don't target any sorted slots, they can't
|
|
|
|
* contribute to the output, so leave them.
|
|
|
|
*/
|
|
|
|
if UNLIKELY(next_target == split_point)
|
|
|
|
break;
|
|
|
|
|
|
|
|
--next_target;
|
|
|
|
split_point = std::partition(sorted_slots.begin(), split_point,
|
2020-11-07 08:36:49 -08:00
|
|
|
[next_target](const EffectSlot *slot) noexcept -> bool
|
|
|
|
{ return slot->Target != *next_target; });
|
2020-11-05 15:35:09 -08:00
|
|
|
} while(split_point - sorted_slots.begin() > 1);
|
|
|
|
}
|
2020-02-21 22:17:29 -08:00
|
|
|
}
|
|
|
|
|
2020-11-07 08:36:49 -08:00
|
|
|
for(const EffectSlot *slot : sorted_slots)
|
2020-02-21 22:17:29 -08:00
|
|
|
{
|
2020-11-07 08:36:49 -08:00
|
|
|
EffectState *state{slot->mEffectState};
|
2020-02-21 22:17:29 -08:00
|
|
|
state->process(SamplesToDo, slot->Wet.Buffer, state->mOutTarget);
|
2020-11-05 15:35:09 -08:00
|
|
|
}
|
2020-01-14 10:38:24 -08:00
|
|
|
}
|
2018-12-22 22:14:25 -08:00
|
|
|
|
2020-02-21 22:17:29 -08:00
|
|
|
/* Signal the event handler if there are any events to read. */
|
|
|
|
RingBuffer *ring{ctx->mAsyncEvents.get()};
|
|
|
|
if(ring->readSpace() > 0)
|
|
|
|
ctx->mEventSem.post();
|
2020-01-14 10:38:24 -08:00
|
|
|
}
|
2018-11-23 13:12:48 -08:00
|
|
|
}
|
|
|
|
|
2015-09-18 00:48:43 -07:00
|
|
|
|
2020-06-09 11:52:48 -07:00
|
|
|
void ApplyDistanceComp(const al::span<FloatBufferLine> Samples, const size_t SamplesToDo,
|
2020-12-27 11:30:45 -08:00
|
|
|
const DistanceComp::ChanData *distcomp)
|
2017-05-24 23:21:08 -07:00
|
|
|
{
|
2018-12-23 17:56:01 -08:00
|
|
|
ASSUME(SamplesToDo > 0);
|
|
|
|
|
2019-06-08 01:39:28 -07:00
|
|
|
for(auto &chanbuffer : Samples)
|
2017-05-24 23:21:08 -07:00
|
|
|
{
|
2020-04-08 06:17:04 -07:00
|
|
|
const float gain{distcomp->Gain};
|
2020-06-09 11:52:48 -07:00
|
|
|
const size_t base{distcomp->Length};
|
2020-04-08 06:17:04 -07:00
|
|
|
float *distbuf{al::assume_aligned<16>(distcomp->Buffer)};
|
2019-06-08 01:39:28 -07:00
|
|
|
++distcomp;
|
2017-05-24 23:21:08 -07:00
|
|
|
|
2019-02-17 00:43:10 -08:00
|
|
|
if(base < 1)
|
2017-05-24 23:21:08 -07:00
|
|
|
continue;
|
|
|
|
|
2020-04-08 06:17:04 -07:00
|
|
|
float *inout{al::assume_aligned<16>(chanbuffer.data())};
|
2019-02-17 00:43:10 -08:00
|
|
|
auto inout_end = inout + SamplesToDo;
|
2019-08-04 11:59:14 -07:00
|
|
|
if LIKELY(SamplesToDo >= base)
|
2017-05-24 23:21:08 -07:00
|
|
|
{
|
2019-02-17 00:43:10 -08:00
|
|
|
auto delay_end = std::rotate(inout, inout_end - base, inout_end);
|
|
|
|
std::swap_ranges(inout, delay_end, distbuf);
|
2017-05-24 23:21:08 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-02-17 00:43:10 -08:00
|
|
|
auto delay_start = std::swap_ranges(inout, inout_end, distbuf);
|
|
|
|
std::rotate(distbuf, delay_start, distbuf + base);
|
2017-05-24 23:21:08 -07:00
|
|
|
}
|
2019-02-17 00:43:10 -08:00
|
|
|
std::transform(inout, inout_end, inout, std::bind(std::multiplies<float>{}, _1, gain));
|
2017-05-24 23:21:08 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-15 20:48:21 -08:00
|
|
|
void ApplyDither(const al::span<FloatBufferLine> Samples, uint *dither_seed,
|
2020-06-09 11:52:48 -07:00
|
|
|
const float quant_scale, const size_t SamplesToDo)
|
2017-06-17 01:11:21 -07:00
|
|
|
{
|
2020-06-09 11:52:48 -07:00
|
|
|
ASSUME(SamplesToDo > 0);
|
|
|
|
|
2018-11-23 13:12:48 -08:00
|
|
|
/* Dithering. Generate whitenoise (uniform distribution of random values
|
|
|
|
* between -1 and +1) and add it to the sample values, after scaling up to
|
|
|
|
* the desired quantization depth amd before rounding.
|
2017-06-17 01:11:21 -07:00
|
|
|
*/
|
2020-04-08 06:17:04 -07:00
|
|
|
const float invscale{1.0f / quant_scale};
|
2020-12-15 20:48:21 -08:00
|
|
|
uint seed{*dither_seed};
|
2020-06-09 11:52:48 -07:00
|
|
|
auto dither_sample = [&seed,invscale,quant_scale](const float sample) noexcept -> float
|
2017-06-17 01:11:21 -07:00
|
|
|
{
|
2020-06-09 11:52:48 -07:00
|
|
|
float val{sample * quant_scale};
|
2020-12-15 20:48:21 -08:00
|
|
|
uint rng0{dither_rng(&seed)};
|
|
|
|
uint rng1{dither_rng(&seed)};
|
2020-06-09 11:52:48 -07:00
|
|
|
val += static_cast<float>(rng0*(1.0/UINT_MAX) - rng1*(1.0/UINT_MAX));
|
|
|
|
return fast_roundf(val) * invscale;
|
2018-11-23 13:12:48 -08:00
|
|
|
};
|
2020-06-09 11:52:48 -07:00
|
|
|
for(FloatBufferLine &inout : Samples)
|
|
|
|
std::transform(inout.begin(), inout.begin()+SamplesToDo, inout.begin(), dither_sample);
|
2017-06-17 01:11:21 -07:00
|
|
|
*dither_seed = seed;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
/* Base template left undefined. Should be marked =delete, but Clang 3.8.1
|
|
|
|
* chokes on that given the inline specializations.
|
|
|
|
*/
|
|
|
|
template<typename T>
|
2019-10-13 09:37:07 -07:00
|
|
|
inline T SampleConv(float) noexcept;
|
2018-11-20 02:12:04 -08:00
|
|
|
|
2019-10-13 09:37:07 -07:00
|
|
|
template<> inline float SampleConv(float val) noexcept
|
2017-05-23 00:02:04 -07:00
|
|
|
{ return val; }
|
2019-10-13 09:37:07 -07:00
|
|
|
template<> inline int32_t SampleConv(float val) noexcept
|
2015-09-07 03:27:25 -07:00
|
|
|
{
|
2019-03-25 22:52:15 -07:00
|
|
|
/* Floats have a 23-bit mantissa, plus an implied 1 bit and a sign bit.
|
|
|
|
* This means a normalized float has at most 25 bits of signed precision.
|
|
|
|
* When scaling and clamping for a signed 32-bit integer, these following
|
|
|
|
* values are the best a float can give.
|
2015-09-07 03:27:25 -07:00
|
|
|
*/
|
2019-03-25 22:52:15 -07:00
|
|
|
return fastf2i(clampf(val*2147483648.0f, -2147483648.0f, 2147483520.0f));
|
2017-05-23 00:02:04 -07:00
|
|
|
}
|
2019-10-13 09:37:07 -07:00
|
|
|
template<> inline int16_t SampleConv(float val) noexcept
|
|
|
|
{ return static_cast<int16_t>(fastf2i(clampf(val*32768.0f, -32768.0f, 32767.0f))); }
|
|
|
|
template<> inline int8_t SampleConv(float val) noexcept
|
|
|
|
{ return static_cast<int8_t>(fastf2i(clampf(val*128.0f, -128.0f, 127.0f))); }
|
2017-05-23 00:02:04 -07:00
|
|
|
|
|
|
|
/* Define unsigned output variations. */
|
2019-10-13 09:37:07 -07:00
|
|
|
template<> inline uint32_t SampleConv(float val) noexcept
|
|
|
|
{ return static_cast<uint32_t>(SampleConv<int32_t>(val)) + 2147483648u; }
|
|
|
|
template<> inline uint16_t SampleConv(float val) noexcept
|
|
|
|
{ return static_cast<uint16_t>(SampleConv<int16_t>(val) + 32768); }
|
|
|
|
template<> inline uint8_t SampleConv(float val) noexcept
|
|
|
|
{ return static_cast<uint8_t>(SampleConv<int8_t>(val) + 128); }
|
2018-11-20 02:12:04 -08:00
|
|
|
|
|
|
|
template<DevFmtType T>
|
2019-10-13 09:37:07 -07:00
|
|
|
void Write(const al::span<const FloatBufferLine> InBuffer, void *OutBuffer, const size_t Offset,
|
2020-06-09 21:28:09 -07:00
|
|
|
const size_t SamplesToDo, const size_t FrameStep)
|
2018-11-20 02:12:04 -08:00
|
|
|
{
|
2019-12-21 20:43:46 -08:00
|
|
|
ASSUME(FrameStep > 0);
|
2020-06-09 21:28:09 -07:00
|
|
|
ASSUME(SamplesToDo > 0);
|
2019-06-27 08:49:04 -07:00
|
|
|
|
2021-07-24 18:15:02 -07:00
|
|
|
DevFmtType_t<T> *outbase{static_cast<DevFmtType_t<T>*>(OutBuffer) + Offset*FrameStep};
|
|
|
|
size_t c{0};
|
2020-06-09 21:28:09 -07:00
|
|
|
for(const FloatBufferLine &inbuf : InBuffer)
|
2018-11-20 02:12:04 -08:00
|
|
|
{
|
2020-12-05 02:44:19 -08:00
|
|
|
DevFmtType_t<T> *out{outbase++};
|
2019-12-21 20:43:46 -08:00
|
|
|
auto conv_sample = [FrameStep,&out](const float s) noexcept -> void
|
2019-05-25 08:17:37 -07:00
|
|
|
{
|
2020-12-05 02:44:19 -08:00
|
|
|
*out = SampleConv<DevFmtType_t<T>>(s);
|
2019-12-21 20:43:46 -08:00
|
|
|
out += FrameStep;
|
2019-05-25 08:17:37 -07:00
|
|
|
};
|
2019-05-28 16:22:36 -07:00
|
|
|
std::for_each(inbuf.begin(), inbuf.begin()+SamplesToDo, conv_sample);
|
2021-07-24 18:15:02 -07:00
|
|
|
++c;
|
|
|
|
}
|
|
|
|
if(const size_t extra{FrameStep - c})
|
|
|
|
{
|
|
|
|
const auto silence = SampleConv<DevFmtType_t<T>>(0.0f);
|
|
|
|
for(size_t i{0};i < SamplesToDo;++i)
|
|
|
|
{
|
|
|
|
std::fill_n(outbase, extra, silence);
|
|
|
|
outbase += FrameStep;
|
|
|
|
}
|
2020-06-09 21:28:09 -07:00
|
|
|
}
|
2018-11-20 02:12:04 -08:00
|
|
|
}
|
2010-12-01 21:50:49 -08:00
|
|
|
|
2018-11-20 02:12:04 -08:00
|
|
|
} // namespace
|
2010-12-01 21:50:49 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
uint DeviceBase::renderSamples(const uint numSamples)
|
2010-11-21 02:51:18 -08:00
|
|
|
{
|
2021-06-08 10:52:37 -07:00
|
|
|
const uint samplesToDo{minu(numSamples, BufferLineSize)};
|
2018-11-23 13:12:48 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Clear main mixing buffers. */
|
|
|
|
for(FloatBufferLine &buffer : MixBuffer)
|
|
|
|
buffer.fill(0.0f);
|
2010-11-21 02:51:18 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Increment the mix count at the start (lsb should now be 1). */
|
|
|
|
IncrementRef(MixCount);
|
2015-09-12 23:36:24 -07:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Process and mix each context's sources and effects. */
|
|
|
|
ProcessContexts(this, samplesToDo);
|
2012-10-09 06:19:36 -07:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Increment the clock time. Every second's worth of samples is converted
|
|
|
|
* and added to clock base so that large sample counts don't overflow
|
|
|
|
* during conversion. This also guarantees a stable conversion.
|
|
|
|
*/
|
|
|
|
SamplesDone += samplesToDo;
|
|
|
|
ClockBase += std::chrono::seconds{SamplesDone / Frequency};
|
|
|
|
SamplesDone %= Frequency;
|
|
|
|
|
|
|
|
/* Increment the mix count at the end (lsb should now be 0). */
|
|
|
|
IncrementRef(MixCount);
|
2018-11-23 13:12:48 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Apply any needed post-process for finalizing the Dry mix to the RealOut
|
|
|
|
* (Ambisonic decode, UHJ encode, etc).
|
|
|
|
*/
|
|
|
|
postProcess(samplesToDo);
|
2010-11-21 02:51:18 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Apply compression, limiting sample amplitude if needed or desired. */
|
|
|
|
if(Limiter) Limiter->process(samplesToDo, RealOut.Buffer.data());
|
2010-11-21 02:51:18 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Apply delays and attenuation for mismatched speaker distances. */
|
|
|
|
if(ChannelDelays)
|
|
|
|
ApplyDistanceComp(RealOut.Buffer, samplesToDo, ChannelDelays->mChannels.data());
|
2018-12-24 07:30:01 -08:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
/* Apply dithering. The compressor should have left enough headroom for the
|
|
|
|
* dither noise to not saturate.
|
|
|
|
*/
|
|
|
|
if(DitherDepth > 0.0f)
|
|
|
|
ApplyDither(RealOut.Buffer, &DitherSeed, DitherDepth, samplesToDo);
|
2017-07-31 23:49:48 -07:00
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
return samplesToDo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceBase::renderSamples(const al::span<float*> outBuffers, const uint numSamples)
|
|
|
|
{
|
|
|
|
FPUCtl mixer_mode{};
|
|
|
|
uint total{0};
|
|
|
|
while(const uint todo{numSamples - total})
|
|
|
|
{
|
|
|
|
const uint samplesToDo{renderSamples(todo)};
|
|
|
|
|
|
|
|
auto *srcbuf = RealOut.Buffer.data();
|
|
|
|
for(auto *dstbuf : outBuffers)
|
|
|
|
{
|
|
|
|
std::copy_n(srcbuf->data(), samplesToDo, dstbuf + total);
|
|
|
|
++srcbuf;
|
|
|
|
}
|
|
|
|
|
|
|
|
total += samplesToDo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceBase::renderSamples(void *outBuffer, const uint numSamples, const size_t frameStep)
|
|
|
|
{
|
|
|
|
FPUCtl mixer_mode{};
|
|
|
|
uint total{0};
|
|
|
|
while(const uint todo{numSamples - total})
|
|
|
|
{
|
|
|
|
const uint samplesToDo{renderSamples(todo)};
|
2016-03-09 22:57:38 -08:00
|
|
|
|
2020-08-07 06:17:12 -07:00
|
|
|
if LIKELY(outBuffer)
|
2018-03-01 16:16:37 -08:00
|
|
|
{
|
2018-11-23 13:12:48 -08:00
|
|
|
/* Finally, interleave and convert samples, writing to the device's
|
|
|
|
* output buffer.
|
|
|
|
*/
|
2020-08-07 06:17:12 -07:00
|
|
|
switch(FmtType)
|
2011-08-12 15:42:36 -07:00
|
|
|
{
|
2019-12-21 20:43:46 -08:00
|
|
|
#define HANDLE_WRITE(T) case T: \
|
2021-06-08 10:52:37 -07:00
|
|
|
Write<T>(RealOut.Buffer, outBuffer, total, samplesToDo, frameStep); break;
|
2019-12-21 20:43:46 -08:00
|
|
|
HANDLE_WRITE(DevFmtByte)
|
|
|
|
HANDLE_WRITE(DevFmtUByte)
|
|
|
|
HANDLE_WRITE(DevFmtShort)
|
|
|
|
HANDLE_WRITE(DevFmtUShort)
|
|
|
|
HANDLE_WRITE(DevFmtInt)
|
|
|
|
HANDLE_WRITE(DevFmtUInt)
|
|
|
|
HANDLE_WRITE(DevFmtFloat)
|
2018-05-27 01:03:18 -07:00
|
|
|
#undef HANDLE_WRITE
|
2011-08-12 15:42:36 -07:00
|
|
|
}
|
2010-11-21 02:51:18 -08:00
|
|
|
}
|
|
|
|
|
2021-06-08 10:52:37 -07:00
|
|
|
total += samplesToDo;
|
2010-11-21 02:51:18 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-24 08:28:13 -07:00
|
|
|
void DeviceBase::handleDisconnect(const char *msg, ...)
|
2009-08-26 19:15:17 -07:00
|
|
|
{
|
2022-04-24 23:25:49 -07:00
|
|
|
IncrementRef(MixCount);
|
|
|
|
if(Connected.exchange(false, std::memory_order_acq_rel))
|
|
|
|
{
|
|
|
|
AsyncEvent evt{AsyncEvent::Disconnected};
|
2018-02-03 13:54:42 -08:00
|
|
|
|
2022-04-24 23:25:49 -07:00
|
|
|
va_list args;
|
|
|
|
va_start(args, msg);
|
|
|
|
int msglen{vsnprintf(evt.u.disconnect.msg, sizeof(evt.u.disconnect.msg), msg, args)};
|
|
|
|
va_end(args);
|
2018-02-03 13:54:42 -08:00
|
|
|
|
2022-04-24 23:25:49 -07:00
|
|
|
if(msglen < 0 || static_cast<size_t>(msglen) >= sizeof(evt.u.disconnect.msg))
|
|
|
|
evt.u.disconnect.msg[sizeof(evt.u.disconnect.msg)-1] = 0;
|
2018-02-03 13:54:42 -08:00
|
|
|
|
2022-04-24 23:25:49 -07:00
|
|
|
for(ContextBase *ctx : *mContexts.load())
|
2018-12-25 09:32:38 -08:00
|
|
|
{
|
2022-04-24 23:25:49 -07:00
|
|
|
const uint enabledevt{ctx->mEnabledEvts.load(std::memory_order_acquire)};
|
|
|
|
if((enabledevt&AsyncEvent::Disconnected))
|
2018-12-25 09:32:38 -08:00
|
|
|
{
|
2022-04-24 23:25:49 -07:00
|
|
|
RingBuffer *ring{ctx->mAsyncEvents.get()};
|
|
|
|
auto evt_data = ring->getWriteVector().first;
|
|
|
|
if(evt_data.len > 0)
|
|
|
|
{
|
|
|
|
al::construct_at(reinterpret_cast<AsyncEvent*>(evt_data.buf), evt);
|
|
|
|
ring->writeAdvance(1);
|
|
|
|
ctx->mEventSem.post();
|
|
|
|
}
|
2018-12-25 09:32:38 -08:00
|
|
|
}
|
2018-02-03 01:07:06 -08:00
|
|
|
|
2022-04-24 23:25:49 -07:00
|
|
|
if(!ctx->mStopVoicesOnDisconnect)
|
|
|
|
{
|
|
|
|
ProcessVoiceChanges(ctx);
|
|
|
|
continue;
|
|
|
|
}
|
2021-04-26 20:25:24 -07:00
|
|
|
|
2022-04-24 23:25:49 -07:00
|
|
|
auto voicelist = ctx->getVoicesSpanAcquired();
|
|
|
|
auto stop_voice = [](Voice *voice) -> void
|
|
|
|
{
|
|
|
|
voice->mCurrentBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
voice->mLoopBuffer.store(nullptr, std::memory_order_relaxed);
|
|
|
|
voice->mSourceID.store(0u, std::memory_order_relaxed);
|
|
|
|
voice->mPlayState.store(Voice::Stopped, std::memory_order_release);
|
|
|
|
};
|
|
|
|
std::for_each(voicelist.begin(), voicelist.end(), stop_voice);
|
|
|
|
}
|
2009-08-26 19:15:17 -07:00
|
|
|
}
|
2020-08-07 06:17:12 -07:00
|
|
|
IncrementRef(MixCount);
|
2009-08-26 19:15:17 -07:00
|
|
|
}
|