Files

copied
Last update 6 years 3 months by Olivier Gillet
Filesplaitsdspphysical_modelling
..
delay_line.h
modal_voice.cc
modal_voice.h
resonator.cc
resonator.h
string.cc
string.h
string_voice.cc
string_voice.h
string.cc
// Copyright 2016 Olivier Gillet. // // Author: Olivier Gillet (ol.gillet@gmail.com) // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // // See http://creativecommons.org/licenses/MIT/ for more information. // // ----------------------------------------------------------------------------- // // Comb filter / KS string. "Lite" version of the implementation used in Rings. #include "plaits/dsp/physical_modelling/string.h" #include <cmath> #include "stmlib/dsp/dsp.h" #include "stmlib/dsp/parameter_interpolator.h" #include "stmlib/dsp/units.h" #include "stmlib/utils/random.h" #include "plaits/dsp/dsp.h" #include "plaits/resources.h" namespace plaits { using namespace std; using namespace stmlib; void String::Init(BufferAllocator* allocator) { string_.Init(allocator->Allocate<float>(kDelayLineSize)); stretch_.Init(allocator->Allocate<float>(kDelayLineSize / 4)); delay_ = 100.0f; Reset(); } void String::Reset() { string_.Reset(); stretch_.Reset(); iir_damping_filter_.Init(); dc_blocker_.Init(1.0f - 20.0f / kSampleRate); dispersion_noise_ = 0.0f; curved_bridge_ = 0.0f; out_sample_[0] = out_sample_[1] = 0.0f; src_phase_ = 0.0f; } void String::Process( float f0, float non_linearity_amount, float brightness, float damping, const float* in, float* out, size_t size) { if (non_linearity_amount <= 0.0f) { ProcessInternal<STRING_NON_LINEARITY_CURVED_BRIDGE>( f0, -non_linearity_amount, brightness, damping, in, out, size); } else { ProcessInternal<STRING_NON_LINEARITY_DISPERSION>( f0, non_linearity_amount, brightness, damping, in, out, size); } } template<StringNonLinearity non_linearity> void String::ProcessInternal( float f0, float non_linearity_amount, float brightness, float damping, const float* in, float* out, size_t size) { float delay = 1.0f / f0; CONSTRAIN(delay, 4.0f, kDelayLineSize - 4.0f); // If there is not enough delay time in the delay line, we play at the // lowest possible note and we upsample on the fly with a shitty linear // interpolator. We don't care because it's a corner case (f0 < 11.7Hz) float src_ratio = delay * f0; if (src_ratio >= 0.9999f) { // When we are above 11.7 Hz, we make sure that the linear interpolator // does not get in the way. src_phase_ = 1.0f; src_ratio = 1.0f; } float damping_cutoff = min( 12.0f + damping * damping * 60.0f + brightness * 24.0f, 84.0f); float damping_f = min(f0 * SemitonesToRatio(damping_cutoff), 0.499f); // Crossfade to infinite decay. if (damping >= 0.95f) { float to_infinite = 20.0f * (damping - 0.95f); brightness += to_infinite * (1.0f - brightness); damping_f += to_infinite * (0.4999f - damping_f); damping_cutoff += to_infinite * (128.0f - damping_cutoff); } iir_damping_filter_.set_f_q<FREQUENCY_FAST>(damping_f, 0.5f); float damping_compensation = Interpolate(lut_svf_shift, damping_cutoff, 1.0f); // Linearly interpolate delay time. ParameterInterpolator delay_modulation( &delay_, delay * damping_compensation, size); float stretch_point = non_linearity_amount * (2.0f - non_linearity_amount) * 0.225f; float stretch_correction = (160.0f / kSampleRate) * delay; CONSTRAIN(stretch_correction, 1.0f, 2.1f); float noise_amount_sqrt = non_linearity_amount > 0.75f ? 4.0f * (non_linearity_amount - 0.75f) : 0.0f; float noise_amount = noise_amount_sqrt * noise_amount_sqrt * 0.1f; float noise_filter = 0.06f + 0.94f * brightness * brightness; float bridge_curving_sqrt = non_linearity_amount; float bridge_curving = bridge_curving_sqrt * bridge_curving_sqrt * 0.01f; float ap_gain = -0.618f * non_linearity_amount / (0.15f + fabsf(non_linearity_amount)); while (size--) { src_phase_ += src_ratio; if (src_phase_ > 1.0f) { src_phase_ -= 1.0f; float delay = delay_modulation.Next(); float s = 0.0f; if (non_linearity == STRING_NON_LINEARITY_DISPERSION) { float noise = Random::GetFloat() - 0.5f; ONE_POLE(dispersion_noise_, noise, noise_filter) delay *= 1.0f + dispersion_noise_ * noise_amount; } else { delay *= 1.0f - curved_bridge_ * bridge_curving; } if (non_linearity == STRING_NON_LINEARITY_DISPERSION) { float ap_delay = delay * stretch_point; float main_delay = delay - ap_delay * (0.408f - stretch_point * 0.308f) * stretch_correction; if (ap_delay >= 4.0f && main_delay >= 4.0f) { s = string_.Read(main_delay); s = stretch_.Allpass(s, ap_delay, ap_gain); } else { s = string_.ReadHermite(delay); } } else { s = string_.ReadHermite(delay); } if (non_linearity == STRING_NON_LINEARITY_CURVED_BRIDGE) { float value = fabsf(s) - 0.025f; float sign = s > 0.0f ? 1.0f : -1.5f; curved_bridge_ = (fabsf(value) + value) * sign; } s += *in; CONSTRAIN(s, -20.0f, +20.0f); dc_blocker_.Process(&s, 1); s = iir_damping_filter_.Process<FILTER_MODE_LOW_PASS>(s); string_.Write(s); out_sample_[1] = out_sample_[0]; out_sample_[0] = s; } *out++ += Crossfade(out_sample_[1], out_sample_[0], src_phase_); in++; } } } // namespace plaits
Report a bug