You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

216 lines
7.8KB

  1. // Copyright 2013 Olivier Gillet.
  2. //
  3. // Author: Olivier Gillet (ol.gillet@gmail.com)
  4. //
  5. // Permission is hereby granted, free of charge, to any person obtaining a copy
  6. // of this software and associated documentation files (the "Software"), to deal
  7. // in the Software without restriction, including without limitation the rights
  8. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. // copies of the Software, and to permit persons to whom the Software is
  10. // furnished to do so, subject to the following conditions:
  11. //
  12. // The above copyright notice and this permission notice shall be included in
  13. // all copies or substantial portions of the Software.
  14. //
  15. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  20. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  21. // THE SOFTWARE.
  22. //
  23. // See http://creativecommons.org/licenses/MIT/ for more information.
  24. //
  25. // -----------------------------------------------------------------------------
  26. //
  27. // Calibration settings.
  28. #include "elements/cv_scaler.h"
  29. #include <algorithm>
  30. #include "stmlib/dsp/dsp.h"
  31. #include "stmlib/system/storage.h"
  32. #include "elements/dsp/part.h"
  33. #include "elements/dsp/patch.h"
  34. namespace elements {
  35. using namespace std;
  36. using namespace stmlib;
  37. /* static */
  38. Law CvScaler::law_[POT_LAST] = {
  39. LAW_LINEAR, // POT_EXCITER_ENVELOPE_SHAPE,
  40. LAW_LINEAR, // POT_EXCITER_BOW_LEVEL,
  41. LAW_LINEAR, // POT_EXCITER_BOW_TIMBRE,
  42. LAW_QUADRATIC_BIPOLAR, // POT_EXCITER_BOW_TIMBRE_ATTENUVERTER,
  43. LAW_LINEAR, // POT_EXCITER_BLOW_LEVEL,
  44. LAW_LINEAR, // POT_EXCITER_BLOW_META,
  45. LAW_QUADRATIC_BIPOLAR, // POT_EXCITER_BLOW_META_ATTENUVERTER,
  46. LAW_LINEAR, // POT_EXCITER_BLOW_TIMBRE,
  47. LAW_QUADRATIC_BIPOLAR, // POT_EXCITER_BLOW_TIMBRE_ATTENUVERTER,
  48. LAW_LINEAR, // POT_EXCITER_STRIKE_LEVEL,
  49. LAW_LINEAR, // POT_EXCITER_STRIKE_META,
  50. LAW_QUADRATIC_BIPOLAR, // POT_EXCITER_STRIKE_META_ATTENUVERTER,
  51. LAW_LINEAR, // POT_EXCITER_STRIKE_TIMBRE,
  52. LAW_QUADRATIC_BIPOLAR, // POT_EXCITER_STRIKE_TIMBRE_ATTENUVERTER,
  53. LAW_QUANTIZED_NOTE, // POT_RESONATOR_COARSE,
  54. LAW_QUADRATIC_BIPOLAR, // POT_RESONATOR_FINE,
  55. LAW_QUARTIC_BIPOLAR, // POT_RESONATOR_FM_ATTENUVERTER,
  56. LAW_LINEAR, // POT_RESONATOR_GEOMETRY,
  57. LAW_QUADRATIC_BIPOLAR, // POT_RESONATOR_GEOMETRY_ATTENUVERTER,
  58. LAW_LINEAR, // POT_RESONATOR_BRIGHTNESS,
  59. LAW_QUADRATIC_BIPOLAR, // POT_RESONATOR_BRIGHTNESS_ATTENUVERTER,
  60. LAW_LINEAR, // POT_RESONATOR_DAMPING,
  61. LAW_QUADRATIC_BIPOLAR, // POT_RESONATOR_DAMPING_ATTENUVERTER,
  62. LAW_LINEAR, // POT_RESONATOR_POSITION,
  63. LAW_QUADRATIC_BIPOLAR, // POT_RESONATOR_POSITION_ATTENUVERTER,
  64. LAW_LINEAR, // POT_SPACE,
  65. LAW_QUADRATIC_BIPOLAR // POT_SPACE_ATTENUVERTER
  66. };
  67. Storage<1> storage;
  68. void CvScaler::Init() {
  69. pots_.Init();
  70. cv_.Init(false);
  71. gate_input_.Init();
  72. if (!storage.ParsimoniousLoad(&calibration_settings_, &version_token_)) {
  73. calibration_settings_.pitch_offset = 66.67f;
  74. calibration_settings_.pitch_scale = -84.26f;
  75. for (size_t i = 0; i < CV_ADC_CHANNEL_LAST; ++i) {
  76. calibration_settings_.offset[i] = 10.0f * 20.0f / 120.0f / 3.3f;
  77. }
  78. calibration_settings_.boot_in_easter_egg_mode = false;
  79. SaveCalibration();
  80. }
  81. note_ = 0.0f;
  82. modulation_ = 0.0f;
  83. fill(&pot_raw_[0], &pot_raw_[POT_LAST], 0.0f);
  84. fill(&pot_lp_[0], &pot_lp_[POT_LAST], 0.0f);
  85. fill(&pot_quantized_[0], &pot_quantized_[POT_LAST], 0.0f);
  86. }
  87. void CvScaler::SaveCalibration() {
  88. storage.ParsimoniousSave(calibration_settings_, &version_token_);
  89. }
  90. void CvScaler::ReadPanelPots() {
  91. // Read one pot from the front panel, and map it to the correct range.
  92. pots_.Scan();
  93. uint8_t index = pots_.last_read();
  94. switch (law_[index]) {
  95. case LAW_LINEAR:
  96. pot_raw_[index] = static_cast<float>(pots_.value(index)) / 65536.0f;
  97. break;
  98. case LAW_QUADRATIC_BIPOLAR:
  99. {
  100. float x = static_cast<float>(pots_.value(index)) / 32768.0f - 1.0f;
  101. float x2 = x * x * 3.3f;
  102. pot_raw_[index] = x < 0.0f ? -x2 : x2;
  103. }
  104. break;
  105. case LAW_QUARTIC_BIPOLAR:
  106. {
  107. float x = static_cast<float>(pots_.value(index)) / 32768.0f - 1.0f;
  108. float x2 = x * x;
  109. float x4 = x2 * x2 * 3.3f;
  110. pot_raw_[index] = x < 0.0f ? -x4 : x4;
  111. }
  112. break;
  113. case LAW_QUANTIZED_NOTE:
  114. {
  115. // 5 octaves.
  116. float note = 60.0f * static_cast<float>(pots_.value(index)) / 65536.0f;
  117. pot_raw_[index] += 0.5f * (note - pot_raw_[index]);
  118. float n = pot_raw_[index];
  119. float hysteresis = n - pot_quantized_[index] > 0.0f ? -0.3f : +0.3f;
  120. pot_quantized_[index] = static_cast<int32_t>(n + hysteresis + 0.5f);
  121. }
  122. break;
  123. }
  124. // Low-pass filter all front-panel pots.
  125. for (size_t i = 0; i < POT_LAST; ++i) {
  126. pot_lp_[i] += 0.01f * (pot_raw_[i] - pot_lp_[i]);
  127. }
  128. }
  129. #define BIND(destination, NAME, lp_coefficient, min, max) \
  130. { \
  131. const float* offset = calibration_settings_.offset; \
  132. float pot = pot_lp_[POT_ ## NAME]; \
  133. float cv = offset[CV_ADC_ ## NAME] - cv_.float_value(CV_ADC_ ## NAME); \
  134. float value = pot + pot_lp_[POT_ ## NAME ## _ATTENUVERTER] * cv; \
  135. CONSTRAIN(value, min, max); \
  136. if (lp_coefficient != 1.0f) { \
  137. destination += (value - destination) * lp_coefficient; \
  138. } else { \
  139. destination = value; \
  140. } \
  141. }
  142. void CvScaler::Read(
  143. Patch* patch,
  144. PerformanceState* state) {
  145. ReadPanelPots();
  146. patch->exciter_envelope_shape = pot_lp_[POT_EXCITER_ENVELOPE_SHAPE];
  147. patch->exciter_bow_level = pot_lp_[POT_EXCITER_BOW_LEVEL];
  148. patch->exciter_blow_level = pot_lp_[POT_EXCITER_BLOW_LEVEL];
  149. patch->exciter_strike_level = pot_lp_[POT_EXCITER_STRIKE_LEVEL];
  150. BIND(patch->exciter_bow_timbre, EXCITER_BOW_TIMBRE, 1.0f, 0.0f, 0.9995f);
  151. BIND(patch->exciter_blow_meta, EXCITER_BLOW_META, 0.05f, 0.0f, 0.9995f);
  152. BIND(patch->exciter_blow_timbre, EXCITER_BLOW_TIMBRE, 1.0f, 0.0f, 0.9995f);
  153. BIND(patch->exciter_strike_meta, EXCITER_STRIKE_META, 0.05f, 0.0f, 0.9995f);
  154. BIND(patch->exciter_strike_timbre, EXCITER_STRIKE_TIMBRE, 1.0f, 0.0f, 0.995f);
  155. BIND(patch->resonator_geometry, RESONATOR_GEOMETRY, 0.05f, 0.0f, 0.9995f);
  156. BIND(patch->resonator_brightness, RESONATOR_BRIGHTNESS, 1.0f, 0.0f, 0.9995f);
  157. BIND(patch->resonator_damping, RESONATOR_DAMPING, 1.0f, 0.0f, 0.9995f);
  158. BIND(patch->resonator_position, RESONATOR_POSITION, 0.01f, 0.0f, 0.9995f);
  159. BIND(patch->space, SPACE, 0.01f, 0.0f, 2.0f);
  160. float note = calibration_settings_.pitch_offset;
  161. note += cv_.float_value(CV_ADC_V_OCT) * calibration_settings_.pitch_scale;
  162. float interval = note - note_;
  163. // When a pitch difference of more than 1 quartertone is observed, do
  164. // not attempt to slew-limit and jump straight to the right pitch.
  165. if (interval < -0.4f || interval > 0.4f) {
  166. note_ = note;
  167. } else {
  168. note_ += 0.1f * interval;
  169. }
  170. float modulation = pot_lp_[POT_RESONATOR_FM_ATTENUVERTER] * 49.5f *
  171. (calibration_settings_.offset[CV_ADC_FM] - cv_.float_value(CV_ADC_FM));
  172. modulation_ += 0.5f * (modulation - modulation_);
  173. state->modulation = modulation_;
  174. state->note = note_;
  175. state->note += pot_quantized_[POT_RESONATOR_COARSE] + 19.0f;
  176. state->note += pot_lp_[POT_RESONATOR_FINE] * (2.0f / 3.3f);
  177. state->strength = 1.0f - cv_.float_value(CV_ADC_PRESSURE);
  178. CONSTRAIN(state->strength, 0.0f, 1.0f);
  179. CONSTRAIN(state->modulation, -60.0f, 60.0f);
  180. // Scan the gate input 1ms after the other CVs to prevent lag.
  181. state->gate = previous_gate_;
  182. previous_gate_ = gate_;
  183. gate_ = gate_input_.Read();
  184. cv_.Convert();
  185. }
  186. } // namespace elements