|
- // 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.
- //
- // -----------------------------------------------------------------------------
- //
- // Calibration settings.
-
- #include "warps/cv_scaler.h"
-
- #include <algorithm>
- #include <cmath>
-
- #include "stmlib/dsp/dsp.h"
- #include "stmlib/system/storage.h"
- #include "stmlib/utils/random.h"
-
- #include "warps/dsp/parameters.h"
- #include "warps/resources.h"
-
- namespace warps {
-
- using namespace std;
- using namespace stmlib;
-
- void CvScaler::Init(CalibrationData* calibration_data) {
- calibration_data_ = calibration_data;
-
- adc_.Init();
- normalization_probe_.Init();
- note_cv_ = 0.0f;
- note_pot_ = 0.0f;
- fill(&lp_state_[0], &lp_state_[ADC_LAST], 0.0f);
- for (int32_t i = 0; i < 4; ++i) {
- normalization_detector_[i].Init(0.01f, 0.5f);
- }
-
- normalization_probe_enabled_ = true;
- }
-
- float CvScaler::UnwrapPot(float x) const {
- return Interpolate(lut_pot_curve, x, 512.0f);
- }
-
- #define BIND(destination, NAME, unwrap, scale, lp_coefficient, attenuate) \
- { \
- lp_state_[ADC_ ## NAME ## _POT] += 0.33f * lp_coefficient * (adc_.float_value(ADC_ ## NAME ## _POT) - lp_state_[ADC_ ## NAME ## _POT]); \
- lp_state_[ADC_ ## NAME ## _CV] += lp_coefficient * (adc_.float_value(ADC_ ## NAME ## _CV) - lp_state_[ADC_ ## NAME ## _CV]); \
- float pot = lp_state_[ADC_ ## NAME ## _POT]; \
- if (unwrap) pot = UnwrapPot(pot); \
- float cv = calibration_data_->offset[ADC_ ## NAME ## _CV] - lp_state_[ADC_ ## NAME ## _CV]; \
- float value = attenuate ? (pot * pot * cv * scale) : (pot + cv * scale); \
- CONSTRAIN(value, 0.0f, 1.0f); \
- destination = value; \
- }
-
- void CvScaler::DetectNormalization() {
- // Check if the value read by the ADC is correlated with the noise sent to the
- // switch. If so, we can conclude that nothing is connected in the input.
- float x = normalization_probe_value_[1] ? 1.0f : -1.0f;
- for (size_t i = 0; i < 2; ++i) {
- float y = adc_.float_value(ADC_LEVEL_1_CV + i) - calibration_data_->normalization_detection_threshold[i];
- if (y > -0.1f && y < 0.1f) {
- y = y > 0.0f ? -1.0f : 1.0f;
- } else {
- y = 0.0f;
- }
- normalization_detector_[i].Process(x, y);
- }
- // Flip normalization probe for next round.
- normalization_probe_value_[1] = normalization_probe_value_[0];
- normalization_probe_value_[0] = (Random::GetWord() >> 31) & 1;
- bool new_state = normalization_probe_enabled_
- ? normalization_probe_value_[0]
- : normalization_probe_forced_state_;
- normalization_probe_.Write(new_state);
- }
-
- void CvScaler::DetectAudioNormalization(Codec::Frame* in, size_t size) {
- for (int32_t channel = 0; channel < 2; ++channel) {
- int32_t count = 0;
- short* input_samples = &in->l + channel;
- for (size_t i = 0; i < size; i += 16) {
- short s = input_samples[i * 2];
- if (s > 50 && s < 1500) {
- ++count;
- } else if (s > -1500 && s < -50) {
- --count;
- }
- }
- float y = static_cast<float>(count) / static_cast<float>(size >> 4);
- float x = normalization_probe_value_[0] ? -1.0f : 1.0f;
-
- normalization_detector_[channel + 2].Process(x, y);
- if (normalization_detector_[channel + 2].normalized()) {
- for (size_t i = 0; i < size; ++i) {
- input_samples[i * 2] = 0;
- }
- }
- }
- }
-
- void CvScaler::Read(Parameters* p) {
- // Modulation parameters.
- BIND(p->channel_drive[0], LEVEL_1, false, 1.6f, 0.08f, true);
- BIND(p->channel_drive[1], LEVEL_2, false, 1.6f, 0.08f, true);
- BIND(p->modulation_algorithm, ALGORITHM, true, 2.0f, 0.08f, false);
- BIND(p->modulation_parameter, PARAMETER, false, 2.0f, 0.08f, false);
-
- // Prevent wavefolder bleed caused by a slight offset in the pot or ADC.
- if (p->modulation_algorithm <= 0.125f) {
- p->modulation_algorithm = p->modulation_algorithm * 1.08f - 0.01f;
- CONSTRAIN(p->modulation_algorithm, 0.0f, 1.0f);
- }
-
- // Easter egg parameter mappings.
- p->frequency_shift_pot = lp_state_[ADC_ALGORITHM_POT];
- float frequency_shift_cv = -lp_state_[ADC_ALGORITHM_CV];
- frequency_shift_cv += calibration_data_->offset[ADC_ALGORITHM_CV];
-
- p->frequency_shift_cv = frequency_shift_cv * 2.0f;
- CONSTRAIN(p->frequency_shift_cv, -1.0f, 1.0f);
-
- float phase_shift = lp_state_[ADC_ALGORITHM_POT] + frequency_shift_cv * 2.0f;
- CONSTRAIN(phase_shift, 0.0f, 1.0f);
- p->phase_shift = phase_shift;
-
- // Internal oscillator parameters.
- float note;
- note = calibration_data_->pitch_offset;
- note += adc_.float_value(ADC_LEVEL_1_CV) * calibration_data_->pitch_scale;
- float interval = note - note_cv_;
- if (interval < -0.4f || interval > 0.4f) {
- note_cv_ = note;
- } else {
- note_cv_ += 0.1f * interval;
- }
-
- note = 60.0f * adc_.float_value(ADC_LEVEL_1_POT) + 12.0f;
- note_pot_ += 0.1f * (note - note_pot_);
- p->note = note_pot_ + note_cv_;
-
- DetectNormalization();
-
- for (int32_t i = 0; i < 2; ++i) {
- if (normalization_detector_[i].normalized()) {
- float pot = lp_state_[ADC_LEVEL_1_POT + i];
- p->channel_drive[i] = pot * pot;
- }
- }
- if (normalization_detector_[0].normalized()) {
- p->note = note_pot_ + 24.0f;
- }
-
- adc_.Convert();
- }
-
- } // namespace warps
|