Files

copied
Last update 6 years 1 month by Olivier Gillet
Fileswarps
..
bootloader
drivers
dsp
hardware_design
resources
test
tools
__init__.py
cv_scaler.cc
cv_scaler.h
makefile
meter.h
resources.cc
resources.h
settings.cc
settings.h
ui.cc
ui.h
warps.cc
cv_scaler.cc
// 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
Report a bug