2016-03-01 15:47:10 +00:00
|
|
|
#ifndef OALSOUND_H
|
|
|
|
#define OALSOUND_H
|
|
|
|
|
2016-04-26 13:12:56 +00:00
|
|
|
#include <thread>
|
|
|
|
#include <mutex>
|
|
|
|
#include <atomic>
|
|
|
|
#include <condition_variable>
|
2017-04-24 10:46:48 +00:00
|
|
|
#include <unordered_map>
|
2016-04-26 13:12:56 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
#include "i_sound.h"
|
|
|
|
#include "s_sound.h"
|
|
|
|
#include "menu/menu.h"
|
|
|
|
|
|
|
|
#ifndef NO_OPENAL
|
|
|
|
|
2016-02-11 12:07:01 +00:00
|
|
|
#ifdef DYN_OPENAL
|
2016-03-01 15:47:10 +00:00
|
|
|
#define AL_NO_PROTOTYPES
|
2017-04-17 15:05:09 +00:00
|
|
|
#include "thirdparty/al.h"
|
|
|
|
#include "thirdparty/alc.h"
|
|
|
|
#else
|
2016-03-01 15:47:10 +00:00
|
|
|
#include "al.h"
|
|
|
|
#include "alc.h"
|
2017-04-17 15:05:09 +00:00
|
|
|
#endif
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-04-18 18:55:05 +00:00
|
|
|
#include "efx.h"
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
#ifndef ALC_ENUMERATE_ALL_EXT
|
|
|
|
#define ALC_ENUMERATE_ALL_EXT 1
|
|
|
|
#define ALC_DEFAULT_ALL_DEVICES_SPECIFIER 0x1012
|
|
|
|
#define ALC_ALL_DEVICES_SPECIFIER 0x1013
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef ALC_EXT_disconnect
|
|
|
|
#define ALC_EXT_disconnect 1
|
|
|
|
#define ALC_CONNECTED 0x313
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AL_EXT_source_distance_model
|
|
|
|
#define AL_EXT_source_distance_model 1
|
|
|
|
#define AL_SOURCE_DISTANCE_MODEL 0x200
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AL_SOFT_loop_points
|
|
|
|
#define AL_SOFT_loop_points 1
|
|
|
|
#define AL_LOOP_POINTS_SOFT 0x2015
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AL_EXT_float32
|
|
|
|
#define AL_EXT_float32 1
|
|
|
|
#define AL_FORMAT_MONO_FLOAT32 0x10010
|
|
|
|
#define AL_FORMAT_STEREO_FLOAT32 0x10011
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AL_EXT_MCFORMATS
|
|
|
|
#define AL_EXT_MCFORMATS 1
|
|
|
|
#define AL_FORMAT_QUAD8 0x1204
|
|
|
|
#define AL_FORMAT_QUAD16 0x1205
|
|
|
|
#define AL_FORMAT_QUAD32 0x1206
|
|
|
|
#define AL_FORMAT_REAR8 0x1207
|
|
|
|
#define AL_FORMAT_REAR16 0x1208
|
|
|
|
#define AL_FORMAT_REAR32 0x1209
|
|
|
|
#define AL_FORMAT_51CHN8 0x120A
|
|
|
|
#define AL_FORMAT_51CHN16 0x120B
|
|
|
|
#define AL_FORMAT_51CHN32 0x120C
|
|
|
|
#define AL_FORMAT_61CHN8 0x120D
|
|
|
|
#define AL_FORMAT_61CHN16 0x120E
|
|
|
|
#define AL_FORMAT_61CHN32 0x120F
|
|
|
|
#define AL_FORMAT_71CHN8 0x1210
|
|
|
|
#define AL_FORMAT_71CHN16 0x1211
|
|
|
|
#define AL_FORMAT_71CHN32 0x1212
|
|
|
|
#endif
|
|
|
|
|
2016-05-02 08:56:12 +00:00
|
|
|
#ifndef AL_EXT_SOURCE_RADIUS
|
|
|
|
#define AL_EXT_SOURCE_RADIUS 1
|
|
|
|
#define AL_SOURCE_RADIUS 0x1031
|
|
|
|
#endif
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
class OpenALSoundStream;
|
|
|
|
|
|
|
|
class OpenALSoundRenderer : public SoundRenderer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OpenALSoundRenderer();
|
|
|
|
virtual ~OpenALSoundRenderer();
|
|
|
|
|
|
|
|
virtual void SetSfxVolume(float volume);
|
|
|
|
virtual void SetMusicVolume(float volume);
|
2017-03-09 18:54:41 +00:00
|
|
|
virtual std::pair<SoundHandle,bool> LoadSound(uint8_t *sfxdata, int length, bool monoize);
|
|
|
|
virtual std::pair<SoundHandle,bool> LoadSoundRaw(uint8_t *sfxdata, int length, int frequency, int channels, int bits, int loopstart, int loopend = -1, bool monoize = false);
|
2016-03-01 15:47:10 +00:00
|
|
|
virtual void UnloadSound(SoundHandle sfx);
|
|
|
|
virtual unsigned int GetMSLength(SoundHandle sfx);
|
|
|
|
virtual unsigned int GetSampleLength(SoundHandle sfx);
|
|
|
|
virtual float GetOutputRate();
|
|
|
|
|
|
|
|
// Streaming sounds.
|
|
|
|
virtual SoundStream *CreateStream(SoundStreamCallback callback, int buffbytes, int flags, int samplerate, void *userdata);
|
|
|
|
virtual SoundStream *OpenStream(FileReader *reader, int flags);
|
|
|
|
|
|
|
|
// Starts a sound.
|
|
|
|
virtual FISoundChannel *StartSound(SoundHandle sfx, float vol, int pitch, int chanflags, FISoundChannel *reuse_chan);
|
|
|
|
virtual FISoundChannel *StartSound3D(SoundHandle sfx, SoundListener *listener, float vol, FRolloffInfo *rolloff, float distscale, int pitch, int priority, const FVector3 &pos, const FVector3 &vel, int channum, int chanflags, FISoundChannel *reuse_chan);
|
|
|
|
|
|
|
|
// Changes a channel's volume.
|
|
|
|
virtual void ChannelVolume(FISoundChannel *chan, float volume);
|
|
|
|
|
|
|
|
// Stops a sound channel.
|
|
|
|
virtual void StopChannel(FISoundChannel *chan);
|
|
|
|
|
|
|
|
// Returns position of sound on this channel, in samples.
|
|
|
|
virtual unsigned int GetPosition(FISoundChannel *chan);
|
|
|
|
|
|
|
|
// Synchronizes following sound startups.
|
|
|
|
virtual void Sync(bool sync);
|
|
|
|
|
|
|
|
// Pauses or resumes all sound effect channels.
|
|
|
|
virtual void SetSfxPaused(bool paused, int slot);
|
|
|
|
|
|
|
|
// Pauses or resumes *every* channel, including environmental reverb.
|
|
|
|
virtual void SetInactive(SoundRenderer::EInactiveState inactive);
|
|
|
|
|
|
|
|
// Updates the volume, separation, and pitch of a sound channel.
|
|
|
|
virtual void UpdateSoundParams3D(SoundListener *listener, FISoundChannel *chan, bool areasound, const FVector3 &pos, const FVector3 &vel);
|
|
|
|
|
|
|
|
virtual void UpdateListener(SoundListener *);
|
|
|
|
virtual void UpdateSounds();
|
|
|
|
|
|
|
|
virtual void MarkStartTime(FISoundChannel*);
|
|
|
|
virtual float GetAudibility(FISoundChannel*);
|
|
|
|
|
|
|
|
|
|
|
|
virtual bool IsValid();
|
|
|
|
virtual void PrintStatus();
|
|
|
|
virtual void PrintDriversList();
|
|
|
|
virtual FString GatherStats();
|
|
|
|
|
2017-03-05 08:01:16 +00:00
|
|
|
virtual MIDIDevice* CreateMIDIDevice() const override;
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
private:
|
|
|
|
struct {
|
|
|
|
bool EXT_EFX;
|
|
|
|
bool EXT_disconnect;
|
2016-05-02 21:23:07 +00:00
|
|
|
bool SOFT_pause_device;
|
2016-03-01 15:47:10 +00:00
|
|
|
} ALC;
|
|
|
|
struct {
|
|
|
|
bool EXT_source_distance_model;
|
2016-05-02 08:56:12 +00:00
|
|
|
bool EXT_SOURCE_RADIUS;
|
2016-03-01 15:47:10 +00:00
|
|
|
bool SOFT_deferred_updates;
|
|
|
|
bool SOFT_loop_points;
|
|
|
|
} AL;
|
|
|
|
|
|
|
|
// EFX Extension function pointer variables. Loaded after context creation
|
|
|
|
// if EFX is supported. These pointers may be context- or device-dependant,
|
|
|
|
// thus can't be static
|
|
|
|
// Effect objects
|
|
|
|
LPALGENEFFECTS alGenEffects;
|
|
|
|
LPALDELETEEFFECTS alDeleteEffects;
|
|
|
|
LPALISEFFECT alIsEffect;
|
|
|
|
LPALEFFECTI alEffecti;
|
|
|
|
LPALEFFECTIV alEffectiv;
|
|
|
|
LPALEFFECTF alEffectf;
|
|
|
|
LPALEFFECTFV alEffectfv;
|
|
|
|
LPALGETEFFECTI alGetEffecti;
|
|
|
|
LPALGETEFFECTIV alGetEffectiv;
|
|
|
|
LPALGETEFFECTF alGetEffectf;
|
|
|
|
LPALGETEFFECTFV alGetEffectfv;
|
|
|
|
// Filter objects
|
|
|
|
LPALGENFILTERS alGenFilters;
|
|
|
|
LPALDELETEFILTERS alDeleteFilters;
|
|
|
|
LPALISFILTER alIsFilter;
|
|
|
|
LPALFILTERI alFilteri;
|
|
|
|
LPALFILTERIV alFilteriv;
|
|
|
|
LPALFILTERF alFilterf;
|
|
|
|
LPALFILTERFV alFilterfv;
|
|
|
|
LPALGETFILTERI alGetFilteri;
|
|
|
|
LPALGETFILTERIV alGetFilteriv;
|
|
|
|
LPALGETFILTERF alGetFilterf;
|
|
|
|
LPALGETFILTERFV alGetFilterfv;
|
|
|
|
// Auxiliary slot objects
|
|
|
|
LPALGENAUXILIARYEFFECTSLOTS alGenAuxiliaryEffectSlots;
|
|
|
|
LPALDELETEAUXILIARYEFFECTSLOTS alDeleteAuxiliaryEffectSlots;
|
|
|
|
LPALISAUXILIARYEFFECTSLOT alIsAuxiliaryEffectSlot;
|
|
|
|
LPALAUXILIARYEFFECTSLOTI alAuxiliaryEffectSloti;
|
|
|
|
LPALAUXILIARYEFFECTSLOTIV alAuxiliaryEffectSlotiv;
|
|
|
|
LPALAUXILIARYEFFECTSLOTF alAuxiliaryEffectSlotf;
|
|
|
|
LPALAUXILIARYEFFECTSLOTFV alAuxiliaryEffectSlotfv;
|
|
|
|
LPALGETAUXILIARYEFFECTSLOTI alGetAuxiliaryEffectSloti;
|
|
|
|
LPALGETAUXILIARYEFFECTSLOTIV alGetAuxiliaryEffectSlotiv;
|
|
|
|
LPALGETAUXILIARYEFFECTSLOTF alGetAuxiliaryEffectSlotf;
|
|
|
|
LPALGETAUXILIARYEFFECTSLOTFV alGetAuxiliaryEffectSlotfv;
|
|
|
|
|
|
|
|
ALvoid (AL_APIENTRY*alDeferUpdatesSOFT)(void);
|
|
|
|
ALvoid (AL_APIENTRY*alProcessUpdatesSOFT)(void);
|
|
|
|
|
2016-05-02 21:23:07 +00:00
|
|
|
void (ALC_APIENTRY*alcDevicePauseSOFT)(ALCdevice *device);
|
|
|
|
void (ALC_APIENTRY*alcDeviceResumeSOFT)(ALCdevice *device);
|
|
|
|
|
2016-04-26 13:12:56 +00:00
|
|
|
void BackgroundProc();
|
|
|
|
void AddStream(OpenALSoundStream *stream);
|
|
|
|
void RemoveStream(OpenALSoundStream *stream);
|
2016-05-05 19:55:16 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
void LoadReverb(const ReverbContainer *env);
|
2017-04-24 10:46:48 +00:00
|
|
|
void FreeSource(ALuint source);
|
2016-03-01 15:47:10 +00:00
|
|
|
void PurgeStoppedSources();
|
|
|
|
static FSoundChan *FindLowestChannel();
|
2017-04-24 10:46:48 +00:00
|
|
|
void ForceStopChannel(FISoundChannel *chan);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-04-26 13:12:56 +00:00
|
|
|
std::thread StreamThread;
|
|
|
|
std::mutex StreamLock;
|
|
|
|
std::condition_variable StreamWake;
|
|
|
|
std::atomic<bool> QuitThread;
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
ALCdevice *Device;
|
|
|
|
ALCcontext *Context;
|
|
|
|
|
|
|
|
TArray<ALuint> Sources;
|
|
|
|
|
|
|
|
ALfloat SfxVolume;
|
|
|
|
ALfloat MusicVolume;
|
|
|
|
|
|
|
|
int SFXPaused;
|
|
|
|
TArray<ALuint> FreeSfx;
|
|
|
|
TArray<ALuint> PausableSfx;
|
|
|
|
TArray<ALuint> ReverbSfx;
|
|
|
|
TArray<ALuint> SfxGroup;
|
|
|
|
|
2017-04-24 10:46:48 +00:00
|
|
|
int UpdateTimeMS;
|
|
|
|
using SourceTimeMap = std::unordered_map<ALuint,int64_t>;
|
|
|
|
SourceTimeMap FadingSources;
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
const ReverbContainer *PrevEnvironment;
|
|
|
|
|
2017-03-08 17:50:37 +00:00
|
|
|
typedef TMap<uint16_t,ALuint> EffectMap;
|
|
|
|
typedef TMapIterator<uint16_t,ALuint> EffectMapIter;
|
2016-03-01 15:47:10 +00:00
|
|
|
ALuint EnvSlot;
|
|
|
|
ALuint EnvFilters[2];
|
|
|
|
EffectMap EnvEffects;
|
|
|
|
|
|
|
|
bool WasInWater;
|
|
|
|
|
|
|
|
TArray<OpenALSoundStream*> Streams;
|
|
|
|
friend class OpenALSoundStream;
|
|
|
|
|
|
|
|
ALCdevice *InitDevice();
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // NO_OPENAL
|
|
|
|
|
|
|
|
#endif
|