|
- // 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.
- //
- // -----------------------------------------------------------------------------
- //
- // Follower.
-
- #include "streams/follower.h"
-
- #include "stmlib/utils/dsp.h"
-
- #include "streams/gain.h"
- #include "streams/resources.h"
-
- namespace streams {
-
- using namespace stmlib;
-
- void Follower::Init() {
- analysis_low_.Init();
- analysis_low_.set_frequency(45 << 7);
- analysis_low_.set_resonance(0);
- analysis_medium_.Init();
- analysis_medium_.set_frequency(86 << 7);
- analysis_medium_.set_resonance(0);
-
- for (uint8_t i = 0; i < 3; ++i) {
- energy_[i][0] = energy_[i][1] = 0;
- follower_[i] = 0;
- follower_lp_[i] = 0;
- spectrum_[i] = 0;
- }
- centroid_ = 0;
- }
-
- void Follower::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;
-
- analysis_low_.Process(excite);
- analysis_medium_.Process(analysis_low_.hp());
-
- int32_t channel[3];
- channel[0] = analysis_low_.lp();
- channel[1] = analysis_medium_.lp();
- channel[2] = analysis_medium_.hp();
-
- int32_t envelope = 0;
- int32_t centroid_numerator = 0;
- int32_t centroid_denominator = 0;
- for (int32_t i = 0; i < 3; ++i) {
- int32_t energy = channel[i];
- energy *= energy;
-
- // Ride an ascending peak.
- if (energy_[i][0] < energy_[i][1] && energy_[i][1] < energy &&
- energy > follower_[i]) {
- follower_[i] = energy;
- }
- // Otherwise, hold and snap on local maxima.
- if (energy_[i][0] <= energy_[i][1] && energy_[i][1] >= energy) {
- follower_[i] = energy_[i][1];
- }
- energy_[i][0] = energy_[i][1];
- energy_[i][1] = energy;
-
- // Then let a low-pass filter smooth things out.
- int64_t error = follower_[i] - follower_lp_[i];
- if (error > 0) {
- follower_lp_[i] += error * attack_coefficient_[i] >> 31;
- } else {
- follower_lp_[i] += error * decay_coefficient_[i] >> 31;
- }
- envelope += follower_lp_[i] >> 13;
-
- // Integrate more slowly for spectrum estimation.
- if (only_filter_) {
- error = follower_lp_[i] - spectrum_[i];
- spectrum_[i] += error >> 6;
- } else {
- error = follower_[i] - spectrum_[i];
- spectrum_[i] += error >> 10;
- }
- centroid_numerator += i * (spectrum_[i] >> 1) >> 16;
- centroid_denominator += spectrum_[i] >> 16;
- }
-
- if (envelope > 65535) {
- envelope = 65535;
- } else if (envelope < 0) {
- envelope = 0;
- }
-
- uint16_t gain_mod = Interpolate824(lut_square_root, envelope << 16) >> 1;
- int32_t centroid = (centroid_numerator << 15) / (centroid_denominator + 1);
- if (gain_mod > 4096) {
- centroid_ = centroid;
- } else if (gain_mod > 2048) {
- centroid_ += (centroid - centroid_) >> 8;
- }
-
- *gain = gain_mod * kUnityGain >> 15;
- *frequency = frequency_offset_ + (centroid_ * frequency_amount_ >> 15);
-
- if (only_filter_) {
- *gain = *frequency;
- *frequency = 65535;
- }
- }
-
- } // namespace streams
|