798 lines
22 KiB
C
798 lines
22 KiB
C
#ifndef AL_MAIN_H
|
|
#define AL_MAIN_H
|
|
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stddef.h>
|
|
#include <stdarg.h>
|
|
#include <assert.h>
|
|
#include <math.h>
|
|
#include <limits.h>
|
|
|
|
#ifdef HAVE_STRINGS_H
|
|
#include <strings.h>
|
|
#endif
|
|
|
|
#include "AL/al.h"
|
|
#include "AL/alc.h"
|
|
#include "AL/alext.h"
|
|
|
|
#include "logging.h"
|
|
#include "polymorphism.h"
|
|
#include "static_assert.h"
|
|
#include "align.h"
|
|
#include "atomic.h"
|
|
#include "uintmap.h"
|
|
#include "vector.h"
|
|
#include "alstring.h"
|
|
#include "almalloc.h"
|
|
#include "threads.h"
|
|
|
|
#ifndef ALC_SOFT_loopback2
|
|
#define ALC_SOFT_loopback2 1
|
|
#define ALC_AMBISONIC_LAYOUT_SOFT 0xfff0
|
|
#define ALC_AMBISONIC_SCALING_SOFT 0xfff1
|
|
#define ALC_AMBISONIC_ORDER_SOFT 0xfff2
|
|
#define ALC_MAX_AMBISONIC_ORDER_SOFT 0xfff3
|
|
|
|
#define ALC_BFORMAT3D_SOFT 0x1508
|
|
|
|
/* Ambisonic layouts */
|
|
#define ALC_ACN_SOFT 0xfff4
|
|
#define ALC_FUMA_SOFT 0xfff5
|
|
|
|
/* Ambisonic scalings (normalization) */
|
|
/*#define ALC_FUMA_SOFT*/
|
|
#define ALC_SN3D_SOFT 0xfff6
|
|
#define ALC_N3D_SOFT 0xfff7
|
|
#endif
|
|
|
|
#ifndef ALC_SOFT_device_clock
|
|
#define ALC_SOFT_device_clock 1
|
|
typedef int64_t ALCint64SOFT;
|
|
typedef uint64_t ALCuint64SOFT;
|
|
#define ALC_DEVICE_CLOCK_SOFT 0x1600
|
|
#define ALC_DEVICE_LATENCY_SOFT 0x1601
|
|
#define ALC_DEVICE_CLOCK_LATENCY_SOFT 0x1602
|
|
#define AL_SAMPLE_OFFSET_CLOCK_SOFT 0x1202
|
|
#define AL_SEC_OFFSET_CLOCK_SOFT 0x1203
|
|
typedef void (ALC_APIENTRY*LPALCGETINTEGER64VSOFT)(ALCdevice *device, ALCenum pname, ALsizei size, ALCint64SOFT *values);
|
|
#ifdef AL_ALEXT_PROTOTYPES
|
|
ALC_API void ALC_APIENTRY alcGetInteger64vSOFT(ALCdevice *device, ALCenum pname, ALsizei size, ALCint64SOFT *values);
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef AL_SOFT_buffer_samples2
|
|
#define AL_SOFT_buffer_samples2 1
|
|
/* Channel configurations */
|
|
#define AL_MONO_SOFT 0x1500
|
|
#define AL_STEREO_SOFT 0x1501
|
|
#define AL_REAR_SOFT 0x1502
|
|
#define AL_QUAD_SOFT 0x1503
|
|
#define AL_5POINT1_SOFT 0x1504
|
|
#define AL_6POINT1_SOFT 0x1505
|
|
#define AL_7POINT1_SOFT 0x1506
|
|
#define AL_BFORMAT2D_SOFT 0x1507
|
|
#define AL_BFORMAT3D_SOFT 0x1508
|
|
|
|
/* Sample types */
|
|
#define AL_BYTE_SOFT 0x1400
|
|
#define AL_UNSIGNED_BYTE_SOFT 0x1401
|
|
#define AL_SHORT_SOFT 0x1402
|
|
#define AL_UNSIGNED_SHORT_SOFT 0x1403
|
|
#define AL_INT_SOFT 0x1404
|
|
#define AL_UNSIGNED_INT_SOFT 0x1405
|
|
#define AL_FLOAT_SOFT 0x1406
|
|
#define AL_DOUBLE_SOFT 0x1407
|
|
#define AL_BYTE3_SOFT 0x1408
|
|
#define AL_UNSIGNED_BYTE3_SOFT 0x1409
|
|
#define AL_MULAW_SOFT 0x140A
|
|
|
|
/* Storage formats */
|
|
#define AL_MONO8_SOFT 0x1100
|
|
#define AL_MONO16_SOFT 0x1101
|
|
#define AL_MONO32F_SOFT 0x10010
|
|
#define AL_STEREO8_SOFT 0x1102
|
|
#define AL_STEREO16_SOFT 0x1103
|
|
#define AL_STEREO32F_SOFT 0x10011
|
|
#define AL_QUAD8_SOFT 0x1204
|
|
#define AL_QUAD16_SOFT 0x1205
|
|
#define AL_QUAD32F_SOFT 0x1206
|
|
#define AL_REAR8_SOFT 0x1207
|
|
#define AL_REAR16_SOFT 0x1208
|
|
#define AL_REAR32F_SOFT 0x1209
|
|
#define AL_5POINT1_8_SOFT 0x120A
|
|
#define AL_5POINT1_16_SOFT 0x120B
|
|
#define AL_5POINT1_32F_SOFT 0x120C
|
|
#define AL_6POINT1_8_SOFT 0x120D
|
|
#define AL_6POINT1_16_SOFT 0x120E
|
|
#define AL_6POINT1_32F_SOFT 0x120F
|
|
#define AL_7POINT1_8_SOFT 0x1210
|
|
#define AL_7POINT1_16_SOFT 0x1211
|
|
#define AL_7POINT1_32F_SOFT 0x1212
|
|
#define AL_BFORMAT2D_8_SOFT 0x20021
|
|
#define AL_BFORMAT2D_16_SOFT 0x20022
|
|
#define AL_BFORMAT2D_32F_SOFT 0x20023
|
|
#define AL_BFORMAT3D_8_SOFT 0x20031
|
|
#define AL_BFORMAT3D_16_SOFT 0x20032
|
|
#define AL_BFORMAT3D_32F_SOFT 0x20033
|
|
|
|
/* Buffer attributes */
|
|
#define AL_INTERNAL_FORMAT_SOFT 0x2008
|
|
#define AL_BYTE_LENGTH_SOFT 0x2009
|
|
#define AL_SAMPLE_LENGTH_SOFT 0x200A
|
|
#define AL_SEC_LENGTH_SOFT 0x200B
|
|
|
|
#if 0
|
|
typedef void (AL_APIENTRY*LPALBUFFERSAMPLESSOFT)(ALuint,ALuint,ALenum,ALsizei,ALenum,ALenum,const ALvoid*);
|
|
typedef void (AL_APIENTRY*LPALGETBUFFERSAMPLESSOFT)(ALuint,ALsizei,ALsizei,ALenum,ALenum,ALvoid*);
|
|
typedef ALboolean (AL_APIENTRY*LPALISBUFFERFORMATSUPPORTEDSOFT)(ALenum);
|
|
#ifdef AL_ALEXT_PROTOTYPES
|
|
AL_API void AL_APIENTRY alBufferSamplesSOFT(ALuint buffer, ALuint samplerate, ALenum internalformat, ALsizei samples, ALenum channels, ALenum type, const ALvoid *data);
|
|
AL_API void AL_APIENTRY alGetBufferSamplesSOFT(ALuint buffer, ALsizei offset, ALsizei samples, ALenum channels, ALenum type, ALvoid *data);
|
|
AL_API ALboolean AL_APIENTRY alIsBufferFormatSupportedSOFT(ALenum format);
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
|
|
#if defined(_WIN64)
|
|
#define SZFMT "%I64u"
|
|
#elif defined(_WIN32)
|
|
#define SZFMT "%u"
|
|
#else
|
|
#define SZFMT "%zu"
|
|
#endif
|
|
|
|
|
|
#ifdef __GNUC__
|
|
/* Because of a long-standing deficiency in C, you're not allowed to implicitly
|
|
* cast a pointer-to-type-array to a pointer-to-const-type-array. For example,
|
|
*
|
|
* int (*ptr)[10];
|
|
* const int (*cptr)[10] = ptr;
|
|
*
|
|
* is not allowed and most compilers will generate noisy warnings about
|
|
* incompatible types, even though it just makes the array elements const.
|
|
* Clang will allow it if you make the array type a typedef, like this:
|
|
*
|
|
* typedef int int10[10];
|
|
* int10 *ptr;
|
|
* const int10 *cptr = ptr;
|
|
*
|
|
* however GCC does not and still issues the incompatible type warning. The
|
|
* "proper" way to fix it is to add an explicit cast for the constified type,
|
|
* but that removes the vast majority of otherwise useful type-checking you'd
|
|
* get, and runs the risk of improper casts if types are later changed. Leaving
|
|
* it non-const can also be an issue if you use it as a function parameter, and
|
|
* happen to have a const type as input (and also reduce the capabilities of
|
|
* the compiler to better optimize the function).
|
|
*
|
|
* So to work around the problem, we use a macro. The macro first assigns the
|
|
* incoming variable to the specified non-const type to ensure it's the correct
|
|
* type, then casts the variable as the desired constified type. Very ugly, but
|
|
* I'd rather not have hundreds of lines of warnings because I want to tell the
|
|
* compiler that some array(s) can't be changed by the code, or have lots of
|
|
* error-prone casts.
|
|
*/
|
|
#define SAFE_CONST(T, var) __extension__({ \
|
|
T _tmp = (var); \
|
|
(const T)_tmp; \
|
|
})
|
|
#else
|
|
/* Non-GNU-compatible compilers have to use a straight cast with no extra
|
|
* checks, due to the lack of multi-statement expressions.
|
|
*/
|
|
#define SAFE_CONST(T, var) ((const T)(var))
|
|
#endif
|
|
|
|
|
|
#ifdef __GNUC__
|
|
#define LIKELY(x) __builtin_expect(!!(x), !0)
|
|
#define UNLIKELY(x) __builtin_expect(!!(x), 0)
|
|
#else
|
|
#define LIKELY(x) (!!(x))
|
|
#define UNLIKELY(x) (!!(x))
|
|
#endif
|
|
|
|
typedef ALint64SOFT ALint64;
|
|
typedef ALuint64SOFT ALuint64;
|
|
|
|
#ifndef U64
|
|
#if defined(_MSC_VER)
|
|
#define U64(x) ((ALuint64)(x##ui64))
|
|
#elif SIZEOF_LONG == 8
|
|
#define U64(x) ((ALuint64)(x##ul))
|
|
#elif SIZEOF_LONG_LONG == 8
|
|
#define U64(x) ((ALuint64)(x##ull))
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef UINT64_MAX
|
|
#define UINT64_MAX U64(18446744073709551615)
|
|
#endif
|
|
|
|
#ifndef UNUSED
|
|
#if defined(__cplusplus)
|
|
#define UNUSED(x)
|
|
#elif defined(__GNUC__)
|
|
#define UNUSED(x) UNUSED_##x __attribute__((unused))
|
|
#elif defined(__LCLINT__)
|
|
#define UNUSED(x) /*@unused@*/ x
|
|
#else
|
|
#define UNUSED(x) x
|
|
#endif
|
|
#endif
|
|
|
|
/* Calculates the size of a struct with N elements of a flexible array member.
|
|
* GCC and Clang allow offsetof(Type, fam[N]) for this, but MSVC seems to have
|
|
* trouble, so a bit more verbose workaround is needed.
|
|
*/
|
|
#define FAM_SIZE(T, M, N) (offsetof(T, M) + sizeof(((T*)NULL)->M[0])*(N))
|
|
|
|
#if defined(__GNUC__) && defined(__i386__)
|
|
/* force_align_arg_pointer is required for proper function arguments aligning
|
|
* when SSE code is used. Some systems (Windows, QNX) do not guarantee our
|
|
* thread functions will be properly aligned on the stack, even though GCC may
|
|
* generate code with the assumption that it is. */
|
|
#define FORCE_ALIGN __attribute__((force_align_arg_pointer))
|
|
#else
|
|
#define FORCE_ALIGN
|
|
#endif
|
|
|
|
#ifdef HAVE_C99_VLA
|
|
#define DECL_VLA(T, _name, _size) T _name[(_size)]
|
|
#else
|
|
#define DECL_VLA(T, _name, _size) T *_name = alloca((_size) * sizeof(T))
|
|
#endif
|
|
|
|
#ifndef PATH_MAX
|
|
#ifdef MAX_PATH
|
|
#define PATH_MAX MAX_PATH
|
|
#else
|
|
#define PATH_MAX 4096
|
|
#endif
|
|
#endif
|
|
|
|
|
|
static const union {
|
|
ALuint u;
|
|
ALubyte b[sizeof(ALuint)];
|
|
} EndianTest = { 1 };
|
|
#define IS_LITTLE_ENDIAN (EndianTest.b[0] == 1)
|
|
|
|
#define COUNTOF(x) (sizeof(x) / sizeof(0[x]))
|
|
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
struct Hrtf;
|
|
struct HrtfEntry;
|
|
struct DirectHrtfState;
|
|
struct FrontStablizer;
|
|
struct Compressor;
|
|
struct ALCbackend;
|
|
struct ALcontextProps;
|
|
struct ALlistenerProps;
|
|
struct ALvoiceProps;
|
|
struct ALeffectslotProps;
|
|
|
|
|
|
#define DEFAULT_OUTPUT_RATE (44100)
|
|
#define MIN_OUTPUT_RATE (8000)
|
|
|
|
|
|
/* Find the next power-of-2 for non-power-of-2 numbers. */
|
|
inline ALuint NextPowerOf2(ALuint value)
|
|
{
|
|
if(value > 0)
|
|
{
|
|
value--;
|
|
value |= value>>1;
|
|
value |= value>>2;
|
|
value |= value>>4;
|
|
value |= value>>8;
|
|
value |= value>>16;
|
|
}
|
|
return value+1;
|
|
}
|
|
|
|
/** Round up a value to the next multiple. */
|
|
inline size_t RoundUp(size_t value, size_t r)
|
|
{
|
|
value += r-1;
|
|
return value - (value%r);
|
|
}
|
|
|
|
/* Fast float-to-int conversion. Assumes the FPU is already in round-to-zero
|
|
* mode. */
|
|
inline ALint fastf2i(ALfloat f)
|
|
{
|
|
#ifdef HAVE_LRINTF
|
|
return lrintf(f);
|
|
#elif defined(_MSC_VER) && defined(_M_IX86)
|
|
ALint i;
|
|
__asm fld f
|
|
__asm fistp i
|
|
return i;
|
|
#else
|
|
return (ALint)f;
|
|
#endif
|
|
}
|
|
|
|
|
|
enum DevProbe {
|
|
ALL_DEVICE_PROBE,
|
|
CAPTURE_DEVICE_PROBE
|
|
};
|
|
|
|
|
|
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 Channel {
|
|
FrontLeft = 0,
|
|
FrontRight,
|
|
FrontCenter,
|
|
LFE,
|
|
BackLeft,
|
|
BackRight,
|
|
BackCenter,
|
|
SideLeft,
|
|
SideRight,
|
|
|
|
UpperFrontLeft,
|
|
UpperFrontRight,
|
|
UpperBackLeft,
|
|
UpperBackRight,
|
|
LowerFrontLeft,
|
|
LowerFrontRight,
|
|
LowerBackLeft,
|
|
LowerBackRight,
|
|
|
|
Aux0,
|
|
Aux1,
|
|
Aux2,
|
|
Aux3,
|
|
Aux4,
|
|
Aux5,
|
|
Aux6,
|
|
Aux7,
|
|
Aux8,
|
|
Aux9,
|
|
Aux10,
|
|
Aux11,
|
|
Aux12,
|
|
Aux13,
|
|
Aux14,
|
|
Aux15,
|
|
|
|
InvalidChannel
|
|
};
|
|
|
|
|
|
/* Device formats */
|
|
enum DevFmtType {
|
|
DevFmtByte = ALC_BYTE_SOFT,
|
|
DevFmtUByte = ALC_UNSIGNED_BYTE_SOFT,
|
|
DevFmtShort = ALC_SHORT_SOFT,
|
|
DevFmtUShort = ALC_UNSIGNED_SHORT_SOFT,
|
|
DevFmtInt = ALC_INT_SOFT,
|
|
DevFmtUInt = ALC_UNSIGNED_INT_SOFT,
|
|
DevFmtFloat = ALC_FLOAT_SOFT,
|
|
|
|
DevFmtTypeDefault = DevFmtFloat
|
|
};
|
|
enum DevFmtChannels {
|
|
DevFmtMono = ALC_MONO_SOFT,
|
|
DevFmtStereo = ALC_STEREO_SOFT,
|
|
DevFmtQuad = ALC_QUAD_SOFT,
|
|
DevFmtX51 = ALC_5POINT1_SOFT,
|
|
DevFmtX61 = ALC_6POINT1_SOFT,
|
|
DevFmtX71 = ALC_7POINT1_SOFT,
|
|
DevFmtAmbi3D = ALC_BFORMAT3D_SOFT,
|
|
|
|
/* Similar to 5.1, except using rear channels instead of sides */
|
|
DevFmtX51Rear = 0x80000000,
|
|
|
|
DevFmtChannelsDefault = DevFmtStereo
|
|
};
|
|
#define MAX_OUTPUT_CHANNELS (16)
|
|
|
|
ALsizei BytesFromDevFmt(enum DevFmtType type);
|
|
ALsizei ChannelsFromDevFmt(enum DevFmtChannels chans, ALsizei ambiorder);
|
|
inline ALsizei FrameSizeFromDevFmt(enum DevFmtChannels chans, enum DevFmtType type, ALsizei ambiorder)
|
|
{
|
|
return ChannelsFromDevFmt(chans, ambiorder) * BytesFromDevFmt(type);
|
|
}
|
|
|
|
enum AmbiLayout {
|
|
AmbiLayout_FuMa = ALC_FUMA_SOFT, /* FuMa channel order */
|
|
AmbiLayout_ACN = ALC_ACN_SOFT, /* ACN channel order */
|
|
|
|
AmbiLayout_Default = AmbiLayout_ACN
|
|
};
|
|
|
|
enum AmbiNorm {
|
|
AmbiNorm_FuMa = ALC_FUMA_SOFT, /* FuMa normalization */
|
|
AmbiNorm_SN3D = ALC_SN3D_SOFT, /* SN3D normalization */
|
|
AmbiNorm_N3D = ALC_N3D_SOFT, /* N3D normalization */
|
|
|
|
AmbiNorm_Default = AmbiNorm_SN3D
|
|
};
|
|
|
|
|
|
enum DeviceType {
|
|
Playback,
|
|
Capture,
|
|
Loopback
|
|
};
|
|
|
|
|
|
enum RenderMode {
|
|
NormalRender,
|
|
StereoPair,
|
|
HrtfRender
|
|
};
|
|
|
|
|
|
/* The maximum number of Ambisonics coefficients. For a given order (o), the
|
|
* size needed will be (o+1)**2, thus zero-order has 1, first-order has 4,
|
|
* second-order has 9, third-order has 16, and fourth-order has 25.
|
|
*/
|
|
#define MAX_AMBI_ORDER 3
|
|
#define MAX_AMBI_COEFFS ((MAX_AMBI_ORDER+1) * (MAX_AMBI_ORDER+1))
|
|
|
|
/* A bitmask of ambisonic channels with height information. If none of these
|
|
* channels are used/needed, there's no height (e.g. with most surround sound
|
|
* speaker setups). This only specifies up to 4th order, which is the highest
|
|
* order a 32-bit mask value can specify (a 64-bit mask could handle up to 7th
|
|
* order). This is ACN ordering, with bit 0 being ACN 0, etc.
|
|
*/
|
|
#define AMBI_PERIPHONIC_MASK (0xfe7ce4)
|
|
|
|
/* The maximum number of Ambisonic coefficients for 2D (non-periphonic)
|
|
* representation. This is 2 per each order above zero-order, plus 1 for zero-
|
|
* order. Or simply, o*2 + 1.
|
|
*/
|
|
#define MAX_AMBI2D_COEFFS (MAX_AMBI_ORDER*2 + 1)
|
|
|
|
|
|
typedef ALfloat ChannelConfig[MAX_AMBI_COEFFS];
|
|
typedef struct BFChannelConfig {
|
|
ALfloat Scale;
|
|
ALsizei Index;
|
|
} BFChannelConfig;
|
|
|
|
typedef union AmbiConfig {
|
|
/* Ambisonic coefficients for mixing to the dry buffer. */
|
|
ChannelConfig Coeffs[MAX_OUTPUT_CHANNELS];
|
|
/* Coefficient channel mapping for mixing to the dry buffer. */
|
|
BFChannelConfig Map[MAX_OUTPUT_CHANNELS];
|
|
} AmbiConfig;
|
|
|
|
|
|
typedef struct EnumeratedHrtf {
|
|
al_string name;
|
|
|
|
struct HrtfEntry *hrtf;
|
|
} EnumeratedHrtf;
|
|
TYPEDEF_VECTOR(EnumeratedHrtf, vector_EnumeratedHrtf)
|
|
|
|
|
|
/* Maximum delay in samples for speaker distance compensation. */
|
|
#define MAX_DELAY_LENGTH 1024
|
|
|
|
typedef struct DistanceComp {
|
|
ALfloat Gain;
|
|
ALsizei Length; /* Valid range is [0...MAX_DELAY_LENGTH). */
|
|
ALfloat *Buffer;
|
|
} DistanceComp;
|
|
|
|
/* Size for temporary storage of buffer data, in ALfloats. Larger values need
|
|
* more memory, while smaller values may need more iterations. The value needs
|
|
* 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.
|
|
*/
|
|
#define BUFFERSIZE 2048
|
|
|
|
typedef struct DryMixParams {
|
|
AmbiConfig Ambi;
|
|
/* Number of coefficients in each Ambi.Coeffs to mix together (4 for first-
|
|
* order, 9 for second-order, etc). If the count is 0, Ambi.Map is used
|
|
* instead to map each output to a coefficient index.
|
|
*/
|
|
ALsizei CoeffCount;
|
|
|
|
ALfloat (*Buffer)[BUFFERSIZE];
|
|
ALsizei NumChannels;
|
|
ALsizei NumChannelsPerOrder[MAX_AMBI_ORDER+1];
|
|
} DryMixParams;
|
|
|
|
typedef struct BFMixParams {
|
|
AmbiConfig Ambi;
|
|
/* Will only be 4 or 0. */
|
|
ALsizei CoeffCount;
|
|
|
|
ALfloat (*Buffer)[BUFFERSIZE];
|
|
ALsizei NumChannels;
|
|
} BFMixParams;
|
|
|
|
typedef struct RealMixParams {
|
|
enum Channel ChannelName[MAX_OUTPUT_CHANNELS];
|
|
|
|
ALfloat (*Buffer)[BUFFERSIZE];
|
|
ALsizei NumChannels;
|
|
} RealMixParams;
|
|
|
|
struct ALCdevice_struct
|
|
{
|
|
RefCount ref;
|
|
|
|
ALCboolean Connected;
|
|
enum DeviceType Type;
|
|
|
|
ALuint Frequency;
|
|
ALuint UpdateSize;
|
|
ALuint NumUpdates;
|
|
enum DevFmtChannels FmtChans;
|
|
enum DevFmtType FmtType;
|
|
ALboolean IsHeadphones;
|
|
ALsizei AmbiOrder;
|
|
/* For DevFmtAmbi* output only, specifies the channel order and
|
|
* normalization.
|
|
*/
|
|
enum AmbiLayout AmbiLayout;
|
|
enum AmbiNorm AmbiScale;
|
|
|
|
al_string DeviceName;
|
|
|
|
ATOMIC(ALCenum) LastError;
|
|
|
|
// Maximum number of sources that can be created
|
|
ALuint SourcesMax;
|
|
// Maximum number of slots that can be created
|
|
ALuint AuxiliaryEffectSlotMax;
|
|
|
|
ALCuint NumMonoSources;
|
|
ALCuint NumStereoSources;
|
|
ALsizei NumAuxSends;
|
|
|
|
// Map of Buffers for this device
|
|
UIntMap BufferMap;
|
|
|
|
// Map of Effects for this device
|
|
UIntMap EffectMap;
|
|
|
|
// Map of Filters for this device
|
|
UIntMap FilterMap;
|
|
|
|
/* HRTF state and info */
|
|
struct DirectHrtfState *Hrtf;
|
|
al_string HrtfName;
|
|
struct Hrtf *HrtfHandle;
|
|
vector_EnumeratedHrtf HrtfList;
|
|
ALCenum HrtfStatus;
|
|
|
|
/* UHJ encoder state */
|
|
struct Uhj2Encoder *Uhj_Encoder;
|
|
|
|
/* High quality Ambisonic decoder */
|
|
struct BFormatDec *AmbiDecoder;
|
|
|
|
/* Stereo-to-binaural filter */
|
|
struct bs2b *Bs2b;
|
|
|
|
/* First-order ambisonic upsampler for higher-order output */
|
|
struct AmbiUpsampler *AmbiUp;
|
|
|
|
/* Rendering mode. */
|
|
enum RenderMode Render_Mode;
|
|
|
|
// Device flags
|
|
ALuint Flags;
|
|
|
|
ALuint64 ClockBase;
|
|
ALuint SamplesDone;
|
|
|
|
/* Temp storage used for mixer processing. */
|
|
alignas(16) ALfloat TempBuffer[4][BUFFERSIZE];
|
|
|
|
/* The "dry" path corresponds to the main output. */
|
|
DryMixParams Dry;
|
|
|
|
/* First-order ambisonics output, to be upsampled to the dry buffer if different. */
|
|
BFMixParams FOAOut;
|
|
|
|
/* "Real" output, which will be written to the device buffer. May alias the
|
|
* dry buffer.
|
|
*/
|
|
RealMixParams RealOut;
|
|
|
|
struct FrontStablizer *Stablizer;
|
|
|
|
struct Compressor *Limiter;
|
|
|
|
/* The average speaker distance as determined by the ambdec configuration
|
|
* (or alternatively, by the NFC-HOA reference delay). Only used for NFC.
|
|
*/
|
|
ALfloat AvgSpeakerDist;
|
|
|
|
/* Delay buffers used to compensate for speaker distances. */
|
|
DistanceComp ChannelDelay[MAX_OUTPUT_CHANNELS];
|
|
|
|
/* Dithering control. */
|
|
ALfloat DitherDepth;
|
|
ALuint DitherSeed;
|
|
|
|
/* Running count of the mixer invocations, in 31.1 fixed point. This
|
|
* actually increments *twice* when mixing, first at the start and then at
|
|
* the end, so the bottom bit indicates if the device is currently mixing
|
|
* and the upper bits indicates how many mixes have been done.
|
|
*/
|
|
RefCount MixCount;
|
|
|
|
// Contexts created on this device
|
|
ATOMIC(ALCcontext*) ContextList;
|
|
|
|
almtx_t BackendLock;
|
|
struct ALCbackend *Backend;
|
|
|
|
ALCdevice *volatile next;
|
|
};
|
|
|
|
// Frequency was requested by the app or config file
|
|
#define DEVICE_FREQUENCY_REQUEST (1u<<1)
|
|
// Channel configuration was requested by the config file
|
|
#define DEVICE_CHANNELS_REQUEST (1u<<2)
|
|
// Sample type was requested by the config file
|
|
#define DEVICE_SAMPLE_TYPE_REQUEST (1u<<3)
|
|
|
|
// Specifies if the DSP is paused at user request
|
|
#define DEVICE_PAUSED (1u<<30)
|
|
|
|
// Specifies if the device is currently running
|
|
#define DEVICE_RUNNING (1u<<31)
|
|
|
|
|
|
/* Nanosecond resolution for the device clock time. */
|
|
#define DEVICE_CLOCK_RES U64(1000000000)
|
|
|
|
|
|
/* Must be less than 15 characters (16 including terminating null) for
|
|
* compatibility with pthread_setname_np limitations. */
|
|
#define MIXER_THREAD_NAME "alsoft-mixer"
|
|
|
|
#define RECORD_THREAD_NAME "alsoft-record"
|
|
|
|
|
|
struct ALCcontext_struct {
|
|
RefCount ref;
|
|
|
|
struct ALlistener *Listener;
|
|
|
|
UIntMap SourceMap;
|
|
UIntMap EffectSlotMap;
|
|
|
|
ATOMIC(ALenum) LastError;
|
|
|
|
enum DistanceModel DistanceModel;
|
|
ALboolean SourceDistanceModel;
|
|
|
|
ALfloat DopplerFactor;
|
|
ALfloat DopplerVelocity;
|
|
ALfloat SpeedOfSound;
|
|
ALfloat MetersPerUnit;
|
|
|
|
ATOMIC_FLAG PropsClean;
|
|
ATOMIC(ALenum) DeferUpdates;
|
|
|
|
RWLock PropLock;
|
|
|
|
/* Counter for the pre-mixing updates, in 31.1 fixed point (lowest bit
|
|
* indicates if updates are currently happening).
|
|
*/
|
|
RefCount UpdateCount;
|
|
ATOMIC(ALenum) HoldUpdates;
|
|
|
|
ALfloat GainBoost;
|
|
|
|
ATOMIC(struct ALcontextProps*) Update;
|
|
|
|
/* Linked lists of unused property containers, free to use for future
|
|
* updates.
|
|
*/
|
|
ATOMIC(struct ALcontextProps*) FreeContextProps;
|
|
ATOMIC(struct ALlistenerProps*) FreeListenerProps;
|
|
ATOMIC(struct ALvoiceProps*) FreeVoiceProps;
|
|
ATOMIC(struct ALeffectslotProps*) FreeEffectslotProps;
|
|
|
|
struct ALvoice **Voices;
|
|
ALsizei VoiceCount;
|
|
ALsizei MaxVoices;
|
|
|
|
ATOMIC(struct ALeffectslotArray*) ActiveAuxSlots;
|
|
|
|
/* Default effect slot */
|
|
struct ALeffectslot *DefaultSlot;
|
|
|
|
ALCdevice *Device;
|
|
const ALCchar *ExtensionList;
|
|
|
|
ALCcontext *volatile next;
|
|
|
|
/* Memory space used by the listener (and possibly default effect slot) */
|
|
alignas(16) ALCbyte _listener_mem[];
|
|
};
|
|
|
|
ALCcontext *GetContextRef(void);
|
|
|
|
void ALCcontext_IncRef(ALCcontext *context);
|
|
void ALCcontext_DecRef(ALCcontext *context);
|
|
|
|
void AllocateVoices(ALCcontext *context, ALsizei num_voices, ALsizei old_sends);
|
|
|
|
void AppendAllDevicesList(const ALCchar *name);
|
|
void AppendCaptureDeviceList(const ALCchar *name);
|
|
|
|
void ALCdevice_Lock(ALCdevice *device);
|
|
void ALCdevice_Unlock(ALCdevice *device);
|
|
|
|
void ALCcontext_DeferUpdates(ALCcontext *context);
|
|
void ALCcontext_ProcessUpdates(ALCcontext *context);
|
|
|
|
|
|
extern ALint RTPrioLevel;
|
|
void SetRTPriority(void);
|
|
|
|
void SetDefaultChannelOrder(ALCdevice *device);
|
|
void SetDefaultWFXChannelOrder(ALCdevice *device);
|
|
|
|
const ALCchar *DevFmtTypeString(enum DevFmtType type);
|
|
const ALCchar *DevFmtChannelsString(enum DevFmtChannels chans);
|
|
|
|
inline ALint GetChannelIndex(const enum Channel names[MAX_OUTPUT_CHANNELS], enum Channel chan)
|
|
{
|
|
ALint i;
|
|
for(i = 0;i < MAX_OUTPUT_CHANNELS;i++)
|
|
{
|
|
if(names[i] == chan)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
/**
|
|
* GetChannelIdxByName
|
|
*
|
|
* Returns the index for the given channel name (e.g. FrontCenter), or -1 if it
|
|
* doesn't exist.
|
|
*/
|
|
inline ALint GetChannelIdxByName(const RealMixParams *real, enum Channel chan)
|
|
{ return GetChannelIndex(real->ChannelName, chan); }
|
|
|
|
|
|
vector_al_string SearchDataFiles(const char *match, const char *subdir);
|
|
|
|
/* Small hack to use a pointer-to-array types as a normal argument type.
|
|
* Shouldn't be used directly.
|
|
*/
|
|
typedef ALfloat ALfloatBUFFERSIZE[BUFFERSIZE];
|
|
typedef ALfloat ALfloat2[2];
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif
|