diff options
author | kcat <[email protected]> | 2019-07-18 15:42:45 -0700 |
---|---|---|
committer | GitHub <[email protected]> | 2019-07-18 15:42:45 -0700 |
commit | ec855215db20e5fb86fd675580f115dbce50b2e1 (patch) | |
tree | 0a67e052a5180c35743c5cd7059f98b2cc71d977 | |
parent | 2b21a08f89bbf5ccff935ada6b5bbe42501f18c4 (diff) | |
parent | 31055f48d68820b6db53917c60aed99a29856891 (diff) |
Merge pull request #318 from Lopuska/pitchshift_for_vmorpher
pitch shift for formant filters
-rw-r--r-- | Alc/effects/vmorpher.cpp | 161 |
1 files changed, 82 insertions, 79 deletions
diff --git a/Alc/effects/vmorpher.cpp b/Alc/effects/vmorpher.cpp index b6b13f0d..5fb7a269 100644 --- a/Alc/effects/vmorpher.cpp +++ b/Alc/effects/vmorpher.cpp @@ -30,13 +30,16 @@ #include "alAuxEffectSlot.h" #include "alError.h" #include "alu.h" -#include "vecmat.h" namespace { #define MAX_UPDATE_SAMPLES 128 -#define Q_FACTOR 5.0f -#define NUM_FORMANTS 4 +#define NUM_FORMANTS 4 +#define NUM_FILTERS 2 +#define Q_FACTOR 5.0f + +#define VOWEL_A_INDEX 0 +#define VOWEL_B_INDEX 1 #define WAVEFORM_FRACBITS 24 #define WAVEFORM_FRACONE (1<<WAVEFORM_FRACBITS) @@ -82,14 +85,14 @@ struct FormantFilter /* A state variable filter from a topology-preserving transform. * Based on a talk given by Ivan Cohen: https://www.youtube.com/watch?v=esjHXGPyrhg */ - const float g = std::tan(al::MathDefs<float>::Pi() * f0norm); - const float h = 1.0f / (1 + (g / Q_FACTOR) + (g * g)); + const ALfloat g = std::tan(al::MathDefs<float>::Pi() * f0norm); + const ALfloat h = 1.0f / (1 + (g / Q_FACTOR) + (g * g)); for (ALsizei i{0};i < numInput;i++) { - const float H = h * (samplesIn[i] - (1.0f / Q_FACTOR + g) * s1 - s2); - const float B = g * H + s1; - const float L = g * B + s2; + const ALfloat H = h * (samplesIn[i] - (1.0f / Q_FACTOR + g) * s1 - s2); + const ALfloat B = g * H + s1; + const ALfloat L = g * B + s2; s1 = g * H + B; s2 = g * B + L; @@ -105,19 +108,17 @@ struct FormantFilter s2 = 0.0f; } - float f0norm; - float fGain; - float s1; - float s2; + ALfloat f0norm; + ALfloat fGain; + ALfloat s1; + ALfloat s2; }; struct VmorpherState final : public EffectState { struct { - struct { - /* Effect parameters */ - FormantFilter Formants[NUM_FORMANTS]; - } Filters[2]; + /* Effect parameters */ + FormantFilter Formants[NUM_FILTERS][NUM_FORMANTS]; /* Effect gains for each channel */ ALfloat CurrentGains[MAX_OUTPUT_CHANNELS]{}; @@ -144,9 +145,9 @@ ALboolean VmorpherState::deviceUpdate(const ALCdevice* /*device*/) { for(auto &e : mChans) { - std::for_each(std::begin(e.Filters[0].Formants), std::end(e.Filters[0].Formants), + std::for_each(std::begin(e.Formants[VOWEL_A_INDEX]), std::end(e.Formants[VOWEL_A_INDEX]), std::mem_fn(&FormantFilter::clear)); - std::for_each(std::begin(e.Filters[1].Formants), std::end(e.Filters[1].Formants), + std::for_each(std::begin(e.Formants[VOWEL_B_INDEX]), std::end(e.Formants[VOWEL_B_INDEX]), std::mem_fn(&FormantFilter::clear)); std::fill(std::begin(e.CurrentGains), std::end(e.CurrentGains), 0.0f); } @@ -156,10 +157,9 @@ ALboolean VmorpherState::deviceUpdate(const ALCdevice* /*device*/) void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, const EffectProps *props, const EffectTarget target) { - const ALCdevice *device = context->Device; - const ALfloat frequency = static_cast<ALfloat>(device->Frequency); - - const float step{props->Vmorpher.Rate / static_cast<ALfloat>(device->Frequency)}; + const ALCdevice *device{context->Device}; + const ALfloat frequency{static_cast<ALfloat>(device->Frequency)}; + const ALfloat step{props->Vmorpher.Rate / static_cast<ALfloat>(device->Frequency)}; mStep = fastf2i(clampf(step*WAVEFORM_FRACONE, 0.0f, ALfloat{WAVEFORM_FRACONE-1})); if(mStep == 0) @@ -171,8 +171,11 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, else /*if(props->Vmorpher.Waveform == AL_VOCAL_MORPHER_WAVEFORM_TRIANGLE)*/ mGetSamples = Oscillate<Triangle>; - auto& vowelA = mChans[0].Filters[0].Formants; - auto& vowelB = mChans[0].Filters[1].Formants; + auto& vowelA = mChans[0].Formants[VOWEL_A_INDEX]; + auto& vowelB = mChans[0].Formants[VOWEL_B_INDEX]; + + const ALfloat pitchA{fastf2i(std::pow(2.0f, props->Vmorpher.PhonemeACoarseTuning*100.0f / 2400.0f)*FRACTIONONE) * (1.0f/FRACTIONONE)}; + const ALfloat pitchB{fastf2i(std::pow(2.0f, props->Vmorpher.PhonemeBCoarseTuning*100.0f / 2400.0f)*FRACTIONONE) * (1.0f/FRACTIONONE)}; /* Using soprano formant set of values to * better match mid-range frequency space. @@ -182,10 +185,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, switch(props->Vmorpher.PhonemeA) { case AL_VOCAL_MORPHER_PHONEME_A: - vowelA[0].f0norm = 800 / frequency; - vowelA[1].f0norm = 1150 / frequency; - vowelA[2].f0norm = 2900 / frequency; - vowelA[3].f0norm = 3900 / frequency; + vowelA[0].f0norm = (800 * pitchA) / frequency; + vowelA[1].f0norm = (1150 * pitchA) / frequency; + vowelA[2].f0norm = (2900 * pitchA) / frequency; + vowelA[3].f0norm = (3900 * pitchA) / frequency; vowelA[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelA[1].fGain = 0.501187f; /* std::pow(10.0f, -6 / 20.0f); */ @@ -193,10 +196,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelA[3].fGain = 0.100000f; /* std::pow(10.0f, -20 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_E: - vowelA[0].f0norm = 350 / frequency; - vowelA[1].f0norm = 2000 / frequency; - vowelA[2].f0norm = 2800 / frequency; - vowelA[3].f0norm = 3600 / frequency; + vowelA[0].f0norm = (350 * pitchA) / frequency; + vowelA[1].f0norm = (2000 * pitchA) / frequency; + vowelA[2].f0norm = (2800 * pitchA) / frequency; + vowelA[3].f0norm = (3600 * pitchA) / frequency; vowelA[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelA[1].fGain = 0.100000f; /* std::pow(10.0f, -20 / 20.0f); */ @@ -204,10 +207,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelA[3].fGain = 0.009999f; /* std::pow(10.0f, -40 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_I: - vowelA[0].f0norm = 270 / frequency; - vowelA[1].f0norm = 2140 / frequency; - vowelA[2].f0norm = 2950 / frequency; - vowelA[3].f0norm = 3900 / frequency; + vowelA[0].f0norm = (270 * pitchA) / frequency; + vowelA[1].f0norm = (2140 * pitchA) / frequency; + vowelA[2].f0norm = (2950 * pitchA) / frequency; + vowelA[3].f0norm = (3900 * pitchA) / frequency; vowelA[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelA[1].fGain = 0.251188f; /* std::pow(10.0f, -12 / 20.0f); */ @@ -215,10 +218,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelA[3].fGain = 0.050118f; /* std::pow(10.0f, -26 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_O: - vowelA[0].f0norm = 450 / frequency; - vowelA[1].f0norm = 800 / frequency; - vowelA[2].f0norm = 2830 / frequency; - vowelA[3].f0norm = 3800 / frequency; + vowelA[0].f0norm = (450 * pitchA) / frequency; + vowelA[1].f0norm = (800 * pitchA) / frequency; + vowelA[2].f0norm = (2830 * pitchA) / frequency; + vowelA[3].f0norm = (3800 * pitchA) / frequency; vowelA[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelA[1].fGain = 0.281838f; /* std::pow(10.0f, -11 / 20.0f); */ @@ -226,10 +229,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelA[3].fGain = 0.079432f; /* std::pow(10.0f, -22 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_U: - vowelA[0].f0norm = 325 / frequency; - vowelA[1].f0norm = 700 / frequency; - vowelA[2].f0norm = 2700 / frequency; - vowelA[3].f0norm = 3800 / frequency; + vowelA[0].f0norm = (325 * pitchA) / frequency; + vowelA[1].f0norm = (700 * pitchA) / frequency; + vowelA[2].f0norm = (2700 * pitchA) / frequency; + vowelA[3].f0norm = (3800 * pitchA) / frequency; vowelA[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelA[1].fGain = 0.158489f; /* std::pow(10.0f, -16 / 20.0f); */ @@ -241,10 +244,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, switch(props->Vmorpher.PhonemeB) { case AL_VOCAL_MORPHER_PHONEME_A: - vowelB[0].f0norm = 800 / frequency; - vowelB[1].f0norm = 1150 / frequency; - vowelB[2].f0norm = 2900 / frequency; - vowelB[3].f0norm = 3900 / frequency; + vowelB[0].f0norm = (800 * pitchB) / frequency; + vowelB[1].f0norm = (1150 * pitchB) / frequency; + vowelB[2].f0norm = (2900 * pitchB) / frequency; + vowelB[3].f0norm = (3900 * pitchB) / frequency; vowelB[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelB[1].fGain = 0.501187f; /* std::pow(10.0f, -6 / 20.0f); */ @@ -252,10 +255,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelB[3].fGain = 0.100000f; /* std::pow(10.0f, -20 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_E: - vowelB[0].f0norm = 350 / frequency; - vowelB[1].f0norm = 2000 / frequency; - vowelB[2].f0norm = 2800 / frequency; - vowelB[3].f0norm = 3600 / frequency; + vowelB[0].f0norm = (350 * pitchB) / frequency; + vowelB[1].f0norm = (2000 * pitchB) / frequency; + vowelB[2].f0norm = (2800 * pitchB) / frequency; + vowelB[3].f0norm = (3600 * pitchB) / frequency; vowelB[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelB[1].fGain = 0.100000f; /* std::pow(10.0f, -20 / 20.0f); */ @@ -263,10 +266,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelB[3].fGain = 0.009999f; /* std::pow(10.0f, -40 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_I: - vowelB[0].f0norm = 270 / frequency; - vowelB[1].f0norm = 2140 / frequency; - vowelB[2].f0norm = 2950 / frequency; - vowelB[3].f0norm = 3900 / frequency; + vowelB[0].f0norm = (270 * pitchB) / frequency; + vowelB[1].f0norm = (2140 * pitchB) / frequency; + vowelB[2].f0norm = (2950 * pitchB) / frequency; + vowelB[3].f0norm = (3900 * pitchB) / frequency; vowelB[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelB[1].fGain = 0.251188f; /* std::pow(10.0f, -12 / 20.0f); */ @@ -274,10 +277,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelB[3].fGain = 0.050118f; /* std::pow(10.0f, -26 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_O: - vowelB[0].f0norm = 450 / frequency; - vowelB[1].f0norm = 800 / frequency; - vowelB[2].f0norm = 2830 / frequency; - vowelB[3].f0norm = 3800 / frequency; + vowelB[0].f0norm = (450 * pitchB) / frequency; + vowelB[1].f0norm = (800 * pitchB) / frequency; + vowelB[2].f0norm = (2830 * pitchB) / frequency; + vowelB[3].f0norm = (3800 * pitchB) / frequency; vowelB[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelB[1].fGain = 0.281838f; /* std::pow(10.0f, -11 / 20.0f); */ @@ -285,10 +288,10 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, vowelB[3].fGain = 0.079432f; /* std::pow(10.0f, -22 / 20.0f); */ break; case AL_VOCAL_MORPHER_PHONEME_U: - vowelB[0].f0norm = 325 / frequency; - vowelB[1].f0norm = 700 / frequency; - vowelB[2].f0norm = 2700 / frequency; - vowelB[3].f0norm = 3800 / frequency; + vowelB[0].f0norm = (325 * pitchB) / frequency; + vowelB[1].f0norm = (700 * pitchB) / frequency; + vowelB[2].f0norm = (2700 * pitchB) / frequency; + vowelB[3].f0norm = (3800 * pitchB) / frequency; vowelB[0].fGain = 1.000000f; /* std::pow(10.0f, 0 / 20.0f); */ vowelB[1].fGain = 0.158489f; /* std::pow(10.0f, -16 / 20.0f); */ @@ -300,15 +303,15 @@ void VmorpherState::update(const ALCcontext *context, const ALeffectslot *slot, /* Copy the filter coefficients for the other input channels. */ for(ALuint i{1u};i < slot->Wet.Buffer.size();++i) { - mChans[i].Filters[0].Formants[0] = vowelA[0]; - mChans[i].Filters[0].Formants[1] = vowelA[1]; - mChans[i].Filters[0].Formants[2] = vowelA[2]; - mChans[i].Filters[0].Formants[3] = vowelA[3]; - - mChans[i].Filters[1].Formants[0] = vowelB[0]; - mChans[i].Filters[1].Formants[1] = vowelB[1]; - mChans[i].Filters[1].Formants[2] = vowelB[2]; - mChans[i].Filters[1].Formants[3] = vowelB[3]; + mChans[i].Formants[VOWEL_A_INDEX][0] = vowelA[0]; + mChans[i].Formants[VOWEL_A_INDEX][1] = vowelA[1]; + mChans[i].Formants[VOWEL_A_INDEX][2] = vowelA[2]; + mChans[i].Formants[VOWEL_A_INDEX][3] = vowelA[3]; + + mChans[i].Formants[VOWEL_B_INDEX][0] = vowelB[0]; + mChans[i].Formants[VOWEL_B_INDEX][1] = vowelB[1]; + mChans[i].Formants[VOWEL_B_INDEX][2] = vowelB[2]; + mChans[i].Formants[VOWEL_B_INDEX][3] = vowelB[3]; } mOutTarget = target.Main->Buffer; @@ -342,8 +345,8 @@ void VmorpherState::process(const ALsizei samplesToDo, const FloatBufferLine *RE mSampleBufferB[i] = 0.0f; } - auto& vowelA = mChans[c].Filters[0].Formants; - auto& vowelB = mChans[c].Filters[1].Formants; + auto& vowelA = mChans[c].Formants[VOWEL_A_INDEX]; + auto& vowelB = mChans[c].Formants[VOWEL_B_INDEX]; /* Process first vowel. */ vowelA[0].process(&samplesIn[c][base], mSampleBufferA, td); @@ -486,12 +489,12 @@ struct VmorpherStateFactory final : public EffectStateFactory { EffectProps VmorpherStateFactory::getDefaultProps() const noexcept { EffectProps props{}; - props.Vmorpher.Rate = AL_VOCAL_MORPHER_DEFAULT_RATE; - props.Vmorpher.PhonemeA = AL_VOCAL_MORPHER_DEFAULT_PHONEMEA; - props.Vmorpher.PhonemeB = AL_VOCAL_MORPHER_DEFAULT_PHONEMEB; + props.Vmorpher.Rate = AL_VOCAL_MORPHER_DEFAULT_RATE; + props.Vmorpher.PhonemeA = AL_VOCAL_MORPHER_DEFAULT_PHONEMEA; + props.Vmorpher.PhonemeB = 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 = AL_VOCAL_MORPHER_DEFAULT_WAVEFORM; + props.Vmorpher.Waveform = AL_VOCAL_MORPHER_DEFAULT_WAVEFORM; return props; } |