77 lines
2.1 KiB
C
77 lines
2.1 KiB
C
|
|
#include "config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
|
|
#include "alMain.h"
|
|
#include "alError.h"
|
|
#include "evtqueue.h"
|
|
#include "rwlock.h"
|
|
#include "alu.h"
|
|
|
|
#include "midi/base.h"
|
|
|
|
typedef struct DSynth {
|
|
DERIVE_FROM_TYPE(MidiSynth);
|
|
} DSynth;
|
|
|
|
static void DSynth_Construct(DSynth *self, ALCdevice *device);
|
|
static DECLARE_FORWARD(DSynth, MidiSynth, void, Destruct)
|
|
static DECLARE_FORWARD3(DSynth, MidiSynth, ALenum, selectSoundfonts, ALCcontext*, ALsizei, const ALuint*)
|
|
static DECLARE_FORWARD1(DSynth, MidiSynth, void, setGain, ALfloat)
|
|
static DECLARE_FORWARD(DSynth, MidiSynth, void, stop)
|
|
static DECLARE_FORWARD(DSynth, MidiSynth, void, reset)
|
|
static DECLARE_FORWARD1(DSynth, MidiSynth, void, update, ALCdevice*)
|
|
static void DSynth_process(DSynth *self, ALuint SamplesToDo, ALfloat (*restrict DryBuffer)[BUFFERSIZE], ALuint NumChannels);
|
|
DECLARE_DEFAULT_ALLOCATORS(DSynth)
|
|
DEFINE_MIDISYNTH_VTABLE(DSynth);
|
|
|
|
|
|
static void DSynth_Construct(DSynth *self, ALCdevice *device)
|
|
{
|
|
MidiSynth_Construct(STATIC_CAST(MidiSynth, self), device);
|
|
SET_VTABLE2(DSynth, MidiSynth, self);
|
|
}
|
|
|
|
|
|
static void DSynth_processQueue(DSynth *self, ALuint64 time)
|
|
{
|
|
EvtQueue *queue = &STATIC_CAST(MidiSynth, self)->EventQueue;
|
|
|
|
while(queue->pos < queue->size && queue->events[queue->pos].time <= time)
|
|
queue->pos++;
|
|
}
|
|
|
|
static void DSynth_process(DSynth *self, ALuint SamplesToDo, ALfloatBUFFERSIZE*restrict UNUSED(DryBuffer), ALuint UNUSED(NumChannels))
|
|
{
|
|
MidiSynth *synth = STATIC_CAST(MidiSynth, self);
|
|
ALuint64 curtime;
|
|
|
|
if(synth->State != AL_PLAYING)
|
|
return;
|
|
|
|
synth->SamplesDone += SamplesToDo;
|
|
synth->ClockBase += (synth->SamplesDone/synth->SampleRate) * MIDI_CLOCK_RES;
|
|
synth->SamplesDone %= synth->SampleRate;
|
|
|
|
curtime = MidiSynth_getTime(synth);
|
|
DSynth_processQueue(self, maxi64(curtime-1, 0));
|
|
}
|
|
|
|
|
|
MidiSynth *DSynth_create(ALCdevice *device)
|
|
{
|
|
DSynth *synth = DSynth_New(sizeof(*synth));
|
|
if(!synth)
|
|
{
|
|
ERR("Failed to allocate DSynth\n");
|
|
return NULL;
|
|
}
|
|
memset(synth, 0, sizeof(*synth));
|
|
DSynth_Construct(synth, device);
|
|
return STATIC_CAST(MidiSynth, synth);
|
|
}
|