comparison ext/openal-soft/OpenAL32/alBuffer.c @ 0:4a0efb7baf70

* Datasets becomes the new trunk and retires after that :-)
author mvbarracuda@33b003aa-7bff-0310-803a-e67f0ece8222
date Sun, 29 Jun 2008 18:44:17 +0000
parents
children
comparison
equal deleted inserted replaced
-1:000000000000 0:4a0efb7baf70
1 /**
2 * OpenAL cross platform audio library
3 * Copyright (C) 1999-2007 by authors.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
8 *
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
13 *
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
18 * Or go to http://www.gnu.org/copyleft/lgpl.html
19 */
20
21 #define _CRT_SECURE_NO_DEPRECATE // get rid of sprintf security warnings on VS2005
22
23 #include "config.h"
24
25 #include <stdlib.h>
26 #include <stdio.h>
27 #include <assert.h>
28 #include "alMain.h"
29 #include "AL/al.h"
30 #include "AL/alc.h"
31 #include "alError.h"
32 #include "alBuffer.h"
33 #include "alThunk.h"
34
35
36 static void LoadData(ALbuffer *ALBuf, const ALubyte *data, ALsizei size, ALuint freq, ALenum OrigFormat, ALenum NewFormat);
37
38 /*
39 * AL Buffer Functions
40 *
41 * AL Buffers are shared amoung Contexts, so we store the list of generated Buffers
42 * as a global variable in this module. (A valid context is not required to make
43 * AL Buffer function calls
44 *
45 */
46
47 /*
48 * Global Variables
49 */
50
51 static ALbuffer *g_pBuffers = NULL; // Linked List of Buffers
52 static ALuint g_uiBufferCount = 0; // Buffer Count
53
54 static const long g_IMAStep_size[89]={ // IMA ADPCM Stepsize table
55 7, 8, 9, 10, 11, 12, 13, 14, 16, 17, 19, 21, 23, 25, 28, 31,
56 34, 37, 41, 45, 50, 55, 60, 66, 73, 80, 88, 97, 107, 118, 130, 143,
57 157, 173, 190, 209, 230, 253, 279, 307, 337, 371, 408, 449, 494, 544, 598, 658,
58 724, 796, 876, 963, 1060, 1166, 1282, 1411, 1552, 1707, 1878, 2066, 2272, 2499, 2749, 3024,
59 3327, 3660, 4026, 4428, 4871, 5358, 5894, 6484, 7132, 7845, 8630, 9493,10442,11487,12635,13899,
60 15289,16818,18500,20350,22358,24633,27086,29794,32767
61 };
62
63 static const long g_IMACodeword_4[16]={ // IMA4 ADPCM Codeword decode table
64 1, 3, 5, 7, 9, 11, 13, 15,
65 -1,-3,-5,-7,-9,-11,-13,-15,
66 };
67
68 static const long g_IMAIndex_adjust_4[16]={ // IMA4 ADPCM Step index adjust decode table
69 -1,-1,-1,-1, 2, 4, 6, 8,
70 -1,-1,-1,-1, 2, 4, 6, 8
71 };
72
73 /*
74 * alGenBuffers(ALsizei n, ALuint *puiBuffers)
75 *
76 * Generates n AL Buffers, and stores the Buffers Names in the array pointed to by puiBuffers
77 */
78 ALAPI ALvoid ALAPIENTRY alGenBuffers(ALsizei n,ALuint *puiBuffers)
79 {
80 ALCcontext *Context;
81 ALsizei i=0;
82
83 Context = alcGetCurrentContext();
84 SuspendContext(Context);
85
86 // Check that we are actually generation some Buffers
87 if (n > 0)
88 {
89 // Check the pointer is valid (and points to enough memory to store Buffer Names)
90 if (!IsBadWritePtr((void*)puiBuffers, n * sizeof(ALuint)))
91 {
92 ALbuffer **list = &g_pBuffers;
93 while(*list)
94 list = &(*list)->next;
95
96 // Create all the new Buffers
97 while(i < n)
98 {
99 *list = calloc(1, sizeof(ALbuffer));
100 if(!(*list))
101 {
102 alDeleteBuffers(i, puiBuffers);
103 alSetError(AL_OUT_OF_MEMORY);
104 break;
105 }
106
107 puiBuffers[i] = (ALuint)ALTHUNK_ADDENTRY(*list);
108 (*list)->state = UNUSED;
109 g_uiBufferCount++;
110 i++;
111
112 list = &(*list)->next;
113 }
114 }
115 else
116 {
117 // Pointer does not point to enough memory to write Buffer names
118 alSetError(AL_INVALID_VALUE);
119 }
120 }
121
122 ProcessContext(Context);
123
124 return;
125 }
126
127 /*
128 * alDeleteBuffers(ALsizei n, ALuint *puiBuffers)
129 *
130 * Deletes the n AL Buffers pointed to by puiBuffers
131 */
132 ALAPI ALvoid ALAPIENTRY alDeleteBuffers(ALsizei n, const ALuint *puiBuffers)
133 {
134 ALCcontext *Context;
135 ALbuffer *ALBuf;
136 ALsizei i;
137 ALboolean bFailed = AL_FALSE;
138
139 Context = alcGetCurrentContext();
140 SuspendContext(Context);
141
142 // Check we are actually Deleting some Buffers
143 if (n >= 0)
144 {
145 if ((ALuint)n <= g_uiBufferCount)
146 {
147 // Check that all the buffers are valid and can actually be deleted
148 for (i = 0; i < n; i++)
149 {
150 // Check for valid Buffer ID (can be NULL buffer)
151 if (alIsBuffer(puiBuffers[i]))
152 {
153 // If not the NULL buffer, check that the reference count is 0
154 ALBuf = ((ALbuffer *)ALTHUNK_LOOKUPENTRY(puiBuffers[i]));
155 if (ALBuf)
156 {
157 if (ALBuf->refcount != 0)
158 {
159 // Buffer still in use, cannot be deleted
160 alSetError(AL_INVALID_OPERATION);
161 bFailed = AL_TRUE;
162 }
163 }
164 }
165 else
166 {
167 // Invalid Buffer
168 alSetError(AL_INVALID_NAME);
169 bFailed = AL_TRUE;
170 }
171 }
172
173 // If all the Buffers were valid (and have Reference Counts of 0), then we can delete them
174 if (!bFailed)
175 {
176 for (i = 0; i < n; i++)
177 {
178 ALBuf=((ALbuffer *)ALTHUNK_LOOKUPENTRY(puiBuffers[i]));
179 if (ALBuf)
180 {
181 ALbuffer **list = &g_pBuffers;
182 while(*list && *list != ALBuf)
183 list = &(*list)->next;
184
185 if(*list)
186 *list = (*list)->next;
187
188 // Release the memory used to store audio data
189 free(ALBuf->data);
190
191 // Release buffer structure
192 ALTHUNK_REMOVEENTRY(puiBuffers[i]);
193 memset(ALBuf, 0, sizeof(ALbuffer));
194 g_uiBufferCount--;
195 free(ALBuf);
196 }
197 }
198 }
199 }
200 else
201 alSetError(AL_INVALID_NAME);
202 }
203 else
204 alSetError(AL_INVALID_VALUE);
205
206 ProcessContext(Context);
207
208 return;
209
210 }
211
212 /*
213 * alIsBuffer(ALuint uiBuffer)
214 *
215 * Checks if ulBuffer is a valid Buffer Name
216 */
217 ALAPI ALboolean ALAPIENTRY alIsBuffer(ALuint uiBuffer)
218 {
219 ALCcontext *Context;
220 ALboolean result=AL_FALSE;
221 ALbuffer *ALBuf;
222 ALbuffer *TgtALBuf;
223
224 Context = alcGetCurrentContext();
225 SuspendContext(Context);
226
227 if (uiBuffer)
228 {
229 TgtALBuf = (ALbuffer *)ALTHUNK_LOOKUPENTRY(uiBuffer);
230
231 // Check through list of generated buffers for uiBuffer
232 ALBuf = g_pBuffers;
233 while (ALBuf)
234 {
235 if (ALBuf == TgtALBuf)
236 {
237 result = AL_TRUE;
238 break;
239 }
240
241 ALBuf = ALBuf->next;
242 }
243 }
244 else
245 {
246 result = AL_TRUE;
247 }
248
249
250 ProcessContext(Context);
251
252 return result;
253 }
254
255 /*
256 * alBufferData(ALuint buffer,ALenum format,ALvoid *data,ALsizei size,ALsizei freq)
257 *
258 * Fill buffer with audio data
259 */
260 ALAPI ALvoid ALAPIENTRY alBufferData(ALuint buffer,ALenum format,const ALvoid *data,ALsizei size,ALsizei freq)
261 {
262 ALuint *IMAData,IMACode;
263 ALCcontext *Context;
264 ALint Sample,Index;
265 ALint LeftSample,LeftIndex;
266 ALint RightSample,RightIndex;
267 ALuint LeftIMACode,RightIMACode;
268 ALbuffer *ALBuf;
269 ALsizei i,j,k;
270
271 Context = alcGetCurrentContext();
272 SuspendContext(Context);
273
274 if (alIsBuffer(buffer) && (buffer != 0))
275 {
276 ALBuf=((ALbuffer *)ALTHUNK_LOOKUPENTRY(buffer));
277 if ((ALBuf->refcount==0)&&(data))
278 {
279 switch(format)
280 {
281 case AL_FORMAT_MONO8:
282 case AL_FORMAT_MONO16:
283 case AL_FORMAT_MONO_FLOAT32:
284 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_MONO16);
285 break;
286
287 case AL_FORMAT_STEREO8:
288 case AL_FORMAT_STEREO16:
289 case AL_FORMAT_STEREO_FLOAT32:
290 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_STEREO16);
291 break;
292
293 case AL_FORMAT_REAR8:
294 case AL_FORMAT_REAR16:
295 case AL_FORMAT_REAR32: {
296 ALuint NewFormat = AL_FORMAT_QUAD16;
297 ALuint NewChannels = aluChannelsFromFormat(NewFormat);
298 ALuint OrigBytes = ((format==AL_FORMAT_REAR8) ? 1 :
299 ((format==AL_FORMAT_REAR16) ? 2 :
300 4));
301 ALsizei i;
302
303 assert(aluBytesFromFormat(NewFormat) == 2);
304
305 if ((size%(OrigBytes*2)) != 0)
306 {
307 alSetError(AL_INVALID_VALUE);
308 break;
309 }
310
311 switch(OrigBytes)
312 {
313 case 1:
314 size /= sizeof(ALubyte);
315 size *= 2;
316
317 // 8bit Samples are converted to 16 bit here
318 // Allocate 8 extra samples
319 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
320 if (ALBuf->data)
321 {
322 for (i = 0;i < size;i+=4)
323 {
324 ALBuf->data[i+0] = 0;
325 ALBuf->data[i+1] = 0;
326 ALBuf->data[i+2] = (ALshort)((((ALubyte*)data)[i/2+0]-128) << 8);
327 ALBuf->data[i+3] = (ALshort)((((ALubyte*)data)[i/2+1]-128) << 8);
328 }
329 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
330
331 ALBuf->format = NewFormat;
332 ALBuf->eOriginalFormat = format;
333 ALBuf->size = size*1*sizeof(ALshort);
334 ALBuf->frequency = freq;
335 }
336 else
337 alSetError(AL_OUT_OF_MEMORY);
338 break;
339
340 case 2:
341 size /= sizeof(ALshort);
342 size *= 2;
343
344 // Allocate 8 extra samples
345 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
346 if (ALBuf->data)
347 {
348 for (i = 0;i < size;i+=4)
349 {
350 ALBuf->data[i+0] = 0;
351 ALBuf->data[i+1] = 0;
352 ALBuf->data[i+2] = ((ALshort*)data)[i/2+0];
353 ALBuf->data[i+3] = ((ALshort*)data)[i/2+1];
354 }
355 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
356
357 ALBuf->format = NewFormat;
358 ALBuf->eOriginalFormat = format;
359 ALBuf->size = size*1*sizeof(ALshort);
360 ALBuf->frequency = freq;
361 }
362 else
363 alSetError(AL_OUT_OF_MEMORY);
364 break;
365
366 case 4:
367 size /= sizeof(ALfloat);
368 size *= 2;
369
370 // Allocate 8 extra samples
371 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
372 if (ALBuf->data)
373 {
374 ALint smp;
375 for (i = 0;i < size;i+=4)
376 {
377 ALBuf->data[i+0] = 0;
378 ALBuf->data[i+1] = 0;
379 smp = (((ALfloat*)data)[i/2+0] * 32767.5f - 0.5);
380 smp = min(smp, 32767);
381 smp = max(smp, -32768);
382 ALBuf->data[i+2] = (ALshort)smp;
383 smp = (((ALfloat*)data)[i/2+1] * 32767.5f - 0.5);
384 smp = min(smp, 32767);
385 smp = max(smp, -32768);
386 ALBuf->data[i+3] = (ALshort)smp;
387 }
388 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
389
390 ALBuf->format = NewFormat;
391 ALBuf->eOriginalFormat = format;
392 ALBuf->size = size*1*sizeof(ALshort);
393 ALBuf->frequency = freq;
394 }
395 else
396 alSetError(AL_OUT_OF_MEMORY);
397 break;
398
399 default:
400 assert(0);
401 }
402 } break;
403
404 case AL_FORMAT_QUAD8_LOKI:
405 case AL_FORMAT_QUAD16_LOKI:
406 case AL_FORMAT_QUAD8:
407 case AL_FORMAT_QUAD16:
408 case AL_FORMAT_QUAD32:
409 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_QUAD16);
410 break;
411
412 case AL_FORMAT_51CHN8:
413 case AL_FORMAT_51CHN16:
414 case AL_FORMAT_51CHN32:
415 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_51CHN16);
416 break;
417
418 case AL_FORMAT_61CHN8:
419 case AL_FORMAT_61CHN16:
420 case AL_FORMAT_61CHN32:
421 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_61CHN16);
422 break;
423
424 case AL_FORMAT_71CHN8:
425 case AL_FORMAT_71CHN16:
426 case AL_FORMAT_71CHN32:
427 LoadData(ALBuf, data, size, freq, format, AL_FORMAT_71CHN16);
428 break;
429
430 case AL_FORMAT_MONO_IMA4:
431 // Here is where things vary:
432 // nVidia and Apple use 64+1 samples per block => block_size=36 bytes
433 // Most PC sound software uses 2040+1 samples per block -> block_size=1024 bytes
434 if ((size%36) == 0)
435 {
436 // Allocate 8 extra samples (16 bytes)
437 ALBuf->data=realloc(ALBuf->data,16+(size/36)*(65*sizeof(ALshort)));
438 if (ALBuf->data)
439 {
440 ALBuf->format = AL_FORMAT_MONO16;
441 ALBuf->eOriginalFormat = AL_FORMAT_MONO_IMA4;
442 IMAData=(ALuint *)data;
443 for (i=0;i<size/36;i++)
444 {
445 Sample=((ALshort *)IMAData)[0];
446 Index=((ALshort *)IMAData)[1];
447
448 Index=Index<0?0:Index;
449 Index=Index>88?88:Index;
450
451 ALBuf->data[i*65]=(short)Sample;
452
453 IMAData++;
454
455 for (j=1;j<65;j+=8)
456 {
457 IMACode=*IMAData;
458 for (k=0;k<8;k+=2)
459 {
460 Sample+=((g_IMAStep_size[Index]*g_IMACodeword_4[IMACode&15])/8);
461 Index+=g_IMAIndex_adjust_4[IMACode&15];
462 if (Sample<-32768) Sample=-32768;
463 else if (Sample>32767) Sample=32767;
464 if (Index<0) Index=0;
465 else if (Index>88) Index=88;
466 ALBuf->data[i*65+j+k]=(short)Sample;
467 IMACode>>=4;
468
469 Sample+=((g_IMAStep_size[Index]*g_IMACodeword_4[IMACode&15])/8);
470 Index+=g_IMAIndex_adjust_4[IMACode&15];
471 if (Sample<-32768) Sample=-32768;
472 else if (Sample>32767) Sample=32767;
473 if (Index<0) Index=0;
474 else if (Index>88) Index=88;
475 ALBuf->data[i*65+j+k+1]=(short)Sample;
476 IMACode>>=4;
477 }
478 IMAData++;
479 }
480 }
481 memset(&(ALBuf->data[(size/36*65)]), 0, 16);
482 ALBuf->size=size/36*65*sizeof(ALshort);
483 ALBuf->frequency=freq;
484 }
485 else
486 alSetError(AL_OUT_OF_MEMORY);
487 }
488 else
489 alSetError(AL_INVALID_VALUE);
490 break;
491
492 case AL_FORMAT_STEREO_IMA4:
493 // Here is where things vary:
494 // nVidia and Apple use 64+1 samples per channel per block => block_size=72 bytes
495 // Most PC sound software uses 2040+1 samples per channel per block -> block_size=2048 bytes
496 if ((size%72) == 0)
497 {
498 // Allocate 8 extra samples (32 bytes)
499 ALBuf->data=realloc(ALBuf->data,32+(size/72)*(2*65*sizeof(ALshort)));
500 if (ALBuf->data)
501 {
502 ALBuf->format = AL_FORMAT_STEREO16;
503 ALBuf->eOriginalFormat = AL_FORMAT_STEREO_IMA4;
504 IMAData=(ALuint *)data;
505 for (i=0;i<size/72;i++)
506 {
507 LeftSample=((ALshort *)IMAData)[0];
508 LeftIndex=((ALshort *)IMAData)[1];
509
510 LeftIndex=LeftIndex<0?0:LeftIndex;
511 LeftIndex=LeftIndex>88?88:LeftIndex;
512
513 ALBuf->data[i*2*65]=(short)LeftSample;
514
515 IMAData++;
516
517 RightSample=((ALshort *)IMAData)[0];
518 RightIndex=((ALshort *)IMAData)[1];
519
520 RightIndex=RightIndex<0?0:RightIndex;
521 RightIndex=RightIndex>88?88:RightIndex;
522
523 ALBuf->data[i*2*65+1]=(short)RightSample;
524
525 IMAData++;
526
527 for (j=2;j<130;j+=16)
528 {
529 LeftIMACode=IMAData[0];
530 RightIMACode=IMAData[1];
531 for (k=0;k<16;k+=4)
532 {
533 LeftSample+=((g_IMAStep_size[LeftIndex]*g_IMACodeword_4[LeftIMACode&15])/8);
534 LeftIndex+=g_IMAIndex_adjust_4[LeftIMACode&15];
535 if (LeftSample<-32768) LeftSample=-32768;
536 else if (LeftSample>32767) LeftSample=32767;
537 if (LeftIndex<0) LeftIndex=0;
538 else if (LeftIndex>88) LeftIndex=88;
539 ALBuf->data[i*2*65+j+k]=(short)LeftSample;
540 LeftIMACode>>=4;
541
542 RightSample+=((g_IMAStep_size[RightIndex]*g_IMACodeword_4[RightIMACode&15])/8);
543 RightIndex+=g_IMAIndex_adjust_4[RightIMACode&15];
544 if (RightSample<-32768) RightSample=-32768;
545 else if (RightSample>32767) RightSample=32767;
546 if (RightIndex<0) RightIndex=0;
547 else if (RightIndex>88) RightIndex=88;
548 ALBuf->data[i*2*65+j+k+1]=(short)RightSample;
549 RightIMACode>>=4;
550
551 LeftSample+=((g_IMAStep_size[LeftIndex]*g_IMACodeword_4[LeftIMACode&15])/8);
552 LeftIndex+=g_IMAIndex_adjust_4[LeftIMACode&15];
553 if (LeftSample<-32768) LeftSample=-32768;
554 else if (LeftSample>32767) LeftSample=32767;
555 if (LeftIndex<0) LeftIndex=0;
556 else if (LeftIndex>88) LeftIndex=88;
557 ALBuf->data[i*2*65+j+k+2]=(short)LeftSample;
558 LeftIMACode>>=4;
559
560 RightSample+=((g_IMAStep_size[RightIndex]*g_IMACodeword_4[RightIMACode&15])/8);
561 RightIndex+=g_IMAIndex_adjust_4[RightIMACode&15];
562 if (RightSample<-32768) RightSample=-32768;
563 else if (RightSample>32767) RightSample=32767;
564 if (RightIndex<0) RightIndex=0;
565 else if (RightIndex>88) RightIndex=88;
566 ALBuf->data[i*2*65+j+k+3]=(short)RightSample;
567 RightIMACode>>=4;
568 }
569 IMAData+=2;
570 }
571 }
572 memset(&(ALBuf->data[(size/72*2*65)]), 0, 32);
573 ALBuf->size=size/72*2*65*sizeof(ALshort);
574 ALBuf->frequency=freq;
575 }
576 else
577 alSetError(AL_OUT_OF_MEMORY);
578 }
579 else
580 alSetError(AL_INVALID_VALUE);
581 break;
582
583 default:
584 alSetError(AL_INVALID_ENUM);
585 break;
586 }
587 }
588 else
589 {
590 // Buffer is in use, or data is a NULL pointer
591 alSetError(AL_INVALID_VALUE);
592 }
593 }
594 else
595 {
596 // Invalid Buffer Name
597 alSetError(AL_INVALID_NAME);
598 }
599
600 ProcessContext(Context);
601 }
602
603
604 ALAPI void ALAPIENTRY alBufferf(ALuint buffer, ALenum eParam, ALfloat flValue)
605 {
606 ALCcontext *pContext;
607
608 (void)flValue;
609
610 pContext = alcGetCurrentContext();
611 SuspendContext(pContext);
612
613 if (alIsBuffer(buffer) && (buffer != 0))
614 {
615 switch(eParam)
616 {
617 default:
618 alSetError(AL_INVALID_ENUM);
619 break;
620 }
621 }
622 else
623 {
624 alSetError(AL_INVALID_NAME);
625 }
626
627 ProcessContext(pContext);
628 }
629
630
631 ALAPI void ALAPIENTRY alBuffer3f(ALuint buffer, ALenum eParam, ALfloat flValue1, ALfloat flValue2, ALfloat flValue3)
632 {
633 ALCcontext *pContext;
634
635 (void)flValue1;
636 (void)flValue2;
637 (void)flValue3;
638
639 pContext = alcGetCurrentContext();
640 SuspendContext(pContext);
641
642 if (alIsBuffer(buffer) && (buffer != 0))
643 {
644 switch(eParam)
645 {
646 default:
647 alSetError(AL_INVALID_ENUM);
648 break;
649 }
650 }
651 else
652 {
653 alSetError(AL_INVALID_NAME);
654 }
655
656 ProcessContext(pContext);
657 }
658
659
660 ALAPI void ALAPIENTRY alBufferfv(ALuint buffer, ALenum eParam, const ALfloat* flValues)
661 {
662 ALCcontext *pContext;
663
664 (void)flValues;
665
666 pContext = alcGetCurrentContext();
667 SuspendContext(pContext);
668
669 if (alIsBuffer(buffer) && (buffer != 0))
670 {
671 switch(eParam)
672 {
673 default:
674 alSetError(AL_INVALID_ENUM);
675 break;
676 }
677 }
678 else
679 {
680 alSetError(AL_INVALID_NAME);
681 }
682
683 ProcessContext(pContext);
684 }
685
686
687 ALAPI void ALAPIENTRY alBufferi(ALuint buffer, ALenum eParam, ALint lValue)
688 {
689 ALCcontext *pContext;
690
691 (void)lValue;
692
693 pContext = alcGetCurrentContext();
694 SuspendContext(pContext);
695
696 if (alIsBuffer(buffer) && (buffer != 0))
697 {
698 switch(eParam)
699 {
700 default:
701 alSetError(AL_INVALID_ENUM);
702 break;
703 }
704 }
705 else
706 {
707 alSetError(AL_INVALID_NAME);
708 }
709
710 ProcessContext(pContext);
711 }
712
713
714 ALAPI void ALAPIENTRY alBuffer3i( ALuint buffer, ALenum eParam, ALint lValue1, ALint lValue2, ALint lValue3)
715 {
716 ALCcontext *pContext;
717
718 (void)lValue1;
719 (void)lValue2;
720 (void)lValue3;
721
722 pContext = alcGetCurrentContext();
723 SuspendContext(pContext);
724
725 if (alIsBuffer(buffer) && (buffer != 0))
726 {
727 switch(eParam)
728 {
729 default:
730 alSetError(AL_INVALID_ENUM);
731 break;
732 }
733 }
734 else
735 {
736 alSetError(AL_INVALID_NAME);
737 }
738
739 ProcessContext(pContext);
740 }
741
742
743 ALAPI void ALAPIENTRY alBufferiv(ALuint buffer, ALenum eParam, const ALint* plValues)
744 {
745 ALCcontext *pContext;
746
747 (void)plValues;
748
749 pContext = alcGetCurrentContext();
750 SuspendContext(pContext);
751
752 if (alIsBuffer(buffer) && (buffer != 0))
753 {
754 switch(eParam)
755 {
756 default:
757 alSetError(AL_INVALID_ENUM);
758 break;
759 }
760 }
761 else
762 {
763 alSetError(AL_INVALID_NAME);
764 }
765
766 ProcessContext(pContext);
767 }
768
769
770 ALAPI ALvoid ALAPIENTRY alGetBufferf(ALuint buffer, ALenum eParam, ALfloat *pflValue)
771 {
772 ALCcontext *pContext;
773
774 pContext = alcGetCurrentContext();
775 SuspendContext(pContext);
776
777 if (pflValue)
778 {
779 if (alIsBuffer(buffer) && (buffer != 0))
780 {
781 switch(eParam)
782 {
783 default:
784 alSetError(AL_INVALID_ENUM);
785 break;
786 }
787 }
788 else
789 {
790 alSetError(AL_INVALID_NAME);
791 }
792 }
793 else
794 {
795 alSetError(AL_INVALID_VALUE);
796 }
797
798 ProcessContext(pContext);
799 }
800
801
802 ALAPI void ALAPIENTRY alGetBuffer3f(ALuint buffer, ALenum eParam, ALfloat* pflValue1, ALfloat* pflValue2, ALfloat* pflValue3)
803 {
804 ALCcontext *pContext;
805
806 pContext = alcGetCurrentContext();
807 SuspendContext(pContext);
808
809 if ((pflValue1) && (pflValue2) && (pflValue3))
810 {
811 if (alIsBuffer(buffer) && (buffer != 0))
812 {
813 switch(eParam)
814 {
815 default:
816 alSetError(AL_INVALID_ENUM);
817 break;
818 }
819 }
820 else
821 {
822 alSetError(AL_INVALID_NAME);
823 }
824 }
825 else
826 {
827 alSetError(AL_INVALID_VALUE);
828 }
829
830 ProcessContext(pContext);
831 }
832
833
834 ALAPI void ALAPIENTRY alGetBufferfv(ALuint buffer, ALenum eParam, ALfloat* pflValues)
835 {
836 ALCcontext *pContext;
837
838 pContext = alcGetCurrentContext();
839 SuspendContext(pContext);
840
841 if (pflValues)
842 {
843 if (alIsBuffer(buffer) && (buffer != 0))
844 {
845 switch(eParam)
846 {
847 default:
848 alSetError(AL_INVALID_ENUM);
849 break;
850 }
851 }
852 else
853 {
854 alSetError(AL_INVALID_NAME);
855 }
856 }
857 else
858 {
859 alSetError(AL_INVALID_VALUE);
860 }
861
862 ProcessContext(pContext);
863 }
864
865
866 ALAPI ALvoid ALAPIENTRY alGetBufferi(ALuint buffer, ALenum eParam, ALint *plValue)
867 {
868 ALCcontext *pContext;
869 ALbuffer *pBuffer;
870
871 pContext = alcGetCurrentContext();
872 SuspendContext(pContext);
873
874 if (plValue)
875 {
876 if (alIsBuffer(buffer) && (buffer != 0))
877 {
878 pBuffer = ((ALbuffer *)ALTHUNK_LOOKUPENTRY(buffer));
879
880 switch (eParam)
881 {
882 case AL_FREQUENCY:
883 *plValue = pBuffer->frequency;
884 break;
885
886 case AL_BITS:
887 *plValue = aluBytesFromFormat(pBuffer->format) * 8;
888 break;
889
890 case AL_CHANNELS:
891 *plValue = aluChannelsFromFormat(pBuffer->format);
892 break;
893
894 case AL_SIZE:
895 *plValue = pBuffer->size;
896 break;
897
898 default:
899 alSetError(AL_INVALID_ENUM);
900 break;
901 }
902 }
903 else
904 {
905 alSetError(AL_INVALID_NAME);
906 }
907 }
908 else
909 {
910 alSetError(AL_INVALID_VALUE);
911 }
912
913 ProcessContext(pContext);
914 }
915
916
917 ALAPI void ALAPIENTRY alGetBuffer3i(ALuint buffer, ALenum eParam, ALint* plValue1, ALint* plValue2, ALint* plValue3)
918 {
919 ALCcontext *pContext;
920
921 pContext = alcGetCurrentContext();
922 SuspendContext(pContext);
923
924 if ((plValue1) && (plValue2) && (plValue3))
925 {
926 if (alIsBuffer(buffer) && (buffer != 0))
927 {
928 switch(eParam)
929 {
930 default:
931 alSetError(AL_INVALID_ENUM);
932 break;
933 }
934 }
935 else
936 {
937 alSetError(AL_INVALID_NAME);
938 }
939 }
940 else
941 {
942 alSetError(AL_INVALID_VALUE);
943 }
944
945 ProcessContext(pContext);
946 }
947
948
949 ALAPI void ALAPIENTRY alGetBufferiv(ALuint buffer, ALenum eParam, ALint* plValues)
950 {
951 ALCcontext *pContext;
952
953 pContext = alcGetCurrentContext();
954 SuspendContext(pContext);
955
956 if (plValues)
957 {
958 if (alIsBuffer(buffer) && (buffer != 0))
959 {
960 switch (eParam)
961 {
962 case AL_FREQUENCY:
963 case AL_BITS:
964 case AL_CHANNELS:
965 case AL_SIZE:
966 alGetBufferi(buffer, eParam, plValues);
967 break;
968
969 default:
970 alSetError(AL_INVALID_ENUM);
971 break;
972 }
973 }
974 else
975 {
976 alSetError(AL_INVALID_NAME);
977 }
978 }
979 else
980 {
981 alSetError(AL_INVALID_VALUE);
982 }
983
984 ProcessContext(pContext);
985 }
986
987 /*
988 * LoadData
989 *
990 * Loads the specified data into the buffer, using the specified formats.
991 * Currently, the new format must be 16-bit, and must have the same channel
992 * configuration as the original format. This does NOT handle compressed
993 * formats (eg. IMA4).
994 */
995 static void LoadData(ALbuffer *ALBuf, const ALubyte *data, ALsizei size, ALuint freq, ALenum OrigFormat, ALenum NewFormat)
996 {
997 ALuint NewChannels = aluChannelsFromFormat(NewFormat);
998 ALuint OrigBytes = aluBytesFromFormat(OrigFormat);
999 ALuint OrigChannels = aluChannelsFromFormat(OrigFormat);
1000 ALsizei i;
1001
1002 assert(aluBytesFromFormat(NewFormat) == 2);
1003 assert(NewChannels == OrigChannels);
1004
1005 if ((size%(OrigBytes*OrigChannels)) != 0)
1006 {
1007 alSetError(AL_INVALID_VALUE);
1008 return;
1009 }
1010
1011 switch(OrigBytes)
1012 {
1013 case 1:
1014 size /= sizeof(ALubyte);
1015
1016 // 8bit Samples are converted to 16 bit here
1017 // Allocate 8 extra samples
1018 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
1019 if (ALBuf->data)
1020 {
1021 for (i = 0;i < size;i++)
1022 ALBuf->data[i] = (ALshort)((data[i]-128) << 8);
1023 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
1024
1025 ALBuf->format = NewFormat;
1026 ALBuf->eOriginalFormat = OrigFormat;
1027 ALBuf->size = size*1*sizeof(ALshort);
1028 ALBuf->frequency = freq;
1029 }
1030 else
1031 alSetError(AL_OUT_OF_MEMORY);
1032 break;
1033
1034 case 2:
1035 size /= sizeof(ALshort);
1036
1037 // Allocate 8 extra samples
1038 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
1039 if (ALBuf->data)
1040 {
1041 memcpy(ALBuf->data, data, size*1*sizeof(ALshort));
1042 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
1043
1044 ALBuf->format = NewFormat;
1045 ALBuf->eOriginalFormat = OrigFormat;
1046 ALBuf->size = size*1*sizeof(ALshort);
1047 ALBuf->frequency = freq;
1048 }
1049 else
1050 alSetError(AL_OUT_OF_MEMORY);
1051 break;
1052
1053 case 4:
1054 size /= sizeof(ALfloat);
1055
1056 // Allocate 8 extra samples
1057 ALBuf->data = realloc(ALBuf->data, (8*NewChannels + size) * (1*sizeof(ALshort)));
1058 if (ALBuf->data)
1059 {
1060 ALint smp;
1061 for (i = 0;i < size;i++)
1062 {
1063 smp = (((ALfloat*)data)[i] * 32767.5f - 0.5f);
1064 smp = min(smp, 32767);
1065 smp = max(smp, -32768);
1066 ALBuf->data[i] = (ALshort)smp;
1067 }
1068 memset(&(ALBuf->data[size]), 0, 16*NewChannels);
1069
1070 ALBuf->format = NewFormat;
1071 ALBuf->eOriginalFormat = OrigFormat;
1072 ALBuf->size = size*1*sizeof(ALshort);
1073 ALBuf->frequency = freq;
1074 }
1075 else
1076 alSetError(AL_OUT_OF_MEMORY);
1077 break;
1078
1079 default:
1080 assert(0);
1081 }
1082 }
1083
1084
1085 /*
1086 * ReleaseALBuffers()
1087 *
1088 * INTERNAL FN : Called by DLLMain on exit to destroy any buffers that still exist
1089 */
1090 ALvoid ReleaseALBuffers(ALvoid)
1091 {
1092 ALbuffer *ALBuffer;
1093 ALbuffer *ALBufferTemp;
1094
1095 #ifdef _DEBUG
1096 if(g_uiBufferCount > 0)
1097 AL_PRINT("exit() %d Buffer(s) NOT deleted\n", g_uiBufferCount);
1098 #endif
1099
1100 ALBuffer = g_pBuffers;
1101 while(ALBuffer)
1102 {
1103 // Release sample data
1104 free(ALBuffer->data);
1105
1106 // Release Buffer structure
1107 ALBufferTemp = ALBuffer;
1108 ALBuffer = ALBuffer->next;
1109 memset(ALBufferTemp, 0, sizeof(ALbuffer));
1110 free(ALBufferTemp);
1111 }
1112 g_pBuffers = NULL;
1113 g_uiBufferCount = 0;
1114 }