|
- // Copyright 2015 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.
-
- #include "rings/drivers/adc.h"
- #include "rings/drivers/codec.h"
- #include "rings/drivers/debug_pin.h"
- #include "rings/drivers/debug_port.h"
- #include "rings/drivers/system.h"
- #include "rings/drivers/version.h"
- #include "rings/dsp/part.h"
- #include "rings/dsp/strummer.h"
- #include "rings/dsp/string_synth_part.h"
- #include "rings/cv_scaler.h"
- #include "rings/settings.h"
- #include "rings/ui.h"
-
- // #define PROFILE_INTERRUPT 1
-
- using namespace rings;
- using namespace stmlib;
-
- uint16_t reverb_buffer[32768] __attribute__ ((section (".ccmdata")));
-
- Codec codec;
- CvScaler cv_scaler;
- DebugPort debug_port;
- Part part;
- Settings settings;
- StringSynthPart string_synth;
- Strummer strummer;
- Ui ui;
-
- // Default interrupt handlers.
- extern "C" {
-
- int __errno;
-
- void NMI_Handler() { }
- void HardFault_Handler() { while (1); }
- void MemManage_Handler() { while (1); }
- void BusFault_Handler() { while (1); }
- void UsageFault_Handler() { while (1); }
- void SVC_Handler() { }
- void DebugMon_Handler() { }
- void PendSV_Handler() { }
-
- void SysTick_Handler() {
- ui.Poll();
- if (settings.freshly_baked()) {
- if (debug_port.readable()) {
- uint8_t command = debug_port.Read();
- uint8_t response = ui.HandleFactoryTestingRequest(command);
- debug_port.Write(response);
- }
- }
- }
-
- }
-
- float in[kMaxBlockSize];
- float out[kMaxBlockSize];
- float aux[kMaxBlockSize];
-
- const float kNoiseGateThreshold = 0.00003f;
- float in_level = 0.0f;
-
- void FillBuffer(Codec::Frame* input, Codec::Frame* output, size_t size) {
- #ifdef PROFILE_INTERRUPT
- TIC
- #endif // PROFILE_INTERRUPT
- PerformanceState performance_state;
- Patch patch;
-
- cv_scaler.DetectAudioNormalization(input, size);
- cv_scaler.Read(&patch, &performance_state);
-
- if (settings.state().easter_egg) {
- for (size_t i = 0; i < size; ++i) {
- in[i] = static_cast<float>(input[i].r) / 32768.0f;
- }
- strummer.Process(NULL, size, &performance_state);
- string_synth.Process(performance_state, patch, in, out, aux, size);
- } else {
- // Apply noise gate.
- for (size_t i = 0; i < size; ++i) {
- float in_sample = static_cast<float>(input[i].r) / 32768.0f;
- float error, gain;
- error = in_sample * in_sample - in_level;
- in_level += error * (error > 0.0f ? 0.1f : 0.0001f);
- gain = in_level <= kNoiseGateThreshold
- ? (1.0f / kNoiseGateThreshold) * in_level : 1.0f;
- in[i] = gain * in_sample;
- }
- strummer.Process(in, size, &performance_state);
- part.Process(performance_state, patch, in, out, aux, size);
- }
-
- for (size_t i = 0; i < size; ++i) {
- output[i].l = Clip16(static_cast<int32_t>(out[i] * 32768.0f));
- output[i].r = Clip16(static_cast<int32_t>(aux[i] * 32768.0f));
- }
- ui.set_strumming_flag(performance_state.strum);
- #ifdef PROFILE_INTERRUPT
- TOC
- #endif // PROFILE_INTERRUPT
- }
-
- void Init() {
- System sys;
- Version version;
-
- sys.Init(true);
- version.Init();
-
- strummer.Init(0.01f, kSampleRate / kMaxBlockSize);
- part.Init(reverb_buffer);
- string_synth.Init(reverb_buffer);
-
- settings.Init();
- cv_scaler.Init(settings.mutable_calibration_data());
- ui.Init(&settings, &cv_scaler, &part, &string_synth);
-
- if (!codec.Init(!version.revised(), kSampleRate)) {
- ui.Panic();
- }
- if (!codec.Start(kMaxBlockSize, &FillBuffer)) {
- ui.Panic();
- }
- codec.set_line_input_gain(22);
-
- if (settings.freshly_baked()) {
- #ifdef PROFILE_INTERRUPT
- DebugPin::Init();
- #else
- debug_port.Init();
- #endif // PROFILE_INTERRUPT
- }
- sys.StartTimers();
- }
-
- int main(void) {
- Init();
- while (1) {
- ui.DoEvents();
- }
- }
|