|
- // Copyright 2014 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.
- //
- // -----------------------------------------------------------------------------
- //
- // Vactrol.
-
- #include "streams/vactrol.h"
-
- #include "stmlib/utils/dsp.h"
-
- #include "streams/gain.h"
- #include "streams/resources.h"
-
- namespace streams {
-
- using namespace stmlib;
-
- void Vactrol::Init() {
- state_[0] = 0;
- state_[1] = 0;
- state_[2] = 0;
- state_[3] = 0;
-
- excite_ = 0;
- }
-
- void Vactrol::Process(
- int16_t audio,
- int16_t excite,
- uint16_t* gain,
- uint16_t* frequency) {
- // Smooth frequency amount parameters.
- frequency_amount_ += (target_frequency_amount_ - frequency_amount_) >> 8;
- frequency_offset_ += (target_frequency_offset_ - frequency_offset_) >> 8;
-
- int32_t input;
- int32_t error;
- int64_t coefficient = 0;
-
- if (excite < 0) {
- excite = 0;
- }
-
- // Simple plucked mode.
- if (plucked_) {
- if (gate_ == false) {
- if (excite > kSchmittTriggerThreshold) {
- gate_ = true;
- state_[0] = 32767 << 16;
- state_[1] = 32767 << 16;
- }
- } else {
- if (excite < (kSchmittTriggerThreshold >> 1)) {
- gate_ = false;
- }
- }
-
- // Filter the excitation pulses.
- state_[0] -= static_cast<int64_t>(
- state_[0]) * fast_decay_coefficient_ >> 31;
- state_[1] -= static_cast<int64_t>(
- state_[1]) * decay_coefficient_ >> 31;
-
- // VCF envelope.
- error = state_[0] - state_[2];
- coefficient = error > 0
- ? fast_attack_coefficient_ : fast_decay_coefficient_;
- state_[2] += static_cast<int64_t>(error) * coefficient >> 31;
-
- // VCA envelope.
- error = state_[1] - state_[3];
- coefficient = error > 0 ? fast_attack_coefficient_ : decay_coefficient_;
- // Increase the duration of the tail
- int64_t strength = error > 0 ? error : -error;
- coefficient = (coefficient >> 1) + (coefficient * strength >> 31);
- state_[3] += static_cast<int64_t>(error) * coefficient >> 31;
-
- uint16_t vcf_amount = state_[2] >> 16;
- uint16_t vca_mount = Interpolate1022(wav_gompertz, (state_[3] >> 2) * 3);
-
- *gain = kAboveUnityGain * vca_mount >> 15;
- *frequency = frequency_offset_ + \
- (frequency_amount_ * vcf_amount >> 15);
-
- return;
- }
-
- // Low-pass filter the negative edges to prevent fast pulse to immediately
- // decay before the vactrol has started reacting. This allows the EXCITE
- // input to be used for both controlling the vactrol or just plucking it
- // from a trigger.
- error = excite - excite_;
- coefficient = error > 0 ? (1 << 30) : (decay_coefficient_ << 1);
- excite_ += static_cast<int64_t>(error) * coefficient >> 31;
- excite = excite_;
-
- input = frequency_offset_;
- input += frequency_amount_ >> 1;
- input = (65535 + input) >> 1;
- input *= excite;
-
- state_[3] += static_cast<int64_t>(input - state_[3]) * 67976239 >> 31;
-
- error = input - state_[0];
- coefficient = 0;
- if (error > 0) {
- if (state_[1] > 0) {
- coefficient = attack_coefficient_;
- // Increase attack time when the photocell has been desensitized.
- coefficient += coefficient * (255 - (state_[2] >> 23)) >> 6;
- } else {
- coefficient = fast_attack_coefficient_;
- }
- } else {
- if (state_[1] < 0) {
- coefficient = decay_coefficient_;
- } else {
- coefficient = fast_decay_coefficient_;
- }
- }
- // First order.
- state_[0] += static_cast<int64_t>(error) * coefficient >> 31;
-
- // Second order.
- state_[1] += static_cast<int64_t>(error - state_[1]) * coefficient >> 31;
-
- // Memory effect.
- int32_t sensitivity = state_[0];
- if (sensitivity > (1 << 28)) {
- sensitivity = 1 << 31;
- } else {
- sensitivity <<= 3;
- }
- error = sensitivity - state_[2];
- if (error > 0) {
- // Get into the "sensitized" state in 1s.
- state_[2] += static_cast<int64_t>(error) * 138132 >> 31;
- } else {
- // Get out of the "sensitized" state in 60s.
- state_[2] += static_cast<int64_t>(error) * 1151 >> 31;
- }
-
- // Apply non-linearity.
- int32_t index = state_[0] >> 1;
-
- // A little hack to add overshoot...
- index += (state_[3] >> 15) * (state_[1] >> 15) >> 1;
- if (index < 0) {
- index = 0;
- } else if (index >= (1 << 30)) {
- index = (1 << 30) - 1;
- }
- uint16_t amplitude = index < 536870912
- ? Interpolate1022(wav_gompertz, static_cast<uint32_t>(index) << 3)
- : 32767;
- uint16_t cutoff = index >> 14;
- if (cutoff >= 32767) cutoff = 32767;
- cutoff = cutoff * cutoff >> 15;
- *gain = kAboveUnityGain * amplitude >> 15;
- *frequency = frequency_offset_ + \
- (frequency_amount_ * cutoff >> 15);
- }
-
- } // namespace streams
|