Skip to content

Commit

Permalink
Get sidechain signal into Valentine
Browse files Browse the repository at this point in the history
use oversampling, doesn't work

hack
  • Loading branch information
JoseDiazRohena committed Aug 31, 2024
1 parent 0f46792 commit 3d4abc1
Show file tree
Hide file tree
Showing 4 changed files with 71 additions and 30 deletions.
15 changes: 10 additions & 5 deletions libs/tote_bag/dsp/Compressor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,8 @@ void Compressor::setOversampleMultiplier (const int o)
overSampleMultiplier = o - 1;
}

inline void Compressor::makeMonoSidechain (const juce::dsp::AudioBlock<float>& inAudio, juce::AudioBuffer<float>& scSignal)
inline void Compressor::makeMonoSidechain (const juce::dsp::AudioBlock<float>& inAudio,
juce::AudioBuffer<float>& scSignal)
{
scSignal.clear();
auto* scWritePointer = scSignal.getWritePointer (0);
Expand Down Expand Up @@ -101,7 +102,9 @@ inline float Compressor::calculateGain (float inputSample)

else if (abs (doubleOvershoot) <= W)
{
cvOutput = inputSample + ((1.0f / R - 1.0f) * powf (overshoot + W / 2.0f, 2)) / (2.0f * W);
cvOutput =
inputSample
+ ((1.0f / R - 1.0f) * powf (overshoot + W / 2.0f, 2)) / (2.0f * W);
}
else
{
Expand All @@ -121,9 +124,10 @@ inline float Compressor::getMakeupGain()
return -threshold.get() * (1.0f / ratio.get() - 1.0f) / 2.0f;
}

void Compressor::process (juce::dsp::AudioBlock<float>& inAudio)
void Compressor::process (juce::dsp::AudioBlock<float>& inAudio,
juce::dsp::AudioBlock<float>& sidechainInput)
{
makeMonoSidechain (inAudio, analysisSignal);
makeMonoSidechain (sidechainInput, analysisSignal);
auto numSamples = inAudio.getNumSamples();
auto numChannels = inAudio.getNumChannels();

Expand All @@ -132,7 +136,8 @@ void Compressor::process (juce::dsp::AudioBlock<float>& inAudio)
{
auto controlVoltage = juce::Decibels::gainToDecibels (abs (sidechain[sample]));
// TODO: level detector methods should be float or templated
controlVoltage = static_cast<float> (levelDetector.processSampleDecoupledBranched (controlVoltage));
controlVoltage = static_cast<float> (
levelDetector.processSampleDecoupledBranched (controlVoltage));

controlVoltage = calculateGain (controlVoltage);
controlVoltage = juce::Decibels::decibelsToGain (controlVoltage);
Expand Down
15 changes: 7 additions & 8 deletions libs/tote_bag/dsp/Compressor.h
Original file line number Diff line number Diff line change
Expand Up @@ -29,13 +29,15 @@ class Compressor
float inThreshold,
float inKnee);

void makeMonoSidechain (const juce::dsp::AudioBlock<float>& inAudio, juce::AudioBuffer<float>& scSignal);
void makeMonoSidechain (const juce::dsp::AudioBlock<float>& inAudio,
juce::AudioBuffer<float>& scSignal);

void makeKneeCoeffs();

float calculateGain (float analysisSample);

void process (juce::dsp::AudioBlock<float>& inAudio);
void process (juce::dsp::AudioBlock<float>& inAudio,
juce::dsp::AudioBlock<float>& sidechainInput);

float getMakeupGain();

Expand All @@ -51,16 +53,13 @@ class Compressor
private:
juce::WeakReference<FFAU::LevelMeterSource> meterSource;

juce::Atomic<float> ratio { -1.0f },
knee { -1.0f },
msAttack { -1.0f },
msRelease { -1.0f },
threshold { -1.0 };
juce::Atomic<float> ratio {-1.0f}, knee {-1.0f}, msAttack {-1.0f}, msRelease {-1.0f},
threshold {-1.0};

EnvelopeDetector levelDetector;
juce::AudioBuffer<float> analysisSignal;

int overSampleMultiplier { 1 };
int overSampleMultiplier {1};

JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Compressor)
};
66 changes: 51 additions & 15 deletions src/PluginProcessor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,8 @@ ValentineAudioProcessor::ValentineAudioProcessor()
#if !JucePlugin_IsMidiEffect
#if !JucePlugin_IsSynth
.withInput ("Input", juce::AudioChannelSet::stereo(), true)
.withInput ("Sidechain", juce::AudioChannelSet::stereo(), false)

