-
-
Notifications
You must be signed in to change notification settings - Fork 24
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Addin Poly Octave effect * Optimizations * Reset filters when we update their coefficients
- Loading branch information
1 parent
de934aa
commit 5575f39
Showing
5 changed files
with
338 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,296 @@ | ||
#include "PolyOctave.h" | ||
#include "processors/ParameterHelpers.h" | ||
|
||
namespace PolyOctaveTags | ||
{ | ||
const String dryTag = "dry"; | ||
const String upOctaveTag = "up_octave"; | ||
const String up2OctaveTag = "up2_octave"; | ||
} // namespace PolyOctaveTags | ||
|
||
namespace FilterBankHelpers | ||
{ | ||
// Reference for filter-bank design and octave shifting: | ||
// https://aaltodoc.aalto.fi/server/api/core/bitstreams/ff9e52cf-fd79-45eb-b695-93038244ec0e/content | ||
|
||
using float_2 = PolyOctave::ERBFilterBank::float_2; | ||
static_assert (float_2::size == 2); | ||
static constexpr auto vec_size = float_2::size; | ||
|
||
inline float_2 processSample (const chowdsp::IIRFilter<4, float_2>& f, std::array<float_2, 5>& z, float_2 x) | ||
{ | ||
const auto y = z[1] + x * f.b[0]; // (we know that b[0] == 0) | ||
|
||
z[1] = z[2] + x * f.b[1] - y * f.a[1]; | ||
z[2] = z[3] + x * f.b[2] - y * f.a[2]; | ||
z[3] = z[4] + x * f.b[3] - y * f.a[3]; | ||
z[4] = x * f.b[4] - y * f.a[4]; | ||
|
||
return y; | ||
} | ||
|
||
static constexpr auto q_c = 4.0; | ||
static void designERBFilter (size_t erb_index, | ||
double gamma, | ||
double erb_start, | ||
double q_ERB, | ||
double sample_rate, | ||
double (&b_coeffs_cplx_real)[5], | ||
double (&b_coeffs_cplx_imag)[5], | ||
double (&a_coeffs_cplx)[5]) | ||
{ | ||
const auto q_PS = gamma; | ||
|
||
const auto z = erb_start + static_cast<float> (erb_index) * (q_c / q_ERB); | ||
const auto center_target_freq = 228.7 * (std::pow (10.0, z / 21.3) - 1.0); | ||
const auto filter_q = (1.0 / (q_PS * q_ERB)) * (24.7 + 0.108 * center_target_freq); | ||
|
||
double b_coeffs_proto[3]; | ||
double a_coeffs_proto[3]; | ||
chowdsp::CoefficientCalculators::calcSecondOrderBPF (b_coeffs_proto, | ||
a_coeffs_proto, | ||
center_target_freq / gamma, | ||
filter_q * 0.5, | ||
sample_rate); | ||
|
||
auto pole = (std::sqrt (std::pow (std::complex { a_coeffs_proto[1] }, 2.0) - 4.0 * a_coeffs_proto[2]) - a_coeffs_proto[1]) / 2.0; | ||
auto conj_pole = std::conj (pole); | ||
if (std::imag (pole) < 0.0) | ||
pole = conj_pole; | ||
else if (std::imag (conj_pole) < 0.0) | ||
conj_pole = pole; | ||
auto at = -(pole + conj_pole); | ||
auto att = pole * conj_pole; | ||
|
||
auto ar1 = std::real (at); | ||
auto ai1 = std::imag (at); | ||
auto ar2 = std::real (att); | ||
auto ai2 = std::imag (att); | ||
|
||
// a[] = 1 + 2 ar1 z + (ai1^2 + ar1^2 + 2 ar2) z^2 + (2 ai1 ai2 + 2 ar1 ar2) z^3 + (ai2^2 + ar2^2) z^4 | ||
a_coeffs_cplx[0] = 1.0; | ||
a_coeffs_cplx[1] = 2.0 * ar1; | ||
a_coeffs_cplx[2] = ai1 * ai1 + ar1 * ar1 + 2.0 * ar2; | ||
a_coeffs_cplx[3] = 2.0 * ai1 * ai2 + 2.0 * ar1 * ar2; | ||
a_coeffs_cplx[4] = ai2 * ai2 + ar2 * ar2; | ||
|
||
// b_real[] = b0 + (ar1 b0 + b1) z + (ar2 b0 + ar1 b1 + b2) z^2 + (ar2 b1 + ar1 b2) z^3 + ar2 b2 z^4 | ||
b_coeffs_cplx_real[0] = b_coeffs_proto[0]; | ||
b_coeffs_cplx_real[1] = ar1 * b_coeffs_proto[0] + b_coeffs_proto[1]; | ||
b_coeffs_cplx_real[2] = ar2 * b_coeffs_proto[0] + ar1 * b_coeffs_proto[1] + b_coeffs_proto[2]; | ||
b_coeffs_cplx_real[3] = ar2 * b_coeffs_proto[1] + ar1 * b_coeffs_proto[2]; | ||
b_coeffs_cplx_real[4] = ar2 * b_coeffs_proto[2]; | ||
|
||
// b_imag[] = -ai1 b0 z - (ai2 b0 + ai1 b1) z^2 - (ai2 b1 + ai1 br) z^3 - ai2 br z^4 | ||
b_coeffs_cplx_imag[0] = 0.0; | ||
b_coeffs_cplx_imag[1] = -ai1 * b_coeffs_proto[0]; | ||
b_coeffs_cplx_imag[2] = -ai2 * b_coeffs_proto[0] - ai1 * b_coeffs_proto[1]; | ||
b_coeffs_cplx_imag[3] = -ai2 * b_coeffs_proto[1] - ai1 * b_coeffs_proto[2]; | ||
b_coeffs_cplx_imag[4] = -ai2 * b_coeffs_proto[2]; | ||
} | ||
|
||
static void designFilterBank (std::array<PolyOctave::ERBFilterBank, 2>& filterBank, | ||
double gamma, | ||
double erb_start, | ||
double q_ERB, | ||
double sampleRate) | ||
{ | ||
for (size_t kiter = 0; kiter < PolyOctave::ERBFilterBank::numFilterBands; kiter += vec_size) | ||
{ | ||
alignas (16) std::array<float_2::value_type, float_2::size> b_coeffs_cplx_real_simd[5]; | ||
alignas (16) std::array<float_2::value_type, float_2::size> b_coeffs_cplx_imag_simd[5]; | ||
alignas (16) std::array<float_2::value_type, float_2::size> a_coeffs_cplx_simd[5]; | ||
for (size_t i = 0; i < float_2::size; ++i) | ||
{ | ||
const auto k = kiter + i; | ||
|
||
double b_coeffs_cplx_real[5]; | ||
double b_coeffs_cplx_imag[5]; | ||
double a_coeffs_cplx[5]; | ||
designERBFilter (k, gamma, erb_start, q_ERB, sampleRate, b_coeffs_cplx_real, b_coeffs_cplx_imag, a_coeffs_cplx); | ||
|
||
for (size_t c = 0; c < 5; ++c) | ||
{ | ||
b_coeffs_cplx_real_simd[c][i] = static_cast<float_2::value_type> (b_coeffs_cplx_real[c]); | ||
b_coeffs_cplx_imag_simd[c][i] = static_cast<float_2::value_type> (b_coeffs_cplx_imag[c]); | ||
a_coeffs_cplx_simd[c][i] = static_cast<float_2::value_type> (a_coeffs_cplx[c]); | ||
} | ||
} | ||
|
||
for (size_t ch = 0; ch < 2; ++ch) | ||
{ | ||
filterBank[ch].erbFilterReal[kiter / vec_size].reset(); | ||
filterBank[ch].erbFilterReal[kiter / vec_size].setCoefs ({ | ||
xsimd::load_aligned (b_coeffs_cplx_real_simd[0].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_real_simd[1].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_real_simd[2].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_real_simd[3].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_real_simd[4].data()), | ||
}, | ||
{ | ||
xsimd::load_aligned (a_coeffs_cplx_simd[0].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[1].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[2].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[3].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[4].data()), | ||
}); | ||
filterBank[ch].erbFilterImag[kiter / vec_size].reset(); | ||
filterBank[ch].erbFilterImag[kiter / vec_size].setCoefs ({ | ||
xsimd::load_aligned (b_coeffs_cplx_imag_simd[0].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_imag_simd[1].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_imag_simd[2].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_imag_simd[3].data()), | ||
xsimd::load_aligned (b_coeffs_cplx_imag_simd[4].data()), | ||
}, | ||
{ | ||
xsimd::load_aligned (a_coeffs_cplx_simd[0].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[1].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[2].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[3].data()), | ||
xsimd::load_aligned (a_coeffs_cplx_simd[4].data()), | ||
}); | ||
} | ||
} | ||
} | ||
} // namespace FilterBankHelpers | ||
|
||
PolyOctave::PolyOctave (UndoManager* um) | ||
: BaseProcessor ("Poly Octave", createParameterLayout(), um) | ||
{ | ||
using namespace ParameterHelpers; | ||
const auto setupGainParam = [this] (const juce::String& paramID, | ||
chowdsp::SmoothedBufferValue<double>& gain) | ||
{ | ||
gain.setParameterHandle (getParameterPointer<chowdsp::FloatParameter*> (vts, paramID)); | ||
gain.setRampLength (0.05); | ||
gain.mappingFunction = [] (double x) | ||
{ | ||
return 2.0 * x * x; | ||
}; | ||
}; | ||
setupGainParam (PolyOctaveTags::dryTag, dryGain); | ||
setupGainParam (PolyOctaveTags::upOctaveTag, upOctaveGain); | ||
setupGainParam (PolyOctaveTags::up2OctaveTag, up2OctaveGain); | ||
|
||
uiOptions.backgroundColour = Colour { 0xff9fa09d }; | ||
uiOptions.powerColour = Colour { 0xffe70510 }; | ||
uiOptions.info.description = "A \"polyphonic octave generator\" effect."; | ||
uiOptions.info.authors = StringArray { "Jatin Chowdhury" }; | ||
} | ||
|
||
ParamLayout PolyOctave::createParameterLayout() | ||
{ | ||
using namespace ParameterHelpers; | ||
auto params = createBaseParams(); | ||
|
||
createPercentParameter (params, PolyOctaveTags::dryTag, "+0 Oct", 0.5f); | ||
createPercentParameter (params, PolyOctaveTags::upOctaveTag, "+1 Oct", 0.0f); | ||
createPercentParameter (params, PolyOctaveTags::up2OctaveTag, "+2 Oct", 0.0f); | ||
|
||
return { params.begin(), params.end() }; | ||
} | ||
|
||
void PolyOctave::prepare (double sampleRate, int samplesPerBlock) | ||
{ | ||
dryGain.prepare (sampleRate, samplesPerBlock); | ||
up2OctaveGain.prepare (sampleRate, samplesPerBlock); | ||
upOctaveGain.prepare (sampleRate, samplesPerBlock); | ||
|
||
doubleBuffer.setMaxSize (2, samplesPerBlock); | ||
up2OctaveBuffer.setMaxSize (2, samplesPerBlock); | ||
upOctaveBuffer.setMaxSize (2, samplesPerBlock); | ||
|
||
FilterBankHelpers::designFilterBank (octaveUpFilterBank, 2.0, 5.0, 6.0, sampleRate); | ||
FilterBankHelpers::designFilterBank (octaveUp2FilterBank, 3.0, 7.0, 4.0, sampleRate); | ||
|
||
dcBlocker.prepare (2); | ||
dcBlocker.calcCoefs (20.0f, (float) sampleRate); | ||
} | ||
|
||
void PolyOctave::processAudio (AudioBuffer<float>& buffer) | ||
{ | ||
const auto numChannels = buffer.getNumChannels(); | ||
const auto numSamples = buffer.getNumSamples(); | ||
|
||
doubleBuffer.setCurrentSize (numChannels, numSamples); | ||
upOctaveBuffer.setCurrentSize (numChannels, numSamples); | ||
upOctaveBuffer.clear(); | ||
up2OctaveBuffer.setCurrentSize (numChannels, numSamples); | ||
up2OctaveBuffer.clear(); | ||
|
||
chowdsp::BufferMath::copyBufferData (buffer, doubleBuffer); | ||
|
||
for (int ch = 0; ch < numChannels; ++ch) | ||
{ | ||
auto* dryData = doubleBuffer.getReadPointer (ch); | ||
auto* upData = upOctaveBuffer.getWritePointer (ch); | ||
auto* up2Data = up2OctaveBuffer.getWritePointer (ch); | ||
auto& upFilterBank = octaveUpFilterBank[static_cast<size_t> (ch)]; | ||
auto& up2FilterBank = octaveUp2FilterBank[static_cast<size_t> (ch)]; | ||
static constexpr auto eps = std::numeric_limits<double>::epsilon(); | ||
|
||
for (size_t k = 0; k < ERBFilterBank::numFilterBands; k += ERBFilterBank::float_2::size) | ||
{ | ||
const auto filter_idx = k / ERBFilterBank::float_2::size; | ||
auto& realFilter = upFilterBank.erbFilterReal[filter_idx]; | ||
auto& imagFilter = upFilterBank.erbFilterImag[filter_idx]; | ||
chowdsp::ScopedValue z_re { realFilter.z[0] }; | ||
chowdsp::ScopedValue z_im { imagFilter.z[0] }; | ||
for (int n = 0; n < numSamples; ++n) | ||
{ | ||
auto x_re = FilterBankHelpers::processSample (realFilter, z_re.get(), dryData[n]); | ||
auto x_im = FilterBankHelpers::processSample (imagFilter, z_im.get(), dryData[n]); | ||
// auto x_re = upFilterBank.erbFilterReal[filter_idx].processSample (dryData[n]); | ||
// auto x_im = upFilterBank.erbFilterImag[filter_idx].processSample (dryData[n]); | ||
|
||
auto x_re_sq = x_re * x_re; | ||
auto x_im_sq = x_im * x_im; | ||
auto x_abs_sq = x_re_sq + x_im_sq; | ||
|
||
auto x_abs_r = xsimd::select (x_abs_sq > eps, xsimd::rsqrt (x_abs_sq), {}); | ||
upData[n] += xsimd::reduce_add ((x_re_sq - x_im_sq) * x_abs_r); | ||
} | ||
} | ||
|
||
for (size_t k = 0; k < ERBFilterBank::numFilterBands; k += ERBFilterBank::float_2::size) | ||
{ | ||
const auto filter_idx = k / ERBFilterBank::float_2::size; | ||
auto& realFilter = up2FilterBank.erbFilterReal[filter_idx]; | ||
auto& imagFilter = up2FilterBank.erbFilterImag[filter_idx]; | ||
chowdsp::ScopedValue z_re { realFilter.z[0] }; | ||
chowdsp::ScopedValue z_im { imagFilter.z[0] }; | ||
|
||
for (int n = 0; n < numSamples; ++n) | ||
{ | ||
auto x_re = FilterBankHelpers::processSample (realFilter, z_re.get(), dryData[n]); | ||
auto x_im = FilterBankHelpers::processSample (imagFilter, z_im.get(), dryData[n]); | ||
// auto x_re = up2FilterBank.erbFilterReal[filter_idx].processSample (dryData[n]); | ||
// auto x_im = up2FilterBank.erbFilterImag[filter_idx].processSample (dryData[n]); | ||
|
||
auto x_re_sq = x_re * x_re; | ||
auto x_im_sq = x_im * x_im; | ||
auto x_abs_sq = x_re_sq + x_im_sq; | ||
|
||
auto x_abs_sq_r = xsimd::select (x_abs_sq > eps, xsimd::reciprocal (x_abs_sq), {}); | ||
up2Data[n] += xsimd::reduce_add (x_re * (x_re_sq - 3.0 * x_im_sq) * x_abs_sq_r); | ||
} | ||
} | ||
} | ||
|
||
chowdsp::BufferMath::applyGain (upOctaveBuffer, 2.0 / static_cast<double> (ERBFilterBank::numFilterBands)); | ||
upOctaveGain.process (numSamples); | ||
chowdsp::BufferMath::applyGainSmoothedBuffer (upOctaveBuffer, upOctaveGain); | ||
|
||
chowdsp::BufferMath::applyGain (up2OctaveBuffer, 2.0 / static_cast<double> (ERBFilterBank::numFilterBands)); | ||
up2OctaveGain.process (numSamples); | ||
chowdsp::BufferMath::applyGainSmoothedBuffer (up2OctaveBuffer, up2OctaveGain); | ||
|
||
dryGain.process (numSamples); | ||
chowdsp::BufferMath::applyGainSmoothedBuffer (doubleBuffer, dryGain); | ||
chowdsp::BufferMath::addBufferData (up2OctaveBuffer, doubleBuffer); | ||
chowdsp::BufferMath::addBufferData (upOctaveBuffer, doubleBuffer); | ||
|
||
chowdsp::BufferMath::copyBufferData (doubleBuffer, buffer); | ||
|
||
dcBlocker.processBlock (buffer); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
#pragma once | ||
|
||
#include "processors/BaseProcessor.h" | ||
|
||
class PolyOctave : public BaseProcessor | ||
{ | ||
public: | ||
explicit PolyOctave (UndoManager* um); | ||
|
||
ProcessorType getProcessorType() const override { return Other; } | ||
static ParamLayout createParameterLayout(); | ||
|
||
void prepare (double sampleRate, int samplesPerBlock) override; | ||
void processAudio (AudioBuffer<float>& buffer) override; | ||
|
||
struct ERBFilterBank | ||
{ | ||
static constexpr size_t numFilterBands = 44; | ||
using float_2 = xsimd::batch<double>; | ||
std::array<chowdsp::IIRFilter<4, float_2>, numFilterBands / float_2::size> erbFilterReal, erbFilterImag; | ||
}; | ||
|
||
private: | ||
chowdsp::SmoothedBufferValue<double> dryGain {}; | ||
chowdsp::SmoothedBufferValue<double> upOctaveGain {}; | ||
chowdsp::SmoothedBufferValue<double> up2OctaveGain {}; | ||
|
||
chowdsp::Buffer<double> doubleBuffer; | ||
chowdsp::Buffer<double> upOctaveBuffer; | ||
chowdsp::Buffer<double> up2OctaveBuffer; | ||
|
||
std::array<ERBFilterBank, 2> octaveUpFilterBank; | ||
std::array<ERBFilterBank, 2> octaveUp2FilterBank; | ||
|
||
chowdsp::FirstOrderHPF<float> dcBlocker; | ||
|
||
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (PolyOctave) | ||
}; |