|
|
@@ -0,0 +1,272 @@ |
|
|
|
/*
|
|
|
|
* DISTRHO Plugin Framework (DPF)
|
|
|
|
* Copyright (C) 2012-2021 Filipe Coelho <falktx@falktx.com>
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and/or distribute this software for any purpose with
|
|
|
|
* or without fee is hereby granted, provided that the above copyright notice and this
|
|
|
|
* permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
|
|
|
|
* TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
|
|
|
|
* NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "DistrhoPlugin.hpp"
|
|
|
|
|
|
|
|
START_NAMESPACE_DISTRHO
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
Simple plugin to demonstrate how to modify input/output port type in DPF.
|
|
|
|
The plugin outputs sample & hold (S&H) value of input signal.
|
|
|
|
User can specify hold time via parameter and/or Hold Time CV port.
|
|
|
|
*/
|
|
|
|
class ExamplePluginCVPort : public Plugin
|
|
|
|
{
|
|
|
|
static constexpr const float kMaxHoldTime = 1.0f;
|
|
|
|
|
|
|
|
public:
|
|
|
|
ExamplePluginCVPort()
|
|
|
|
: Plugin(1, 0, 0), // 1 parameters, 0 programs, 0 states
|
|
|
|
counter(0),
|
|
|
|
holdTime(0.0f),
|
|
|
|
holdValue(0.0f),
|
|
|
|
sampleRate(getSampleRate()) {}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
/* --------------------------------------------------------------------------------------------------------
|
|
|
|
* Information */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin label.
|
|
|
|
A plugin label follows the same rules as Parameter::symbol, with the exception that it can start with numbers.
|
|
|
|
*/
|
|
|
|
const char* getLabel() const override
|
|
|
|
{
|
|
|
|
return "CVPort";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get an extensive comment/description about the plugin.
|
|
|
|
*/
|
|
|
|
const char* getDescription() const override
|
|
|
|
{
|
|
|
|
return "Simple plugin with CVPort.\nThe plugin does sample & hold processing.";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin author/maker.
|
|
|
|
*/
|
|
|
|
const char* getMaker() const override
|
|
|
|
{
|
|
|
|
return "DISTRHO";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin homepage.
|
|
|
|
*/
|
|
|
|
const char* getHomePage() const override
|
|
|
|
{
|
|
|
|
return "https://github.com/DISTRHO/DPF";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin license name (a single line of text).
|
|
|
|
For commercial plugins this should return some short copyright information.
|
|
|
|
*/
|
|
|
|
const char* getLicense() const override
|
|
|
|
{
|
|
|
|
return "ISC";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin version, in hexadecimal.
|
|
|
|
*/
|
|
|
|
uint32_t getVersion() const override
|
|
|
|
{
|
|
|
|
return d_version(1, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the plugin unique Id.
|
|
|
|
This value is used by LADSPA, DSSI and VST plugin formats.
|
|
|
|
*/
|
|
|
|
int64_t getUniqueId() const override
|
|
|
|
{
|
|
|
|
return d_cconst('d', 'C', 'V', 'P');
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------------------
|
|
|
|
* Init */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Initialize the audio port @a index.@n
|
|
|
|
This function will be called once, shortly after the plugin is created.
|
|
|
|
*/
|
|
|
|
void initAudioPort(bool input, uint32_t index, AudioPort& port) override
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
Note that index is independent for input and output.
|
|
|
|
In other words, input port index starts from 0 and output port index also starts from 0.
|
|
|
|
*/
|
|
|
|
if (input)
|
|
|
|
{
|
|
|
|
switch (index)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
// Audio port doesn't need to specify port.hints.
|
|
|
|
port.name = "Audio Input";
|
|
|
|
port.symbol = "audio_in";
|
|
|
|
return;
|
|
|
|
case 1:
|
|
|
|
port.hints = kAudioPortIsCV;
|
|
|
|
port.name = "Hold Time";
|
|
|
|
port.symbol = "hold_time";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Add more conditions here when increasing DISTRHO_PLUGIN_NUM_INPUTS.
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (index)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
port.hints = kAudioPortIsCV;
|
|
|
|
port.name = "CV Output";
|
|
|
|
port.symbol = "cv_out";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Add more conditions here when increasing DISTRHO_PLUGIN_NUM_OUTPUTS.
|
|
|
|
}
|
|
|
|
|
|
|
|
// It shouldn't reach here, but just in case if index is greater than 0.
|
|
|
|
Plugin::initAudioPort(input, index, port);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Initialize the parameter @a index.
|
|
|
|
This function will be called once, shortly after the plugin is created.
|
|
|
|
*/
|
|
|
|
void initParameter(uint32_t index, Parameter& parameter) override
|
|
|
|
{
|
|
|
|
if (index != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
parameter.name = "Hold Time";
|
|
|
|
parameter.symbol = "hold_time";
|
|
|
|
parameter.hints = kParameterIsAutomable|kParameterIsLogarithmic;
|
|
|
|
parameter.ranges.min = 0.0f;
|
|
|
|
parameter.ranges.max = kMaxHoldTime;
|
|
|
|
parameter.ranges.def = 0.1f;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------------------
|
|
|
|
* Internal data */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get the current value of a parameter.
|
|
|
|
*/
|
|
|
|
float getParameterValue(uint32_t index) const override
|
|
|
|
{
|
|
|
|
if (index != 0)
|
|
|
|
return 0.0f;
|
|
|
|
|
|
|
|
return holdTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Change a parameter value.
|
|
|
|
*/
|
|
|
|
void setParameterValue(uint32_t index, float value) override
|
|
|
|
{
|
|
|
|
if (index != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
holdTime = value;
|
|
|
|
counter = uint32_t(holdTime * sampleRate);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------------------
|
|
|
|
* Process */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Run/process function for plugins without MIDI input.
|
|
|
|
*/
|
|
|
|
void run(const float** inputs, float** outputs, uint32_t frames) override
|
|
|
|
{
|
|
|
|
float cv, time;
|
|
|
|
|
|
|
|
/**
|
|
|
|
- inputs[0] is input audio port.
|
|
|
|
- inputs[1] is hold time CV port.
|
|
|
|
- outputs[0] is output CV port.
|
|
|
|
*/
|
|
|
|
const float* const audioIn = inputs[0];
|
|
|
|
const float* const holdCV = inputs[1];
|
|
|
|
float* const cvOut = outputs[0];
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < frames; ++i)
|
|
|
|
{
|
|
|
|
if (counter == 0)
|
|
|
|
{
|
|
|
|
cv = holdCV[i] > 0.0f ? holdCV[i] : 0.0f;
|
|
|
|
|
|
|
|
time = holdTime + cv;
|
|
|
|
if (time > kMaxHoldTime)
|
|
|
|
time = kMaxHoldTime;
|
|
|
|
|
|
|
|
counter = static_cast<uint32_t>(time * sampleRate + 0.5f);
|
|
|
|
|
|
|
|
holdValue = audioIn[i]; // Refresh hold value.
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
--counter;
|
|
|
|
}
|
|
|
|
|
|
|
|
cvOut[i] = holdValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------------------
|
|
|
|
* Callbacks (optional) */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Optional callback to inform the plugin about a sample rate change.@n
|
|
|
|
This function will only be called when the plugin is deactivated.
|
|
|
|
*/
|
|
|
|
void sampleRateChanged(double newSampleRate) override
|
|
|
|
{
|
|
|
|
sampleRate = newSampleRate;
|
|
|
|
counter = static_cast<uint32_t>(holdTime * sampleRate + 0.5f);
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint32_t counter; // Hold time in samples. Used to count hold time.
|
|
|
|
float holdTime; // Hold time in seconds.
|
|
|
|
float holdValue;
|
|
|
|
float sampleRate;
|
|
|
|
|
|
|
|
/**
|
|
|
|
Set our plugin class as non-copyable and add a leak detector just in case.
|
|
|
|
*/
|
|
|
|
DISTRHO_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(ExamplePluginCVPort)
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------------------------------------------------
|
|
|
|
* Plugin entry point, called by DPF to create a new plugin instance. */
|
|
|
|
|
|
|
|
Plugin* createPlugin()
|
|
|
|
{
|
|
|
|
return new ExamplePluginCVPort();
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
END_NAMESPACE_DISTRHO
|