2007-12-17 17:43:19 -08:00
|
|
|
#ifndef _AL_AUXEFFECTSLOT_H_
|
|
|
|
#define _AL_AUXEFFECTSLOT_H_
|
|
|
|
|
2012-09-14 02:42:36 -07:00
|
|
|
#include "alMain.h"
|
2008-07-25 19:31:12 -07:00
|
|
|
#include "alEffect.h"
|
2007-12-17 17:43:19 -08:00
|
|
|
|
2017-03-23 19:16:32 -07:00
|
|
|
#include "atomic.h"
|
2014-04-19 02:11:04 -07:00
|
|
|
#include "align.h"
|
|
|
|
|
2007-12-17 17:43:19 -08:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2013-10-07 05:41:41 -07:00
|
|
|
struct ALeffectStateVtable;
|
|
|
|
struct ALeffectslot;
|
2013-05-21 12:47:18 -07:00
|
|
|
|
2013-10-07 05:41:41 -07:00
|
|
|
typedef struct ALeffectState {
|
2016-08-25 03:42:43 -07:00
|
|
|
RefCount Ref;
|
2013-10-07 05:41:41 -07:00
|
|
|
const struct ALeffectStateVtable *vtbl;
|
2016-03-17 10:10:26 -07:00
|
|
|
|
|
|
|
ALfloat (*OutBuffer)[BUFFERSIZE];
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei OutChannels;
|
2013-10-07 05:41:41 -07:00
|
|
|
} ALeffectState;
|
2013-05-21 04:18:02 -07:00
|
|
|
|
2016-08-25 03:42:43 -07:00
|
|
|
void ALeffectState_Construct(ALeffectState *state);
|
2016-05-12 18:26:33 -07:00
|
|
|
void ALeffectState_Destruct(ALeffectState *state);
|
|
|
|
|
2013-05-21 04:18:02 -07:00
|
|
|
struct ALeffectStateVtable {
|
2013-10-29 20:08:03 -07:00
|
|
|
void (*const Destruct)(ALeffectState *state);
|
2013-05-29 11:17:45 -07:00
|
|
|
|
|
|
|
ALboolean (*const deviceUpdate)(ALeffectState *state, ALCdevice *device);
|
2016-05-13 18:28:01 -07:00
|
|
|
void (*const update)(ALeffectState *state, const ALCdevice *device, const struct ALeffectslot *slot, const union ALeffectProps *props);
|
2017-05-02 04:54:59 -07:00
|
|
|
void (*const process)(ALeffectState *state, ALsizei samplesToDo, const ALfloat (*restrict samplesIn)[BUFFERSIZE], ALfloat (*restrict samplesOut)[BUFFERSIZE], ALsizei numChannels);
|
2013-05-25 21:04:00 -07:00
|
|
|
|
2014-03-21 23:56:18 -07:00
|
|
|
void (*const Delete)(void *ptr);
|
2013-05-21 04:18:02 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
#define DEFINE_ALEFFECTSTATE_VTABLE(T) \
|
2013-10-29 20:08:03 -07:00
|
|
|
DECLARE_THUNK(T, ALeffectState, void, Destruct) \
|
|
|
|
DECLARE_THUNK1(T, ALeffectState, ALboolean, deviceUpdate, ALCdevice*) \
|
2016-05-13 18:28:01 -07:00
|
|
|
DECLARE_THUNK3(T, ALeffectState, void, update, const ALCdevice*, const ALeffectslot*, const ALeffectProps*) \
|
2017-05-02 04:54:59 -07:00
|
|
|
DECLARE_THUNK4(T, ALeffectState, void, process, ALsizei, const ALfloatBUFFERSIZE*restrict, ALfloatBUFFERSIZE*restrict, ALsizei) \
|
2014-03-21 23:56:18 -07:00
|
|
|
static void T##_ALeffectState_Delete(void *ptr) \
|
2014-04-19 02:38:32 -07:00
|
|
|
{ return T##_Delete(STATIC_UPCAST(T, ALeffectState, (ALeffectState*)ptr)); } \
|
2013-05-21 05:02:25 -07:00
|
|
|
\
|
2013-05-21 04:18:02 -07:00
|
|
|
static const struct ALeffectStateVtable T##_ALeffectState_vtable = { \
|
2013-05-21 13:02:56 -07:00
|
|
|
T##_ALeffectState_Destruct, \
|
2013-05-29 11:17:45 -07:00
|
|
|
\
|
|
|
|
T##_ALeffectState_deviceUpdate, \
|
|
|
|
T##_ALeffectState_update, \
|
|
|
|
T##_ALeffectState_process, \
|
|
|
|
\
|
2013-05-25 21:04:00 -07:00
|
|
|
T##_ALeffectState_Delete, \
|
2013-05-21 04:18:02 -07:00
|
|
|
}
|
|
|
|
|
2013-05-21 12:47:18 -07:00
|
|
|
|
2013-10-07 05:41:41 -07:00
|
|
|
struct ALeffectStateFactoryVtable;
|
2013-05-21 12:47:18 -07:00
|
|
|
|
2013-10-07 05:41:41 -07:00
|
|
|
typedef struct ALeffectStateFactory {
|
2013-05-21 12:47:18 -07:00
|
|
|
const struct ALeffectStateFactoryVtable *vtbl;
|
2013-10-07 05:41:41 -07:00
|
|
|
} ALeffectStateFactory;
|
|
|
|
|
|
|
|
struct ALeffectStateFactoryVtable {
|
|
|
|
ALeffectState *(*const create)(ALeffectStateFactory *factory);
|
2013-05-21 12:47:18 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
#define DEFINE_ALEFFECTSTATEFACTORY_VTABLE(T) \
|
2013-10-29 20:08:03 -07:00
|
|
|
DECLARE_THUNK(T, ALeffectStateFactory, ALeffectState*, create) \
|
2013-05-21 12:47:18 -07:00
|
|
|
\
|
|
|
|
static const struct ALeffectStateFactoryVtable T##_ALeffectStateFactory_vtable = { \
|
|
|
|
T##_ALeffectStateFactory_create, \
|
|
|
|
}
|
2012-09-14 02:42:36 -07:00
|
|
|
|
2009-05-29 13:30:50 -07:00
|
|
|
|
2016-01-28 00:02:46 -08:00
|
|
|
#define MAX_EFFECT_CHANNELS (4)
|
|
|
|
|
|
|
|
|
2017-03-27 23:16:23 -07:00
|
|
|
struct ALeffectslotArray {
|
|
|
|
ALsizei count;
|
|
|
|
struct ALeffectslot *slot[];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-05-12 18:26:33 -07:00
|
|
|
struct ALeffectslotProps {
|
2017-03-08 03:38:28 -08:00
|
|
|
ALfloat Gain;
|
|
|
|
ALboolean AuxSendAuto;
|
2016-05-12 18:26:33 -07:00
|
|
|
|
2017-03-08 03:38:28 -08:00
|
|
|
ALenum Type;
|
2016-05-12 18:26:33 -07:00
|
|
|
ALeffectProps Props;
|
|
|
|
|
2017-03-08 03:38:28 -08:00
|
|
|
ALeffectState *State;
|
2016-05-12 18:26:33 -07:00
|
|
|
|
|
|
|
ATOMIC(struct ALeffectslotProps*) next;
|
|
|
|
};
|
|
|
|
|
2007-12-18 14:22:59 -08:00
|
|
|
|
2016-05-12 18:26:33 -07:00
|
|
|
typedef struct ALeffectslot {
|
2016-06-03 05:46:29 -07:00
|
|
|
ALfloat Gain;
|
|
|
|
ALboolean AuxSendAuto;
|
2007-12-18 15:47:24 -08:00
|
|
|
|
2016-05-12 18:26:33 -07:00
|
|
|
struct {
|
|
|
|
ALenum Type;
|
|
|
|
ALeffectProps Props;
|
|
|
|
|
|
|
|
ALeffectState *State;
|
|
|
|
} Effect;
|
Implement AL_EFFECT_REVERB
Here is a quick description of how the reverb effect works:
+--->---+*(4)
| V new sample
+-----+---+---+ |
|extra|ltr|ref| <- +*(1)
+-----+---+---+
(3,5)*| |*(2)
+-->|
V
out sample
1) Apply master reverb gain to incoming sample and place it at the head of the
buffer. The master reverb gainhf was already applied when the source was
initially mixed.
2) Copy the delayed reflection sample to an output sample and apply the
reflection gain.
3) Apply the late reverb gain to the late reverb sample
4) Copy the end of the buffer, applying a decay gain and the decay hf ratio,
and add to the late reverb.
5) Copy the late reverb sample, adding to the output sample.
Then the head and sampling points are shifted forward, and done again for each
new sample. The extra buffer length is determined by the Reverb Density
property. A value of 0 gives a length of 0.1 seconds (long, with fairly
distinct echos) , and 1 gives 0.075 seconds (short, indistinct echos).
The decay gain is calculated such that after a number of loops to satisfy the
Decay Time, a sample will be 1/32768th as powerful (virtually insignificant to
the resulting output, and only getting further reduced). It is calculated as:
DecayGain = pow(1.0f/32768.0f, 1.0/(DecayTime/ExtraLength));
Things to note: Reverb Diffusion is not currently handled, nor is Decay HF
Limit. Decay HF Ratios above 1 probably give incorrect results. Also, this
method likely sucks, but it's the best I can come up with before release. :)
2008-01-18 21:25:40 -08:00
|
|
|
|
2017-03-23 19:16:32 -07:00
|
|
|
ATOMIC_FLAG PropsClean;
|
|
|
|
|
2011-08-29 23:05:47 -07:00
|
|
|
RefCount ref;
|
2008-01-16 13:20:09 -08:00
|
|
|
|
2016-05-12 18:26:33 -07:00
|
|
|
ATOMIC(struct ALeffectslotProps*) Update;
|
|
|
|
ATOMIC(struct ALeffectslotProps*) FreeList;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
ALfloat Gain;
|
|
|
|
ALboolean AuxSendAuto;
|
|
|
|
|
|
|
|
ALenum EffectType;
|
|
|
|
ALeffectState *EffectState;
|
|
|
|
|
|
|
|
ALfloat RoomRolloff; /* Added to the source's room rolloff, not multiplied. */
|
|
|
|
ALfloat DecayTime;
|
2017-05-19 18:59:04 -07:00
|
|
|
ALfloat DecayHFRatio;
|
|
|
|
ALboolean DecayHFLimit;
|
2017-05-27 22:33:40 -07:00
|
|
|
ALfloat AirAbsorptionGainHF;
|
2016-05-12 18:26:33 -07:00
|
|
|
} Params;
|
|
|
|
|
2012-04-19 22:28:01 -07:00
|
|
|
/* Self ID */
|
|
|
|
ALuint id;
|
2016-01-28 00:02:46 -08:00
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei NumChannels;
|
2016-04-14 21:50:36 -07:00
|
|
|
BFChannelConfig ChanMap[MAX_EFFECT_CHANNELS];
|
2016-01-28 00:02:46 -08:00
|
|
|
/* Wet buffer configuration is ACN channel order with N3D scaling:
|
|
|
|
* * Channel 0 is the unattenuated mono signal.
|
|
|
|
* * Channel 1 is OpenAL -X
|
|
|
|
* * Channel 2 is OpenAL Y
|
|
|
|
* * Channel 3 is OpenAL -Z
|
|
|
|
* Consequently, effects that only want to work with mono input can use
|
|
|
|
* channel 0 by itself. Effects that want multichannel can process the
|
2016-04-15 10:50:46 -07:00
|
|
|
* ambisonics signal and make a B-Format pan (ComputeFirstOrderGains) for
|
|
|
|
* first-order device output (FOAOut).
|
2016-01-28 00:02:46 -08:00
|
|
|
*/
|
|
|
|
alignas(16) ALfloat WetBuffer[MAX_EFFECT_CHANNELS][BUFFERSIZE];
|
2013-10-07 05:41:41 -07:00
|
|
|
} ALeffectslot;
|
2007-12-17 17:43:19 -08:00
|
|
|
|
2016-05-29 02:47:54 -07:00
|
|
|
inline void LockEffectSlotsRead(ALCcontext *context)
|
|
|
|
{ LockUIntMapRead(&context->EffectSlotMap); }
|
|
|
|
inline void UnlockEffectSlotsRead(ALCcontext *context)
|
|
|
|
{ UnlockUIntMapRead(&context->EffectSlotMap); }
|
|
|
|
inline void LockEffectSlotsWrite(ALCcontext *context)
|
|
|
|
{ LockUIntMapWrite(&context->EffectSlotMap); }
|
|
|
|
inline void UnlockEffectSlotsWrite(ALCcontext *context)
|
|
|
|
{ UnlockUIntMapWrite(&context->EffectSlotMap); }
|
|
|
|
|
2013-11-04 13:51:19 -08:00
|
|
|
inline struct ALeffectslot *LookupEffectSlot(ALCcontext *context, ALuint id)
|
2016-05-29 02:47:54 -07:00
|
|
|
{ return (struct ALeffectslot*)LookupUIntMapKeyNoLock(&context->EffectSlotMap, id); }
|
2013-11-04 13:51:19 -08:00
|
|
|
inline struct ALeffectslot *RemoveEffectSlot(ALCcontext *context, ALuint id)
|
2016-05-29 02:47:54 -07:00
|
|
|
{ return (struct ALeffectslot*)RemoveUIntMapKeyNoLock(&context->EffectSlotMap, id); }
|
2007-12-17 17:43:19 -08:00
|
|
|
|
2012-01-20 16:23:15 -08:00
|
|
|
ALenum InitEffectSlot(ALeffectslot *slot);
|
2016-05-12 18:26:33 -07:00
|
|
|
void DeinitEffectSlot(ALeffectslot *slot);
|
2016-05-15 01:19:05 -07:00
|
|
|
void UpdateEffectSlotProps(ALeffectslot *slot);
|
2016-08-25 06:17:36 -07:00
|
|
|
void UpdateAllEffectSlotProps(ALCcontext *context);
|
2008-01-15 16:22:39 -08:00
|
|
|
ALvoid ReleaseALAuxiliaryEffectSlots(ALCcontext *Context);
|
2007-12-17 17:43:19 -08:00
|
|
|
|
2009-05-29 13:30:50 -07:00
|
|
|
|
2013-05-23 18:50:07 -07:00
|
|
|
ALeffectStateFactory *ALnullStateFactory_getFactory(void);
|
2013-05-21 12:47:18 -07:00
|
|
|
ALeffectStateFactory *ALreverbStateFactory_getFactory(void);
|
|
|
|
ALeffectStateFactory *ALchorusStateFactory_getFactory(void);
|
2013-10-03 07:55:12 -07:00
|
|
|
ALeffectStateFactory *ALcompressorStateFactory_getFactory(void);
|
2013-05-21 12:47:18 -07:00
|
|
|
ALeffectStateFactory *ALdistortionStateFactory_getFactory(void);
|
|
|
|
ALeffectStateFactory *ALechoStateFactory_getFactory(void);
|
|
|
|
ALeffectStateFactory *ALequalizerStateFactory_getFactory(void);
|
|
|
|
ALeffectStateFactory *ALflangerStateFactory_getFactory(void);
|
|
|
|
ALeffectStateFactory *ALmodulatorStateFactory_getFactory(void);
|
|
|
|
|
|
|
|
ALeffectStateFactory *ALdedicatedStateFactory_getFactory(void);
|
|
|
|
|
2009-05-29 13:30:50 -07:00
|
|
|
|
2012-03-13 14:58:34 -07:00
|
|
|
ALenum InitializeEffect(ALCdevice *Device, ALeffectslot *EffectSlot, ALeffect *effect);
|
2009-05-29 13:30:50 -07:00
|
|
|
|
2013-05-21 12:47:18 -07:00
|
|
|
void InitEffectFactoryMap(void);
|
|
|
|
void DeinitEffectFactoryMap(void);
|
|
|
|
|
2007-12-17 17:43:19 -08:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|