428 lines
13 KiB
C++
428 lines
13 KiB
C++
/*
|
|
* DISTRHO Nekobi Plugin, based on Nekobee by Sean Bolton and others.
|
|
* Copyright (C) 2004 Sean Bolton and others
|
|
* Copyright (C) 2013-2015 Filipe Coelho <falktx@falktx.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of
|
|
* the License, or any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* For a full copy of the GNU General Public License see the LICENSE file.
|
|
*/
|
|
|
|
#include "DistrhoPluginNekobi.hpp"
|
|
|
|
extern "C" {
|
|
|
|
#include "nekobee-src/nekobee_synth.c"
|
|
#include "nekobee-src/nekobee_voice.c"
|
|
#include "nekobee-src/nekobee_voice_render.c"
|
|
#include "nekobee-src/minblep_tables.c"
|
|
|
|
// -----------------------------------------------------------------------
|
|
// mutual exclusion
|
|
|
|
bool dssp_voicelist_mutex_trylock(nekobee_synth_t* const synth)
|
|
{
|
|
/* Attempt the mutex lock */
|
|
if (pthread_mutex_trylock(&synth->voicelist_mutex) != 0)
|
|
{
|
|
synth->voicelist_mutex_grab_failed = 1;
|
|
return false;
|
|
}
|
|
|
|
/* Clean up if a previous mutex grab failed */
|
|
if (synth->voicelist_mutex_grab_failed)
|
|
{
|
|
nekobee_synth_all_voices_off(synth);
|
|
synth->voicelist_mutex_grab_failed = 0;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool dssp_voicelist_mutex_unlock(nekobee_synth_t* const synth)
|
|
{
|
|
return (pthread_mutex_unlock(&synth->voicelist_mutex) == 0);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
// nekobee_handle_raw_event
|
|
|
|
void nekobee_handle_raw_event(nekobee_synth_t* const synth, const uint8_t size, const uint8_t* const data)
|
|
{
|
|
if (size != 3)
|
|
return;
|
|
|
|
switch (data[0] & 0xf0)
|
|
{
|
|
case 0x80:
|
|
nekobee_synth_note_off(synth, data[1], data[2]);
|
|
break;
|
|
case 0x90:
|
|
if (data[2] > 0)
|
|
nekobee_synth_note_on(synth, data[1], data[2]);
|
|
else
|
|
nekobee_synth_note_off(synth, data[1], 64); /* shouldn't happen, but... */
|
|
break;
|
|
case 0xB0:
|
|
nekobee_synth_control_change(synth, data[1], data[2]);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
} /* extern "C" */
|
|
|
|
START_NAMESPACE_DISTRHO
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
DistrhoPluginNekobi::DistrhoPluginNekobi()
|
|
: Plugin(paramCount, 0, 0) // 0 programs, 0 states
|
|
{
|
|
nekobee_init_tables();
|
|
|
|
// init synth
|
|
fSynth.sample_rate = getSampleRate();
|
|
fSynth.deltat = 1.0f / (float)getSampleRate();
|
|
fSynth.nugget_remains = 0;
|
|
|
|
fSynth.note_id = 0;
|
|
fSynth.polyphony = XSYNTH_DEFAULT_POLYPHONY;
|
|
fSynth.voices = XSYNTH_DEFAULT_POLYPHONY;
|
|
fSynth.monophonic = XSYNTH_MONO_MODE_ONCE;
|
|
fSynth.glide = 0;
|
|
fSynth.last_noteon_pitch = 0.0f;
|
|
fSynth.vcf_accent = 0.0f;
|
|
fSynth.vca_accent = 0.0f;
|
|
|
|
for (int i=0; i<8; ++i)
|
|
fSynth.held_keys[i] = -1;
|
|
|
|
fSynth.voice = nekobee_voice_new();
|
|
fSynth.voicelist_mutex_grab_failed = 0;
|
|
pthread_mutex_init(&fSynth.voicelist_mutex, nullptr);
|
|
|
|
fSynth.channel_pressure = 0;
|
|
fSynth.pitch_wheel_sensitivity = 0;
|
|
fSynth.pitch_wheel = 0;
|
|
|
|
for (int i=0; i<128; ++i)
|
|
{
|
|
fSynth.key_pressure[i] = 0;
|
|
fSynth.cc[i] = 0;
|
|
}
|
|
fSynth.cc[7] = 127; // full volume
|
|
|
|
fSynth.mod_wheel = 1.0f;
|
|
fSynth.pitch_bend = 1.0f;
|
|
fSynth.cc_volume = 1.0f;
|
|
|
|
// Default values
|
|
fParams.waveform = 0.0f;
|
|
fParams.tuning = 0.0f;
|
|
fParams.cutoff = 25.0f;
|
|
fParams.resonance = 25.0f;
|
|
fParams.envMod = 50.0f;
|
|
fParams.decay = 75.0f;
|
|
fParams.accent = 25.0f;
|
|
fParams.volume = 75.0f;
|
|
fParams.bypass = false;
|
|
|
|
// Internal stuff
|
|
fSynth.waveform = 0.0f;
|
|
fSynth.tuning = 1.0f;
|
|
fSynth.cutoff = 5.0f;
|
|
fSynth.resonance = 0.8f;
|
|
fSynth.envmod = 0.3f;
|
|
fSynth.decay = 0.0002f;
|
|
fSynth.accent = 0.3f;
|
|
fSynth.volume = 0.75f;
|
|
|
|
// reset
|
|
deactivate();
|
|
}
|
|
|
|
DistrhoPluginNekobi::~DistrhoPluginNekobi()
|
|
{
|
|
std::free(fSynth.voice);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Init
|
|
|
|
void DistrhoPluginNekobi::initParameter(uint32_t index, Parameter& parameter)
|
|
{
|
|
switch (index)
|
|
{
|
|
case paramWaveform:
|
|
parameter.hints = kParameterIsAutomable|kParameterIsInteger;
|
|
parameter.name = "Waveform";
|
|
parameter.symbol = "waveform";
|
|
parameter.ranges.def = 0.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 1.0f;
|
|
parameter.enumValues.count = 2;
|
|
parameter.enumValues.restrictedMode = true;
|
|
{
|
|
ParameterEnumerationValue* const enumValues = new ParameterEnumerationValue[2];
|
|
enumValues[0].value = 0.0f;
|
|
enumValues[0].label = "Square";
|
|
enumValues[1].value = 1.0f;
|
|
enumValues[1].label = "Triangle";
|
|
parameter.enumValues.values = enumValues;
|
|
}
|
|
break;
|
|
case paramTuning:
|
|
parameter.hints = kParameterIsAutomable; // was 0.5 <-> 2.0, log
|
|
parameter.name = "Tuning";
|
|
parameter.symbol = "tuning";
|
|
parameter.ranges.def = 0.0f;
|
|
parameter.ranges.min = -12.0f;
|
|
parameter.ranges.max = 12.0f;
|
|
break;
|
|
case paramCutoff:
|
|
parameter.hints = kParameterIsAutomable; // modified x2.5
|
|
parameter.name = "Cutoff";
|
|
parameter.symbol = "cutoff";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 25.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 100.0f;
|
|
break;
|
|
case paramResonance:
|
|
parameter.hints = kParameterIsAutomable; // modified x100
|
|
parameter.name = "VCF Resonance";
|
|
parameter.symbol = "resonance";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 25.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 95.0f;
|
|
break;
|
|
case paramEnvMod:
|
|
parameter.hints = kParameterIsAutomable; // modified x100
|
|
parameter.name = "Env Mod";
|
|
parameter.symbol = "env_mod";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 50.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 100.0f;
|
|
break;
|
|
case paramDecay:
|
|
parameter.hints = kParameterIsAutomable; // was 0.000009 <-> 0.0005, log
|
|
parameter.name = "Decay";
|
|
parameter.symbol = "decay";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 75.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 100.0f;
|
|
break;
|
|
case paramAccent:
|
|
parameter.hints = kParameterIsAutomable; // modified x100
|
|
parameter.name = "Accent";
|
|
parameter.symbol = "accent";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 25.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 100.0f;
|
|
break;
|
|
case paramVolume:
|
|
parameter.hints = kParameterIsAutomable; // modified x100
|
|
parameter.name = "Volume";
|
|
parameter.symbol = "volume";
|
|
parameter.unit = "%";
|
|
parameter.ranges.def = 75.0f;
|
|
parameter.ranges.min = 0.0f;
|
|
parameter.ranges.max = 100.0f;
|
|
break;
|
|
case paramBypass:
|
|
parameter.initDesignation(kParameterDesignationBypass);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Internal data
|
|
|
|
float DistrhoPluginNekobi::getParameterValue(uint32_t index) const
|
|
{
|
|
switch (index)
|
|
{
|
|
case paramWaveform:
|
|
return fParams.waveform;
|
|
case paramTuning:
|
|
return fParams.tuning;
|
|
case paramCutoff:
|
|
return fParams.cutoff;
|
|
case paramResonance:
|
|
return fParams.resonance;
|
|
case paramEnvMod:
|
|
return fParams.envMod;
|
|
case paramDecay:
|
|
return fParams.decay;
|
|
case paramAccent:
|
|
return fParams.accent;
|
|
case paramVolume:
|
|
return fParams.volume;
|
|
case paramBypass:
|
|
return fParams.bypass ? 1.0f : 0.0f;
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
void DistrhoPluginNekobi::setParameterValue(uint32_t index, float value)
|
|
{
|
|
switch (index)
|
|
{
|
|
case paramWaveform:
|
|
fParams.waveform = value;
|
|
fSynth.waveform = value;
|
|
DISTRHO_SAFE_ASSERT(fSynth.waveform == 0.0f || fSynth.waveform == 1.0f);
|
|
break;
|
|
case paramTuning:
|
|
fParams.tuning = value;
|
|
fSynth.tuning = (value+12.0f)/24.0f * 1.5 + 0.5f; // FIXME: log?
|
|
DISTRHO_SAFE_ASSERT(fSynth.tuning >= 0.5f && fSynth.tuning <= 2.0f);
|
|
break;
|
|
case paramCutoff:
|
|
fParams.cutoff = value;
|
|
fSynth.cutoff = value/2.5f;
|
|
DISTRHO_SAFE_ASSERT(fSynth.cutoff >= 0.0f && fSynth.cutoff <= 40.0f);
|
|
break;
|
|
case paramResonance:
|
|
fParams.resonance = value;
|
|
fSynth.resonance = value/100.0f;
|
|
DISTRHO_SAFE_ASSERT(fSynth.resonance >= 0.0f && fSynth.resonance <= 0.95f);
|
|
break;
|
|
case paramEnvMod:
|
|
fParams.envMod = value;
|
|
fSynth.envmod = value/100.0f;
|
|
DISTRHO_SAFE_ASSERT(fSynth.envmod >= 0.0f && fSynth.envmod <= 1.0f);
|
|
break;
|
|
case paramDecay:
|
|
fParams.decay = value;
|
|
fSynth.decay = value/100.0f * 0.000491f + 0.000009f; // FIXME: log?
|
|
DISTRHO_SAFE_ASSERT(fSynth.decay >= 0.000009f && fSynth.decay <= 0.0005f);
|
|
break;
|
|
case paramAccent:
|
|
fParams.accent = value;
|
|
fSynth.accent = value/100.0f;
|
|
DISTRHO_SAFE_ASSERT(fSynth.accent >= 0.0f && fSynth.accent <= 1.0f);
|
|
break;
|
|
case paramVolume:
|
|
fParams.volume = value;
|
|
fSynth.volume = value/100.0f;
|
|
DISTRHO_SAFE_ASSERT(fSynth.volume >= 0.0f && fSynth.volume <= 1.0f);
|
|
break;
|
|
case paramBypass: {
|
|
const bool bypass = (value > 0.5f);
|
|
if (fParams.bypass != bypass)
|
|
{
|
|
fParams.bypass = bypass;
|
|
nekobee_synth_all_voices_off(&fSynth);
|
|
}
|
|
} break;
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Process
|
|
|
|
void DistrhoPluginNekobi::activate()
|
|
{
|
|
fSynth.nugget_remains = 0;
|
|
fSynth.note_id = 0;
|
|
|
|
if (fSynth.voice != nullptr)
|
|
nekobee_synth_all_voices_off(&fSynth);
|
|
}
|
|
|
|
void DistrhoPluginNekobi::deactivate()
|
|
{
|
|
if (fSynth.voice != nullptr)
|
|
nekobee_synth_all_voices_off(&fSynth);
|
|
}
|
|
|
|
void DistrhoPluginNekobi::run(const float**, float** outputs, uint32_t frames, const MidiEvent* midiEvents, uint32_t midiEventCount)
|
|
{
|
|
uint32_t framesDone = 0;
|
|
uint32_t curEventIndex = 0;
|
|
uint32_t burstSize;
|
|
|
|
float* out = outputs[0];
|
|
|
|
if (fSynth.voice == nullptr || ! dssp_voicelist_mutex_trylock(&fSynth))
|
|
{
|
|
std::memset(out, 0, sizeof(float)*frames);
|
|
return;
|
|
}
|
|
|
|
// ignore midi input if bypassed
|
|
if (fParams.bypass)
|
|
midiEventCount = 0;
|
|
|
|
while (framesDone < frames)
|
|
{
|
|
if (fSynth.nugget_remains == 0)
|
|
fSynth.nugget_remains = XSYNTH_NUGGET_SIZE;
|
|
|
|
/* process any ready events */
|
|
while (curEventIndex < midiEventCount && framesDone == midiEvents[curEventIndex].frame)
|
|
{
|
|
if (midiEvents[curEventIndex].size > MidiEvent::kDataSize)
|
|
continue;
|
|
|
|
nekobee_handle_raw_event(&fSynth, midiEvents[curEventIndex].size, midiEvents[curEventIndex].data);
|
|
curEventIndex++;
|
|
}
|
|
|
|
/* calculate the sample count (burstSize) for the next nekobee_voice_render() call to be the smallest of:
|
|
* - control calculation quantization size (XSYNTH_NUGGET_SIZE, in samples)
|
|
* - the number of samples remaining in an already-begun nugget (synth->nugget_remains)
|
|
* - the number of samples until the next event is ready
|
|
* - the number of samples left in this run
|
|
*/
|
|
burstSize = XSYNTH_NUGGET_SIZE;
|
|
|
|
/* we're still in the middle of a nugget, so reduce the burst size
|
|
* to end when the nugget ends */
|
|
if (fSynth.nugget_remains < burstSize)
|
|
burstSize = fSynth.nugget_remains;
|
|
|
|
/* reduce burst size to end when next event is ready */
|
|
if (curEventIndex < midiEventCount && midiEvents[curEventIndex].frame - framesDone < burstSize)
|
|
burstSize = midiEvents[curEventIndex].frame - framesDone;
|
|
|
|
/* reduce burst size to end at end of this run */
|
|
if (frames - framesDone < burstSize)
|
|
burstSize = frames - framesDone;
|
|
|
|
/* render the burst */
|
|
nekobee_synth_render_voices(&fSynth, out + framesDone, burstSize, (burstSize == fSynth.nugget_remains));
|
|
framesDone += burstSize;
|
|
fSynth.nugget_remains -= burstSize;
|
|
}
|
|
|
|
dssp_voicelist_mutex_unlock(&fSynth);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
Plugin* createPlugin()
|
|
{
|
|
return new DistrhoPluginNekobi();
|
|
}
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
END_NAMESPACE_DISTRHO
|