Permalink
Cannot retrieve contributors at this time
| // Copyright 2012 Emilie Gillet. | |
| // | |
| // Author: Emilie Gillet (emilie.o.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. | |
| // | |
| // ----------------------------------------------------------------------------- | |
| // | |
| // Oscillator - analog style waveforms. | |
| #ifndef BRAIDS_ANALOG_OSCILLATOR_H_ | |
| #define BRAIDS_ANALOG_OSCILLATOR_H_ | |
| #include "stmlib/stmlib.h" | |
| #include <cstring> | |
| #include <cstdio> | |
| #include "braids/resources.h" | |
| namespace braids { | |
| enum AnalogOscillatorShape { | |
| OSC_SHAPE_SAW, | |
| OSC_SHAPE_VARIABLE_SAW, | |
| OSC_SHAPE_CSAW, | |
| OSC_SHAPE_SQUARE, | |
| OSC_SHAPE_TRIANGLE, | |
| OSC_SHAPE_SINE, | |
| OSC_SHAPE_TRIANGLE_FOLD, | |
| OSC_SHAPE_SINE_FOLD, | |
| OSC_SHAPE_BUZZ | |
| }; | |
| enum SyncMode { | |
| OSCILLATOR_SYNC_MODE_OFF, | |
| OSCILLATOR_SYNC_MODE_MASTER, | |
| OSCILLATOR_SYNC_MODE_SLAVE | |
| }; | |
| class AnalogOscillator { | |
| public: | |
| typedef void (AnalogOscillator::*RenderFn)( | |
| const uint8_t*, | |
| int16_t*, | |
| uint8_t*, | |
| size_t); | |
| AnalogOscillator() { } | |
| ~AnalogOscillator() { } | |
| inline void Init() { | |
| phase_ = 0; | |
| phase_increment_ = 1; | |
| high_ = false; | |
| parameter_ = previous_parameter_ = 0; | |
| aux_parameter_ = 0; | |
| discontinuity_depth_ = -16383; | |
| pitch_ = 60 << 7; | |
| next_sample_ = 0; | |
| } | |
| inline void set_shape(AnalogOscillatorShape shape) { | |
| shape_ = shape; | |
| } | |
| inline void set_pitch(int16_t pitch) { | |
| pitch_ = pitch; | |
| } | |
| inline void set_parameter(int16_t parameter) { | |
| parameter_ = parameter; | |
| } | |
| inline void set_aux_parameter(int16_t parameter) { | |
| aux_parameter_ = parameter; | |
| } | |
| inline uint32_t phase_increment() const { | |
| return phase_increment_; | |
| } | |
| inline void Reset() { | |
| phase_ = -phase_increment_; | |
| } | |
| void Render( | |
| const uint8_t* sync_in, | |
| int16_t* buffer, | |
| uint8_t* sync_out, | |
| size_t size); | |
| private: | |
| void RenderSquare(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderSaw(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderVariableSaw(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderCSaw(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderTriangle(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderSine(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderTriangleFold(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderSineFold(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| void RenderBuzz(const uint8_t*, int16_t*, uint8_t*, size_t); | |
| uint32_t ComputePhaseIncrement(int16_t midi_pitch); | |
| inline int32_t ThisBlepSample(uint32_t t) { | |
| if (t > 65535) { | |
| t = 65535; | |
| } | |
| return t * t >> 18; | |
| } | |
| inline int32_t NextBlepSample(uint32_t t) { | |
| if (t > 65535) { | |
| t = 65535; | |
| } | |
| t = 65535 - t; | |
| return -static_cast<int32_t>(t * t >> 18); | |
| } | |
| uint32_t phase_; | |
| uint32_t phase_increment_; | |
| uint32_t previous_phase_increment_; | |
| bool high_; | |
| int16_t parameter_; | |
| int16_t previous_parameter_; | |
| int16_t aux_parameter_; | |
| int16_t discontinuity_depth_; | |
| int16_t pitch_; | |
| int32_t next_sample_; | |
| AnalogOscillatorShape shape_; | |
| AnalogOscillatorShape previous_shape_; | |
| static RenderFn fn_table_[]; | |
| DISALLOW_COPY_AND_ASSIGN(AnalogOscillator); | |
| }; | |
| } // namespace braids | |
| #endif // BRAIDS_ANALOG_OSCILLATOR_H_ |