#endif
.withOutput ("Output", juce::AudioChannelSet::stereo(), true)
#endif
Expand All @@ -83,6 +85,9 @@ ValentineAudioProcessor::ValentineAudioProcessor()
, presetManager (this)
, processedDelayLine (32)
, cleanDelayLine (32)
, sidechainOversampler (2,
detail::kOversampleFactor,
Oversampling::filterHalfBandPolyphaseIIR)
#endif
{
initializeDSP();
Expand Down Expand Up @@ -253,6 +258,8 @@ void ValentineAudioProcessor::prepareToPlay (double sampleRate, int samplesPerBl

oversampler->reset();
oversampler->initProcessing (static_cast<size_t> (samplesPerBlock));
sidechainOversampler.reset();
sidechainOversampler.initProcessing (static_cast<size_t> (samplesPerBlock));

saturator->reset (sampleRate);
boundedSaturator->reset (sampleRate);
Expand All @@ -264,7 +271,7 @@ void ValentineAudioProcessor::prepareToPlay (double sampleRate, int samplesPerBl

const auto rmsWindow =
juce::roundToInt (detail::kRmsTime * 0.001f * sampleRate / samplesPerBlock);
inputMeterSource.resize (getTotalNumInputChannels(), rmsWindow);
inputMeterSource.resize (getMainBusNumInputChannels(), rmsWindow);

grMeterSource.resize (1, rmsWindow);
ffCompressor->setMeterSource (&grMeterSource);
Expand Down Expand Up @@ -306,8 +313,12 @@ void ValentineAudioProcessor::processBlock (juce::AudioBuffer<float>& buffer,
juce::MidiBuffer&)
{
juce::ScopedNoDenormals noDenormals;
auto totalNumInputChannels = getTotalNumInputChannels();
const auto numMainBusInputChannels = getMainBusNumInputChannels();
auto totalNumOutputChannels = getTotalNumOutputChannels();

auto mainBusInputOutput = getBusBuffer (buffer, true, 0);
auto sidechainInput = getBusBuffer (buffer, true, 1);

auto bufferSize = buffer.getNumSamples();
auto currentSamplesPerBlock = bufferSize;

Expand All @@ -321,19 +332,19 @@ void ValentineAudioProcessor::processBlock (juce::AudioBuffer<float>& buffer,
// oversampling latency. Copying into the process buffer after this, then, would undo the
// purpose of the delay: maintaining phase coherence between processed and unprocessed
// signals.
processBuffer.setSize (totalNumOutputChannels,
processBuffer.setSize (numMainBusInputChannels,
currentSamplesPerBlock,
true,
true,
true);
for (auto channel = 0; channel < totalNumInputChannels; ++channel)
for (auto channel = 0; channel < numMainBusInputChannels; ++channel)
processBuffer.copyFrom (channel,
0,
buffer.getReadPointer (channel),
buffer.getNumSamples());
mainBusInputOutput.getReadPointer (channel),
mainBusInputOutput.getNumSamples());

prepareInputBuffer (buffer,
totalNumInputChannels,
prepareInputBuffer (mainBusInputOutput,
numMainBusInputChannels,
totalNumOutputChannels,
currentSamplesPerBlock);

Expand All @@ -342,7 +353,7 @@ void ValentineAudioProcessor::processBlock (juce::AudioBuffer<float>& buffer,
return;
}

inputMeterSource.measureBlock (buffer);
inputMeterSource.measureBlock (mainBusInputOutput);

// "Downsample" and Bitcrush processing
if (crushOn.get())
Expand All @@ -357,16 +368,38 @@ void ValentineAudioProcessor::processBlock (juce::AudioBuffer<float>& buffer,
}

auto g = compressValue.get();
for (int i = 0; i < totalNumOutputChannels; ++i)
processBuffer.applyGainRamp (i, 0, bufferSize, currentGain, g);

if (sidechainInput.getNumChannels() > 0)
{
for (int i = 0; i < totalNumOutputChannels; ++i)
{
sidechainInput.applyGainRamp (i, 0, bufferSize, currentGain, g);
}
}
else
{
for (int i = 0; i < totalNumOutputChannels; ++i)
processBuffer.applyGainRamp (i, 0, bufferSize, currentGain, g);
}

// for (int i = 0; i < totalNumOutputChannels; ++i)
// processBuffer.applyGainRamp (i, 0, bufferSize, currentGain, g);
currentGain = g;

// Upsample then do non-linear processing
juce::dsp::AudioBlock<float> processBlock (processBuffer);
juce::dsp::AudioBlock<float> highSampleRateBlock =
oversampler->processSamplesUp (processBlock);

ffCompressor->process (highSampleRateBlock);
if (sidechainInput.getNumChannels() > 0)
{
juce::dsp::AudioBlock<float> highSampleRateSideChain =
sidechainOversampler.processSamplesUp (sidechainInput);
ffCompressor->process (highSampleRateBlock, highSampleRateSideChain);
}
{
ffCompressor->process (highSampleRateBlock, highSampleRateBlock);
}

if (saturateOn.get())
{
Expand Down Expand Up @@ -426,13 +459,16 @@ void ValentineAudioProcessor::processBlock (juce::AudioBuffer<float>& buffer,
for (int i = 0; i < totalNumOutputChannels; ++i)
{
auto processed = processBuffer.getSample (i, j);
auto unprocessed = buffer.getSample (i, j);
auto unprocessed = mainBusInputOutput.getSample (i, j);

buffer.setSample (i, j, mix * processed + (1.0f - currentMix) * unprocessed);
mainBusInputOutput.setSample (i,
j,
mix * processed
+ (1.0f - currentMix) * unprocessed);
}
}

outputMeterSource.measureBlock (buffer);
outputMeterSource.measureBlock (mainBusInputOutput);
}

void ValentineAudioProcessor::processBlockBypassed (juce::AudioBuffer<float>& buffer,
Expand Down
5 changes: 3 additions & 2 deletions src/PluginProcessor.h
Original file line number Diff line number Diff line change
Expand Up @@ -199,8 +199,8 @@ class ValentineAudioProcessor : public juce::AudioProcessor,
FFCompParameterDefaults[static_cast<size_t> (VParameter::outputClipEnable)]
> 0.5f};
bool clipOnState = clipOn.get();
juce::Atomic<bool> saturateOn = FFCompParameterDefaults[static_cast<size_t> (VParameter::saturateEnable)]
> 0.5f;
juce::Atomic<bool> saturateOn =
FFCompParameterDefaults[static_cast<size_t> (VParameter::saturateEnable)] > 0.5f;
bool saturateOnState = saturateOn.get();

// This is used for, yes you guessed it, processing
Expand All @@ -209,6 +209,7 @@ class ValentineAudioProcessor : public juce::AudioProcessor,
// DSP objects
std::unique_ptr<tote_bag::audio_helpers::SimpleOnePole<float>> dryWetFilter;
std::unique_ptr<Oversampling> oversampler;
Oversampling sidechainOversampler;
std::unique_ptr<Compressor> ffCompressor;
std::unique_ptr<Saturation> saturator;
std::unique_ptr<Saturation> boundedSaturator;
Expand Down

0 comments on commit 3d4abc1

Please sign in to comment.