Skip to content
Permalink
be7b73c0d3
Go to file
 
 
Cannot retrieve contributors at this time
executable file 162 lines (133 sloc) 4.23 KB
// 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_