added generic voice allocator

This commit is contained in:
Christopher Herb
2023-07-07 14:42:10 +02:00
parent 01596ca8e8
commit d9bcd96276
2 changed files with 173 additions and 0 deletions

25
synth/note_event.h Normal file
View 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
View 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++;
}
}
}
};
}