2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2021-05-14 10:15:42 +08:00
|
|
|
#include <stdexcept>
|
|
|
|
|
2021-04-28 12:43:51 +08:00
|
|
|
#include "AL/al.h"
|
|
|
|
#include "AL/efx.h"
|
|
|
|
|
2021-05-14 10:15:42 +08:00
|
|
|
#include "alc/effects/base.h"
|
2021-04-28 12:43:51 +08:00
|
|
|
#include "aloptional.h"
|
|
|
|
#include "effects.h"
|
|
|
|
|
2022-04-25 12:02:45 +08:00
|
|
|
#ifdef ALSOFT_EAX
|
|
|
|
#include <cassert>
|
|
|
|
|
|
|
|
#include "alnumeric.h"
|
|
|
|
|
|
|
|
#include "al/eax_exception.h"
|
|
|
|
#include "al/eax_utils.h"
|
|
|
|
#endif // ALSOFT_EAX
|
|
|
|
|
2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
al::optional<VMorpherPhenome> PhenomeFromEnum(ALenum val)
|
|
|
|
{
|
|
|
|
#define HANDLE_PHENOME(x) case AL_VOCAL_MORPHER_PHONEME_ ## x: \
|
|
|
|
return al::make_optional(VMorpherPhenome::x)
|
|
|
|
switch(val)
|
|
|
|
{
|
|
|
|
HANDLE_PHENOME(A);
|
|
|
|
HANDLE_PHENOME(E);
|
|
|
|
HANDLE_PHENOME(I);
|
|
|
|
HANDLE_PHENOME(O);
|
|
|
|
HANDLE_PHENOME(U);
|
|
|
|
HANDLE_PHENOME(AA);
|
|
|
|
HANDLE_PHENOME(AE);
|
|
|
|
HANDLE_PHENOME(AH);
|
|
|
|
HANDLE_PHENOME(AO);
|
|
|
|
HANDLE_PHENOME(EH);
|
|
|
|
HANDLE_PHENOME(ER);
|
|
|
|
HANDLE_PHENOME(IH);
|
|
|
|
HANDLE_PHENOME(IY);
|
|
|
|
HANDLE_PHENOME(UH);
|
|
|
|
HANDLE_PHENOME(UW);
|
|
|
|
HANDLE_PHENOME(B);
|
|
|
|
HANDLE_PHENOME(D);
|
|
|
|
HANDLE_PHENOME(F);
|
|
|
|
HANDLE_PHENOME(G);
|
|
|
|
HANDLE_PHENOME(J);
|
|
|
|
HANDLE_PHENOME(K);
|
|
|
|
HANDLE_PHENOME(L);
|
|
|
|
HANDLE_PHENOME(M);
|
|
|
|
HANDLE_PHENOME(N);
|
|
|
|
HANDLE_PHENOME(P);
|
|
|
|
HANDLE_PHENOME(R);
|
|
|
|
HANDLE_PHENOME(S);
|
|
|
|
HANDLE_PHENOME(T);
|
|
|
|
HANDLE_PHENOME(V);
|
|
|
|
HANDLE_PHENOME(Z);
|
|
|
|
}
|
|
|
|
return al::nullopt;
|
|
|
|
#undef HANDLE_PHENOME
|
|
|
|
}
|
|
|
|
ALenum EnumFromPhenome(VMorpherPhenome phenome)
|
|
|
|
{
|
|
|
|
#define HANDLE_PHENOME(x) case VMorpherPhenome::x: return AL_VOCAL_MORPHER_PHONEME_ ## x
|
|
|
|
switch(phenome)
|
|
|
|
{
|
|
|
|
HANDLE_PHENOME(A);
|
|
|
|
HANDLE_PHENOME(E);
|
|
|
|
HANDLE_PHENOME(I);
|
|
|
|
HANDLE_PHENOME(O);
|
|
|
|
HANDLE_PHENOME(U);
|
|
|
|
HANDLE_PHENOME(AA);
|
|
|
|
HANDLE_PHENOME(AE);
|
|
|
|
HANDLE_PHENOME(AH);
|
|
|
|
HANDLE_PHENOME(AO);
|
|
|
|
HANDLE_PHENOME(EH);
|
|
|
|
HANDLE_PHENOME(ER);
|
|
|
|
HANDLE_PHENOME(IH);
|
|
|
|
HANDLE_PHENOME(IY);
|
|
|
|
HANDLE_PHENOME(UH);
|
|
|
|
HANDLE_PHENOME(UW);
|
|
|
|
HANDLE_PHENOME(B);
|
|
|
|
HANDLE_PHENOME(D);
|
|
|
|
HANDLE_PHENOME(F);
|
|
|
|
HANDLE_PHENOME(G);
|
|
|
|
HANDLE_PHENOME(J);
|
|
|
|
HANDLE_PHENOME(K);
|
|
|
|
HANDLE_PHENOME(L);
|
|
|
|
HANDLE_PHENOME(M);
|
|
|
|
HANDLE_PHENOME(N);
|
|
|
|
HANDLE_PHENOME(P);
|
|
|
|
HANDLE_PHENOME(R);
|
|
|
|
HANDLE_PHENOME(S);
|
|
|
|
HANDLE_PHENOME(T);
|
|
|
|
HANDLE_PHENOME(V);
|
|
|
|
HANDLE_PHENOME(Z);
|
|
|
|
}
|
|
|
|
throw std::runtime_error{"Invalid phenome: "+std::to_string(static_cast<int>(phenome))};
|
|
|
|
#undef HANDLE_PHENOME
|
|
|
|
}
|
|
|
|
|
|
|
|
al::optional<VMorpherWaveform> WaveformFromEmum(ALenum value)
|
|
|
|
{
|
|
|
|
switch(value)
|
|
|
|
{
|
|
|
|
case AL_VOCAL_MORPHER_WAVEFORM_SINUSOID: return al::make_optional(VMorpherWaveform::Sinusoid);
|
|
|
|
case AL_VOCAL_MORPHER_WAVEFORM_TRIANGLE: return al::make_optional(VMorpherWaveform::Triangle);
|
|
|
|
case AL_VOCAL_MORPHER_WAVEFORM_SAWTOOTH: return al::make_optional(VMorpherWaveform::Sawtooth);
|
|
|
|
}
|
|
|
|
return al::nullopt;
|
|
|
|
}
|
|
|
|
ALenum EnumFromWaveform(VMorpherWaveform type)
|
|
|
|
{
|
|
|
|
switch(type)
|
|
|
|
{
|
|
|
|
case VMorpherWaveform::Sinusoid: return AL_VOCAL_MORPHER_WAVEFORM_SINUSOID;
|
|
|
|
case VMorpherWaveform::Triangle: return AL_VOCAL_MORPHER_WAVEFORM_TRIANGLE;
|
|
|
|
case VMorpherWaveform::Sawtooth: return AL_VOCAL_MORPHER_WAVEFORM_SAWTOOTH;
|
|
|
|
}
|
|
|
|
throw std::runtime_error{"Invalid vocal morpher waveform: " +
|
|
|
|
std::to_string(static_cast<int>(type))};
|
|
|
|
}
|
|
|
|
|
|
|
|
void Vmorpher_setParami(EffectProps *props, ALenum param, int val)
|
|
|
|
{
|
|
|
|
switch(param)
|
|
|
|
{
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEA:
|
|
|
|
if(auto phenomeopt = PhenomeFromEnum(val))
|
|
|
|
props->Vmorpher.PhonemeA = *phenomeopt;
|
|
|
|
else
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher phoneme-a out of range: 0x%04x", val};
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEA_COARSE_TUNING:
|
|
|
|
if(!(val >= AL_VOCAL_MORPHER_MIN_PHONEMEA_COARSE_TUNING && val <= AL_VOCAL_MORPHER_MAX_PHONEMEA_COARSE_TUNING))
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher phoneme-a coarse tuning out of range"};
|
|
|
|
props->Vmorpher.PhonemeACoarseTuning = val;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEB:
|
|
|
|
if(auto phenomeopt = PhenomeFromEnum(val))
|
|
|
|
props->Vmorpher.PhonemeB = *phenomeopt;
|
|
|
|
else
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher phoneme-b out of range: 0x%04x", val};
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEB_COARSE_TUNING:
|
|
|
|
if(!(val >= AL_VOCAL_MORPHER_MIN_PHONEMEB_COARSE_TUNING && val <= AL_VOCAL_MORPHER_MAX_PHONEMEB_COARSE_TUNING))
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher phoneme-b coarse tuning out of range"};
|
|
|
|
props->Vmorpher.PhonemeBCoarseTuning = val;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_WAVEFORM:
|
|
|
|
if(auto formopt = WaveformFromEmum(val))
|
|
|
|
props->Vmorpher.Waveform = *formopt;
|
|
|
|
else
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher waveform out of range: 0x%04x", val};
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher integer property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Vmorpher_setParamiv(EffectProps*, ALenum param, const int*)
|
|
|
|
{
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher integer-vector property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
void Vmorpher_setParamf(EffectProps *props, ALenum param, float val)
|
|
|
|
{
|
|
|
|
switch(param)
|
|
|
|
{
|
|
|
|
case AL_VOCAL_MORPHER_RATE:
|
|
|
|
if(!(val >= AL_VOCAL_MORPHER_MIN_RATE && val <= AL_VOCAL_MORPHER_MAX_RATE))
|
|
|
|
throw effect_exception{AL_INVALID_VALUE, "Vocal morpher rate out of range"};
|
|
|
|
props->Vmorpher.Rate = val;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher float property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Vmorpher_setParamfv(EffectProps *props, ALenum param, const float *vals)
|
|
|
|
{ Vmorpher_setParamf(props, param, vals[0]); }
|
|
|
|
|
|
|
|
void Vmorpher_getParami(const EffectProps *props, ALenum param, int* val)
|
|
|
|
{
|
|
|
|
switch(param)
|
|
|
|
{
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEA:
|
|
|
|
*val = EnumFromPhenome(props->Vmorpher.PhonemeA);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEA_COARSE_TUNING:
|
|
|
|
*val = props->Vmorpher.PhonemeACoarseTuning;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEB:
|
|
|
|
*val = EnumFromPhenome(props->Vmorpher.PhonemeB);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_PHONEMEB_COARSE_TUNING:
|
|
|
|
*val = props->Vmorpher.PhonemeBCoarseTuning;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_VOCAL_MORPHER_WAVEFORM:
|
|
|
|
*val = EnumFromWaveform(props->Vmorpher.Waveform);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher integer property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Vmorpher_getParamiv(const EffectProps*, ALenum param, int*)
|
|
|
|
{
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher integer-vector property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
void Vmorpher_getParamf(const EffectProps *props, ALenum param, float *val)
|
|
|
|
{
|
|
|
|
switch(param)
|
|
|
|
{
|
|
|
|
case AL_VOCAL_MORPHER_RATE:
|
|
|
|
*val = props->Vmorpher.Rate;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw effect_exception{AL_INVALID_ENUM, "Invalid vocal morpher float property 0x%04x",
|
|
|
|
param};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Vmorpher_getParamfv(const EffectProps *props, ALenum param, float *vals)
|
|
|
|
{ Vmorpher_getParamf(props, param, vals); }
|
|
|
|
|
|
|
|
EffectProps genDefaultProps() noexcept
|
|
|
|
{
|
|
|
|
EffectProps props{};
|
|
|
|
props.Vmorpher.Rate = AL_VOCAL_MORPHER_DEFAULT_RATE;
|
|
|
|
props.Vmorpher.PhonemeA = *PhenomeFromEnum(AL_VOCAL_MORPHER_DEFAULT_PHONEMEA);
|
|
|
|
props.Vmorpher.PhonemeB = *PhenomeFromEnum(AL_VOCAL_MORPHER_DEFAULT_PHONEMEB);
|
|
|
|
props.Vmorpher.PhonemeACoarseTuning = AL_VOCAL_MORPHER_DEFAULT_PHONEMEA_COARSE_TUNING;
|
|
|
|
props.Vmorpher.PhonemeBCoarseTuning = AL_VOCAL_MORPHER_DEFAULT_PHONEMEB_COARSE_TUNING;
|
|
|
|
props.Vmorpher.Waveform = *WaveformFromEmum(AL_VOCAL_MORPHER_DEFAULT_WAVEFORM);
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
DEFINE_ALEFFECT_VTABLE(Vmorpher);
|
|
|
|
|
|
|
|
const EffectProps VmorpherEffectProps{genDefaultProps()};
|
2022-04-25 12:02:45 +08:00
|
|
|
|
|
|
|
#ifdef ALSOFT_EAX
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using EaxVocalMorpherEffectDirtyFlagsValue = std::uint_least8_t;
|
|
|
|
|
|
|
|
struct EaxVocalMorpherEffectDirtyFlags
|
|
|
|
{
|
|
|
|
using EaxIsBitFieldStruct = bool;
|
|
|
|
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue ulPhonemeA : 1;
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue lPhonemeACoarseTuning : 1;
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue ulPhonemeB : 1;
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue lPhonemeBCoarseTuning : 1;
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue ulWaveform : 1;
|
|
|
|
EaxVocalMorpherEffectDirtyFlagsValue flRate : 1;
|
|
|
|
}; // EaxPitchShifterEffectDirtyFlags
|
|
|
|
|
|
|
|
|
|
|
|
class EaxVocalMorpherEffect final :
|
|
|
|
public EaxEffect
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
EaxVocalMorpherEffect();
|
|
|
|
|
|
|
|
void dispatch(const EaxEaxCall& eax_call) override;
|
|
|
|
|
|
|
|
// [[nodiscard]]
|
|
|
|
bool apply_deferred() override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
EAXVOCALMORPHERPROPERTIES eax_{};
|
|
|
|
EAXVOCALMORPHERPROPERTIES eax_d_{};
|
|
|
|
EaxVocalMorpherEffectDirtyFlags eax_dirty_flags_{};
|
|
|
|
|
|
|
|
void set_eax_defaults();
|
|
|
|
|
|
|
|
void set_efx_phoneme_a();
|
|
|
|
void set_efx_phoneme_a_coarse_tuning();
|
|
|
|
void set_efx_phoneme_b();
|
|
|
|
void set_efx_phoneme_b_coarse_tuning();
|
|
|
|
void set_efx_waveform();
|
|
|
|
void set_efx_rate();
|
|
|
|
void set_efx_defaults();
|
|
|
|
|
|
|
|
void get(const EaxEaxCall& eax_call);
|
|
|
|
|
|
|
|
void validate_phoneme_a(unsigned long ulPhonemeA);
|
|
|
|
void validate_phoneme_a_coarse_tuning(long lPhonemeACoarseTuning);
|
|
|
|
void validate_phoneme_b(unsigned long ulPhonemeB);
|
|
|
|
void validate_phoneme_b_coarse_tuning(long lPhonemeBCoarseTuning);
|
|
|
|
void validate_waveform(unsigned long ulWaveform);
|
|
|
|
void validate_rate(float flRate);
|
|
|
|
void validate_all(const EAXVOCALMORPHERPROPERTIES& all);
|
|
|
|
|
|
|
|
void defer_phoneme_a(unsigned long ulPhonemeA);
|
|
|
|
void defer_phoneme_a_coarse_tuning(long lPhonemeACoarseTuning);
|
|
|
|
void defer_phoneme_b(unsigned long ulPhonemeB);
|
|
|
|
void defer_phoneme_b_coarse_tuning(long lPhonemeBCoarseTuning);
|
|
|
|
void defer_waveform(unsigned long ulWaveform);
|
|
|
|
void defer_rate(float flRate);
|
|
|
|
void defer_all(const EAXVOCALMORPHERPROPERTIES& all);
|
|
|
|
|
|
|
|
void defer_phoneme_a(const EaxEaxCall& eax_call);
|
|
|
|
void defer_phoneme_a_coarse_tuning(const EaxEaxCall& eax_call);
|
|
|
|
void defer_phoneme_b(const EaxEaxCall& eax_call);
|
|
|
|
void defer_phoneme_b_coarse_tuning(const EaxEaxCall& eax_call);
|
|
|
|
void defer_waveform(const EaxEaxCall& eax_call);
|
|
|
|
void defer_rate(const EaxEaxCall& eax_call);
|
|
|
|
void defer_all(const EaxEaxCall& eax_call);
|
|
|
|
|
|
|
|
void set(const EaxEaxCall& eax_call);
|
|
|
|
}; // EaxVocalMorpherEffect
|
|
|
|
|
|
|
|
|
|
|
|
class EaxVocalMorpherEffectException :
|
|
|
|
public EaxException
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit EaxVocalMorpherEffectException(
|
|
|
|
const char* message)
|
|
|
|
:
|
|
|
|
EaxException{"EAX_VOCAL_MORPHER_EFFECT", message}
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}; // EaxVocalMorpherEffectException
|
|
|
|
|
|
|
|
|
|
|
|
EaxVocalMorpherEffect::EaxVocalMorpherEffect()
|
|
|
|
: EaxEffect{AL_EFFECT_VOCAL_MORPHER}
|
|
|
|
{
|
|
|
|
set_eax_defaults();
|
|
|
|
set_efx_defaults();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::dispatch(const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
eax_call.is_get() ? get(eax_call) : set(eax_call);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_eax_defaults()
|
|
|
|
{
|
|
|
|
eax_.ulPhonemeA = EAXVOCALMORPHER_DEFAULTPHONEMEA;
|
|
|
|
eax_.lPhonemeACoarseTuning = EAXVOCALMORPHER_DEFAULTPHONEMEACOARSETUNING;
|
|
|
|
eax_.ulPhonemeB = EAXVOCALMORPHER_DEFAULTPHONEMEB;
|
|
|
|
eax_.lPhonemeBCoarseTuning = EAXVOCALMORPHER_DEFAULTPHONEMEBCOARSETUNING;
|
|
|
|
eax_.ulWaveform = EAXVOCALMORPHER_DEFAULTWAVEFORM;
|
|
|
|
eax_.flRate = EAXVOCALMORPHER_DEFAULTRATE;
|
|
|
|
|
|
|
|
eax_d_ = eax_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_phoneme_a()
|
|
|
|
{
|
|
|
|
const auto phoneme_a = clamp(
|
|
|
|
static_cast<ALint>(eax_.ulPhonemeA),
|
|
|
|
AL_VOCAL_MORPHER_MIN_PHONEMEA,
|
|
|
|
AL_VOCAL_MORPHER_MAX_PHONEMEA);
|
|
|
|
|
|
|
|
const auto efx_phoneme_a = PhenomeFromEnum(phoneme_a);
|
|
|
|
assert(efx_phoneme_a.has_value());
|
|
|
|
al_effect_props_.Vmorpher.PhonemeA = *efx_phoneme_a;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_phoneme_a_coarse_tuning()
|
|
|
|
{
|
|
|
|
const auto phoneme_a_coarse_tuning = clamp(
|
|
|
|
static_cast<ALint>(eax_.lPhonemeACoarseTuning),
|
|
|
|
AL_VOCAL_MORPHER_MIN_PHONEMEA_COARSE_TUNING,
|
|
|
|
AL_VOCAL_MORPHER_MAX_PHONEMEA_COARSE_TUNING);
|
|
|
|
|
|
|
|
al_effect_props_.Vmorpher.PhonemeACoarseTuning = phoneme_a_coarse_tuning;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_phoneme_b()
|
|
|
|
{
|
|
|
|
const auto phoneme_b = clamp(
|
|
|
|
static_cast<ALint>(eax_.ulPhonemeB),
|
|
|
|
AL_VOCAL_MORPHER_MIN_PHONEMEB,
|
|
|
|
AL_VOCAL_MORPHER_MAX_PHONEMEB);
|
|
|
|
|
|
|
|
const auto efx_phoneme_b = PhenomeFromEnum(phoneme_b);
|
|
|
|
assert(efx_phoneme_b.has_value());
|
|
|
|
al_effect_props_.Vmorpher.PhonemeB = *efx_phoneme_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_phoneme_b_coarse_tuning()
|
|
|
|
{
|
|
|
|
const auto phoneme_b_coarse_tuning = clamp(
|
|
|
|
static_cast<ALint>(eax_.lPhonemeBCoarseTuning),
|
|
|
|
AL_VOCAL_MORPHER_MIN_PHONEMEB_COARSE_TUNING,
|
|
|
|
AL_VOCAL_MORPHER_MAX_PHONEMEB_COARSE_TUNING);
|
|
|
|
|
|
|
|
al_effect_props_.Vmorpher.PhonemeBCoarseTuning = phoneme_b_coarse_tuning;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_waveform()
|
|
|
|
{
|
|
|
|
const auto waveform = clamp(
|
|
|
|
static_cast<ALint>(eax_.ulWaveform),
|
|
|
|
AL_VOCAL_MORPHER_MIN_WAVEFORM,
|
|
|
|
AL_VOCAL_MORPHER_MAX_WAVEFORM);
|
|
|
|
|
|
|
|
const auto wfx_waveform = WaveformFromEmum(waveform);
|
|
|
|
assert(wfx_waveform.has_value());
|
|
|
|
al_effect_props_.Vmorpher.Waveform = *wfx_waveform;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_rate()
|
|
|
|
{
|
|
|
|
const auto rate = clamp(
|
|
|
|
eax_.flRate,
|
|
|
|
AL_VOCAL_MORPHER_MIN_RATE,
|
|
|
|
AL_VOCAL_MORPHER_MAX_RATE);
|
|
|
|
|
|
|
|
al_effect_props_.Vmorpher.Rate = rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set_efx_defaults()
|
|
|
|
{
|
|
|
|
set_efx_phoneme_a();
|
|
|
|
set_efx_phoneme_a_coarse_tuning();
|
|
|
|
set_efx_phoneme_b();
|
|
|
|
set_efx_phoneme_b_coarse_tuning();
|
|
|
|
set_efx_waveform();
|
|
|
|
set_efx_rate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::get(const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
switch(eax_call.get_property_id())
|
|
|
|
{
|
|
|
|
case EAXVOCALMORPHER_NONE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_ALLPARAMETERS:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEA:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.ulPhonemeA);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEACOARSETUNING:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.lPhonemeACoarseTuning);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEB:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.ulPhonemeB);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEBCOARSETUNING:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.lPhonemeBCoarseTuning);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_WAVEFORM:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.ulWaveform);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_RATE:
|
|
|
|
eax_call.set_value<EaxVocalMorpherEffectException>(eax_.flRate);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw EaxVocalMorpherEffectException{"Unsupported property id."};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_phoneme_a(
|
|
|
|
unsigned long ulPhonemeA)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Phoneme A",
|
|
|
|
ulPhonemeA,
|
|
|
|
EAXVOCALMORPHER_MINPHONEMEA,
|
|
|
|
EAXVOCALMORPHER_MAXPHONEMEA);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_phoneme_a_coarse_tuning(
|
|
|
|
long lPhonemeACoarseTuning)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Phoneme A Coarse Tuning",
|
|
|
|
lPhonemeACoarseTuning,
|
|
|
|
EAXVOCALMORPHER_MINPHONEMEACOARSETUNING,
|
|
|
|
EAXVOCALMORPHER_MAXPHONEMEACOARSETUNING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_phoneme_b(
|
|
|
|
unsigned long ulPhonemeB)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Phoneme B",
|
|
|
|
ulPhonemeB,
|
|
|
|
EAXVOCALMORPHER_MINPHONEMEB,
|
|
|
|
EAXVOCALMORPHER_MAXPHONEMEB);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_phoneme_b_coarse_tuning(
|
|
|
|
long lPhonemeBCoarseTuning)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Phoneme B Coarse Tuning",
|
|
|
|
lPhonemeBCoarseTuning,
|
|
|
|
EAXVOCALMORPHER_MINPHONEMEBCOARSETUNING,
|
|
|
|
EAXVOCALMORPHER_MAXPHONEMEBCOARSETUNING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_waveform(
|
|
|
|
unsigned long ulWaveform)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Waveform",
|
|
|
|
ulWaveform,
|
|
|
|
EAXVOCALMORPHER_MINWAVEFORM,
|
|
|
|
EAXVOCALMORPHER_MAXWAVEFORM);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_rate(
|
|
|
|
float flRate)
|
|
|
|
{
|
|
|
|
eax_validate_range<EaxVocalMorpherEffectException>(
|
|
|
|
"Rate",
|
|
|
|
flRate,
|
|
|
|
EAXVOCALMORPHER_MINRATE,
|
|
|
|
EAXVOCALMORPHER_MAXRATE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::validate_all(
|
|
|
|
const EAXVOCALMORPHERPROPERTIES& all)
|
|
|
|
{
|
|
|
|
validate_phoneme_a(all.ulPhonemeA);
|
|
|
|
validate_phoneme_a_coarse_tuning(all.lPhonemeACoarseTuning);
|
|
|
|
validate_phoneme_b(all.ulPhonemeB);
|
|
|
|
validate_phoneme_b_coarse_tuning(all.lPhonemeBCoarseTuning);
|
|
|
|
validate_waveform(all.ulWaveform);
|
|
|
|
validate_rate(all.flRate);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_a(
|
|
|
|
unsigned long ulPhonemeA)
|
|
|
|
{
|
|
|
|
eax_d_.ulPhonemeA = ulPhonemeA;
|
|
|
|
eax_dirty_flags_.ulPhonemeA = (eax_.ulPhonemeA != eax_d_.ulPhonemeA);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_a_coarse_tuning(
|
|
|
|
long lPhonemeACoarseTuning)
|
|
|
|
{
|
|
|
|
eax_d_.lPhonemeACoarseTuning = lPhonemeACoarseTuning;
|
|
|
|
eax_dirty_flags_.lPhonemeACoarseTuning = (eax_.lPhonemeACoarseTuning != eax_d_.lPhonemeACoarseTuning);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_b(
|
|
|
|
unsigned long ulPhonemeB)
|
|
|
|
{
|
|
|
|
eax_d_.ulPhonemeB = ulPhonemeB;
|
|
|
|
eax_dirty_flags_.ulPhonemeB = (eax_.ulPhonemeB != eax_d_.ulPhonemeB);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_b_coarse_tuning(
|
|
|
|
long lPhonemeBCoarseTuning)
|
|
|
|
{
|
|
|
|
eax_d_.lPhonemeBCoarseTuning = lPhonemeBCoarseTuning;
|
|
|
|
eax_dirty_flags_.lPhonemeBCoarseTuning = (eax_.lPhonemeBCoarseTuning != eax_d_.lPhonemeBCoarseTuning);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_waveform(
|
|
|
|
unsigned long ulWaveform)
|
|
|
|
{
|
|
|
|
eax_d_.ulWaveform = ulWaveform;
|
|
|
|
eax_dirty_flags_.ulWaveform = (eax_.ulWaveform != eax_d_.ulWaveform);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_rate(
|
|
|
|
float flRate)
|
|
|
|
{
|
|
|
|
eax_d_.flRate = flRate;
|
|
|
|
eax_dirty_flags_.flRate = (eax_.flRate != eax_d_.flRate);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_all(
|
|
|
|
const EAXVOCALMORPHERPROPERTIES& all)
|
|
|
|
{
|
|
|
|
defer_phoneme_a(all.ulPhonemeA);
|
|
|
|
defer_phoneme_a_coarse_tuning(all.lPhonemeACoarseTuning);
|
|
|
|
defer_phoneme_b(all.ulPhonemeB);
|
|
|
|
defer_phoneme_b_coarse_tuning(all.lPhonemeBCoarseTuning);
|
|
|
|
defer_waveform(all.ulWaveform);
|
|
|
|
defer_rate(all.flRate);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_a(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& phoneme_a = eax_call.get_value<EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::ulPhonemeA)>();
|
|
|
|
|
|
|
|
validate_phoneme_a(phoneme_a);
|
|
|
|
defer_phoneme_a(phoneme_a);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_a_coarse_tuning(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& phoneme_a_coarse_tuning = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::lPhonemeACoarseTuning)
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_phoneme_a_coarse_tuning(phoneme_a_coarse_tuning);
|
|
|
|
defer_phoneme_a_coarse_tuning(phoneme_a_coarse_tuning);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_b(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& phoneme_b = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::ulPhonemeB)
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_phoneme_b(phoneme_b);
|
|
|
|
defer_phoneme_b(phoneme_b);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_phoneme_b_coarse_tuning(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& phoneme_b_coarse_tuning = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::lPhonemeBCoarseTuning)
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_phoneme_b_coarse_tuning(phoneme_b_coarse_tuning);
|
|
|
|
defer_phoneme_b_coarse_tuning(phoneme_b_coarse_tuning);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_waveform(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& waveform = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::ulWaveform)
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_waveform(waveform);
|
|
|
|
defer_waveform(waveform);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_rate(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& rate = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const decltype(EAXVOCALMORPHERPROPERTIES::flRate)
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_rate(rate);
|
|
|
|
defer_rate(rate);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::defer_all(
|
|
|
|
const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
const auto& all = eax_call.get_value<
|
|
|
|
EaxVocalMorpherEffectException,
|
|
|
|
const EAXVOCALMORPHERPROPERTIES
|
|
|
|
>();
|
|
|
|
|
|
|
|
validate_all(all);
|
|
|
|
defer_all(all);
|
|
|
|
}
|
|
|
|
|
|
|
|
// [[nodiscard]]
|
|
|
|
bool EaxVocalMorpherEffect::apply_deferred()
|
|
|
|
{
|
|
|
|
if (eax_dirty_flags_ == EaxVocalMorpherEffectDirtyFlags{})
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
eax_ = eax_d_;
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.ulPhonemeA)
|
|
|
|
{
|
|
|
|
set_efx_phoneme_a();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.lPhonemeACoarseTuning)
|
|
|
|
{
|
|
|
|
set_efx_phoneme_a_coarse_tuning();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.ulPhonemeB)
|
|
|
|
{
|
|
|
|
set_efx_phoneme_b();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.lPhonemeBCoarseTuning)
|
|
|
|
{
|
|
|
|
set_efx_phoneme_b_coarse_tuning();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.ulWaveform)
|
|
|
|
{
|
|
|
|
set_efx_waveform();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eax_dirty_flags_.flRate)
|
|
|
|
{
|
|
|
|
set_efx_rate();
|
|
|
|
}
|
|
|
|
|
|
|
|
eax_dirty_flags_ = EaxVocalMorpherEffectDirtyFlags{};
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EaxVocalMorpherEffect::set(const EaxEaxCall& eax_call)
|
|
|
|
{
|
|
|
|
switch(eax_call.get_property_id())
|
|
|
|
{
|
|
|
|
case EAXVOCALMORPHER_NONE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_ALLPARAMETERS:
|
|
|
|
defer_all(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEA:
|
|
|
|
defer_phoneme_a(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEACOARSETUNING:
|
|
|
|
defer_phoneme_a_coarse_tuning(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEB:
|
|
|
|
defer_phoneme_b(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_PHONEMEBCOARSETUNING:
|
|
|
|
defer_phoneme_b_coarse_tuning(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_WAVEFORM:
|
|
|
|
defer_waveform(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAXVOCALMORPHER_RATE:
|
|
|
|
defer_rate(eax_call);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw EaxVocalMorpherEffectException{"Unsupported property id."};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
EaxEffectUPtr eax_create_eax_vocal_morpher_effect()
|
|
|
|
{
|
|
|
|
return std::make_unique<EaxVocalMorpherEffect>();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // ALSOFT_EAX
|