aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChris Robinson <[email protected]>2013-11-28 03:57:26 -0800
committerChris Robinson <[email protected]>2013-11-28 03:57:45 -0800
commit7794e5366747a402d386f5fd559d0e433ed7cffb (patch)
tree3905b66ebad51cb48a3ebdbaa405defc23961a2f
parentbb0207d784ee2d5ebbdb6d7c26a4a626e62dbbed (diff)
Be a bit safer with MIDI state changes
-rw-r--r--OpenAL32/Include/alMidi.h6
-rw-r--r--OpenAL32/alMidi.c54
2 files changed, 30 insertions, 30 deletions
diff --git a/OpenAL32/Include/alMidi.h b/OpenAL32/Include/alMidi.h
index f2b29df0..d648e179 100644
--- a/OpenAL32/Include/alMidi.h
+++ b/OpenAL32/Include/alMidi.h
@@ -21,6 +21,12 @@ typedef struct MidiSynth {
ALuint SampleRate;
ALdouble SamplesPerTick;
+ /* NOTE: This rwlock is for the state and soundfont. The EventQueue and
+ * related must instead use the device lock as they're used in the mixer
+ * thread.
+ */
+ RWLock Lock;
+
volatile ALenum State;
const struct MidiSynthVtable *vtbl;
diff --git a/OpenAL32/alMidi.c b/OpenAL32/alMidi.c
index 0cbe11b2..3bc61aae 100644
--- a/OpenAL32/alMidi.c
+++ b/OpenAL32/alMidi.c
@@ -30,13 +30,15 @@ static void MidiSynth_Construct(MidiSynth *self, ALCdevice *device)
{
InitEvtQueue(&self->EventQueue);
+ RWLockInit(&self->Lock);
+
+ self->State = AL_INITIAL;
+
self->LastEvtTime = 0;
self->NextEvtTime = UINT64_MAX;
self->SamplesSinceLast = 0.0;
self->SamplesToNext = 0.0;
- self->State = AL_INITIAL;
-
self->SampleRate = device->Frequency;
self->SamplesPerTick = (ALdouble)self->SampleRate / TICKS_PER_SECOND;
MidiSynth_updateSpeed(self);
@@ -49,7 +51,9 @@ static void MidiSynth_Destruct(MidiSynth *self)
static inline void MidiSynth_setState(MidiSynth *self, ALenum state)
{
+ WriteLock(&self->Lock);
ExchangeInt(&self->State, state);
+ WriteUnlock(&self->Lock);
}
ALuint64 MidiSynth_getTime(const MidiSynth *self)
@@ -107,11 +111,6 @@ static ALenum MidiSynth_insertEvent(MidiSynth *self, ALuint64 time, ALuint event
typedef struct FSynth {
DERIVE_FROM_TYPE(MidiSynth);
- /* NOTE: This rwlock is for setting the soundfont. The EventQueue and
- * related must use the device lock as they're used in the mixer thread.
- */
- RWLock Lock;
-
fluid_settings_t *Settings;
fluid_synth_t *Synth;
int FontID;
@@ -133,8 +132,6 @@ static void FSynth_Construct(FSynth *self, ALCdevice *device)
MidiSynth_Construct(STATIC_CAST(MidiSynth, self), device);
SET_VTABLE2(FSynth, MidiSynth, self);
- RWLockInit(&self->Lock);
-
self->Settings = NULL;
self->Synth = NULL;
self->FontID = FLUID_FAILED;
@@ -184,21 +181,11 @@ static ALboolean FSynth_init(FSynth *self, ALCdevice *device)
static ALenum FSynth_loadSoundfont(FSynth *self, const char *filename)
{
- ALenum state;
int fontid;
- WriteLock(&self->Lock);
- state = STATIC_CAST(MidiSynth, self)->State;
- if(state == AL_PLAYING || state == AL_PAUSED)
- {
- WriteUnlock(&self->Lock);
- return AL_INVALID_OPERATION;
- }
-
fontid = fluid_synth_sfload(self->Synth, filename, 1);
if(fontid == FLUID_FAILED)
{
- WriteUnlock(&self->Lock);
ERR("Failed to load soundfont '%s'\n", filename);
return AL_INVALID_VALUE;
}
@@ -206,14 +193,13 @@ static ALenum FSynth_loadSoundfont(FSynth *self, const char *filename)
if(self->FontID != FLUID_FAILED)
fluid_synth_sfunload(self->Synth, self->FontID, 1);
self->FontID = fontid;
- WriteUnlock(&self->Lock);
return AL_NO_ERROR;
}
static void FSynth_setState(FSynth *self, ALenum state)
{
- WriteLock(&self->Lock);
+ WriteLock(&STATIC_CAST(MidiSynth, self)->Lock);
if(state == AL_PLAYING)
{
if(self->FontID == FLUID_FAILED)
@@ -229,8 +215,8 @@ static void FSynth_setState(FSynth *self, ALenum state)
}
}
}
- MidiSynth_setState(STATIC_CAST(MidiSynth, self), state);
- WriteUnlock(&self->Lock);
+ ExchangeInt(&STATIC_CAST(MidiSynth, self)->State, state);
+ WriteUnlock(&STATIC_CAST(MidiSynth, self)->Lock);
}
static void FSynth_update(FSynth *self, ALCdevice *device)
@@ -290,11 +276,12 @@ static void FSynth_processQueue(FSynth *self, ALuint64 time)
static void FSynth_process(FSynth *self, ALuint SamplesToDo, ALfloat (*restrict DryBuffer)[BUFFERSIZE])
{
MidiSynth *synth = STATIC_CAST(MidiSynth, self);
+ ALenum state = synth->State;
ALuint total = 0;
- if(synth->State != AL_PLAYING)
+ if(state != AL_PLAYING)
{
- if(synth->State == AL_PAUSED)
+ if(state == AL_PAUSED)
fluid_synth_write_float(self->Synth, SamplesToDo, DryBuffer[FrontLeft], 0, 1,
DryBuffer[FrontRight], 0, 1);
return;
@@ -468,21 +455,28 @@ MidiSynth *SynthCreate(ALCdevice *device)
AL_API void AL_APIENTRY alMidiSoundfontSOFT(const char *filename)
{
- ALCdevice *device;
ALCcontext *context;
ALenum err;
context = GetContextRef();
if(!context) return;
- device = context->Device;
if(!(filename && filename[0]))
alSetError(context, AL_INVALID_VALUE);
else
{
- err = V(device->Synth,loadSoundfont)(filename);
- if(err != AL_NO_ERROR)
- alSetError(context, err);
+ ALCdevice *device = context->Device;
+ MidiSynth *synth = device->Synth;
+ WriteLock(&synth->Lock);
+ if(synth->State == AL_PLAYING || synth->State == AL_PAUSED)
+ alSetError(context, AL_INVALID_OPERATION);
+ else
+ {
+ err = V(synth,loadSoundfont)(filename);
+ if(err != AL_NO_ERROR)
+ alSetError(context, err);
+ }
+ WriteUnlock(&synth->Lock);
}
ALCcontext_DecRef(context);