Files
-
blinds / hardware_design / pcb / blinds_v60.brd
-
blinds / hardware_design / pcb / blinds_v60.sch
-
braids / hardware_design / pcb / braids_v50.brd
-
braids / hardware_design / pcb / braids_v50.sch
-
branches / hardware_design / pcb / branches_v40.brd
-
branches / hardware_design / pcb / branches_v40.sch
-
clouds / hardware_design / pcb / clouds_v30.brd
-
clouds / hardware_design / pcb / clouds_v30.sch
-
ears / hardware_design / panel / ears_panel_v30.brd
-
ears / hardware_design / panel / ears_panel_v30.sch
-
ears / hardware_design / pcb / ears_v40.brd
-
ears / hardware_design / pcb / ears_v40.sch
-
edges / hardware_design / pcb / edges_expander_v01.brd
-
edges / hardware_design / pcb / edges_expander_v01.sch
-
edges / hardware_design / pcb / edges_v20.brd
-
edges / hardware_design / pcb / edges_v20.sch
-
elements / hardware_design / pcb / elements_v02.brd
-
elements / hardware_design / pcb / elements_v02.sch
-
frames / hardware_design / pcb / frames_v03.brd
-
frames / hardware_design / pcb / frames_v03.sch
-
grids / hardware_design / pcb / grids_v02.brd
-
grids / hardware_design / pcb / grids_v02.sch
-
kinks / hardware_design / pcb / kinks_v41.brd
-
kinks / hardware_design / pcb / kinks_v41.sch
-
links / hardware_design / pcb / links_v40.brd
-
links / hardware_design / pcb / links_v40.sch
-
marbles / hardware_design / pcb / marbles_v70.brd
-
marbles / hardware_design / pcb / marbles_v70.sch
-
peaks / hardware_design / pcb / peaks_v30.brd
-
peaks / hardware_design / pcb / peaks_v30.sch
-
plaits / hardware_design / pcb / plaits_v50.brd
-
plaits / hardware_design / pcb / plaits_v50.sch
-
rings / hardware_design / pcb / rings_v30.brd
-
rings / hardware_design / pcb / rings_v30.sch
-
ripples / hardware_design / pcb / ripples_v40.brd
-
ripples / hardware_design / pcb / ripples_v40.sch
-
shades / hardware_design / pcb / shades_v30.brd
-
shades / hardware_design / pcb / shades_v30.sch
-
shelves / hardware_design / pcb / shelves_expander_v10.brd
-
shelves / hardware_design / pcb / shelves_expander_v10.sch
-
shelves / hardware_design / pcb / shelves_v05.brd
-
shelves / hardware_design / pcb / shelves_v05.sch
-
stages / hardware_design / pcb / stages_v70.brd
-
stages / hardware_design / pcb / stages_v70.sch
-
streams / hardware_design / pcb / streams_v02_bargraph.brd
-
streams / hardware_design / pcb / streams_v02_bargraph.sch
-
streams / hardware_design / pcb / streams_v05.brd
-
streams / hardware_design / pcb / streams_v05.sch
-
tides / hardware_design / pcb / tides_v40.brd
-
tides / hardware_design / pcb / tides_v40.sch
-
veils / hardware_design / pcb / veils_v40.brd
-
veils / hardware_design / pcb / veils_v40.sch
-
volts / hardware_design / pcb / volts_v01.brd
-
volts / hardware_design / pcb / volts_v01.sch
-
warps / hardware_design / pcb / warps_v30.brd
-
warps / hardware_design / pcb / warps_v30.sch
-
yarns / hardware_design / pcb / yarns_v03.brd
-
yarns / hardware_design / pcb / yarns_v03.sch
Last update 6 years 3 months
by
Olivier Gillet
fm_engine.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. // // ----------------------------------------------------------------------------- // // Classic 2-op FM found in Braids, Rings and Elements. #include "plaits/dsp/engine/fm_engine.h" #include "stmlib/dsp/parameter_interpolator.h" #include "plaits/resources.h" namespace plaits { using namespace stmlib; void FMEngine::Init(BufferAllocator* allocator) { carrier_phase_ = 0; modulator_phase_ = 0; sub_phase_ = 0; previous_carrier_frequency_ = a0; previous_modulator_frequency_ = a0; previous_amount_ = 0.0f; previous_feedback_ = 0.0f; previous_sample_ = 0.0f; } void FMEngine::Reset() { } inline float FMEngine::SinePM(uint32_t phase, float fm) const { phase += (static_cast<uint32_t>((fm + 4.0f) * 536870912.0f)) << 3; uint32_t integral = phase >> 22; float fractional = static_cast<float>(phase << 10) / 4294967296.0f; float a = lut_sine[integral]; float b = lut_sine[integral + 1]; return a + (b - a) * fractional; } const size_t kOversampling = 4; const size_t kFirHalfSize = 4; static const float fir_coefficient[kFirHalfSize] = { 0.02442415f, 0.09297315f, 0.16712938f, 0.21547332f, }; class Downsampler { public: Downsampler(float* state) { head_ = *state; tail_ = 0.0f; state_ = state; } ~Downsampler() { *state_ = head_; } inline void Accumulate(int i, float sample) { head_ += sample * fir_coefficient[3 - (i & 3)]; tail_ += sample * fir_coefficient[i & 3]; } inline float Read() { float value = head_; head_ = tail_; tail_ = 0.0f; return value; } private: float head_; float tail_; float* state_; DISALLOW_COPY_AND_ASSIGN(Downsampler); }; void FMEngine::Render( const EngineParameters& parameters, float* out, float* aux, size_t size, bool* already_enveloped) { // 4x oversampling const float note = parameters.note - 24.0f; const float ratio = Interpolate( lut_fm_frequency_quantizer, parameters.harmonics, 128.0f); float modulator_note = note + ratio; float target_modulator_frequency = NoteToFrequency(modulator_note); CONSTRAIN(target_modulator_frequency, 0.0f, 0.5f); // Reduce the maximum FM index for high pitched notes, to prevent aliasing. float hf_taming = 1.0f - (modulator_note - 72.0f) * 0.025f; CONSTRAIN(hf_taming, 0.0f, 1.0f); hf_taming *= hf_taming; ParameterInterpolator carrier_frequency( &previous_carrier_frequency_, NoteToFrequency(note), size); ParameterInterpolator modulator_frequency( &previous_modulator_frequency_, target_modulator_frequency, size); ParameterInterpolator amount_modulation( &previous_amount_, 2.0f * parameters.timbre * parameters.timbre * hf_taming, size); ParameterInterpolator feedback_modulation( &previous_feedback_, 2.0f * parameters.morph - 1.0f, size); Downsampler carrier_downsampler(&carrier_fir_); Downsampler sub_downsampler(&sub_fir_); while (size--) { const float amount = amount_modulation.Next(); const float feedback = feedback_modulation.Next(); float phase_feedback = feedback < 0.0f ? 0.5f * feedback * feedback : 0.0f; const uint32_t carrier_increment = static_cast<uint32_t>( 4294967296.0f * carrier_frequency.Next()); float _modulator_frequency = modulator_frequency.Next(); for (size_t j = 0; j < kOversampling; ++j) { modulator_phase_ += static_cast<uint32_t>(4294967296.0f * \ _modulator_frequency * (1.0f + previous_sample_ * phase_feedback)); carrier_phase_ += carrier_increment; sub_phase_ += carrier_increment >> 1; float modulator_fb = feedback > 0.0f ? 0.25f * feedback * feedback : 0.0f; float modulator = SinePM( modulator_phase_, modulator_fb * previous_sample_); float carrier = SinePM(carrier_phase_, amount * modulator); float sub = SinePM(sub_phase_, amount * carrier * 0.25f); ONE_POLE(previous_sample_, carrier, 0.05f); carrier_downsampler.Accumulate(j, carrier); sub_downsampler.Accumulate(j, sub); } *out++ = carrier_downsampler.Read(); *aux++ = sub_downsampler.Read(); } } } // namespace plaits