1274 lines
36 KiB
C
1274 lines
36 KiB
C
/**
|
|
* 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 <stdio.h>
|
|
#include <assert.h>
|
|
#include "alMain.h"
|
|
#include "AL/al.h"
|
|
#include "AL/alc.h"
|
|
#include "alError.h"
|
|
#include "alBuffer.h"
|
|
#include "alDatabuffer.h"
|
|
#include "alThunk.h"
|
|
|
|
|
|
static void LoadData(ALbuffer *ALBuf, const ALubyte *data, ALsizei size, ALuint freq, ALenum OrigFormat, ALenum NewFormat);
|
|
static void ConvertData(ALfloat *dst, const ALvoid *src, ALint origBytes, ALsizei len);
|
|
static void ConvertDataRear(ALfloat *dst, const ALvoid *src, ALint origBytes, ALsizei len);
|
|
static void ConvertDataIMA4(ALfloat *dst, const ALvoid *src, ALint origChans, ALsizei len);
|
|
static void ConvertDataMULaw(ALfloat *dst, const ALvoid *src, ALsizei len);
|
|
static void ConvertDataMULawRear(ALfloat *dst, const ALvoid *src, ALsizei len);
|
|
|
|
/*
|
|
* Global Variables
|
|
*/
|
|
|
|
static const long g_IMAStep_size[89]={ // IMA ADPCM Stepsize table
|
|
7, 8, 9, 10, 11, 12, 13, 14, 16, 17, 19, 21, 23, 25, 28, 31,
|
|
34, 37, 41, 45, 50, 55, 60, 66, 73, 80, 88, 97, 107, 118, 130, 143,
|
|
157, 173, 190, 209, 230, 253, 279, 307, 337, 371, 408, 449, 494, 544, 598, 658,
|
|
724, 796, 876, 963, 1060, 1166, 1282, 1411, 1552, 1707, 1878, 2066, 2272, 2499, 2749, 3024,
|
|
3327, 3660, 4026, 4428, 4871, 5358, 5894, 6484, 7132, 7845, 8630, 9493,10442,11487,12635,13899,
|
|
15289,16818,18500,20350,22358,24633,27086,29794,32767
|
|
};
|
|
|
|
static const long g_IMACodeword_4[16]={ // IMA4 ADPCM Codeword decode table
|
|
1, 3, 5, 7, 9, 11, 13, 15,
|
|
-1,-3,-5,-7,-9,-11,-13,-15,
|
|
};
|
|
|
|
static const long g_IMAIndex_adjust_4[16]={ // IMA4 ADPCM Step index adjust decode table
|
|
-1,-1,-1,-1, 2, 4, 6, 8,
|
|
-1,-1,-1,-1, 2, 4, 6, 8
|
|
};
|
|
|
|
static const ALshort muLawDecompressionTable[256] = {
|
|
-32124,-31100,-30076,-29052,-28028,-27004,-25980,-24956,
|
|
-23932,-22908,-21884,-20860,-19836,-18812,-17788,-16764,
|
|
-15996,-15484,-14972,-14460,-13948,-13436,-12924,-12412,
|
|
-11900,-11388,-10876,-10364, -9852, -9340, -8828, -8316,
|
|
-7932, -7676, -7420, -7164, -6908, -6652, -6396, -6140,
|
|
-5884, -5628, -5372, -5116, -4860, -4604, -4348, -4092,
|
|
-3900, -3772, -3644, -3516, -3388, -3260, -3132, -3004,
|
|
-2876, -2748, -2620, -2492, -2364, -2236, -2108, -1980,
|
|
-1884, -1820, -1756, -1692, -1628, -1564, -1500, -1436,
|
|
-1372, -1308, -1244, -1180, -1116, -1052, -988, -924,
|
|
-876, -844, -812, -780, -748, -716, -684, -652,
|
|
-620, -588, -556, -524, -492, -460, -428, -396,
|
|
-372, -356, -340, -324, -308, -292, -276, -260,
|
|
-244, -228, -212, -196, -180, -164, -148, -132,
|
|
-120, -112, -104, -96, -88, -80, -72, -64,
|
|
-56, -48, -40, -32, -24, -16, -8, 0,
|
|
32124, 31100, 30076, 29052, 28028, 27004, 25980, 24956,
|
|
23932, 22908, 21884, 20860, 19836, 18812, 17788, 16764,
|
|
15996, 15484, 14972, 14460, 13948, 13436, 12924, 12412,
|
|
11900, 11388, 10876, 10364, 9852, 9340, 8828, 8316,
|
|
7932, 7676, 7420, 7164, 6908, 6652, 6396, 6140,
|
|
5884, 5628, 5372, 5116, 4860, 4604, 4348, 4092,
|
|
3900, 3772, 3644, 3516, 3388, 3260, 3132, 3004,
|
|
2876, 2748, 2620, 2492, 2364, 2236, 2108, 1980,
|
|
1884, 1820, 1756, 1692, 1628, 1564, 1500, 1436,
|
|
1372, 1308, 1244, 1180, 1116, 1052, 988, 924,
|
|
876, 844, 812, 780, 748, 716, 684, 652,
|
|
620, 588, 556, 524, 492, 460, 428, 396,
|
|
372, 356, 340, 324, 308, 292, 276, 260,
|
|
244, 228, 212, 196, 180, 164, 148, 132,
|
|
120, 112, 104, 96, 88, 80, 72, 64,
|
|
56, 48, 40, 32, 24, 16, 8, 0
|
|
};
|
|
|
|
/*
|
|
* alGenBuffers(ALsizei n, ALuint *puiBuffers)
|
|
*
|
|
* Generates n AL Buffers, and stores the Buffers Names in the array pointed to by puiBuffers
|
|
*/
|
|
ALAPI ALvoid ALAPIENTRY alGenBuffers(ALsizei n,ALuint *puiBuffers)
|
|
{
|
|
ALCcontext *Context;
|
|
ALsizei i=0;
|
|
|
|
Context = GetContextSuspended();
|
|
if(!Context) return;
|
|
|
|
// Check that we are actually generation some Buffers
|
|
if (n > 0)
|
|
{
|
|
ALCdevice *device = Context->Device;
|
|
|
|
// Check the pointer is valid (and points to enough memory to store Buffer Names)
|
|
if (!IsBadWritePtr((void*)puiBuffers, n * sizeof(ALuint)))
|
|
{
|
|
ALbuffer **list = &device->Buffers;
|
|
while(*list)
|
|
list = &(*list)->next;
|
|
|
|
// Create all the new Buffers
|
|
while(i < n)
|
|
{
|
|
*list = calloc(1, sizeof(ALbuffer));
|
|
if(!(*list))
|
|
{
|
|
alDeleteBuffers(i, puiBuffers);
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
break;
|
|
}
|
|
|
|
puiBuffers[i] = (ALuint)ALTHUNK_ADDENTRY(*list);
|
|
(*list)->buffer = puiBuffers[i];
|
|
|
|
device->BufferCount++;
|
|
i++;
|
|
|
|
list = &(*list)->next;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Pointer does not point to enough memory to write Buffer names
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
}
|
|
|
|
ProcessContext(Context);
|
|
}
|
|
|
|
/*
|
|
* alDeleteBuffers(ALsizei n, ALuint *puiBuffers)
|
|
*
|
|
* Deletes the n AL Buffers pointed to by puiBuffers
|
|
*/
|
|
ALAPI ALvoid ALAPIENTRY alDeleteBuffers(ALsizei n, const ALuint *puiBuffers)
|
|
{
|
|
ALCcontext *Context;
|
|
ALbuffer *ALBuf;
|
|
ALsizei i;
|
|
ALboolean bFailed = AL_FALSE;
|
|
|
|
Context = GetContextSuspended();
|
|
if(!Context) return;
|
|
|
|
// Check we are actually Deleting some Buffers
|
|
if (n >= 0)
|
|
{
|
|
ALCdevice *device = Context->Device;
|
|
|
|
// Check that all the buffers are valid and can actually be deleted
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
// Check for valid Buffer ID (can be NULL buffer)
|
|
if (alIsBuffer(puiBuffers[i]))
|
|
{
|
|
// If not the NULL buffer, check that the reference count is 0
|
|
ALBuf = ((ALbuffer *)ALTHUNK_LOOKUPENTRY(puiBuffers[i]));
|
|
if (ALBuf)
|
|
{
|
|
if (ALBuf->refcount != 0)
|
|
{
|
|
// Buffer still in use, cannot be deleted
|
|
alSetError(AL_INVALID_OPERATION);
|
|
bFailed = AL_TRUE;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Invalid Buffer
|
|
alSetError(AL_INVALID_NAME);
|
|
bFailed = AL_TRUE;
|
|
}
|
|
}
|
|
|
|
// If all the Buffers were valid (and have Reference Counts of 0), then we can delete them
|
|
if (!bFailed)
|
|
{
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
if (puiBuffers[i] && alIsBuffer(puiBuffers[i]))
|
|
{
|
|
ALbuffer **list = &device->Buffers;
|
|
|
|
ALBuf=((ALbuffer *)ALTHUNK_LOOKUPENTRY(puiBuffers[i]));
|
|
while(*list && *list != ALBuf)
|
|
list = &(*list)->next;
|
|
|
|
if(*list)
|
|
*list = (*list)->next;
|
|
|
|
// Release the memory used to store audio data
|
|
free(ALBuf->data);
|
|
|
|
// Release buffer structure
|
|
ALTHUNK_REMOVEENTRY(puiBuffers[i]);
|
|
memset(ALBuf, 0, sizeof(ALbuffer));
|
|
device->BufferCount--;
|
|
free(ALBuf);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
alSetError(AL_INVALID_VALUE);
|
|
|
|
ProcessContext(Context);
|
|
}
|
|
|
|
/*
|
|
* alIsBuffer(ALuint uiBuffer)
|
|
*
|
|
* Checks if ulBuffer is a valid Buffer Name
|
|
*/
|
|
ALAPI ALboolean ALAPIENTRY alIsBuffer(ALuint uiBuffer)
|
|
{
|
|
ALCcontext *Context;
|
|
ALboolean result=AL_FALSE;
|
|
ALbuffer *ALBuf;
|
|
ALbuffer *TgtALBuf;
|
|
|
|
Context = GetContextSuspended();
|
|
if(!Context) return AL_FALSE;
|
|
|
|
if (uiBuffer)
|
|
{
|
|
ALCdevice *device = Context->Device;
|
|
|
|
TgtALBuf = (ALbuffer *)ALTHUNK_LOOKUPENTRY(uiBuffer);
|
|
|
|
// Check through list of generated buffers for uiBuffer
|
|
ALBuf = device->Buffers;
|
|
while (ALBuf)
|
|
{
|
|
if (ALBuf == TgtALBuf)
|
|
{
|
|
result = AL_TRUE;
|
|
break;
|
|
}
|
|
|
|
ALBuf = ALBuf->next;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
result = AL_TRUE;
|
|
}
|
|
|
|
|
|
ProcessContext(Context);
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* alBufferData(ALuint buffer,ALenum format,ALvoid *data,ALsizei size,ALsizei freq)
|
|
*
|
|
* Fill buffer with audio data
|
|
*/
|
|
ALAPI ALvoid ALAPIENTRY alBufferData(ALuint buffer,ALenum format,const ALvoid *data,ALsizei size,ALsizei freq)
|
|
{
|
|
ALCcontext *Context;
|
|
ALbuffer *ALBuf;
|
|
ALvoid *temp;
|
|
|
|
Context = GetContextSuspended();
|
|
if(!Context) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
ALBuf=((ALbuffer *)ALTHUNK_LOOKUPENTRY(buffer));
|
|
|
|
if(Context->SampleSource)
|
|
{
|
|
ALuint offset;
|
|
|
|
if(Context->SampleSource->state == MAPPED)
|
|
{
|
|
alSetError(AL_INVALID_OPERATION);
|
|
ProcessContext(Context);
|
|
return;
|
|
}
|
|
|
|
offset = (ALuint)data;
|
|
data = Context->SampleSource->data + offset;
|
|
}
|
|
|
|
if ((ALBuf->refcount==0)&&(data))
|
|
{
|
|
switch(format)
|
|
{
|
|
case AL_FORMAT_MONO8:
|
|
case AL_FORMAT_MONO16:
|
|
case AL_FORMAT_MONO_FLOAT32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_MONO_FLOAT32);
|
|
break;
|
|
|
|
case AL_FORMAT_STEREO8:
|
|
case AL_FORMAT_STEREO16:
|
|
case AL_FORMAT_STEREO_FLOAT32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_STEREO_FLOAT32);
|
|
break;
|
|
|
|
case AL_FORMAT_REAR8:
|
|
case AL_FORMAT_REAR16:
|
|
case AL_FORMAT_REAR32: {
|
|
ALuint NewFormat = AL_FORMAT_QUAD32;
|
|
ALuint NewChannels = aluChannelsFromFormat(NewFormat);
|
|
ALuint NewBytes = aluBytesFromFormat(NewFormat);
|
|
ALuint OrigBytes = ((format==AL_FORMAT_REAR8) ? 1 :
|
|
((format==AL_FORMAT_REAR16) ? 2 :
|
|
4));
|
|
|
|
assert(aluBytesFromFormat(NewFormat) == 2);
|
|
|
|
if((size%(OrigBytes*2)) != 0)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
size /= OrigBytes;
|
|
size *= 2;
|
|
|
|
// Samples are converted here
|
|
temp = realloc(ALBuf->data, (BUFFER_PADDING*NewChannels + size) * NewBytes);
|
|
if(temp)
|
|
{
|
|
ALBuf->data = temp;
|
|
ConvertDataRear(ALBuf->data, data, OrigBytes, size);
|
|
|
|
ALBuf->format = NewFormat;
|
|
ALBuf->eOriginalFormat = format;
|
|
ALBuf->size = size*NewBytes;
|
|
ALBuf->frequency = freq;
|
|
}
|
|
else
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
} break;
|
|
|
|
case AL_FORMAT_QUAD8_LOKI:
|
|
case AL_FORMAT_QUAD16_LOKI:
|
|
case AL_FORMAT_QUAD8:
|
|
case AL_FORMAT_QUAD16:
|
|
case AL_FORMAT_QUAD32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_QUAD32);
|
|
break;
|
|
|
|
case AL_FORMAT_51CHN8:
|
|
case AL_FORMAT_51CHN16:
|
|
case AL_FORMAT_51CHN32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_51CHN32);
|
|
break;
|
|
|
|
case AL_FORMAT_61CHN8:
|
|
case AL_FORMAT_61CHN16:
|
|
case AL_FORMAT_61CHN32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_61CHN32);
|
|
break;
|
|
|
|
case AL_FORMAT_71CHN8:
|
|
case AL_FORMAT_71CHN16:
|
|
case AL_FORMAT_71CHN32:
|
|
LoadData(ALBuf, data, size, freq, format, AL_FORMAT_71CHN32);
|
|
break;
|
|
|
|
case AL_FORMAT_MONO_IMA4:
|
|
case AL_FORMAT_STEREO_IMA4: {
|
|
int OrigChans = ((format==AL_FORMAT_MONO_IMA4) ? 1 : 2);
|
|
ALuint NewFormat = ((OrigChans==1) ? AL_FORMAT_MONO_FLOAT32 :
|
|
AL_FORMAT_STEREO_FLOAT32);
|
|
ALuint NewBytes = aluBytesFromFormat(NewFormat);
|
|
|
|
// Here is where things vary:
|
|
// nVidia and Apple use 64+1 samples per channel per block => block_size=36*chans bytes
|
|
// Most PC sound software uses 2040+1 samples per channel per block -> block_size=1024*chans bytes
|
|
if((size%(36*OrigChans)) != 0)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
size /= 36;
|
|
size *= 65;
|
|
|
|
// Allocate extra padding samples
|
|
temp = realloc(ALBuf->data, (BUFFER_PADDING*OrigChans + size)*NewBytes);
|
|
if(temp)
|
|
{
|
|
ALBuf->data = temp;
|
|
ConvertDataIMA4(ALBuf->data, data, OrigChans, size/65);
|
|
|
|
ALBuf->format = NewFormat;
|
|
ALBuf->eOriginalFormat = format;
|
|
ALBuf->size = size*NewBytes;
|
|
ALBuf->frequency = freq;
|
|
}
|
|
else
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
} break;
|
|
|
|
case AL_FORMAT_MONO_MULAW:
|
|
case AL_FORMAT_STEREO_MULAW:
|
|
case AL_FORMAT_QUAD_MULAW:
|
|
case AL_FORMAT_51CHN_MULAW:
|
|
case AL_FORMAT_61CHN_MULAW:
|
|
case AL_FORMAT_71CHN_MULAW: {
|
|
int Channels = ((format==AL_FORMAT_MONO_MULAW) ? 1 :
|
|
((format==AL_FORMAT_STEREO_MULAW) ? 2 :
|
|
((format==AL_FORMAT_QUAD_MULAW) ? 4 :
|
|
((format==AL_FORMAT_51CHN_MULAW) ? 6 :
|
|
((format==AL_FORMAT_61CHN_MULAW) ? 7 : 8)))));
|
|
ALuint NewFormat = ((Channels==1) ? AL_FORMAT_MONO_FLOAT32 :
|
|
((Channels==2) ? AL_FORMAT_STEREO_FLOAT32 :
|
|
((Channels==4) ? AL_FORMAT_QUAD32 :
|
|
((Channels==6) ? AL_FORMAT_51CHN32 :
|
|
((Channels==7) ? AL_FORMAT_61CHN32 :
|
|
AL_FORMAT_71CHN32)))));
|
|
ALuint NewBytes = aluBytesFromFormat(NewFormat);
|
|
|
|
if((size%(1*Channels)) != 0)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
// Allocate extra padding samples
|
|
temp = realloc(ALBuf->data, (BUFFER_PADDING*Channels + size)*NewBytes);
|
|
if(temp)
|
|
{
|
|
ALBuf->data = temp;
|
|
ConvertDataMULaw(ALBuf->data, data, size);
|
|
|
|
ALBuf->format = NewFormat;
|
|
ALBuf->eOriginalFormat = format;
|
|
ALBuf->size = size*NewBytes;
|
|
ALBuf->frequency = freq;
|
|
}
|
|
else
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
} break;
|
|
|
|
case AL_FORMAT_REAR_MULAW: {
|
|
int OrigChans = 2;
|
|
ALuint NewFormat = AL_FORMAT_QUAD32;
|
|
ALuint NewBytes = aluBytesFromFormat(NewFormat);
|
|
ALuint NewChannels = aluChannelsFromFormat(NewFormat);
|
|
|
|
if((size%(1*OrigChans)) != 0)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
size *= 2;
|
|
|
|
// Allocate extra padding samples
|
|
temp = realloc(ALBuf->data, (BUFFER_PADDING*NewChannels + size)*NewBytes);
|
|
if(temp)
|
|
{
|
|
ALBuf->data = temp;
|
|
ConvertDataMULawRear(ALBuf->data, data, size);
|
|
|
|
ALBuf->format = NewFormat;
|
|
ALBuf->eOriginalFormat = format;
|
|
ALBuf->size = size*NewBytes;
|
|
ALBuf->frequency = freq;
|
|
}
|
|
else
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
} break;
|
|
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Buffer is in use, or data is a NULL pointer
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Invalid Buffer Name
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(Context);
|
|
}
|
|
|
|
/*
|
|
* alBufferSubDataEXT(ALuint buffer,ALenum format,ALvoid *data,ALsizei offset,ALsizei length)
|
|
*
|
|
* Fill buffer with audio data
|
|
*/
|
|
ALvoid ALAPIENTRY alBufferSubDataEXT(ALuint buffer,ALenum format,const ALvoid *data,ALsizei offset,ALsizei length)
|
|
{
|
|
ALCcontext *Context;
|
|
ALbuffer *ALBuf;
|
|
|
|
Context = GetContextSuspended();
|
|
if(!Context) return;
|
|
|
|
if(alIsBuffer(buffer) && buffer != 0)
|
|
{
|
|
ALBuf = (ALbuffer*)ALTHUNK_LOOKUPENTRY(buffer);
|
|
|
|
if(Context->SampleSource)
|
|
{
|
|
ALuint offset;
|
|
|
|
if(Context->SampleSource->state == MAPPED)
|
|
{
|
|
alSetError(AL_INVALID_OPERATION);
|
|
ProcessContext(Context);
|
|
return;
|
|
}
|
|
|
|
offset = (ALuint)data;
|
|
data = Context->SampleSource->data + offset;
|
|
}
|
|
|
|
if(ALBuf->data == NULL)
|
|
{
|
|
// buffer does not have any data
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
else if(length < 0 || offset < 0 || (length > 0 && data == NULL))
|
|
{
|
|
// data is NULL or offset/length is negative
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
else
|
|
{
|
|
switch(format)
|
|
{
|
|
case AL_FORMAT_REAR8:
|
|
case AL_FORMAT_REAR16:
|
|
case AL_FORMAT_REAR32: {
|
|
ALuint OrigBytes = ((format==AL_FORMAT_REAR8) ? 1 :
|
|
((format==AL_FORMAT_REAR16) ? 2 :
|
|
4));
|
|
ALuint NewBytes = aluBytesFromFormat(ALBuf->format);
|
|
|
|
if(ALBuf->eOriginalFormat != AL_FORMAT_REAR8 &&
|
|
ALBuf->eOriginalFormat != AL_FORMAT_REAR16 &&
|
|
ALBuf->eOriginalFormat != AL_FORMAT_REAR32)
|
|
{
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
|
|
if(ALBuf->size/4/NewBytes < (ALuint)offset+length)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
ConvertDataRear(&ALBuf->data[offset*4], data, OrigBytes, length*2);
|
|
} break;
|
|
|
|
case AL_FORMAT_MONO_IMA4:
|
|
case AL_FORMAT_STEREO_IMA4: {
|
|
int Channels = aluChannelsFromFormat(ALBuf->format);
|
|
ALuint Bytes = aluBytesFromFormat(ALBuf->format);
|
|
|
|
if(ALBuf->eOriginalFormat != format)
|
|
{
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
|
|
if((offset%65) != 0 || (length%65) != 0 ||
|
|
ALBuf->size/Channels/Bytes < (ALuint)offset+length)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
ConvertDataIMA4(&ALBuf->data[offset*Channels], data, Channels, length/65*Channels);
|
|
} break;
|
|
|
|
case AL_FORMAT_MONO_MULAW:
|
|
case AL_FORMAT_STEREO_MULAW:
|
|
case AL_FORMAT_QUAD_MULAW:
|
|
case AL_FORMAT_REAR_MULAW:
|
|
case AL_FORMAT_51CHN_MULAW:
|
|
case AL_FORMAT_61CHN_MULAW:
|
|
case AL_FORMAT_71CHN_MULAW: {
|
|
int Channels = aluChannelsFromFormat(ALBuf->format);
|
|
ALuint Bytes = aluBytesFromFormat(ALBuf->format);
|
|
|
|
if(ALBuf->eOriginalFormat != format)
|
|
{
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
|
|
if(ALBuf->size/Channels/Bytes < (ALuint)offset+length)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
if(ALBuf->eOriginalFormat == AL_FORMAT_REAR_MULAW)
|
|
ConvertDataMULawRear(&ALBuf->data[offset*Channels], data, length*2);
|
|
else
|
|
ConvertDataMULaw(&ALBuf->data[offset*Channels], data, length*Channels);
|
|
} break;
|
|
|
|
default: {
|
|
ALuint Channels = aluChannelsFromFormat(format);
|
|
ALuint Bytes = aluBytesFromFormat(format);
|
|
ALuint NewBytes = aluBytesFromFormat(ALBuf->format);
|
|
|
|
if(Channels != aluChannelsFromFormat(ALBuf->format))
|
|
{
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
|
|
if(ALBuf->size/Channels/NewBytes < (ALuint)offset+length)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
break;
|
|
}
|
|
|
|
ConvertData(&ALBuf->data[offset*Channels], data, Bytes, length*Channels);
|
|
} break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Invalid Buffer Name
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(Context);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBufferf(ALuint buffer, ALenum eParam, ALfloat flValue)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)flValue;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBuffer3f(ALuint buffer, ALenum eParam, ALfloat flValue1, ALfloat flValue2, ALfloat flValue3)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)flValue1;
|
|
(void)flValue2;
|
|
(void)flValue3;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBufferfv(ALuint buffer, ALenum eParam, const ALfloat* flValues)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)flValues;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBufferi(ALuint buffer, ALenum eParam, ALint lValue)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)lValue;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBuffer3i( ALuint buffer, ALenum eParam, ALint lValue1, ALint lValue2, ALint lValue3)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)lValue1;
|
|
(void)lValue2;
|
|
(void)lValue3;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alBufferiv(ALuint buffer, ALenum eParam, const ALint* plValues)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
(void)plValues;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI ALvoid ALAPIENTRY alGetBufferf(ALuint buffer, ALenum eParam, ALfloat *pflValue)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (pflValue)
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alGetBuffer3f(ALuint buffer, ALenum eParam, ALfloat* pflValue1, ALfloat* pflValue2, ALfloat* pflValue3)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if ((pflValue1) && (pflValue2) && (pflValue3))
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alGetBufferfv(ALuint buffer, ALenum eParam, ALfloat* pflValues)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (pflValues)
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI ALvoid ALAPIENTRY alGetBufferi(ALuint buffer, ALenum eParam, ALint *plValue)
|
|
{
|
|
ALCcontext *pContext;
|
|
ALbuffer *pBuffer;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (plValue)
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
pBuffer = ((ALbuffer *)ALTHUNK_LOOKUPENTRY(buffer));
|
|
|
|
switch (eParam)
|
|
{
|
|
case AL_FREQUENCY:
|
|
*plValue = pBuffer->frequency;
|
|
break;
|
|
|
|
case AL_BITS:
|
|
*plValue = aluBytesFromFormat(pBuffer->format) * 8;
|
|
break;
|
|
|
|
case AL_CHANNELS:
|
|
*plValue = aluChannelsFromFormat(pBuffer->format);
|
|
break;
|
|
|
|
case AL_SIZE:
|
|
*plValue = pBuffer->size;
|
|
break;
|
|
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alGetBuffer3i(ALuint buffer, ALenum eParam, ALint* plValue1, ALint* plValue2, ALint* plValue3)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if ((plValue1) && (plValue2) && (plValue3))
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch(eParam)
|
|
{
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
|
|
ALAPI void ALAPIENTRY alGetBufferiv(ALuint buffer, ALenum eParam, ALint* plValues)
|
|
{
|
|
ALCcontext *pContext;
|
|
|
|
pContext = GetContextSuspended();
|
|
if(!pContext) return;
|
|
|
|
if (plValues)
|
|
{
|
|
if (alIsBuffer(buffer) && (buffer != 0))
|
|
{
|
|
switch (eParam)
|
|
{
|
|
case AL_FREQUENCY:
|
|
case AL_BITS:
|
|
case AL_CHANNELS:
|
|
case AL_SIZE:
|
|
alGetBufferi(buffer, eParam, plValues);
|
|
break;
|
|
|
|
default:
|
|
alSetError(AL_INVALID_ENUM);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_NAME);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
}
|
|
|
|
ProcessContext(pContext);
|
|
}
|
|
|
|
/*
|
|
* LoadData
|
|
*
|
|
* Loads the specified data into the buffer, using the specified formats.
|
|
* Currently, the new format must be 32-bit float, and must have the same
|
|
* channel configuration as the original format. This does NOT handle
|
|
* compressed formats (eg. IMA4).
|
|
*/
|
|
static void LoadData(ALbuffer *ALBuf, const ALubyte *data, ALsizei size, ALuint freq, ALenum OrigFormat, ALenum NewFormat)
|
|
{
|
|
ALuint NewBytes = aluBytesFromFormat(NewFormat);
|
|
ALuint NewChannels = aluChannelsFromFormat(NewFormat);
|
|
ALuint OrigBytes = aluBytesFromFormat(OrigFormat);
|
|
ALuint OrigChannels = aluChannelsFromFormat(OrigFormat);
|
|
ALvoid *temp;
|
|
|
|
assert(NewBytes == 4);
|
|
assert(NewChannels == OrigChannels);
|
|
|
|
if ((size%(OrigBytes*OrigChannels)) != 0)
|
|
{
|
|
alSetError(AL_INVALID_VALUE);
|
|
return;
|
|
}
|
|
|
|
// Samples are converted here
|
|
size /= OrigBytes;
|
|
temp = realloc(ALBuf->data, (BUFFER_PADDING*NewChannels + size) * NewBytes);
|
|
if(temp)
|
|
{
|
|
ALBuf->data = temp;
|
|
ConvertData(ALBuf->data, data, OrigBytes, size);
|
|
|
|
ALBuf->format = NewFormat;
|
|
ALBuf->eOriginalFormat = OrigFormat;
|
|
ALBuf->size = size*NewBytes;
|
|
ALBuf->frequency = freq;
|
|
}
|
|
else
|
|
alSetError(AL_OUT_OF_MEMORY);
|
|
}
|
|
|
|
static void ConvertData(ALfloat *dst, const ALvoid *src, ALint origBytes, ALsizei len)
|
|
{
|
|
ALsizei i;
|
|
ALint smp;
|
|
switch(origBytes)
|
|
{
|
|
case 1:
|
|
for(i = 0;i < len;i++)
|
|
{
|
|
smp = ((ALubyte*)src)[i];
|
|
dst[i] = ((smp < 0x80) ? ((smp-128)/128.0f) : ((smp-128)/127.0f));
|
|
}
|
|
break;
|
|
|
|
case 2:
|
|
for(i = 0;i < len;i++)
|
|
{
|
|
smp = ((ALshort*)src)[i];
|
|
dst[i] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
}
|
|
break;
|
|
|
|
case 4:
|
|
for(i = 0;i < len;i++)
|
|
dst[i] = ((ALfloat*)src)[i];
|
|
break;
|
|
|
|
default:
|
|
assert(0);
|
|
}
|
|
}
|
|
|
|
static void ConvertDataRear(ALfloat *dst, const ALvoid *src, ALint origBytes, ALsizei len)
|
|
{
|
|
ALsizei i;
|
|
ALint smp;
|
|
switch(origBytes)
|
|
{
|
|
case 1:
|
|
for(i = 0;i < len;i+=4)
|
|
{
|
|
dst[i+0] = 0;
|
|
dst[i+1] = 0;
|
|
smp = ((ALubyte*)src)[i/2+0];
|
|
dst[i+2] = ((smp < 0x80) ? ((smp-128)/128.0f) : ((smp-128)/127.0f));
|
|
smp = ((ALubyte*)src)[i/2+1];
|
|
dst[i+3] = ((smp < 0x80) ? ((smp-128)/128.0f) : ((smp-128)/127.0f));
|
|
}
|
|
break;
|
|
|
|
case 2:
|
|
for(i = 0;i < len;i+=4)
|
|
{
|
|
dst[i+0] = 0;
|
|
dst[i+1] = 0;
|
|
smp = ((ALshort*)src)[i/2+0];
|
|
dst[i+2] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
smp = ((ALshort*)src)[i/2+1];
|
|
dst[i+3] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
}
|
|
break;
|
|
|
|
case 4:
|
|
for(i = 0;i < len;i+=4)
|
|
{
|
|
dst[i+0] = 0;
|
|
dst[i+1] = 0;
|
|
dst[i+2] = ((ALfloat*)src)[i/2+0];
|
|
dst[i+3] = ((ALfloat*)src)[i/2+1];
|
|
}
|
|
break;
|
|
|
|
default:
|
|
assert(0);
|
|
}
|
|
}
|
|
|
|
static void ConvertDataIMA4(ALfloat *dst, const ALvoid *src, ALint origChans, ALsizei len)
|
|
{
|
|
const ALuint *IMAData;
|
|
ALint Sample[2],Index[2];
|
|
ALuint IMACode[2];
|
|
ALsizei i,j,k,c;
|
|
|
|
assert(origChans <= 2);
|
|
|
|
IMAData = src;
|
|
for(i = 0;i < len/origChans;i++)
|
|
{
|
|
for(c = 0;c < origChans;c++)
|
|
{
|
|
Sample[c] = ((ALshort*)IMAData)[0];
|
|
Index[c] = ((ALshort*)IMAData)[1];
|
|
|
|
Index[c] = ((Index[c]<0) ? 0 : Index[c]);
|
|
Index[c] = ((Index[c]>88) ? 88 : Index[c]);
|
|
|
|
dst[i*65*origChans + c] = ((Sample[c] < 0) ? (Sample[c]/32768.0f) : (Sample[c]/32767.0f));
|
|
|
|
IMAData++;
|
|
}
|
|
|
|
for(j = 1;j < 65;j += 8)
|
|
{
|
|
for(c = 0;c < origChans;c++)
|
|
IMACode[c] = *(IMAData++);
|
|
|
|
for(k = 0;k < 8;k++)
|
|
{
|
|
for(c = 0;c < origChans;c++)
|
|
{
|
|
Sample[c] += ((g_IMAStep_size[Index[c]]*g_IMACodeword_4[IMACode[c]&15])/8);
|
|
Index[c] += g_IMAIndex_adjust_4[IMACode[c]&15];
|
|
|
|
if(Sample[c] < -32768) Sample[c] = -32768;
|
|
else if(Sample[c] > 32767) Sample[c] = 32767;
|
|
|
|
if(Index[c]<0) Index[c] = 0;
|
|
else if(Index[c]>88) Index[c] = 88;
|
|
|
|
dst[(i*65+j+k)*origChans + c] = ((Sample[c] < 0) ? (Sample[c]/32768.0f) : (Sample[c]/32767.0f));
|
|
IMACode[c] >>= 4;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void ConvertDataMULaw(ALfloat *dst, const ALvoid *src, ALsizei len)
|
|
{
|
|
ALsizei i;
|
|
ALint smp;
|
|
for(i = 0;i < len;i++)
|
|
{
|
|
smp = muLawDecompressionTable[((ALubyte*)src)[i]];
|
|
dst[i] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
}
|
|
}
|
|
|
|
static void ConvertDataMULawRear(ALfloat *dst, const ALvoid *src, ALsizei len)
|
|
{
|
|
ALsizei i;
|
|
ALint smp;
|
|
for(i = 0;i < len;i+=4)
|
|
{
|
|
dst[i+0] = 0;
|
|
dst[i+1] = 0;
|
|
smp = muLawDecompressionTable[((ALubyte*)src)[i/2+0]];
|
|
dst[i+2] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
smp = muLawDecompressionTable[((ALubyte*)src)[i/2+1]];
|
|
dst[i+3] = ((smp < 0) ? (smp/32768.0f) : (smp/32767.0f));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* ReleaseALBuffers()
|
|
*
|
|
* INTERNAL FN : Called by DLLMain on exit to destroy any buffers that still exist
|
|
*/
|
|
ALvoid ReleaseALBuffers(ALCdevice *device)
|
|
{
|
|
ALbuffer *ALBuffer;
|
|
ALbuffer *ALBufferTemp;
|
|
|
|
ALBuffer = device->Buffers;
|
|
while(ALBuffer)
|
|
{
|
|
// Release sample data
|
|
free(ALBuffer->data);
|
|
|
|
// Release Buffer structure
|
|
ALBufferTemp = ALBuffer;
|
|
ALBuffer = ALBuffer->next;
|
|
memset(ALBufferTemp, 0, sizeof(ALbuffer));
|
|
free(ALBufferTemp);
|
|
}
|
|
device->Buffers = NULL;
|
|
device->BufferCount = 0;
|
|
}
|