2007-11-13 18:02:18 -08:00
|
|
|
#ifndef AL_MAIN_H
|
|
|
|
#define AL_MAIN_H
|
|
|
|
|
|
|
|
#include <string.h>
|
2008-01-15 12:45:24 -08:00
|
|
|
#include <stdio.h>
|
2009-06-07 15:42:15 -07:00
|
|
|
#include <stdarg.h>
|
2012-04-23 19:46:05 -07:00
|
|
|
#include <assert.h>
|
2012-09-28 04:20:55 -07:00
|
|
|
#include <math.h>
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2008-08-08 08:12:41 -07:00
|
|
|
#ifdef HAVE_FENV_H
|
|
|
|
#include <fenv.h>
|
|
|
|
#endif
|
|
|
|
|
2009-05-16 23:26:39 -07:00
|
|
|
#include "AL/al.h"
|
|
|
|
#include "AL/alc.h"
|
|
|
|
#include "AL/alext.h"
|
|
|
|
|
2012-08-18 11:02:54 -07:00
|
|
|
/* Define int64_t and uint64_t types */
|
|
|
|
#if defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L
|
|
|
|
#include <inttypes.h>
|
|
|
|
#elif defined(_WIN32) && defined(__GNUC__)
|
|
|
|
#include <stdint.h>
|
|
|
|
#elif defined(_WIN32)
|
|
|
|
typedef __int64 int64_t;
|
|
|
|
typedef unsigned __int64 uint64_t;
|
|
|
|
#else
|
|
|
|
/* Fallback if nothing above works */
|
|
|
|
#include <inttypes.h>
|
|
|
|
#endif
|
|
|
|
|
2011-07-16 16:59:20 -07:00
|
|
|
#ifndef AL_SOFT_deferred_updates
|
|
|
|
#define AL_SOFT_deferred_updates 1
|
|
|
|
#define AL_DEFERRED_UPDATES_SOFT 0xC002
|
|
|
|
typedef ALvoid (AL_APIENTRY*LPALDEFERUPDATESSOFT)(void);
|
|
|
|
typedef ALvoid (AL_APIENTRY*LPALPROCESSUPDATESSOFT)(void);
|
|
|
|
#ifdef AL_ALEXT_PROTOTYPES
|
|
|
|
AL_API ALvoid AL_APIENTRY alDeferUpdatesSOFT(void);
|
|
|
|
AL_API ALvoid AL_APIENTRY alProcessUpdatesSOFT(void);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2012-08-18 11:02:54 -07:00
|
|
|
#ifndef AL_SOFT_source_latency
|
|
|
|
#define AL_SOFT_source_latency 1
|
|
|
|
#define AL_SAMPLE_OFFSET_LATENCY_SOFT 0x1200
|
2012-08-20 15:57:27 -07:00
|
|
|
#define AL_SEC_OFFSET_LATENCY_SOFT 0x1201
|
2012-08-18 11:02:54 -07:00
|
|
|
typedef int64_t ALint64SOFT;
|
|
|
|
typedef uint64_t ALuint64SOFT;
|
2012-10-13 00:56:39 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALSOURCEDSOFT)(ALuint,ALenum,ALdouble);
|
|
|
|
typedef void (AL_APIENTRY*LPALSOURCE3DSOFT)(ALuint,ALenum,ALdouble,ALdouble,ALdouble);
|
|
|
|
typedef void (AL_APIENTRY*LPALSOURCEDVSOFT)(ALuint,ALenum,const ALdouble*);
|
2012-08-20 15:26:35 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCEDSOFT)(ALuint,ALenum,ALdouble*);
|
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCE3DSOFT)(ALuint,ALenum,ALdouble*,ALdouble*,ALdouble*);
|
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCEDVSOFT)(ALuint,ALenum,ALdouble*);
|
2012-10-13 00:56:39 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALSOURCEI64SOFT)(ALuint,ALenum,ALint64SOFT);
|
|
|
|
typedef void (AL_APIENTRY*LPALSOURCE3I64SOFT)(ALuint,ALenum,ALint64SOFT,ALint64SOFT,ALint64SOFT);
|
|
|
|
typedef void (AL_APIENTRY*LPALSOURCEI64VSOFT)(ALuint,ALenum,const ALint64SOFT*);
|
2012-08-18 11:02:54 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCEI64SOFT)(ALuint,ALenum,ALint64SOFT*);
|
2012-08-20 14:48:08 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCE3I64SOFT)(ALuint,ALenum,ALint64SOFT*,ALint64SOFT*,ALint64SOFT*);
|
2012-08-18 11:02:54 -07:00
|
|
|
typedef void (AL_APIENTRY*LPALGETSOURCEI64VSOFT)(ALuint,ALenum,ALint64SOFT*);
|
|
|
|
#ifdef AL_ALEXT_PROTOTYPES
|
2012-10-13 00:56:39 -07:00
|
|
|
AL_API void AL_APIENTRY alSourcedSOFT(ALuint source, ALenum param, ALdouble value);
|
|
|
|
AL_API void AL_APIENTRY alSource3dSOFT(ALuint source, ALenum param, ALdouble value1, ALdouble value2, ALdouble value3);
|
|
|
|
AL_API void AL_APIENTRY alSourcedvSOFT(ALuint source, ALenum param, const ALdouble *values);
|
2012-08-20 15:26:35 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcedSOFT(ALuint source, ALenum param, ALdouble *value);
|
|
|
|
AL_API void AL_APIENTRY alGetSource3dSOFT(ALuint source, ALenum param, ALdouble *value1, ALdouble *value2, ALdouble *value3);
|
|
|
|
AL_API void AL_APIENTRY alGetSourcedvSOFT(ALuint source, ALenum param, ALdouble *values);
|
2012-10-13 00:56:39 -07:00
|
|
|
AL_API void AL_APIENTRY alSourcei64SOFT(ALuint source, ALenum param, ALint64SOFT value);
|
|
|
|
AL_API void AL_APIENTRY alSource3i64SOFT(ALuint source, ALenum param, ALint64SOFT value1, ALint64SOFT value2, ALint64SOFT value3);
|
|
|
|
AL_API void AL_APIENTRY alSourcei64vSOFT(ALuint source, ALenum param, const ALint64SOFT *values);
|
2012-08-20 14:50:43 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcei64SOFT(ALuint source, ALenum param, ALint64SOFT *value);
|
2012-08-20 14:48:08 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSource3i64SOFT(ALuint source, ALenum param, ALint64SOFT *value1, ALint64SOFT *value2, ALint64SOFT *value3);
|
2012-08-18 11:02:54 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcei64vSOFT(ALuint source, ALenum param, ALint64SOFT *values);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2010-07-30 20:23:55 -07:00
|
|
|
|
2009-12-24 15:41:45 -08:00
|
|
|
#if defined(HAVE_STDINT_H)
|
|
|
|
#include <stdint.h>
|
|
|
|
typedef int64_t ALint64;
|
|
|
|
typedef uint64_t ALuint64;
|
|
|
|
#elif defined(HAVE___INT64)
|
|
|
|
typedef __int64 ALint64;
|
|
|
|
typedef unsigned __int64 ALuint64;
|
|
|
|
#elif (SIZEOF_LONG == 8)
|
|
|
|
typedef long ALint64;
|
|
|
|
typedef unsigned long ALuint64;
|
|
|
|
#elif (SIZEOF_LONG_LONG == 8)
|
|
|
|
typedef long long ALint64;
|
|
|
|
typedef unsigned long long ALuint64;
|
|
|
|
#endif
|
|
|
|
|
2011-06-15 23:46:35 -07:00
|
|
|
typedef ptrdiff_t ALintptrEXT;
|
|
|
|
typedef ptrdiff_t ALsizeiptrEXT;
|
|
|
|
|
2012-08-19 22:31:55 -07:00
|
|
|
#define MAKEU64(x,y) (((ALuint64)(x)<<32)|(ALuint64)(y))
|
|
|
|
|
2010-01-12 08:48:25 -08:00
|
|
|
#ifdef HAVE_GCC_FORMAT
|
|
|
|
#define PRINTF_STYLE(x, y) __attribute__((format(printf, (x), (y))))
|
|
|
|
#else
|
|
|
|
#define PRINTF_STYLE(x, y)
|
|
|
|
#endif
|
|
|
|
|
2011-08-28 17:21:01 -07:00
|
|
|
|
2012-02-15 21:47:35 -08:00
|
|
|
static const union {
|
|
|
|
ALuint u;
|
|
|
|
ALubyte b[sizeof(ALuint)];
|
|
|
|
} EndianTest = { 1 };
|
|
|
|
#define IS_LITTLE_ENDIAN (EndianTest.b[0] == 1)
|
|
|
|
|
2012-02-19 12:07:40 -08:00
|
|
|
#define COUNTOF(x) (sizeof((x))/sizeof((x)[0]))
|
2012-02-15 21:47:35 -08:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#ifdef _WIN32
|
2008-01-15 12:45:24 -08:00
|
|
|
|
2012-10-06 21:36:15 -07:00
|
|
|
#define WIN32_LEAN_AND_MEAN
|
2007-11-13 18:02:18 -08:00
|
|
|
#include <windows.h>
|
|
|
|
|
2011-08-28 17:21:01 -07:00
|
|
|
typedef DWORD pthread_key_t;
|
|
|
|
int pthread_key_create(pthread_key_t *key, void (*callback)(void*));
|
|
|
|
int pthread_key_delete(pthread_key_t key);
|
|
|
|
void *pthread_getspecific(pthread_key_t key);
|
|
|
|
int pthread_setspecific(pthread_key_t key, void *val);
|
2009-09-12 16:45:46 -07:00
|
|
|
|
2011-06-12 04:36:24 -07:00
|
|
|
#define HAVE_DYNLOAD 1
|
2011-06-12 04:45:03 -07:00
|
|
|
void *LoadLib(const char *name);
|
|
|
|
void CloseLib(void *handle);
|
|
|
|
void *GetSymbol(void *handle, const char *name);
|
2011-06-12 04:36:24 -07:00
|
|
|
|
2012-03-01 06:07:00 -08:00
|
|
|
WCHAR *strdupW(const WCHAR *str);
|
|
|
|
|
2011-08-20 03:59:46 -07:00
|
|
|
typedef LONG pthread_once_t;
|
|
|
|
#define PTHREAD_ONCE_INIT 0
|
|
|
|
void pthread_once(pthread_once_t *once, void (*callback)(void));
|
|
|
|
|
2011-09-20 14:36:42 -07:00
|
|
|
static __inline int sched_yield(void)
|
|
|
|
{ SwitchToThread(); return 0; }
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#else
|
|
|
|
|
2009-11-01 10:03:05 -08:00
|
|
|
#include <unistd.h>
|
2007-11-13 18:02:18 -08:00
|
|
|
#include <assert.h>
|
|
|
|
#include <pthread.h>
|
2008-01-11 06:01:51 -08:00
|
|
|
#include <sys/time.h>
|
2008-01-11 08:15:44 -08:00
|
|
|
#include <time.h>
|
|
|
|
#include <errno.h>
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2009-12-04 01:33:50 -08:00
|
|
|
#define IsBadWritePtr(a,b) ((a) == NULL && (b) != 0)
|
2007-11-13 18:02:18 -08:00
|
|
|
|
|
|
|
typedef pthread_mutex_t CRITICAL_SECTION;
|
2011-05-29 03:08:26 -07:00
|
|
|
void InitializeCriticalSection(CRITICAL_SECTION *cs);
|
|
|
|
void DeleteCriticalSection(CRITICAL_SECTION *cs);
|
2011-05-29 03:28:34 -07:00
|
|
|
void EnterCriticalSection(CRITICAL_SECTION *cs);
|
|
|
|
void LeaveCriticalSection(CRITICAL_SECTION *cs);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-06-12 05:04:36 -07:00
|
|
|
ALuint timeGetTime(void);
|
2011-09-20 14:36:42 -07:00
|
|
|
void Sleep(ALuint t);
|
2011-06-12 04:36:24 -07:00
|
|
|
|
|
|
|
#if defined(HAVE_DLFCN_H)
|
|
|
|
#define HAVE_DYNLOAD 1
|
2011-06-12 04:45:03 -07:00
|
|
|
void *LoadLib(const char *name);
|
|
|
|
void CloseLib(void *handle);
|
|
|
|
void *GetSymbol(void *handle, const char *name);
|
2011-06-12 04:36:24 -07:00
|
|
|
#endif
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#endif
|
|
|
|
|
2012-01-01 15:00:03 -08:00
|
|
|
typedef void *volatile XchgPtr;
|
|
|
|
|
2011-08-29 10:51:58 -07:00
|
|
|
#if defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 1))
|
2011-08-28 15:44:03 -07:00
|
|
|
typedef ALuint RefCount;
|
|
|
|
static __inline RefCount IncrementRef(volatile RefCount *ptr)
|
|
|
|
{ return __sync_add_and_fetch(ptr, 1); }
|
|
|
|
static __inline RefCount DecrementRef(volatile RefCount *ptr)
|
|
|
|
{ return __sync_sub_and_fetch(ptr, 1); }
|
|
|
|
|
2011-08-29 21:30:12 -07:00
|
|
|
static __inline int ExchangeInt(volatile int *ptr, int newval)
|
|
|
|
{
|
|
|
|
return __sync_lock_test_and_set(ptr, newval);
|
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline void *ExchangePtr(XchgPtr *ptr, void *newval)
|
2011-08-29 23:55:24 -07:00
|
|
|
{
|
|
|
|
return __sync_lock_test_and_set(ptr, newval);
|
|
|
|
}
|
2011-08-29 21:30:12 -07:00
|
|
|
static __inline ALboolean CompExchangeInt(volatile int *ptr, int oldval, int newval)
|
|
|
|
{
|
|
|
|
return __sync_bool_compare_and_swap(ptr, oldval, newval);
|
2011-08-29 00:50:55 -07:00
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline ALboolean CompExchangePtr(XchgPtr *ptr, void *oldval, void *newval)
|
2011-09-02 02:57:21 -07:00
|
|
|
{
|
|
|
|
return __sync_bool_compare_and_swap(ptr, oldval, newval);
|
|
|
|
}
|
2011-08-29 00:50:55 -07:00
|
|
|
|
2011-09-23 12:42:13 -07:00
|
|
|
#elif defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__))
|
|
|
|
|
|
|
|
static __inline int xaddl(volatile int *dest, int incr)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
__asm__ __volatile__("lock; xaddl %0,(%1)"
|
|
|
|
: "=r" (ret)
|
|
|
|
: "r" (dest), "0" (incr)
|
|
|
|
: "memory");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef int RefCount;
|
|
|
|
static __inline RefCount IncrementRef(volatile RefCount *ptr)
|
|
|
|
{ return xaddl(ptr, 1)+1; }
|
|
|
|
static __inline RefCount DecrementRef(volatile RefCount *ptr)
|
|
|
|
{ return xaddl(ptr, -1)-1; }
|
|
|
|
|
|
|
|
static __inline int ExchangeInt(volatile int *dest, int newval)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
__asm__ __volatile__("lock; xchgl %0,(%1)"
|
|
|
|
: "=r" (ret)
|
|
|
|
: "r" (dest), "0" (newval)
|
|
|
|
: "memory");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline ALboolean CompExchangeInt(volatile int *dest, int oldval, int newval)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
__asm__ __volatile__("lock; cmpxchgl %2,(%1)"
|
|
|
|
: "=a" (ret)
|
|
|
|
: "r" (dest), "r" (newval), "0" (oldval)
|
|
|
|
: "memory");
|
|
|
|
return ret == oldval;
|
|
|
|
}
|
|
|
|
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline void *ExchangePtr(XchgPtr *dest, void *newval)
|
2011-09-23 12:42:13 -07:00
|
|
|
{
|
|
|
|
void *ret;
|
|
|
|
__asm__ __volatile__(
|
|
|
|
#ifdef __i386__
|
|
|
|
"lock; xchgl %0,(%1)"
|
|
|
|
#else
|
|
|
|
"lock; xchgq %0,(%1)"
|
|
|
|
#endif
|
|
|
|
: "=r" (ret)
|
|
|
|
: "r" (dest), "0" (newval)
|
|
|
|
: "memory"
|
|
|
|
);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline ALboolean CompExchangePtr(XchgPtr *dest, void *oldval, void *newval)
|
2011-09-23 12:42:13 -07:00
|
|
|
{
|
|
|
|
void *ret;
|
|
|
|
__asm__ __volatile__(
|
|
|
|
#ifdef __i386__
|
|
|
|
"lock; cmpxchgl %2,(%1)"
|
|
|
|
#else
|
|
|
|
"lock; cmpxchgq %2,(%1)"
|
|
|
|
#endif
|
|
|
|
: "=a" (ret)
|
|
|
|
: "r" (dest), "r" (newval), "0" (oldval)
|
|
|
|
: "memory"
|
|
|
|
);
|
|
|
|
return ret == oldval;
|
|
|
|
}
|
|
|
|
|
2011-08-28 15:44:03 -07:00
|
|
|
#elif defined(_WIN32)
|
|
|
|
|
|
|
|
typedef LONG RefCount;
|
|
|
|
static __inline RefCount IncrementRef(volatile RefCount *ptr)
|
2011-08-29 01:01:20 -07:00
|
|
|
{ return InterlockedIncrement(ptr); }
|
2011-08-28 15:44:03 -07:00
|
|
|
static __inline RefCount DecrementRef(volatile RefCount *ptr)
|
|
|
|
{ return InterlockedDecrement(ptr); }
|
|
|
|
|
2011-08-30 17:55:52 -07:00
|
|
|
extern ALbyte LONG_size_does_not_match_int[(sizeof(LONG)==sizeof(int))?1:-1];
|
2011-08-29 21:30:12 -07:00
|
|
|
|
|
|
|
static __inline int ExchangeInt(volatile int *ptr, int newval)
|
|
|
|
{
|
|
|
|
union {
|
|
|
|
volatile int *i;
|
|
|
|
volatile LONG *l;
|
|
|
|
} u = { ptr };
|
|
|
|
return InterlockedExchange(u.l, newval);
|
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline void *ExchangePtr(XchgPtr *ptr, void *newval)
|
2011-08-29 23:55:24 -07:00
|
|
|
{
|
2011-08-30 17:55:52 -07:00
|
|
|
return InterlockedExchangePointer(ptr, newval);
|
2011-08-29 23:55:24 -07:00
|
|
|
}
|
2011-08-29 21:30:12 -07:00
|
|
|
static __inline ALboolean CompExchangeInt(volatile int *ptr, int oldval, int newval)
|
|
|
|
{
|
|
|
|
union {
|
|
|
|
volatile int *i;
|
|
|
|
volatile LONG *l;
|
|
|
|
} u = { ptr };
|
|
|
|
return InterlockedCompareExchange(u.l, newval, oldval) == oldval;
|
2011-08-29 00:50:55 -07:00
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline ALboolean CompExchangePtr(XchgPtr *ptr, void *oldval, void *newval)
|
2011-09-02 02:57:21 -07:00
|
|
|
{
|
2011-09-11 09:39:29 -07:00
|
|
|
return InterlockedCompareExchangePointer(ptr, newval, oldval) == oldval;
|
2011-09-02 02:57:21 -07:00
|
|
|
}
|
2011-08-29 00:50:55 -07:00
|
|
|
|
2011-08-28 15:44:03 -07:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
|
|
|
|
#include <libkern/OSAtomic.h>
|
|
|
|
|
|
|
|
typedef int32_t RefCount;
|
|
|
|
static __inline RefCount IncrementRef(volatile RefCount *ptr)
|
|
|
|
{ return OSAtomicIncrement32Barrier(ptr); }
|
|
|
|
static __inline RefCount DecrementRef(volatile RefCount *ptr)
|
|
|
|
{ return OSAtomicDecrement32Barrier(ptr); }
|
|
|
|
|
2011-08-29 21:30:12 -07:00
|
|
|
static __inline int ExchangeInt(volatile int *ptr, int newval)
|
|
|
|
{
|
|
|
|
/* Really? No regular old atomic swap? */
|
|
|
|
int oldval;
|
|
|
|
do {
|
|
|
|
oldval = *ptr;
|
|
|
|
} while(!OSAtomicCompareAndSwap32Barrier(oldval, newval, ptr));
|
|
|
|
return oldval;
|
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline void *ExchangePtr(XchgPtr *ptr, void *newval)
|
2011-08-29 23:55:24 -07:00
|
|
|
{
|
|
|
|
void *oldval;
|
|
|
|
do {
|
|
|
|
oldval = *ptr;
|
|
|
|
} while(!OSAtomicCompareAndSwapPtrBarrier(oldval, newval, ptr));
|
|
|
|
return oldval;
|
|
|
|
}
|
2011-08-29 21:30:12 -07:00
|
|
|
static __inline ALboolean CompExchangeInt(volatile int *ptr, int oldval, int newval)
|
|
|
|
{
|
|
|
|
return OSAtomicCompareAndSwap32Barrier(oldval, newval, ptr);
|
2011-08-29 00:50:55 -07:00
|
|
|
}
|
2012-01-01 15:00:03 -08:00
|
|
|
static __inline ALboolean CompExchangePtr(XchgPtr *ptr, void *oldval, void *newval)
|
2011-09-02 02:57:21 -07:00
|
|
|
{
|
|
|
|
return OSAtomicCompareAndSwapPtrBarrier(oldval, newval, ptr);
|
|
|
|
}
|
2011-08-29 00:50:55 -07:00
|
|
|
|
2011-08-28 15:44:03 -07:00
|
|
|
#else
|
|
|
|
#error "No atomic functions available on this platform!"
|
2011-08-29 10:51:58 -07:00
|
|
|
typedef ALuint RefCount;
|
2011-08-28 15:44:03 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2011-08-29 19:15:22 -07:00
|
|
|
typedef struct {
|
|
|
|
volatile RefCount read_count;
|
|
|
|
volatile RefCount write_count;
|
|
|
|
volatile ALenum read_lock;
|
|
|
|
volatile ALenum read_entry_lock;
|
|
|
|
volatile ALenum write_lock;
|
|
|
|
} RWLock;
|
|
|
|
|
2011-08-31 01:00:19 -07:00
|
|
|
void RWLockInit(RWLock *lock);
|
2011-08-29 19:15:22 -07:00
|
|
|
void ReadLock(RWLock *lock);
|
|
|
|
void ReadUnlock(RWLock *lock);
|
|
|
|
void WriteLock(RWLock *lock);
|
|
|
|
void WriteUnlock(RWLock *lock);
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct UIntMap {
|
|
|
|
struct {
|
|
|
|
ALuint key;
|
|
|
|
ALvoid *value;
|
|
|
|
} *array;
|
|
|
|
ALsizei size;
|
|
|
|
ALsizei maxsize;
|
2011-08-30 20:33:47 -07:00
|
|
|
ALsizei limit;
|
2011-08-29 19:15:22 -07:00
|
|
|
RWLock lock;
|
|
|
|
} UIntMap;
|
|
|
|
extern UIntMap TlsDestructor;
|
|
|
|
|
2011-08-30 20:33:47 -07:00
|
|
|
void InitUIntMap(UIntMap *map, ALsizei limit);
|
2011-08-29 19:15:22 -07:00
|
|
|
void ResetUIntMap(UIntMap *map);
|
|
|
|
ALenum InsertUIntMapEntry(UIntMap *map, ALuint key, ALvoid *value);
|
2011-10-06 06:39:13 -07:00
|
|
|
ALvoid *RemoveUIntMapKey(UIntMap *map, ALuint key);
|
2011-08-29 19:15:22 -07:00
|
|
|
ALvoid *LookupUIntMapKey(UIntMap *map, ALuint key);
|
|
|
|
|
2011-08-29 20:21:28 -07:00
|
|
|
static __inline void LockUIntMapRead(UIntMap *map)
|
|
|
|
{ ReadLock(&map->lock); }
|
|
|
|
static __inline void UnlockUIntMapRead(UIntMap *map)
|
|
|
|
{ ReadUnlock(&map->lock); }
|
|
|
|
static __inline void LockUIntMapWrite(UIntMap *map)
|
|
|
|
{ WriteLock(&map->lock); }
|
|
|
|
static __inline void UnlockUIntMapWrite(UIntMap *map)
|
|
|
|
{ WriteUnlock(&map->lock); }
|
2011-08-29 19:15:22 -07:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
2008-02-08 19:55:51 -08:00
|
|
|
extern "C" {
|
2007-11-13 18:02:18 -08:00
|
|
|
#endif
|
|
|
|
|
2012-09-14 02:52:37 -07:00
|
|
|
struct Hrtf;
|
2008-07-24 00:41:25 -07:00
|
|
|
|
|
|
|
|
2012-09-14 02:52:37 -07:00
|
|
|
#define DEFAULT_OUTPUT_RATE (44100)
|
|
|
|
#define MIN_OUTPUT_RATE (8000)
|
2011-09-18 09:52:40 -07:00
|
|
|
|
|
|
|
|
2009-11-19 09:50:15 -08:00
|
|
|
// Find the next power-of-2 for non-power-of-2 numbers.
|
|
|
|
static __inline ALuint NextPowerOf2(ALuint value)
|
|
|
|
{
|
|
|
|
ALuint powerOf2 = 1;
|
|
|
|
|
|
|
|
if(value)
|
|
|
|
{
|
|
|
|
value--;
|
|
|
|
while(value)
|
|
|
|
{
|
|
|
|
value >>= 1;
|
|
|
|
powerOf2 <<= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return powerOf2;
|
|
|
|
}
|
|
|
|
|
2011-09-29 03:51:46 -07:00
|
|
|
/* Fast float-to-int conversion. Assumes the FPU is already in round-to-zero
|
|
|
|
* mode. */
|
|
|
|
static __inline ALint fastf2i(ALfloat f)
|
|
|
|
{
|
2012-09-28 04:20:55 -07:00
|
|
|
#ifdef HAVE_LRINTF
|
|
|
|
return lrintf(f);
|
|
|
|
#elif defined(_MSC_VER) && defined(_M_IX86)
|
2011-09-29 03:51:46 -07:00
|
|
|
ALint i;
|
|
|
|
__asm fld f
|
|
|
|
__asm fistp i
|
|
|
|
#else
|
2012-09-28 04:20:55 -07:00
|
|
|
return (ALint)f;
|
2011-09-29 03:51:46 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fast float-to-uint conversion. Assumes the FPU is already in round-to-zero
|
|
|
|
* mode. */
|
|
|
|
static __inline ALuint fastf2u(ALfloat f)
|
|
|
|
{ return fastf2i(f); }
|
|
|
|
|
2009-11-19 09:50:15 -08:00
|
|
|
|
2011-08-20 03:59:46 -07:00
|
|
|
enum DevProbe {
|
|
|
|
ALL_DEVICE_PROBE,
|
|
|
|
CAPTURE_DEVICE_PROBE
|
|
|
|
};
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
typedef struct {
|
2011-08-24 14:24:48 -07:00
|
|
|
ALCenum (*OpenPlayback)(ALCdevice*, const ALCchar*);
|
2007-11-13 18:02:18 -08:00
|
|
|
void (*ClosePlayback)(ALCdevice*);
|
2009-09-16 00:24:44 -07:00
|
|
|
ALCboolean (*ResetPlayback)(ALCdevice*);
|
2012-03-05 07:11:09 -08:00
|
|
|
ALCboolean (*StartPlayback)(ALCdevice*);
|
2009-09-16 00:24:44 -07:00
|
|
|
void (*StopPlayback)(ALCdevice*);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-08-24 14:44:15 -07:00
|
|
|
ALCenum (*OpenCapture)(ALCdevice*, const ALCchar*);
|
2007-11-13 18:02:18 -08:00
|
|
|
void (*CloseCapture)(ALCdevice*);
|
|
|
|
void (*StartCapture)(ALCdevice*);
|
|
|
|
void (*StopCapture)(ALCdevice*);
|
2011-09-14 02:01:35 -07:00
|
|
|
ALCenum (*CaptureSamples)(ALCdevice*, void*, ALCuint);
|
2007-11-13 18:02:18 -08:00
|
|
|
ALCuint (*AvailableSamples)(ALCdevice*);
|
2012-08-17 13:38:52 -07:00
|
|
|
|
2012-08-18 15:58:04 -07:00
|
|
|
void (*Lock)(ALCdevice*);
|
|
|
|
void (*Unlock)(ALCdevice*);
|
|
|
|
|
2012-08-17 13:38:52 -07:00
|
|
|
ALint64 (*GetLatency)(ALCdevice*);
|
2007-11-13 18:02:18 -08:00
|
|
|
} BackendFuncs;
|
|
|
|
|
2011-08-20 03:59:46 -07:00
|
|
|
struct BackendInfo {
|
|
|
|
const char *name;
|
|
|
|
ALCboolean (*Init)(BackendFuncs*);
|
|
|
|
void (*Deinit)(void);
|
|
|
|
void (*Probe)(enum DevProbe);
|
|
|
|
BackendFuncs Funcs;
|
2009-08-27 06:09:33 -07:00
|
|
|
};
|
|
|
|
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_alsa_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_alsa_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_alsa_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_oss_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_oss_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_oss_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_solaris_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_solaris_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_solaris_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_sndio_init(BackendFuncs *func_list);
|
2011-06-22 19:29:13 -07:00
|
|
|
void alc_sndio_deinit(void);
|
|
|
|
void alc_sndio_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alcMMDevApiInit(BackendFuncs *func_list);
|
2011-05-15 20:26:25 -07:00
|
|
|
void alcMMDevApiDeinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alcMMDevApiProbe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alcDSoundInit(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alcDSoundDeinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alcDSoundProbe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alcWinMMInit(BackendFuncs *FuncList);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alcWinMMDeinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alcWinMMProbe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_pa_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_pa_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_pa_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_wave_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_wave_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_wave_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_pulse_init(BackendFuncs *func_list);
|
2009-08-26 23:45:00 -07:00
|
|
|
void alc_pulse_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_pulse_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_ca_init(BackendFuncs *func_list);
|
2011-03-15 04:58:56 -07:00
|
|
|
void alc_ca_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_ca_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_opensl_init(BackendFuncs *func_list);
|
2011-06-12 04:41:42 -07:00
|
|
|
void alc_opensl_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_opensl_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_null_init(BackendFuncs *func_list);
|
2010-05-28 16:41:52 -07:00
|
|
|
void alc_null_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_null_probe(enum DevProbe type);
|
2011-08-17 06:05:26 -07:00
|
|
|
ALCboolean alc_loopback_init(BackendFuncs *func_list);
|
2011-03-11 00:13:44 -08:00
|
|
|
void alc_loopback_deinit(void);
|
2011-06-14 04:02:58 -07:00
|
|
|
void alc_loopback_probe(enum DevProbe type);
|
2007-12-06 22:22:11 -08:00
|
|
|
|
|
|
|
|
2012-09-14 02:42:36 -07:00
|
|
|
enum DistanceModel {
|
|
|
|
InverseDistanceClamped = AL_INVERSE_DISTANCE_CLAMPED,
|
|
|
|
LinearDistanceClamped = AL_LINEAR_DISTANCE_CLAMPED,
|
|
|
|
ExponentDistanceClamped = AL_EXPONENT_DISTANCE_CLAMPED,
|
|
|
|
InverseDistance = AL_INVERSE_DISTANCE,
|
|
|
|
LinearDistance = AL_LINEAR_DISTANCE,
|
|
|
|
ExponentDistance = AL_EXPONENT_DISTANCE,
|
|
|
|
DisableDistance = AL_NONE,
|
|
|
|
|
|
|
|
DefaultDistanceModel = InverseDistanceClamped
|
|
|
|
};
|
|
|
|
|
|
|
|
enum Resampler {
|
|
|
|
PointResampler,
|
|
|
|
LinearResampler,
|
|
|
|
CubicResampler,
|
|
|
|
|
|
|
|
ResamplerMax,
|
|
|
|
};
|
|
|
|
|
2012-09-14 02:14:29 -07:00
|
|
|
enum Channel {
|
|
|
|
FrontLeft = 0,
|
|
|
|
FrontRight,
|
|
|
|
FrontCenter,
|
|
|
|
LFE,
|
|
|
|
BackLeft,
|
|
|
|
BackRight,
|
|
|
|
BackCenter,
|
|
|
|
SideLeft,
|
|
|
|
SideRight,
|
|
|
|
|
|
|
|
MaxChannels,
|
|
|
|
};
|
|
|
|
|
2012-09-14 02:42:36 -07:00
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
/* Device formats */
|
|
|
|
enum DevFmtType {
|
2011-11-01 16:00:47 -07:00
|
|
|
DevFmtByte = ALC_BYTE_SOFT,
|
|
|
|
DevFmtUByte = ALC_UNSIGNED_BYTE_SOFT,
|
|
|
|
DevFmtShort = ALC_SHORT_SOFT,
|
|
|
|
DevFmtUShort = ALC_UNSIGNED_SHORT_SOFT,
|
2012-02-14 11:44:57 -08:00
|
|
|
DevFmtInt = ALC_INT_SOFT,
|
|
|
|
DevFmtUInt = ALC_UNSIGNED_INT_SOFT,
|
2012-03-13 15:32:44 -07:00
|
|
|
DevFmtFloat = ALC_FLOAT_SOFT,
|
|
|
|
|
|
|
|
DevFmtTypeDefault = DevFmtFloat
|
2010-12-04 19:50:00 -08:00
|
|
|
};
|
|
|
|
enum DevFmtChannels {
|
2011-11-01 16:00:47 -07:00
|
|
|
DevFmtMono = ALC_MONO_SOFT,
|
|
|
|
DevFmtStereo = ALC_STEREO_SOFT,
|
|
|
|
DevFmtQuad = ALC_QUAD_SOFT,
|
|
|
|
DevFmtX51 = ALC_5POINT1_SOFT,
|
|
|
|
DevFmtX61 = ALC_6POINT1_SOFT,
|
|
|
|
DevFmtX71 = ALC_7POINT1_SOFT,
|
2011-05-28 19:35:32 -07:00
|
|
|
|
|
|
|
/* Similar to 5.1, except using the side channels instead of back */
|
2012-03-13 15:32:44 -07:00
|
|
|
DevFmtX51Side = 0x80000000,
|
|
|
|
|
|
|
|
DevFmtChannelsDefault = DevFmtStereo
|
2010-12-04 19:50:00 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
ALuint BytesFromDevFmt(enum DevFmtType type);
|
|
|
|
ALuint ChannelsFromDevFmt(enum DevFmtChannels chans);
|
|
|
|
static __inline ALuint FrameSizeFromDevFmt(enum DevFmtChannels chans,
|
|
|
|
enum DevFmtType type)
|
|
|
|
{
|
|
|
|
return ChannelsFromDevFmt(chans) * BytesFromDevFmt(type);
|
|
|
|
}
|
|
|
|
|
2010-05-01 19:59:41 -07:00
|
|
|
|
2011-03-12 20:37:22 -08:00
|
|
|
extern const struct EffectList {
|
|
|
|
const char *name;
|
|
|
|
int type;
|
|
|
|
const char *ename;
|
2011-03-12 20:54:49 -08:00
|
|
|
ALenum val;
|
2011-03-12 20:37:22 -08:00
|
|
|
} EffectList[];
|
|
|
|
|
|
|
|
|
2012-02-23 15:25:30 -08:00
|
|
|
enum DeviceType {
|
|
|
|
Playback,
|
|
|
|
Capture,
|
|
|
|
Loopback
|
|
|
|
};
|
|
|
|
|
2012-09-14 02:14:29 -07:00
|
|
|
|
|
|
|
/* Size for temporary storage of buffer data, in ALfloats. Larger values need
|
2012-10-05 06:42:26 -07:00
|
|
|
* more memory, while smaller values may need more iterations. The value needs
|
2012-09-14 02:14:29 -07:00
|
|
|
* to be a sensible size, however, as it constrains the max stepping value used
|
|
|
|
* for mixing, as well as the maximum number of samples per mixing iteration.
|
2012-09-26 18:04:22 -07:00
|
|
|
*
|
|
|
|
* The mixer requires being able to do two samplings per mixing loop. With the
|
2012-09-28 03:30:57 -07:00
|
|
|
* cubic resampler (which requires 3 padding samples), this limits a 2048
|
|
|
|
* buffer size to about 2044. This means that buffer_freq*source_pitch cannot
|
|
|
|
* exceed device_freq*2044 for a 32-bit buffer.
|
2012-09-14 02:14:29 -07:00
|
|
|
*/
|
|
|
|
#ifndef BUFFERSIZE
|
2012-09-28 03:30:57 -07:00
|
|
|
#define BUFFERSIZE 2048
|
2012-09-14 02:14:29 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
struct ALCdevice_struct
|
|
|
|
{
|
2011-09-10 02:43:07 -07:00
|
|
|
volatile RefCount ref;
|
|
|
|
|
2012-02-23 15:25:30 -08:00
|
|
|
ALCboolean Connected;
|
|
|
|
enum DeviceType Type;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-07-06 05:07:36 -07:00
|
|
|
CRITICAL_SECTION Mutex;
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
ALuint Frequency;
|
2008-02-12 19:38:27 -08:00
|
|
|
ALuint UpdateSize;
|
2009-09-16 22:58:54 -07:00
|
|
|
ALuint NumUpdates;
|
2010-12-04 19:50:00 -08:00
|
|
|
enum DevFmtChannels FmtChans;
|
|
|
|
enum DevFmtType FmtType;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 22:50:11 -07:00
|
|
|
ALCchar *DeviceName;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-09-10 05:19:08 -07:00
|
|
|
volatile ALCenum LastError;
|
2009-12-28 23:19:13 -08:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
// Maximum number of sources that can be created
|
|
|
|
ALuint MaxNoOfSources;
|
2009-06-07 14:53:22 -07:00
|
|
|
// Maximum number of slots that can be created
|
|
|
|
ALuint AuxiliaryEffectSlotMax;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2010-03-25 18:39:16 -07:00
|
|
|
ALCuint NumMonoSources;
|
|
|
|
ALCuint NumStereoSources;
|
2009-07-06 03:09:01 -07:00
|
|
|
ALuint NumAuxSends;
|
|
|
|
|
2010-05-01 19:59:41 -07:00
|
|
|
// Map of Buffers for this device
|
|
|
|
UIntMap BufferMap;
|
2009-08-15 09:14:08 -07:00
|
|
|
|
2010-05-18 17:41:06 -07:00
|
|
|
// Map of Effects for this device
|
|
|
|
UIntMap EffectMap;
|
2009-08-15 09:39:18 -07:00
|
|
|
|
2010-05-18 17:54:45 -07:00
|
|
|
// Map of Filters for this device
|
|
|
|
UIntMap FilterMap;
|
2009-08-15 09:39:18 -07:00
|
|
|
|
2011-09-18 09:52:40 -07:00
|
|
|
/* HRTF filter tables */
|
|
|
|
const struct Hrtf *Hrtf;
|
|
|
|
|
2009-09-15 19:06:47 -07:00
|
|
|
// Stereo-to-binaural filter
|
|
|
|
struct bs2b *Bs2b;
|
|
|
|
ALCint Bs2bLevel;
|
|
|
|
|
2011-05-01 18:18:37 -07:00
|
|
|
// Device flags
|
|
|
|
ALuint Flags;
|
2011-05-01 13:19:23 -07:00
|
|
|
|
2012-06-28 18:49:49 -07:00
|
|
|
enum Channel DevChannels[MaxChannels];
|
2009-12-02 04:03:51 -08:00
|
|
|
|
2012-06-28 18:49:49 -07:00
|
|
|
enum Channel Speaker2Chan[MaxChannels];
|
|
|
|
ALfloat SpeakerAngle[MaxChannels];
|
2010-04-08 15:58:11 -07:00
|
|
|
ALuint NumChan;
|
|
|
|
|
2012-10-05 06:42:26 -07:00
|
|
|
/* Temp storage used for mixing. +1 for the predictive sample. */
|
|
|
|
ALIGN(16) ALfloat SampleData1[BUFFERSIZE+1];
|
|
|
|
ALIGN(16) ALfloat SampleData2[BUFFERSIZE+1];
|
|
|
|
|
2012-08-29 01:40:42 -07:00
|
|
|
// Dry path buffer mix
|
2012-09-11 06:32:42 -07:00
|
|
|
ALIGN(16) ALfloat DryBuffer[MaxChannels][BUFFERSIZE];
|
2012-08-29 01:40:42 -07:00
|
|
|
|
|
|
|
ALIGN(16) ALfloat ClickRemoval[MaxChannels];
|
|
|
|
ALIGN(16) ALfloat PendingClicks[MaxChannels];
|
2010-08-14 21:30:14 -07:00
|
|
|
|
2012-01-19 19:30:03 -08:00
|
|
|
/* Default effect slot */
|
|
|
|
struct ALeffectslot *DefaultSlot;
|
|
|
|
|
2009-10-20 11:54:04 -07:00
|
|
|
// Contexts created on this device
|
2011-09-10 07:18:29 -07:00
|
|
|
ALCcontext *volatile ContextList;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
|
|
|
BackendFuncs *Funcs;
|
|
|
|
void *ExtraData; // For the backend's use
|
2008-01-14 10:39:54 -08:00
|
|
|
|
2011-09-12 03:57:53 -07:00
|
|
|
ALCdevice *volatile next;
|
2007-11-13 18:02:18 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
#define ALCdevice_OpenPlayback(a,b) ((a)->Funcs->OpenPlayback((a), (b)))
|
|
|
|
#define ALCdevice_ClosePlayback(a) ((a)->Funcs->ClosePlayback((a)))
|
2009-09-16 00:24:44 -07:00
|
|
|
#define ALCdevice_ResetPlayback(a) ((a)->Funcs->ResetPlayback((a)))
|
2012-03-05 07:11:09 -08:00
|
|
|
#define ALCdevice_StartPlayback(a) ((a)->Funcs->StartPlayback((a)))
|
2009-09-16 00:24:44 -07:00
|
|
|
#define ALCdevice_StopPlayback(a) ((a)->Funcs->StopPlayback((a)))
|
2009-08-13 19:36:14 -07:00
|
|
|
#define ALCdevice_OpenCapture(a,b) ((a)->Funcs->OpenCapture((a), (b)))
|
2007-11-13 18:02:18 -08:00
|
|
|
#define ALCdevice_CloseCapture(a) ((a)->Funcs->CloseCapture((a)))
|
|
|
|
#define ALCdevice_StartCapture(a) ((a)->Funcs->StartCapture((a)))
|
|
|
|
#define ALCdevice_StopCapture(a) ((a)->Funcs->StopCapture((a)))
|
|
|
|
#define ALCdevice_CaptureSamples(a,b,c) ((a)->Funcs->CaptureSamples((a), (b), (c)))
|
|
|
|
#define ALCdevice_AvailableSamples(a) ((a)->Funcs->AvailableSamples((a)))
|
2012-08-18 15:58:04 -07:00
|
|
|
#define ALCdevice_Lock(a) ((a)->Funcs->Lock((a)))
|
|
|
|
#define ALCdevice_Unlock(a) ((a)->Funcs->Unlock((a)))
|
2012-08-17 13:38:52 -07:00
|
|
|
#define ALCdevice_GetLatency(a) ((a)->Funcs->GetLatency((a)))
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-05-03 02:29:26 -07:00
|
|
|
// Frequency was requested by the app or config file
|
2011-10-06 01:16:07 -07:00
|
|
|
#define DEVICE_FREQUENCY_REQUEST (1<<1)
|
2011-05-03 17:11:07 -07:00
|
|
|
// Channel configuration was requested by the config file
|
2011-10-06 01:16:07 -07:00
|
|
|
#define DEVICE_CHANNELS_REQUEST (1<<2)
|
2012-02-15 16:26:32 -08:00
|
|
|
// Sample type was requested by the config file
|
|
|
|
#define DEVICE_SAMPLE_TYPE_REQUEST (1<<3)
|
2011-05-01 18:18:37 -07:00
|
|
|
|
2012-08-09 05:38:07 -07:00
|
|
|
// Stereo sources cover 120-degree angles around +/-90
|
|
|
|
#define DEVICE_WIDE_STEREO (1<<16)
|
|
|
|
|
2011-06-15 01:59:07 -07:00
|
|
|
// Specifies if the device is currently running
|
|
|
|
#define DEVICE_RUNNING (1<<31)
|
2011-05-01 18:18:37 -07:00
|
|
|
|
2011-09-24 18:34:45 -07:00
|
|
|
#define LookupBuffer(m, k) ((struct ALbuffer*)LookupUIntMapKey(&(m)->BufferMap, (k)))
|
|
|
|
#define LookupEffect(m, k) ((struct ALeffect*)LookupUIntMapKey(&(m)->EffectMap, (k)))
|
|
|
|
#define LookupFilter(m, k) ((struct ALfilter*)LookupUIntMapKey(&(m)->FilterMap, (k)))
|
2011-10-06 06:39:13 -07:00
|
|
|
#define RemoveBuffer(m, k) ((struct ALbuffer*)RemoveUIntMapKey(&(m)->BufferMap, (k)))
|
|
|
|
#define RemoveEffect(m, k) ((struct ALeffect*)RemoveUIntMapKey(&(m)->EffectMap, (k)))
|
|
|
|
#define RemoveFilter(m, k) ((struct ALfilter*)RemoveUIntMapKey(&(m)->FilterMap, (k)))
|
2011-09-24 18:34:45 -07:00
|
|
|
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
struct ALCcontext_struct
|
|
|
|
{
|
2011-08-28 15:44:03 -07:00
|
|
|
volatile RefCount ref;
|
|
|
|
|
2012-10-09 04:48:12 -07:00
|
|
|
struct ALlistener *Listener;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2010-05-01 19:59:41 -07:00
|
|
|
UIntMap SourceMap;
|
2010-05-12 02:20:14 -07:00
|
|
|
UIntMap EffectSlotMap;
|
2008-01-15 16:22:39 -08:00
|
|
|
|
2011-08-31 00:19:27 -07:00
|
|
|
ALenum LastError;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-08-31 00:19:27 -07:00
|
|
|
volatile ALenum UpdateSources;
|
2009-12-09 12:14:53 -08:00
|
|
|
|
2011-08-31 00:19:27 -07:00
|
|
|
volatile enum DistanceModel DistanceModel;
|
|
|
|
volatile ALboolean SourceDistanceModel;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-08-31 00:19:27 -07:00
|
|
|
volatile ALfloat DopplerFactor;
|
|
|
|
volatile ALfloat DopplerVelocity;
|
2012-04-19 22:50:11 -07:00
|
|
|
volatile ALfloat SpeedOfSound;
|
2011-08-31 00:19:27 -07:00
|
|
|
volatile ALenum DeferUpdates;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2010-06-06 00:17:50 -07:00
|
|
|
struct ALsource **ActiveSources;
|
|
|
|
ALsizei ActiveSourceCount;
|
|
|
|
ALsizei MaxActiveSources;
|
|
|
|
|
2011-08-30 20:13:42 -07:00
|
|
|
struct ALeffectslot **ActiveEffectSlots;
|
|
|
|
ALsizei ActiveEffectSlotCount;
|
|
|
|
ALsizei MaxActiveEffectSlots;
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
ALCdevice *Device;
|
2008-07-22 12:39:10 -07:00
|
|
|
const ALCchar *ExtensionList;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2011-09-10 07:18:29 -07:00
|
|
|
ALCcontext *volatile next;
|
2007-11-13 18:02:18 -08:00
|
|
|
};
|
|
|
|
|
2011-09-24 18:34:45 -07:00
|
|
|
#define LookupSource(m, k) ((struct ALsource*)LookupUIntMapKey(&(m)->SourceMap, (k)))
|
|
|
|
#define LookupEffectSlot(m, k) ((struct ALeffectslot*)LookupUIntMapKey(&(m)->EffectSlotMap, (k)))
|
2011-10-06 06:39:13 -07:00
|
|
|
#define RemoveSource(m, k) ((struct ALsource*)RemoveUIntMapKey(&(m)->SourceMap, (k)))
|
|
|
|
#define RemoveEffectSlot(m, k) ((struct ALeffectslot*)RemoveUIntMapKey(&(m)->EffectSlotMap, (k)))
|
2011-09-24 18:34:45 -07:00
|
|
|
|
2012-09-14 03:10:12 -07:00
|
|
|
|
2011-09-11 09:28:30 -07:00
|
|
|
ALCcontext *GetContextRef(void);
|
|
|
|
|
2011-08-29 13:22:07 -07:00
|
|
|
void ALCcontext_IncRef(ALCcontext *context);
|
|
|
|
void ALCcontext_DecRef(ALCcontext *context);
|
|
|
|
|
2012-05-09 16:28:16 -07:00
|
|
|
void AppendAllDevicesList(const ALCchar *name);
|
2009-08-27 01:47:41 -07:00
|
|
|
void AppendCaptureDeviceList(const ALCchar *name);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-08-18 15:58:04 -07:00
|
|
|
void ALCdevice_LockDefault(ALCdevice *device);
|
|
|
|
void ALCdevice_UnlockDefault(ALCdevice *device);
|
2012-08-19 12:31:59 -07:00
|
|
|
ALint64 ALCdevice_GetLatencyDefault(ALCdevice *device);
|
2012-01-09 23:37:03 -08:00
|
|
|
|
|
|
|
static __inline void LockContext(ALCcontext *context)
|
2012-08-18 15:58:04 -07:00
|
|
|
{ ALCdevice_Lock(context->Device); }
|
2012-01-09 23:37:03 -08:00
|
|
|
static __inline void UnlockContext(ALCcontext *context)
|
2012-08-18 15:58:04 -07:00
|
|
|
{ ALCdevice_Unlock(context->Device); }
|
2012-01-09 23:37:03 -08:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-08-15 05:50:40 -07:00
|
|
|
void *al_malloc(size_t alignment, size_t size);
|
|
|
|
void *al_calloc(size_t alignment, size_t size);
|
|
|
|
void al_free(void *ptr);
|
|
|
|
|
2012-09-16 01:35:16 -07:00
|
|
|
typedef struct {
|
|
|
|
int state;
|
2012-09-16 03:19:53 -07:00
|
|
|
#ifdef HAVE_SSE
|
|
|
|
int sse_state;
|
|
|
|
#endif
|
2012-09-16 01:35:16 -07:00
|
|
|
} FPUCtl;
|
|
|
|
void SetMixerFPUMode(FPUCtl *ctl);
|
|
|
|
void RestoreFPUMode(const FPUCtl *ctl);
|
2012-09-14 01:10:19 -07:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
ALvoid *StartThread(ALuint (*func)(ALvoid*), ALvoid *ptr);
|
|
|
|
ALuint StopThread(ALvoid *thread);
|
|
|
|
|
2007-12-16 18:36:06 -08:00
|
|
|
typedef struct RingBuffer RingBuffer;
|
|
|
|
RingBuffer *CreateRingBuffer(ALsizei frame_size, ALsizei length);
|
|
|
|
void DestroyRingBuffer(RingBuffer *ring);
|
|
|
|
ALsizei RingBufferSize(RingBuffer *ring);
|
|
|
|
void WriteRingBuffer(RingBuffer *ring, const ALubyte *data, ALsizei len);
|
|
|
|
void ReadRingBuffer(RingBuffer *ring, ALubyte *data, ALsizei len);
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
void ReadALConfig(void);
|
|
|
|
void FreeALConfig(void);
|
2009-12-28 13:08:15 -08:00
|
|
|
int ConfigValueExists(const char *blockName, const char *keyName);
|
2007-11-13 18:02:18 -08:00
|
|
|
const char *GetConfigValue(const char *blockName, const char *keyName, const char *def);
|
2009-12-26 07:42:57 -08:00
|
|
|
int GetConfigValueBool(const char *blockName, const char *keyName, int def);
|
2011-09-19 11:29:18 -07:00
|
|
|
int ConfigValueStr(const char *blockName, const char *keyName, const char **ret);
|
2011-09-18 16:16:55 -07:00
|
|
|
int ConfigValueInt(const char *blockName, const char *keyName, int *ret);
|
|
|
|
int ConfigValueUInt(const char *blockName, const char *keyName, unsigned int *ret);
|
|
|
|
int ConfigValueFloat(const char *blockName, const char *keyName, float *ret);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2010-05-12 07:27:12 -07:00
|
|
|
void SetRTPriority(void);
|
2009-12-01 23:15:09 -08:00
|
|
|
|
2009-12-07 04:19:33 -08:00
|
|
|
void SetDefaultChannelOrder(ALCdevice *device);
|
|
|
|
void SetDefaultWFXChannelOrder(ALCdevice *device);
|
|
|
|
|
2011-05-15 04:03:15 -07:00
|
|
|
const ALCchar *DevFmtTypeString(enum DevFmtType type);
|
|
|
|
const ALCchar *DevFmtChannelsString(enum DevFmtChannels chans);
|
|
|
|
|
2012-09-11 01:59:42 -07:00
|
|
|
#define HRIR_BITS (7)
|
2011-06-03 01:06:00 -07:00
|
|
|
#define HRIR_LENGTH (1<<HRIR_BITS)
|
2011-06-17 16:22:39 -07:00
|
|
|
#define HRIR_MASK (HRIR_LENGTH-1)
|
2012-07-20 15:08:11 -07:00
|
|
|
#define HRTFDELAY_BITS (20)
|
|
|
|
#define HRTFDELAY_FRACONE (1<<HRTFDELAY_BITS)
|
|
|
|
#define HRTFDELAY_MASK (HRTFDELAY_FRACONE-1)
|
2011-09-18 09:52:40 -07:00
|
|
|
const struct Hrtf *GetHrtf(ALCdevice *device);
|
2012-09-11 01:59:42 -07:00
|
|
|
void FreeHrtfs(void);
|
|
|
|
ALuint GetHrtfIrSize (const struct Hrtf *Hrtf);
|
2011-07-16 16:24:01 -07:00
|
|
|
ALfloat CalcHrtfDelta(ALfloat oldGain, ALfloat newGain, const ALfloat olddir[3], const ALfloat newdir[3]);
|
2011-09-18 09:52:40 -07:00
|
|
|
void GetLerpedHrtfCoeffs(const struct Hrtf *Hrtf, ALfloat elevation, ALfloat azimuth, ALfloat gain, ALfloat (*coeffs)[2], ALuint *delays);
|
|
|
|
ALuint GetMovingHrtfCoeffs(const struct Hrtf *Hrtf, ALfloat elevation, ALfloat azimuth, ALfloat gain, ALfloat delta, ALint counter, ALfloat (*coeffs)[2], ALuint *delays, ALfloat (*coeffStep)[2], ALint *delayStep);
|
2011-05-01 13:21:56 -07:00
|
|
|
|
2012-10-07 04:31:31 -07:00
|
|
|
void al_print(const char *type, const char *func, const char *fmt, ...) PRINTF_STYLE(3,4);
|
|
|
|
#define AL_PRINT(T, ...) al_print((T), __FUNCTION__, __VA_ARGS__)
|
2010-01-12 09:05:57 -08:00
|
|
|
|
2011-08-20 03:59:46 -07:00
|
|
|
extern FILE *LogFile;
|
2011-07-10 21:30:25 -07:00
|
|
|
enum LogLevel {
|
|
|
|
NoLog,
|
|
|
|
LogError,
|
|
|
|
LogWarning,
|
2011-09-20 12:24:23 -07:00
|
|
|
LogTrace,
|
|
|
|
LogRef
|
2011-07-10 21:30:25 -07:00
|
|
|
};
|
|
|
|
extern enum LogLevel LogLevel;
|
|
|
|
|
2011-09-20 12:24:23 -07:00
|
|
|
#define TRACEREF(...) do { \
|
|
|
|
if(LogLevel >= LogRef) \
|
2012-10-07 04:31:31 -07:00
|
|
|
AL_PRINT("(--)", __VA_ARGS__); \
|
2011-09-20 12:24:23 -07:00
|
|
|
} while(0)
|
|
|
|
|
2011-07-10 21:30:25 -07:00
|
|
|
#define TRACE(...) do { \
|
|
|
|
if(LogLevel >= LogTrace) \
|
2012-10-07 04:31:31 -07:00
|
|
|
AL_PRINT("(II)", __VA_ARGS__); \
|
2011-07-10 21:30:25 -07:00
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define WARN(...) do { \
|
|
|
|
if(LogLevel >= LogWarning) \
|
2012-10-07 04:31:31 -07:00
|
|
|
AL_PRINT("(WW)", __VA_ARGS__); \
|
2011-07-10 21:30:25 -07:00
|
|
|
} while(0)
|
|
|
|
|
2011-07-13 01:43:00 -07:00
|
|
|
#define ERR(...) do { \
|
2011-07-10 21:30:25 -07:00
|
|
|
if(LogLevel >= LogError) \
|
2012-10-07 04:31:31 -07:00
|
|
|
AL_PRINT("(EE)", __VA_ARGS__); \
|
2011-07-10 21:30:25 -07:00
|
|
|
} while(0)
|
|
|
|
|
|
|
|
|
2011-08-20 03:59:46 -07:00
|
|
|
extern ALint RTPrioLevel;
|
|
|
|
|
2012-08-13 08:53:36 -07:00
|
|
|
|
|
|
|
extern ALuint CPUCapFlags;
|
|
|
|
enum {
|
2012-08-15 01:29:19 -07:00
|
|
|
CPU_CAP_SSE = 1<<0,
|
|
|
|
CPU_CAP_NEON = 1<<1,
|
2012-08-13 08:53:36 -07:00
|
|
|
};
|
|
|
|
|
2012-08-13 10:37:49 -07:00
|
|
|
void FillCPUCaps(ALuint capfilter);
|
2012-08-13 08:53:36 -07:00
|
|
|
|
|
|
|
|
2012-04-23 19:46:05 -07:00
|
|
|
/**
|
|
|
|
* Starts a try block. Must not be nested within another try block within the
|
|
|
|
* same function.
|
|
|
|
*/
|
|
|
|
#define al_try do { \
|
|
|
|
int _al_err=0; \
|
|
|
|
_al_try_label: \
|
|
|
|
if(_al_err == 0)
|
|
|
|
/**
|
|
|
|
* After a try or another catch block, runs the next block if the given value
|
|
|
|
* was thrown.
|
|
|
|
*/
|
|
|
|
#define al_catch(val) else if(_al_err == (val))
|
|
|
|
/**
|
|
|
|
* After a try or catch block, runs the next block for any value thrown and not
|
|
|
|
* caught.
|
|
|
|
*/
|
|
|
|
#define al_catchany() else
|
|
|
|
/** Marks the end of the final catch (or the try) block. */
|
|
|
|
#define al_endtry } while(0)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The given integer value is "thrown" so as to be caught by a catch block.
|
|
|
|
* Must be called in a try block within the same function. The value must not
|
|
|
|
* be 0.
|
|
|
|
*/
|
|
|
|
#define al_throw(e) do { \
|
|
|
|
_al_err = (e); \
|
|
|
|
assert(_al_err != 0); \
|
|
|
|
goto _al_try_label; \
|
|
|
|
} while(0)
|
|
|
|
/** Sets an AL error on the given context, before throwing the error code. */
|
|
|
|
#define al_throwerr(ctx, err) do { \
|
|
|
|
alSetError((ctx), (err)); \
|
|
|
|
al_throw((err)); \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Throws an AL_INVALID_VALUE error with the given ctx if the given condition
|
|
|
|
* is false.
|
|
|
|
*/
|
|
|
|
#define CHECK_VALUE(ctx, cond) do { \
|
|
|
|
if(!(cond)) \
|
|
|
|
al_throwerr((ctx), AL_INVALID_VALUE); \
|
|
|
|
} while(0)
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|