#include "config.h" #include #include #include "alMain.h" #include "alMidi.h" #include "alThunk.h" #include "alError.h" #include "midi/base.h" extern inline struct ALsoundfont *LookupSfont(ALCdevice *device, ALuint id); extern inline struct ALsoundfont *RemoveSfont(ALCdevice *device, ALuint id); AL_API void AL_APIENTRY alGenSoundfontsSOFT(ALsizei n, ALuint *ids) { ALCdevice *device; ALCcontext *context; ALsizei cur = 0; ALenum err; context = GetContextRef(); if(!context) return; if(!(n >= 0)) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); device = context->Device; for(cur = 0;cur < n;cur++) { ALsoundfont *sfont = calloc(1, sizeof(ALsoundfont)); if(!sfont) { alDeleteSoundfontsSOFT(cur, ids); SET_ERROR_AND_GOTO(context, AL_OUT_OF_MEMORY, done); } ALsoundfont_Construct(sfont); err = NewThunkEntry(&sfont->id); if(err == AL_NO_ERROR) err = InsertUIntMapEntry(&device->SfontMap, sfont->id, sfont); if(err != AL_NO_ERROR) { FreeThunkEntry(sfont->id); ALsoundfont_Destruct(sfont); memset(sfont, 0, sizeof(ALsoundfont)); free(sfont); alDeleteSoundfontsSOFT(cur, ids); SET_ERROR_AND_GOTO(context, err, done); } ids[cur] = sfont->id; } done: ALCcontext_DecRef(context); } AL_API ALvoid AL_APIENTRY alDeleteSoundfontsSOFT(ALsizei n, const ALuint *ids) { ALCdevice *device; ALCcontext *context; ALsoundfont *sfont; ALsizei i; context = GetContextRef(); if(!context) return; if(!(n >= 0)) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); device = context->Device; for(i = 0;i < n;i++) { if(!ids[i]) continue; /* Check for valid soundfont ID */ if((sfont=LookupSfont(device, ids[i])) == NULL) SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done); if(sfont->Mapped != AL_FALSE || sfont->ref != 0) SET_ERROR_AND_GOTO(context, AL_INVALID_OPERATION, done); } for(i = 0;i < n;i++) { if((sfont=RemoveSfont(device, ids[i])) == NULL) continue; FreeThunkEntry(sfont->id); ALsoundfont_Destruct(sfont); memset(sfont, 0, sizeof(*sfont)); free(sfont); } done: ALCcontext_DecRef(context); } AL_API ALboolean AL_APIENTRY alIsSoundfontSOFT(ALuint id) { ALCcontext *context; ALboolean ret; context = GetContextRef(); if(!context) return AL_FALSE; ret = ((!id || LookupSfont(context->Device, id)) ? AL_TRUE : AL_FALSE); ALCcontext_DecRef(context); return ret; } AL_API ALvoid AL_APIENTRY alSoundfontSamplesSOFT(ALuint sfid, ALenum type, ALsizei count, const ALvoid *samples) { ALCdevice *device; ALCcontext *context; ALsoundfont *sfont; context = GetContextRef(); if(!context) return; device = context->Device; if(!(sfont=LookupSfont(device, sfid))) SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done); if(type != AL_SHORT_SOFT) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); if(count <= 0) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); // TODO: Lock and check ref count { void *temp = realloc(sfont->Samples, count * sizeof(ALshort)); if(!temp) SET_ERROR_AND_GOTO(context, AL_OUT_OF_MEMORY, done); sfont->Samples = temp; sfont->NumSamples = count; if(samples) memcpy(sfont->Samples, samples, count * sizeof(ALshort)); } done: ALCcontext_DecRef(context); } AL_API ALvoid* AL_APIENTRY alSoundfontMapSamplesSOFT(ALuint sfid, ALsizei offset, ALsizei length) { ALCdevice *device; ALCcontext *context; ALsoundfont *sfont; ALvoid *ptr = NULL; context = GetContextRef(); if(!context) return NULL; device = context->Device; if(!(sfont=LookupSfont(device, sfid))) SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done); if(offset < 0 || (ALuint)offset > sfont->NumSamples*sizeof(ALshort)) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); if(length <= 0 || (ALuint)length > (sfont->NumSamples*sizeof(ALshort) - offset)) SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done); if(ExchangeInt(&sfont->Mapped, AL_TRUE) == AL_TRUE) SET_ERROR_AND_GOTO(context, AL_INVALID_OPERATION, done); ptr = (ALbyte*)sfont->Samples + offset; done: ALCcontext_DecRef(context); return ptr; } AL_API ALvoid AL_APIENTRY alSoundfontUnmapSamplesSOFT(ALuint sfid) { ALCdevice *device; ALCcontext *context; ALsoundfont *sfont; context = GetContextRef(); if(!context) return; device = context->Device; if(!(sfont=LookupSfont(device, sfid))) SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done); if(ExchangeInt(&sfont->Mapped, AL_FALSE) == AL_FALSE) SET_ERROR_AND_GOTO(context, AL_INVALID_OPERATION, done); done: ALCcontext_DecRef(context); } /* ReleaseALSoundfonts * * Called to destroy any soundfonts that still exist on the device */ void ReleaseALSoundfonts(ALCdevice *device) { ALsizei i; for(i = 0;i < device->SfontMap.size;i++) { ALsoundfont *temp = device->SfontMap.array[i].value; device->SfontMap.array[i].value = NULL; FreeThunkEntry(temp->id); ALsoundfont_Destruct(temp); memset(temp, 0, sizeof(*temp)); free(temp); } }