123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133 |
- /**
- * OpenAL cross platform audio library
- * Copyright (C) 1999-2007 by authors.
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- * Or go to http://www.gnu.org/copyleft/lgpl.html
- */
- #include "config.h"
- #include <stdlib.h>
- #include <math.h>
- #include <float.h>
- #include "alMain.h"
- #include "AL/al.h"
- #include "AL/alc.h"
- #include "alError.h"
- #include "alSource.h"
- #include "alBuffer.h"
- #include "alThunk.h"
- #include "alAuxEffectSlot.h"
- static ALvoid InitSourceParams(ALsource *Source);
- static ALvoid GetSourceOffset(ALsource *Source, ALenum eName, ALdouble *Offsets, ALdouble updateLen);
- static ALboolean ApplyOffset(ALsource *Source);
- static ALint GetByteOffset(ALsource *Source);
- static ALint FramesFromBytes(ALint offset, ALenum format, ALint channels);
- #define LookupSource(m, k) ((ALsource*)LookupUIntMapKey(&(m), (k)))
- #define LookupBuffer(m, k) ((ALbuffer*)LookupUIntMapKey(&(m), (k)))
- #define LookupFilter(m, k) ((ALfilter*)LookupUIntMapKey(&(m), (k)))
- #define LookupEffectSlot(m, k) ((ALeffectslot*)LookupUIntMapKey(&(m), (k)))
- AL_API ALvoid AL_APIENTRY alGenSources(ALsizei n,ALuint *sources)
- {
- ALCcontext *Context;
- ALCdevice *Device;
- ALsizei i=0;
- Context = GetContextSuspended();
- if(!Context) return;
- if(n > 0)
- {
- Device = Context->Device;
- // Check that enough memory has been allocted in the 'sources' array for n Sources
- if(!IsBadWritePtr((void*)sources, n * sizeof(ALuint)))
- {
- // Check that the requested number of sources can be generated
- if((Context->SourceMap.size + n) <= (ALsizei)Device->MaxNoOfSources)
- {
- ALenum err;
- // Add additional sources to the list
- while(i < n)
- {
- ALsource *source = calloc(1, sizeof(ALsource));
- if(!source)
- {
- alSetError(Context, AL_OUT_OF_MEMORY);
- alDeleteSources(i, sources);
- break;
- }
- source->source = (ALuint)ALTHUNK_ADDENTRY(source);
- err = InsertUIntMapEntry(&Context->SourceMap, source->source,
- source);
- if(err != AL_NO_ERROR)
- {
- ALTHUNK_REMOVEENTRY(source->source);
- memset(source, 0, sizeof(ALsource));
- free(source);
- alSetError(Context, err);
- alDeleteSources(i, sources);
- break;
- }
- sources[i++] = source->source;
- InitSourceParams(source);
- }
- }
- else
- {
- // Not enough resources to create the Sources
- alSetError(Context, AL_INVALID_VALUE);
- }
- }
- else
- {
- // Bad pointer
- alSetError(Context, AL_INVALID_VALUE);
- }
- }
- ProcessContext(Context);
- }
- AL_API ALvoid AL_APIENTRY alDeleteSources(ALsizei n, const ALuint *sources)
- {
- ALCcontext *Context;
- ALCdevice *Device;
- ALsource *Source;
- ALsizei i, j;
- ALbufferlistitem *BufferList;
- ALboolean bSourcesValid = AL_TRUE;
- Context = GetContextSuspended();
- if(!Context) return;
- if(n >= 0)
- {
- Device = Context->Device;
- // Check that all Sources are valid (and can therefore be deleted)
- for (i = 0; i < n; i++)
- {
- if(LookupSource(Context->SourceMap, sources[i]) == NULL)
- {
- alSetError(Context, AL_INVALID_NAME);
- bSourcesValid = AL_FALSE;
- break;
- }
- }
- if(bSourcesValid)
- {
- // All Sources are valid, and can be deleted
- for(i = 0; i < n; i++)
- {
- // Recheck that the Source is valid, because there could be duplicated Source names
- if((Source=LookupSource(Context->SourceMap, sources[i])) != NULL)
- {
- for(j = 0;j < Context->ActiveSourceCount;j++)
- {
- if(Context->ActiveSources[j] == Source)
- {
- ALsizei end = --(Context->ActiveSourceCount);
- Context->ActiveSources[j] = Context->ActiveSources[end];
- break;
- }
- }
- // For each buffer in the source's queue, decrement its reference counter and remove it
- while(Source->queue != NULL)
- {
- BufferList = Source->queue;
- // Decrement buffer's reference counter
- if(BufferList->buffer != NULL)
- BufferList->buffer->refcount--;
- // Update queue to point to next element in list
- Source->queue = BufferList->next;
- // Release memory allocated for buffer list item
- free(BufferList);
- }
- for(j = 0;j < MAX_SENDS;++j)
- {
- if(Source->Send[j].Slot)
- Source->Send[j].Slot->refcount--;
- Source->Send[j].Slot = NULL;
- }
- // Remove Source from list of Sources
- RemoveUIntMapKey(&Context->SourceMap, Source->source);
- ALTHUNK_REMOVEENTRY(Source->source);
- memset(Source,0,sizeof(ALsource));
- free(Source);
- }
- }
- }
- }
- else
- alSetError(Context, AL_INVALID_VALUE);
- ProcessContext(Context);
- }
- AL_API ALboolean AL_APIENTRY alIsSource(ALuint source)
- {
- ALCcontext *Context;
- ALboolean result;
- Context = GetContextSuspended();
- if(!Context) return AL_FALSE;
- result = (LookupSource(Context->SourceMap, source) ? AL_TRUE : AL_FALSE);
- ProcessContext(Context);
- return result;
- }
- AL_API ALvoid AL_APIENTRY alSourcef(ALuint source, ALenum eParam, ALfloat flValue)
- {
- ALCcontext *pContext;
- ALsource *Source;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_PITCH:
- if(flValue >= 0.0f)
- {
- Source->flPitch = flValue;
- if(Source->flPitch < 0.001f)
- Source->flPitch = 0.001f;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_CONE_INNER_ANGLE:
- if(flValue >= 0.0f && flValue <= 360.0f)
- {
- Source->flInnerAngle = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_CONE_OUTER_ANGLE:
- if(flValue >= 0.0f && flValue <= 360.0f)
- {
- Source->flOuterAngle = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_GAIN:
- if(flValue >= 0.0f)
- {
- Source->flGain = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_MAX_DISTANCE:
- if(flValue >= 0.0f)
- {
- Source->flMaxDistance = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_ROLLOFF_FACTOR:
- if(flValue >= 0.0f)
- {
- Source->flRollOffFactor = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_REFERENCE_DISTANCE:
- if(flValue >= 0.0f)
- {
- Source->flRefDistance = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_MIN_GAIN:
- if(flValue >= 0.0f && flValue <= 1.0f)
- {
- Source->flMinGain = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_MAX_GAIN:
- if(flValue >= 0.0f && flValue <= 1.0f)
- {
- Source->flMaxGain = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_CONE_OUTER_GAIN:
- if(flValue >= 0.0f && flValue <= 1.0f)
- {
- Source->flOuterGain = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_CONE_OUTER_GAINHF:
- if(flValue >= 0.0f && flValue <= 1.0f)
- {
- Source->OuterGainHF = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_AIR_ABSORPTION_FACTOR:
- if(flValue >= 0.0f && flValue <= 10.0f)
- {
- Source->AirAbsorptionFactor = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_ROOM_ROLLOFF_FACTOR:
- if(flValue >= 0.0f && flValue <= 10.0f)
- {
- Source->RoomRolloffFactor = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_DOPPLER_FACTOR:
- if(flValue >= 0.0f && flValue <= 1.0f)
- {
- Source->DopplerFactor = flValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- if(flValue >= 0.0f)
- {
- Source->lOffsetType = eParam;
- // Store Offset (convert Seconds into Milliseconds)
- if(eParam == AL_SEC_OFFSET)
- Source->lOffset = (ALint)(flValue * 1000.0f);
- else
- Source->lOffset = (ALint)flValue;
- if ((Source->state == AL_PLAYING) || (Source->state == AL_PAUSED))
- {
- if(ApplyOffset(Source) == AL_FALSE)
- alSetError(pContext, AL_INVALID_VALUE);
- }
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- {
- // Invalid Source Name
- alSetError(pContext, AL_INVALID_NAME);
- }
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alSource3f(ALuint source, ALenum eParam, ALfloat flValue1,ALfloat flValue2,ALfloat flValue3)
- {
- ALCcontext *pContext;
- ALsource *Source;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_POSITION:
- Source->vPosition[0] = flValue1;
- Source->vPosition[1] = flValue2;
- Source->vPosition[2] = flValue3;
- Source->NeedsUpdate = AL_TRUE;
- break;
- case AL_VELOCITY:
- Source->vVelocity[0] = flValue1;
- Source->vVelocity[1] = flValue2;
- Source->vVelocity[2] = flValue3;
- Source->NeedsUpdate = AL_TRUE;
- break;
- case AL_DIRECTION:
- Source->vOrientation[0] = flValue1;
- Source->vOrientation[1] = flValue2;
- Source->vOrientation[2] = flValue3;
- Source->NeedsUpdate = AL_TRUE;
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alSourcefv(ALuint source, ALenum eParam, const ALfloat *pflValues)
- {
- ALCcontext *pContext;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(pflValues)
- {
- if(LookupSource(pContext->SourceMap, source) != NULL)
- {
- switch(eParam)
- {
- case AL_PITCH:
- case AL_CONE_INNER_ANGLE:
- case AL_CONE_OUTER_ANGLE:
- case AL_GAIN:
- case AL_MAX_DISTANCE:
- case AL_ROLLOFF_FACTOR:
- case AL_REFERENCE_DISTANCE:
- case AL_MIN_GAIN:
- case AL_MAX_GAIN:
- case AL_CONE_OUTER_GAIN:
- case AL_CONE_OUTER_GAINHF:
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- case AL_AIR_ABSORPTION_FACTOR:
- case AL_ROOM_ROLLOFF_FACTOR:
- alSourcef(source, eParam, pflValues[0]);
- break;
- case AL_POSITION:
- case AL_VELOCITY:
- case AL_DIRECTION:
- alSource3f(source, eParam, pflValues[0], pflValues[1], pflValues[2]);
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alSourcei(ALuint source,ALenum eParam,ALint lValue)
- {
- ALCcontext *pContext;
- ALsource *Source;
- ALbufferlistitem *BufferListItem;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- ALCdevice *device = pContext->Device;
- switch(eParam)
- {
- case AL_MAX_DISTANCE:
- case AL_ROLLOFF_FACTOR:
- case AL_CONE_INNER_ANGLE:
- case AL_CONE_OUTER_ANGLE:
- case AL_REFERENCE_DISTANCE:
- alSourcef(source, eParam, (ALfloat)lValue);
- break;
- case AL_SOURCE_RELATIVE:
- if(lValue == AL_FALSE || lValue == AL_TRUE)
- {
- Source->bHeadRelative = (ALboolean)lValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_LOOPING:
- if(lValue == AL_FALSE || lValue == AL_TRUE)
- Source->bLooping = (ALboolean)lValue;
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_BUFFER:
- if(Source->state == AL_STOPPED || Source->state == AL_INITIAL)
- {
- ALbuffer *buffer = NULL;
- if(lValue == 0 ||
- (buffer=LookupBuffer(device->BufferMap, lValue)) != NULL)
- {
- // Remove all elements in the queue
- while(Source->queue != NULL)
- {
- BufferListItem = Source->queue;
- Source->queue = BufferListItem->next;
- // Decrement reference counter for buffer
- if(BufferListItem->buffer)
- BufferListItem->buffer->refcount--;
- // Release memory for buffer list item
- free(BufferListItem);
- // Decrement the number of buffers in the queue
- Source->BuffersInQueue--;
- }
- // Add the buffer to the queue (as long as it is NOT the NULL buffer)
- if(lValue != 0)
- {
- // Source is now in STATIC mode
- Source->lSourceType = AL_STATIC;
- // Add the selected buffer to the queue
- BufferListItem = malloc(sizeof(ALbufferlistitem));
- BufferListItem->buffer = buffer;
- BufferListItem->next = NULL;
- Source->queue = BufferListItem;
- Source->BuffersInQueue = 1;
- // Increment reference counter for buffer
- buffer->refcount++;
- }
- else
- {
- // Source is now in UNDETERMINED mode
- Source->lSourceType = AL_UNDETERMINED;
- }
- Source->BuffersPlayed = 0;
- // Update AL_BUFFER parameter
- Source->Buffer = buffer;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- }
- else
- alSetError(pContext, AL_INVALID_OPERATION);
- break;
- case AL_SOURCE_STATE:
- // Query only
- alSetError(pContext, AL_INVALID_OPERATION);
- break;
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- if(lValue >= 0)
- {
- Source->lOffsetType = eParam;
- // Store Offset (convert Seconds into Milliseconds)
- if(eParam == AL_SEC_OFFSET)
- Source->lOffset = lValue * 1000;
- else
- Source->lOffset = lValue;
- if(Source->state == AL_PLAYING || Source->state == AL_PAUSED)
- {
- if(ApplyOffset(Source) == AL_FALSE)
- alSetError(pContext, AL_INVALID_VALUE);
- }
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_DIRECT_FILTER: {
- ALfilter *filter = NULL;
- if(lValue == 0 ||
- (filter=LookupFilter(pContext->Device->FilterMap, lValue)) != NULL)
- {
- if(!filter)
- {
- Source->DirectFilter.type = AL_FILTER_NULL;
- Source->DirectFilter.filter = 0;
- }
- else
- memcpy(&Source->DirectFilter, filter, sizeof(*filter));
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- } break;
- case AL_DIRECT_FILTER_GAINHF_AUTO:
- if(lValue == AL_TRUE || lValue == AL_FALSE)
- {
- Source->DryGainHFAuto = lValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
- if(lValue == AL_TRUE || lValue == AL_FALSE)
- {
- Source->WetGainAuto = lValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
- if(lValue == AL_TRUE || lValue == AL_FALSE)
- {
- Source->WetGainHFAuto = lValue;
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- case AL_DISTANCE_MODEL:
- if(lValue == AL_NONE ||
- lValue == AL_INVERSE_DISTANCE ||
- lValue == AL_INVERSE_DISTANCE_CLAMPED ||
- lValue == AL_LINEAR_DISTANCE ||
- lValue == AL_LINEAR_DISTANCE_CLAMPED ||
- lValue == AL_EXPONENT_DISTANCE ||
- lValue == AL_EXPONENT_DISTANCE_CLAMPED)
- {
- Source->DistanceModel = lValue;
- if(pContext->SourceDistanceModel)
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- ProcessContext(pContext);
- }
- AL_API void AL_APIENTRY alSource3i(ALuint source, ALenum eParam, ALint lValue1, ALint lValue2, ALint lValue3)
- {
- ALCcontext *pContext;
- ALsource *Source;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- ALCdevice *device = pContext->Device;
- switch (eParam)
- {
- case AL_POSITION:
- case AL_VELOCITY:
- case AL_DIRECTION:
- alSource3f(source, eParam, (ALfloat)lValue1, (ALfloat)lValue2, (ALfloat)lValue3);
- break;
- case AL_AUXILIARY_SEND_FILTER: {
- ALeffectslot *ALEffectSlot = NULL;
- ALfilter *ALFilter = NULL;
- if((ALuint)lValue2 < device->NumAuxSends &&
- (lValue1 == 0 ||
- (ALEffectSlot=LookupEffectSlot(pContext->EffectSlotMap, lValue1)) != NULL) &&
- (lValue3 == 0 ||
- (ALFilter=LookupFilter(device->FilterMap, lValue3)) != NULL))
- {
- /* Release refcount on the previous slot, and add one for
- * the new slot */
- if(Source->Send[lValue2].Slot)
- Source->Send[lValue2].Slot->refcount--;
- Source->Send[lValue2].Slot = ALEffectSlot;
- if(Source->Send[lValue2].Slot)
- Source->Send[lValue2].Slot->refcount++;
- if(!ALFilter)
- {
- /* Disable filter */
- Source->Send[lValue2].WetFilter.type = 0;
- Source->Send[lValue2].WetFilter.filter = 0;
- }
- else
- memcpy(&Source->Send[lValue2].WetFilter, ALFilter, sizeof(*ALFilter));
- Source->NeedsUpdate = AL_TRUE;
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- } break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- ProcessContext(pContext);
- }
- AL_API void AL_APIENTRY alSourceiv(ALuint source, ALenum eParam, const ALint* plValues)
- {
- ALCcontext *pContext;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(plValues)
- {
- if(LookupSource(pContext->SourceMap, source) != NULL)
- {
- switch(eParam)
- {
- case AL_SOURCE_RELATIVE:
- case AL_CONE_INNER_ANGLE:
- case AL_CONE_OUTER_ANGLE:
- case AL_LOOPING:
- case AL_BUFFER:
- case AL_SOURCE_STATE:
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- case AL_MAX_DISTANCE:
- case AL_ROLLOFF_FACTOR:
- case AL_REFERENCE_DISTANCE:
- case AL_DIRECT_FILTER:
- case AL_DIRECT_FILTER_GAINHF_AUTO:
- case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
- case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
- case AL_DISTANCE_MODEL:
- alSourcei(source, eParam, plValues[0]);
- break;
- case AL_POSITION:
- case AL_VELOCITY:
- case AL_DIRECTION:
- case AL_AUXILIARY_SEND_FILTER:
- alSource3i(source, eParam, plValues[0], plValues[1], plValues[2]);
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alGetSourcef(ALuint source, ALenum eParam, ALfloat *pflValue)
- {
- ALCcontext *pContext;
- ALsource *Source;
- ALdouble Offsets[2];
- ALdouble updateLen;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(pflValue)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_PITCH:
- *pflValue = Source->flPitch;
- break;
- case AL_GAIN:
- *pflValue = Source->flGain;
- break;
- case AL_MIN_GAIN:
- *pflValue = Source->flMinGain;
- break;
- case AL_MAX_GAIN:
- *pflValue = Source->flMaxGain;
- break;
- case AL_MAX_DISTANCE:
- *pflValue = Source->flMaxDistance;
- break;
- case AL_ROLLOFF_FACTOR:
- *pflValue = Source->flRollOffFactor;
- break;
- case AL_CONE_OUTER_GAIN:
- *pflValue = Source->flOuterGain;
- break;
- case AL_CONE_OUTER_GAINHF:
- *pflValue = Source->OuterGainHF;
- break;
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- updateLen = (ALdouble)pContext->Device->UpdateSize /
- pContext->Device->Frequency;
- GetSourceOffset(Source, eParam, Offsets, updateLen);
- *pflValue = Offsets[0];
- break;
- case AL_CONE_INNER_ANGLE:
- *pflValue = Source->flInnerAngle;
- break;
- case AL_CONE_OUTER_ANGLE:
- *pflValue = Source->flOuterAngle;
- break;
- case AL_REFERENCE_DISTANCE:
- *pflValue = Source->flRefDistance;
- break;
- case AL_AIR_ABSORPTION_FACTOR:
- *pflValue = Source->AirAbsorptionFactor;
- break;
- case AL_ROOM_ROLLOFF_FACTOR:
- *pflValue = Source->RoomRolloffFactor;
- break;
- case AL_DOPPLER_FACTOR:
- *pflValue = Source->DopplerFactor;
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alGetSource3f(ALuint source, ALenum eParam, ALfloat* pflValue1, ALfloat* pflValue2, ALfloat* pflValue3)
- {
- ALCcontext *pContext;
- ALsource *Source;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(pflValue1 && pflValue2 && pflValue3)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_POSITION:
- *pflValue1 = Source->vPosition[0];
- *pflValue2 = Source->vPosition[1];
- *pflValue3 = Source->vPosition[2];
- break;
- case AL_VELOCITY:
- *pflValue1 = Source->vVelocity[0];
- *pflValue2 = Source->vVelocity[1];
- *pflValue3 = Source->vVelocity[2];
- break;
- case AL_DIRECTION:
- *pflValue1 = Source->vOrientation[0];
- *pflValue2 = Source->vOrientation[1];
- *pflValue3 = Source->vOrientation[2];
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alGetSourcefv(ALuint source, ALenum eParam, ALfloat *pflValues)
- {
- ALCcontext *pContext;
- ALsource *Source;
- ALdouble Offsets[2];
- ALdouble updateLen;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(pflValues)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_PITCH:
- case AL_GAIN:
- case AL_MIN_GAIN:
- case AL_MAX_GAIN:
- case AL_MAX_DISTANCE:
- case AL_ROLLOFF_FACTOR:
- case AL_DOPPLER_FACTOR:
- case AL_CONE_OUTER_GAIN:
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- case AL_CONE_INNER_ANGLE:
- case AL_CONE_OUTER_ANGLE:
- case AL_REFERENCE_DISTANCE:
- case AL_CONE_OUTER_GAINHF:
- case AL_AIR_ABSORPTION_FACTOR:
- case AL_ROOM_ROLLOFF_FACTOR:
- alGetSourcef(source, eParam, pflValues);
- break;
- case AL_SAMPLE_RW_OFFSETS_EXT:
- case AL_BYTE_RW_OFFSETS_EXT:
- updateLen = (ALdouble)pContext->Device->UpdateSize /
- pContext->Device->Frequency;
- GetSourceOffset(Source, eParam, Offsets, updateLen);
- pflValues[0] = Offsets[0];
- pflValues[1] = Offsets[1];
- break;
- case AL_POSITION:
- pflValues[0] = Source->vPosition[0];
- pflValues[1] = Source->vPosition[1];
- pflValues[2] = Source->vPosition[2];
- break;
- case AL_VELOCITY:
- pflValues[0] = Source->vVelocity[0];
- pflValues[1] = Source->vVelocity[1];
- pflValues[2] = Source->vVelocity[2];
- break;
- case AL_DIRECTION:
- pflValues[0] = Source->vOrientation[0];
- pflValues[1] = Source->vOrientation[1];
- pflValues[2] = Source->vOrientation[2];
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alGetSourcei(ALuint source, ALenum eParam, ALint *plValue)
- {
- ALCcontext *pContext;
- ALsource *Source;
- ALdouble Offsets[2];
- ALdouble updateLen;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(plValue)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_MAX_DISTANCE:
- *plValue = (ALint)Source->flMaxDistance;
- break;
- case AL_ROLLOFF_FACTOR:
- *plValue = (ALint)Source->flRollOffFactor;
- break;
- case AL_REFERENCE_DISTANCE:
- *plValue = (ALint)Source->flRefDistance;
- break;
- case AL_SOURCE_RELATIVE:
- *plValue = Source->bHeadRelative;
- break;
- case AL_CONE_INNER_ANGLE:
- *plValue = (ALint)Source->flInnerAngle;
- break;
- case AL_CONE_OUTER_ANGLE:
- *plValue = (ALint)Source->flOuterAngle;
- break;
- case AL_LOOPING:
- *plValue = Source->bLooping;
- break;
- case AL_BUFFER:
- *plValue = (Source->Buffer ? Source->Buffer->buffer : 0);
- break;
- case AL_SOURCE_STATE:
- *plValue = Source->state;
- break;
- case AL_BUFFERS_QUEUED:
- *plValue = Source->BuffersInQueue;
- break;
- case AL_BUFFERS_PROCESSED:
- if(Source->bLooping || Source->lSourceType != AL_STREAMING)
- {
- /* Buffers on a looping source are in a perpetual state
- * of PENDING, so don't report any as PROCESSED */
- *plValue = 0;
- }
- else
- *plValue = Source->BuffersPlayed;
- break;
- case AL_SOURCE_TYPE:
- *plValue = Source->lSourceType;
- break;
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- updateLen = (ALdouble)pContext->Device->UpdateSize /
- pContext->Device->Frequency;
- GetSourceOffset(Source, eParam, Offsets, updateLen);
- *plValue = (ALint)Offsets[0];
- break;
- case AL_DIRECT_FILTER:
- *plValue = Source->DirectFilter.filter;
- break;
- case AL_DIRECT_FILTER_GAINHF_AUTO:
- *plValue = Source->DryGainHFAuto;
- break;
- case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
- *plValue = Source->WetGainAuto;
- break;
- case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
- *plValue = Source->WetGainHFAuto;
- break;
- case AL_DOPPLER_FACTOR:
- *plValue = (ALint)Source->DopplerFactor;
- break;
- case AL_DISTANCE_MODEL:
- *plValue = Source->DistanceModel;
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API void AL_APIENTRY alGetSource3i(ALuint source, ALenum eParam, ALint* plValue1, ALint* plValue2, ALint* plValue3)
- {
- ALCcontext *pContext;
- ALsource *Source;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(plValue1 && plValue2 && plValue3)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_POSITION:
- *plValue1 = (ALint)Source->vPosition[0];
- *plValue2 = (ALint)Source->vPosition[1];
- *plValue3 = (ALint)Source->vPosition[2];
- break;
- case AL_VELOCITY:
- *plValue1 = (ALint)Source->vVelocity[0];
- *plValue2 = (ALint)Source->vVelocity[1];
- *plValue3 = (ALint)Source->vVelocity[2];
- break;
- case AL_DIRECTION:
- *plValue1 = (ALint)Source->vOrientation[0];
- *plValue2 = (ALint)Source->vOrientation[1];
- *plValue3 = (ALint)Source->vOrientation[2];
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API void AL_APIENTRY alGetSourceiv(ALuint source, ALenum eParam, ALint* plValues)
- {
- ALCcontext *pContext;
- ALsource *Source;
- ALdouble Offsets[2];
- ALdouble updateLen;
- pContext = GetContextSuspended();
- if(!pContext) return;
- if(plValues)
- {
- if((Source=LookupSource(pContext->SourceMap, source)) != NULL)
- {
- switch(eParam)
- {
- case AL_SOURCE_RELATIVE:
- case AL_CONE_INNER_ANGLE:
- case AL_CONE_OUTER_ANGLE:
- case AL_LOOPING:
- case AL_BUFFER:
- case AL_SOURCE_STATE:
- case AL_BUFFERS_QUEUED:
- case AL_BUFFERS_PROCESSED:
- case AL_SEC_OFFSET:
- case AL_SAMPLE_OFFSET:
- case AL_BYTE_OFFSET:
- case AL_MAX_DISTANCE:
- case AL_ROLLOFF_FACTOR:
- case AL_DOPPLER_FACTOR:
- case AL_REFERENCE_DISTANCE:
- case AL_SOURCE_TYPE:
- case AL_DIRECT_FILTER:
- case AL_DIRECT_FILTER_GAINHF_AUTO:
- case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
- case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
- case AL_DISTANCE_MODEL:
- alGetSourcei(source, eParam, plValues);
- break;
- case AL_SAMPLE_RW_OFFSETS_EXT:
- case AL_BYTE_RW_OFFSETS_EXT:
- updateLen = (ALdouble)pContext->Device->UpdateSize /
- pContext->Device->Frequency;
- GetSourceOffset(Source, eParam, Offsets, updateLen);
- plValues[0] = (ALint)Offsets[0];
- plValues[1] = (ALint)Offsets[1];
- break;
- case AL_POSITION:
- plValues[0] = (ALint)Source->vPosition[0];
- plValues[1] = (ALint)Source->vPosition[1];
- plValues[2] = (ALint)Source->vPosition[2];
- break;
- case AL_VELOCITY:
- plValues[0] = (ALint)Source->vVelocity[0];
- plValues[1] = (ALint)Source->vVelocity[1];
- plValues[2] = (ALint)Source->vVelocity[2];
- break;
- case AL_DIRECTION:
- plValues[0] = (ALint)Source->vOrientation[0];
- plValues[1] = (ALint)Source->vOrientation[1];
- plValues[2] = (ALint)Source->vOrientation[2];
- break;
- default:
- alSetError(pContext, AL_INVALID_ENUM);
- break;
- }
- }
- else
- alSetError(pContext, AL_INVALID_NAME);
- }
- else
- alSetError(pContext, AL_INVALID_VALUE);
- ProcessContext(pContext);
- }
- AL_API ALvoid AL_APIENTRY alSourcePlay(ALuint source)
- {
- alSourcePlayv(1, &source);
- }
- AL_API ALvoid AL_APIENTRY alSourcePlayv(ALsizei n, const ALuint *sources)
- {
- ALCcontext *Context;
- ALsource *Source;
- ALbufferlistitem *BufferList;
- ALsizei i, j;
- Context = GetContextSuspended();
- if(!Context) return;
- if(!sources)
- {
- alSetError(Context, AL_INVALID_VALUE);
- goto done;
- }
- // Check that all the Sources are valid
- for(i = 0;i < n;i++)
- {
- if(!LookupSource(Context->SourceMap, sources[i]))
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- }
- if(Context->ActiveSourceCount+n < n)
- {
- alSetError(Context, AL_OUT_OF_MEMORY);
- goto done;
- }
- while(Context->MaxActiveSources < Context->ActiveSourceCount+n)
- {
- void *temp = NULL;
- ALsizei newcount;
- newcount = Context->MaxActiveSources << 1;
- if(newcount > 0)
- temp = realloc(Context->ActiveSources,
- sizeof(*Context->ActiveSources) * newcount);
- if(!temp)
- {
- alSetError(Context, AL_OUT_OF_MEMORY);
- goto done;
- }
- Context->ActiveSources = temp;
- Context->MaxActiveSources = newcount;
- }
- for(i = 0;i < n;i++)
- {
- Source = (ALsource*)ALTHUNK_LOOKUPENTRY(sources[i]);
- // Check that there is a queue containing at least one non-null, non zero length AL Buffer
- BufferList = Source->queue;
- while(BufferList)
- {
- if(BufferList->buffer != NULL && BufferList->buffer->size)
- break;
- BufferList = BufferList->next;
- }
- if(!BufferList)
- {
- Source->BuffersPlayed = Source->BuffersInQueue;
- continue;
- }
- for(j = 0;j < OUTPUTCHANNELS;j++)
- Source->DryGains[j] = 0.0f;
- for(j = 0;j < MAX_SENDS;j++)
- Source->WetGains[j] = 0.0f;
- if(Source->state != AL_PAUSED)
- {
- Source->state = AL_PLAYING;
- Source->position = 0;
- Source->position_fraction = 0;
- Source->BuffersPlayed = 0;
- Source->Buffer = Source->queue->buffer;
- }
- else
- Source->state = AL_PLAYING;
- // Check if an Offset has been set
- if(Source->lOffset)
- ApplyOffset(Source);
- if(Source->BuffersPlayed == 0 && Source->position == 0 &&
- Source->position_fraction == 0)
- Source->FirstStart = AL_TRUE;
- else
- Source->FirstStart = AL_FALSE;
- // If device is disconnected, go right to stopped
- if(!Context->Device->Connected)
- {
- Source->state = AL_STOPPED;
- Source->BuffersPlayed = Source->BuffersInQueue;
- Source->position = 0;
- Source->position_fraction = 0;
- }
- else
- {
- for(j = 0;j < Context->ActiveSourceCount;j++)
- {
- if(Context->ActiveSources[j] == Source)
- break;
- }
- if(j == Context->ActiveSourceCount)
- Context->ActiveSources[Context->ActiveSourceCount++] = Source;
- }
- }
- done:
- ProcessContext(Context);
- }
- AL_API ALvoid AL_APIENTRY alSourcePause(ALuint source)
- {
- alSourcePausev(1, &source);
- }
- AL_API ALvoid AL_APIENTRY alSourcePausev(ALsizei n, const ALuint *sources)
- {
- ALCcontext *Context;
- ALsource *Source;
- ALsizei i;
- Context = GetContextSuspended();
- if(!Context) return;
- if(!sources)
- {
- alSetError(Context, AL_INVALID_VALUE);
- goto done;
- }
- // Check all the Sources are valid
- for(i = 0;i < n;i++)
- {
- if(!LookupSource(Context->SourceMap, sources[i]))
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- }
- for(i = 0;i < n;i++)
- {
- Source = (ALsource*)ALTHUNK_LOOKUPENTRY(sources[i]);
- if(Source->state == AL_PLAYING)
- Source->state = AL_PAUSED;
- }
- done:
- ProcessContext(Context);
- }
- AL_API ALvoid AL_APIENTRY alSourceStop(ALuint source)
- {
- alSourceStopv(1, &source);
- }
- AL_API ALvoid AL_APIENTRY alSourceStopv(ALsizei n, const ALuint *sources)
- {
- ALCcontext *Context;
- ALsource *Source;
- ALsizei i;
- Context = GetContextSuspended();
- if(!Context) return;
- if(!sources)
- {
- alSetError(Context, AL_INVALID_VALUE);
- goto done;
- }
- // Check all the Sources are valid
- for(i = 0;i < n;i++)
- {
- if(!LookupSource(Context->SourceMap, sources[i]))
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- }
- for(i = 0;i < n;i++)
- {
- Source = (ALsource*)ALTHUNK_LOOKUPENTRY(sources[i]);
- if(Source->state != AL_INITIAL)
- {
- Source->state = AL_STOPPED;
- Source->BuffersPlayed = Source->BuffersInQueue;
- }
- Source->lOffset = 0;
- }
- done:
- ProcessContext(Context);
- }
- AL_API ALvoid AL_APIENTRY alSourceRewind(ALuint source)
- {
- alSourceRewindv(1, &source);
- }
- AL_API ALvoid AL_APIENTRY alSourceRewindv(ALsizei n, const ALuint *sources)
- {
- ALCcontext *Context;
- ALsource *Source;
- ALsizei i;
- Context = GetContextSuspended();
- if(!Context) return;
- if(!sources)
- {
- alSetError(Context, AL_INVALID_VALUE);
- goto done;
- }
- // Check all the Sources are valid
- for(i = 0;i < n;i++)
- {
- if(!LookupSource(Context->SourceMap, sources[i]))
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- }
- for(i = 0;i < n;i++)
- {
- Source = (ALsource*)ALTHUNK_LOOKUPENTRY(sources[i]);
- if(Source->state != AL_INITIAL)
- {
- Source->state = AL_INITIAL;
- Source->position = 0;
- Source->position_fraction = 0;
- Source->BuffersPlayed = 0;
- if(Source->queue)
- Source->Buffer = Source->queue->buffer;
- }
- Source->lOffset = 0;
- }
- done:
- ProcessContext(Context);
- }
- AL_API ALvoid AL_APIENTRY alSourceQueueBuffers(ALuint source, ALsizei n, const ALuint *buffers)
- {
- ALCcontext *Context;
- ALCdevice *device;
- ALsource *Source;
- ALbuffer *buffer;
- ALsizei i;
- ALbufferlistitem *BufferListStart;
- ALbufferlistitem *BufferList;
- ALboolean HadFormat;
- ALint Frequency;
- ALint Format;
- if(n == 0)
- return;
- Context = GetContextSuspended();
- if(!Context) return;
- // Check that all buffers are valid or zero and that the source is valid
- // Check that this is a valid source
- if((Source=LookupSource(Context->SourceMap, source)) == NULL)
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- // Check that this is not a STATIC Source
- if(Source->lSourceType == AL_STATIC)
- {
- // Invalid Source Type (can't queue on a Static Source)
- alSetError(Context, AL_INVALID_OPERATION);
- goto done;
- }
- device = Context->Device;
- Frequency = -1;
- Format = -1;
- HadFormat = AL_FALSE;
- // Check existing Queue (if any) for a valid Buffers and get its frequency and format
- BufferList = Source->queue;
- while(BufferList)
- {
- if(BufferList->buffer)
- {
- Frequency = BufferList->buffer->frequency;
- Format = BufferList->buffer->format;
- HadFormat = AL_TRUE;
- break;
- }
- BufferList = BufferList->next;
- }
- for(i = 0;i < n;i++)
- {
- if(!buffers[i])
- continue;
- if((buffer=LookupBuffer(device->BufferMap, buffers[i])) == NULL)
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- if(Frequency == -1 && Format == -1)
- {
- Frequency = buffer->frequency;
- Format = buffer->format;
- }
- else if(Frequency != buffer->frequency || Format != buffer->format)
- {
- alSetError(Context, AL_INVALID_OPERATION);
- goto done;
- }
- }
- // Change Source Type
- Source->lSourceType = AL_STREAMING;
- buffer = (ALbuffer*)ALTHUNK_LOOKUPENTRY(buffers[0]);
- // All buffers are valid - so add them to the list
- BufferListStart = malloc(sizeof(ALbufferlistitem));
- BufferListStart->buffer = buffer;
- BufferListStart->next = NULL;
- // Increment reference counter for buffer
- if(buffer) buffer->refcount++;
- BufferList = BufferListStart;
- for(i = 1;i < n;i++)
- {
- buffer = (ALbuffer*)ALTHUNK_LOOKUPENTRY(buffers[i]);
- BufferList->next = malloc(sizeof(ALbufferlistitem));
- BufferList->next->buffer = buffer;
- BufferList->next->next = NULL;
- // Increment reference counter for buffer
- if(buffer) buffer->refcount++;
- BufferList = BufferList->next;
- }
- if(Source->queue == NULL)
- {
- Source->queue = BufferListStart;
- // Update Current Buffer
- Source->Buffer = BufferListStart->buffer;
- }
- else
- {
- // Find end of queue
- BufferList = Source->queue;
- while(BufferList->next != NULL)
- BufferList = BufferList->next;
- BufferList->next = BufferListStart;
- }
- // Update number of buffers in queue
- Source->BuffersInQueue += n;
- // If no previous format, mark the source dirty now that it may have one
- if(!HadFormat)
- Source->NeedsUpdate = AL_TRUE;
- done:
- ProcessContext(Context);
- }
- // Implementation assumes that n is the number of buffers to be removed from the queue and buffers is
- // an array of buffer IDs that are to be filled with the names of the buffers removed
- AL_API ALvoid AL_APIENTRY alSourceUnqueueBuffers( ALuint source, ALsizei n, ALuint* buffers )
- {
- ALCcontext *Context;
- ALsource *Source;
- ALsizei i;
- ALbufferlistitem *BufferList;
- if(n == 0)
- return;
- Context = GetContextSuspended();
- if(!Context) return;
- if((Source=LookupSource(Context->SourceMap, source)) == NULL)
- {
- alSetError(Context, AL_INVALID_NAME);
- goto done;
- }
- if(Source->bLooping || Source->lSourceType != AL_STREAMING ||
- (ALuint)n > Source->BuffersPlayed)
- {
- // Some buffers can't be unqueue because they have not been processed
- alSetError(Context, AL_INVALID_VALUE);
- goto done;
- }
- for(i = 0;i < n;i++)
- {
- BufferList = Source->queue;
- Source->queue = BufferList->next;
- if(BufferList->buffer)
- {
- // Record name of buffer
- buffers[i] = BufferList->buffer->buffer;
- // Decrement buffer reference counter
- BufferList->buffer->refcount--;
- }
- else
- buffers[i] = 0;
- // Release memory for buffer list item
- free(BufferList);
- Source->BuffersInQueue--;
- }
- if(Source->state != AL_PLAYING)
- {
- if(Source->queue)
- Source->Buffer = Source->queue->buffer;
- else
- Source->Buffer = NULL;
- }
- Source->BuffersPlayed -= n;
- done:
- ProcessContext(Context);
- }
- static ALvoid InitSourceParams(ALsource *Source)
- {
- Source->flInnerAngle = 360.0f;
- Source->flOuterAngle = 360.0f;
- Source->flPitch = 1.0f;
- Source->vPosition[0] = 0.0f;
- Source->vPosition[1] = 0.0f;
- Source->vPosition[2] = 0.0f;
- Source->vOrientation[0] = 0.0f;
- Source->vOrientation[1] = 0.0f;
- Source->vOrientation[2] = 0.0f;
- Source->vVelocity[0] = 0.0f;
- Source->vVelocity[1] = 0.0f;
- Source->vVelocity[2] = 0.0f;
- Source->flRefDistance = 1.0f;
- Source->flMaxDistance = FLT_MAX;
- Source->flRollOffFactor = 1.0f;
- Source->bLooping = AL_FALSE;
- Source->flGain = 1.0f;
- Source->flMinGain = 0.0f;
- Source->flMaxGain = 1.0f;
- Source->flOuterGain = 0.0f;
- Source->OuterGainHF = 1.0f;
- Source->DryGainHFAuto = AL_TRUE;
- Source->WetGainAuto = AL_TRUE;
- Source->WetGainHFAuto = AL_TRUE;
- Source->AirAbsorptionFactor = 0.0f;
- Source->RoomRolloffFactor = 0.0f;
- Source->DopplerFactor = 1.0f;
- Source->DistanceModel = AL_INVERSE_DISTANCE_CLAMPED;
- Source->Resampler = DefaultResampler;
- Source->state = AL_INITIAL;
- Source->lSourceType = AL_UNDETERMINED;
- Source->NeedsUpdate = AL_TRUE;
- Source->Buffer = NULL;
- }
- /*
- GetSourceOffset
- Gets the current playback position in the given Source, in the appropriate format (Bytes, Samples or MilliSeconds)
- The offset is relative to the start of the queue (not the start of the current buffer)
- */
- static ALvoid GetSourceOffset(ALsource *Source, ALenum name, ALdouble *offset, ALdouble updateLen)
- {
- ALbufferlistitem *BufferList;
- ALbuffer *Buffer = NULL;
- ALfloat BufferFreq;
- ALint Channels, Bytes;
- ALuint readPos, writePos;
- ALenum OriginalFormat;
- ALuint TotalBufferDataSize;
- ALuint i;
- // Find the first non-NULL Buffer in the Queue
- BufferList = Source->queue;
- while(BufferList)
- {
- if(BufferList->buffer)
- {
- Buffer = BufferList->buffer;
- break;
- }
- BufferList = BufferList->next;
- }
- if((Source->state != AL_PLAYING && Source->state != AL_PAUSED) || !Buffer)
- {
- offset[0] = 0.0;
- offset[1] = 0.0;
- return;
- }
- // Get Current Buffer Size and frequency (in milliseconds)
- BufferFreq = (ALfloat)Buffer->frequency;
- OriginalFormat = Buffer->eOriginalFormat;
- Channels = aluChannelsFromFormat(Buffer->format);
- Bytes = aluBytesFromFormat(Buffer->format);
- // Get Current BytesPlayed (NOTE : This is the byte offset into the *current* buffer)
- readPos = Source->position * Channels * Bytes;
- // Add byte length of any processed buffers in the queue
- TotalBufferDataSize = 0;
- BufferList = Source->queue;
- for(i = 0;BufferList;i++)
- {
- if(BufferList->buffer)
- {
- if(i < Source->BuffersPlayed)
- readPos += BufferList->buffer->size;
- TotalBufferDataSize += BufferList->buffer->size;
- }
- BufferList = BufferList->next;
- }
- if(Source->state == AL_PLAYING)
- writePos = readPos + ((ALuint)(updateLen*BufferFreq) * Channels * Bytes);
- else
- writePos = readPos;
- if(Source->bLooping)
- {
- readPos %= TotalBufferDataSize;
- writePos %= TotalBufferDataSize;
- }
- else
- {
- // Clamp positions to TotalBufferDataSize
- if(readPos > TotalBufferDataSize)
- readPos = TotalBufferDataSize;
- if(writePos > TotalBufferDataSize)
- writePos = TotalBufferDataSize;
- }
- switch(name)
- {
- case AL_SEC_OFFSET:
- offset[0] = (ALdouble)readPos / (Channels * Bytes * BufferFreq);
- offset[1] = (ALdouble)writePos / (Channels * Bytes * BufferFreq);
- break;
- case AL_SAMPLE_OFFSET:
- case AL_SAMPLE_RW_OFFSETS_EXT:
- offset[0] = (ALdouble)(readPos / (Channels * Bytes));
- offset[1] = (ALdouble)(writePos / (Channels * Bytes));
- break;
- case AL_BYTE_OFFSET:
- case AL_BYTE_RW_OFFSETS_EXT:
- // Take into account the original format of the Buffer
- if((OriginalFormat == AL_FORMAT_MONO_IMA4) ||
- (OriginalFormat == AL_FORMAT_STEREO_IMA4))
- {
- // Round down to nearest ADPCM block
- offset[0] = (ALdouble)((readPos / (65 * Bytes * Channels)) * 36 * Channels);
- if(Source->state == AL_PLAYING)
- {
- // Round up to nearest ADPCM block
- offset[1] = (ALdouble)(((writePos + (65 * Bytes * Channels) - 1) / (65 * Bytes * Channels)) * 36 * Channels);
- }
- else
- offset[1] = offset[0];
- }
- else if(OriginalFormat == AL_FORMAT_MONO_MULAW ||
- OriginalFormat == AL_FORMAT_STEREO_MULAW ||
- OriginalFormat == AL_FORMAT_QUAD_MULAW ||
- OriginalFormat == AL_FORMAT_51CHN_MULAW ||
- OriginalFormat == AL_FORMAT_61CHN_MULAW ||
- OriginalFormat == AL_FORMAT_71CHN_MULAW)
- {
- offset[0] = (ALdouble)(readPos / Bytes * 1);
- offset[1] = (ALdouble)(writePos / Bytes * 1);
- }
- else if(OriginalFormat == AL_FORMAT_REAR_MULAW)
- {
- offset[0] = (ALdouble)(readPos / 2 / Bytes * 1);
- offset[1] = (ALdouble)(writePos / 2 / Bytes * 1);
- }
- else if(OriginalFormat == AL_FORMAT_REAR8)
- {
- offset[0] = (ALdouble)(readPos / 2 / Bytes * 1);
- offset[1] = (ALdouble)(writePos / 2 / Bytes * 1);
- }
- else if(OriginalFormat == AL_FORMAT_REAR16)
- {
- offset[0] = (ALdouble)(readPos / 2 / Bytes * 2);
- offset[1] = (ALdouble)(writePos / 2 / Bytes * 2);
- }
- else if(OriginalFormat == AL_FORMAT_REAR32)
- {
- offset[0] = (ALdouble)(readPos / 2 / Bytes * 4);
- offset[1] = (ALdouble)(writePos / 2 / Bytes * 4);
- }
- else
- {
- ALuint OrigBytes = aluBytesFromFormat(OriginalFormat);
- offset[0] = (ALdouble)(readPos / Bytes * OrigBytes);
- offset[1] = (ALdouble)(writePos / Bytes * OrigBytes);
- }
- break;
- }
- }
- /*
- ApplyOffset
- Apply a playback offset to the Source. This function will update the queue (to correctly
- mark buffers as 'pending' or 'processed' depending upon the new offset.
- */
- static ALboolean ApplyOffset(ALsource *Source)
- {
- ALbufferlistitem *BufferList;
- ALbuffer *Buffer;
- ALint lBufferSize, lTotalBufferSize;
- ALint BuffersPlayed;
- ALint lByteOffset;
- // Get true byte offset
- lByteOffset = GetByteOffset(Source);
- // If the offset is invalid, don't apply it
- if(lByteOffset == -1)
- return AL_FALSE;
- // Sort out the queue (pending and processed states)
- BufferList = Source->queue;
- lTotalBufferSize = 0;
- BuffersPlayed = 0;
- while(BufferList)
- {
- Buffer = BufferList->buffer;
- lBufferSize = Buffer ? Buffer->size : 0;
- if(lTotalBufferSize+lBufferSize <= lByteOffset)
- {
- // Offset is past this buffer so increment BuffersPlayed
- BuffersPlayed++;
- }
- else if(lTotalBufferSize <= lByteOffset)
- {
- // Offset is within this buffer
- // Set Current Buffer
- Source->Buffer = BufferList->buffer;
- Source->BuffersPlayed = BuffersPlayed;
- // SW Mixer Positions are in Samples
- Source->position = (lByteOffset - lTotalBufferSize) /
- aluFrameSizeFromFormat(Buffer->format);
- return AL_TRUE;
- }
- // Increment the TotalBufferSize
- lTotalBufferSize += lBufferSize;
- // Move on to next buffer in the Queue
- BufferList = BufferList->next;
- }
- // Offset is out of range of the buffer queue
- return AL_FALSE;
- }
- /*
- GetByteOffset
- Returns the 'true' byte offset into the Source's queue (from the Sample, Byte or Millisecond
- offset supplied by the application). This takes into account the fact that the buffer format
- may have been modifed by AL (e.g 8bit samples are converted to float)
- */
- static ALint GetByteOffset(ALsource *Source)
- {
- ALbuffer *Buffer = NULL;
- ALbufferlistitem *BufferList;
- ALfloat BufferFreq;
- ALint Channels, Bytes;
- ALint ByteOffset = -1;
- // Find the first non-NULL Buffer in the Queue
- BufferList = Source->queue;
- while(BufferList)
- {
- if(BufferList->buffer)
- {
- Buffer = BufferList->buffer;
- break;
- }
- BufferList = BufferList->next;
- }
- if(!Buffer)
- {
- Source->lOffset = 0;
- return -1;
- }
- BufferFreq = ((ALfloat)Buffer->frequency);
- Channels = aluChannelsFromFormat(Buffer->format);
- Bytes = aluBytesFromFormat(Buffer->format);
- // Determine the ByteOffset (and ensure it is block aligned)
- switch(Source->lOffsetType)
- {
- case AL_BYTE_OFFSET:
- // Take into consideration the original format
- ByteOffset = FramesFromBytes(Source->lOffset, Buffer->eOriginalFormat,
- Channels);
- ByteOffset *= Channels * Bytes;
- break;
- case AL_SAMPLE_OFFSET:
- ByteOffset = Source->lOffset * Channels * Bytes;
- break;
- case AL_SEC_OFFSET:
- // Note - lOffset is internally stored as Milliseconds
- ByteOffset = (ALint)(Source->lOffset / 1000.0f * BufferFreq);
- ByteOffset *= Channels * Bytes;
- break;
- }
- // Clear Offset
- Source->lOffset = 0;
- return ByteOffset;
- }
- static ALint FramesFromBytes(ALint offset, ALenum format, ALint channels)
- {
- if(format==AL_FORMAT_MONO_IMA4 || format==AL_FORMAT_STEREO_IMA4)
- {
- // Round down to nearest ADPCM block
- offset /= 36 * channels;
- // Multiply by compression rate (65 sample frames per block)
- offset *= 65;
- }
- else if(format==AL_FORMAT_MONO_MULAW || format==AL_FORMAT_STEREO_MULAW ||
- format==AL_FORMAT_QUAD_MULAW || format==AL_FORMAT_51CHN_MULAW ||
- format==AL_FORMAT_61CHN_MULAW || format==AL_FORMAT_71CHN_MULAW)
- {
- /* muLaw has 1 byte per sample */
- offset /= 1 * channels;
- }
- else if(format == AL_FORMAT_REAR_MULAW)
- {
- /* Rear is 2 channels */
- offset /= 1 * 2;
- }
- else if(format == AL_FORMAT_REAR8)
- offset /= 1 * 2;
- else if(format == AL_FORMAT_REAR16)
- offset /= 2 * 2;
- else if(format == AL_FORMAT_REAR32)
- offset /= 4 * 2;
- else
- {
- ALuint bytes = aluBytesFromFormat(format);
- offset /= bytes * channels;
- }
- return offset;
- }
- ALvoid ReleaseALSources(ALCcontext *Context)
- {
- ALsizei pos;
- ALuint j;
- for(pos = 0;pos < Context->SourceMap.size;pos++)
- {
- ALsource *temp = Context->SourceMap.array[pos].value;
- Context->SourceMap.array[pos].value = NULL;
- // For each buffer in the source's queue, decrement its reference counter and remove it
- while(temp->queue != NULL)
- {
- ALbufferlistitem *BufferList = temp->queue;
- // Decrement buffer's reference counter
- if(BufferList->buffer != NULL)
- BufferList->buffer->refcount--;
- // Update queue to point to next element in list
- temp->queue = BufferList->next;
- // Release memory allocated for buffer list item
- free(BufferList);
- }
- for(j = 0;j < MAX_SENDS;++j)
- {
- if(temp->Send[j].Slot)
- temp->Send[j].Slot->refcount--;
- }
- // Release source structure
- ALTHUNK_REMOVEENTRY(temp->source);
- memset(temp, 0, sizeof(ALsource));
- free(temp);
- }
- }
|