added generic voice allocator
This commit is contained in:
25
synth/note_event.h
Normal file
25
synth/note_event.h
Normal file
@@ -0,0 +1,25 @@
|
||||
#pragma once
|
||||
|
||||
namespace trnr::lib::synth {
|
||||
|
||||
enum note_event_type {
|
||||
note_on = 0,
|
||||
note_off
|
||||
};
|
||||
|
||||
class note_event {
|
||||
public:
|
||||
note_event_type type;
|
||||
int midi_note;
|
||||
float velocity;
|
||||
int offset;
|
||||
|
||||
note_event(note_event_type type, int _midi_note, float _velocity, int _offset)
|
||||
: type { type }
|
||||
, midi_note { _midi_note }
|
||||
, velocity { _velocity }
|
||||
, offset { _offset }
|
||||
{
|
||||
}
|
||||
};
|
||||
}
|
||||
148
synth/voice_allocator.h
Normal file
148
synth/voice_allocator.h
Normal file
@@ -0,0 +1,148 @@
|
||||
#pragma once
|
||||
#include <vector>
|
||||
#include "note_event.h"
|
||||
|
||||
namespace trnr::lib::synth {
|
||||
|
||||
template <typename t_voice>
|
||||
class voice_allocator {
|
||||
public:
|
||||
std::vector<t_voice> voices;
|
||||
|
||||
voice_allocator(const double& _samplerate)
|
||||
: samplerate { _samplerate }
|
||||
, voices(8, t_voice(_samplerate))
|
||||
{
|
||||
}
|
||||
|
||||
void set_voice_count(const int& voice_count)
|
||||
{
|
||||
voices.resize(voice_count, voices.at(0));
|
||||
}
|
||||
|
||||
void note_on(const note_event& event)
|
||||
{
|
||||
t_voice* voice = get_free_voice(event.midi_note);
|
||||
|
||||
if (voice == nullptr) {
|
||||
voice = steal_voice();
|
||||
}
|
||||
|
||||
if (voice != nullptr) {
|
||||
voice->note_on(event.midi_note, event.velocity);
|
||||
}
|
||||
}
|
||||
|
||||
void note_off(const note_event& event)
|
||||
{
|
||||
for (auto it = voices.begin(); it != voices.end(); it++) {
|
||||
if ((*it).MidiNote == event.midi_note) {
|
||||
(*it).note_off();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void access(std::function<void(t_voice&)> f)
|
||||
{
|
||||
std::for_each(voices.begin(), voices.end(), f);
|
||||
}
|
||||
|
||||
void process_samples(double** _outputs, int _start_index, int _block_size)
|
||||
{
|
||||
for (int s = _start_index; s < _start_index + _block_size; s++) {
|
||||
|
||||
process_events(s);
|
||||
|
||||
float voices_signal = 0.;
|
||||
std::for_each(voices.begin(), voices.end(), [&voices_signal](t_voice& voice) { voices_signal += (voice.ProcessSample() / 3.); });
|
||||
|
||||
_outputs[0][s] = voices_signal;
|
||||
_outputs[1][s] = voices_signal;
|
||||
}
|
||||
}
|
||||
|
||||
void add_event(note_event event)
|
||||
{
|
||||
input_queue.push_back(event);
|
||||
}
|
||||
|
||||
bool voices_active()
|
||||
{
|
||||
bool voices_active = false;
|
||||
|
||||
for (auto it = voices.begin(); it != voices.end(); it++) {
|
||||
bool busy = (*it).is_busy();
|
||||
voices_active |= busy;
|
||||
}
|
||||
|
||||
return voices_active;
|
||||
}
|
||||
|
||||
void set_samplerate(double _samplerate)
|
||||
{
|
||||
this->samplerate = _samplerate;
|
||||
for (int i = 0; i < voices.size(); i++) {
|
||||
voices.at(i).set_samplerate(_samplerate);
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
double samplerate;
|
||||
std::vector<note_event> input_queue;
|
||||
|
||||
t_voice* get_free_voice(float frequency)
|
||||
{
|
||||
t_voice* voice = nullptr;
|
||||
|
||||
for (auto it = voices.begin(); it != voices.end(); it++) {
|
||||
if (!(*it).is_busy()) {
|
||||
voice = &*it;
|
||||
}
|
||||
}
|
||||
|
||||
return voice;
|
||||
}
|
||||
|
||||
t_voice* steal_voice()
|
||||
{
|
||||
t_voice* free_voice = nullptr;
|
||||
|
||||
for (auto it = voices.begin(); it != voices.end(); it++) {
|
||||
if (!(*it).gate) {
|
||||
free_voice = &*it;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (free_voice == nullptr) {
|
||||
free_voice = &voices.at(0);
|
||||
}
|
||||
|
||||
return free_voice;
|
||||
}
|
||||
|
||||
void process_events(int _start_index)
|
||||
{
|
||||
auto iterator = input_queue.begin();
|
||||
while (iterator != input_queue.end()) {
|
||||
|
||||
note_event& event = *iterator;
|
||||
if (event.offset == _start_index) {
|
||||
|
||||
switch (event.type) {
|
||||
case note_event_type::note_on:
|
||||
note_on(event);
|
||||
break;
|
||||
case note_event_type::note_off:
|
||||
note_off(event);
|
||||
break;
|
||||
}
|
||||
|
||||
iterator = input_queue.erase(iterator);
|
||||
} else {
|
||||
iterator++;
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user