From 46301a087cfa106979dd1bab5574737020c9f94f Mon Sep 17 00:00:00 2001 From: Chris Robinson Date: Wed, 14 Nov 2018 06:17:47 -0800 Subject: [PATCH] Use C++ a bit more with alc.cpp --- Alc/alc.cpp | 246 +++++++++++++++++--------------------- OpenAL32/Include/alMain.h | 11 +- 2 files changed, 116 insertions(+), 141 deletions(-) diff --git a/Alc/alc.cpp b/Alc/alc.cpp index b46f1986..ab8f8fbe 100644 --- a/Alc/alc.cpp +++ b/Alc/alc.cpp @@ -30,6 +30,8 @@ #include #include +#include +#include #include #include @@ -58,6 +60,8 @@ #include "backends/base.h" +namespace { + /************************************************ * Backends ************************************************/ @@ -66,7 +70,7 @@ struct BackendInfo { ALCbackendFactory* (*getFactory)(void); }; -static struct BackendInfo BackendList[] = { +struct BackendInfo BackendList[] = { #ifdef HAVE_JACK { "jack", ALCjackBackendFactory_getFactory }, #endif @@ -115,18 +119,18 @@ static struct BackendInfo BackendList[] = { { "wave", ALCwaveBackendFactory_getFactory }, #endif }; -static ALsizei BackendListSize = COUNTOF(BackendList); +ALsizei BackendListSize = COUNTOF(BackendList); #undef EmptyFuncs -static struct BackendInfo PlaybackBackend; -static struct BackendInfo CaptureBackend; +struct BackendInfo PlaybackBackend; +struct BackendInfo CaptureBackend; /************************************************ * Functions, enums, and errors ************************************************/ #define DECL(x) { #x, (ALCvoid*)(x) } -static const struct { +constexpr struct { const ALCchar *funcName; ALCvoid *address; } alcFunctions[] = { @@ -305,7 +309,7 @@ static const struct { #undef DECL #define DECL(x) { #x, (x) } -static const struct { +constexpr struct { const ALCchar *enumName; ALCenum value; } alcEnumerations[] = { @@ -688,12 +692,12 @@ static const struct { }; #undef DECL -static const ALCchar alcNoError[] = "No Error"; -static const ALCchar alcErrInvalidDevice[] = "Invalid Device"; -static const ALCchar alcErrInvalidContext[] = "Invalid Context"; -static const ALCchar alcErrInvalidEnum[] = "Invalid Enum"; -static const ALCchar alcErrInvalidValue[] = "Invalid Value"; -static const ALCchar alcErrOutOfMemory[] = "Out of Memory"; +constexpr ALCchar alcNoError[] = "No Error"; +constexpr ALCchar alcErrInvalidDevice[] = "Invalid Device"; +constexpr ALCchar alcErrInvalidContext[] = "Invalid Context"; +constexpr ALCchar alcErrInvalidEnum[] = "Invalid Enum"; +constexpr ALCchar alcErrInvalidValue[] = "Invalid Value"; +constexpr ALCchar alcErrOutOfMemory[] = "Out of Memory"; /************************************************ @@ -701,17 +705,17 @@ static const ALCchar alcErrOutOfMemory[] = "Out of Memory"; ************************************************/ /* Enumerated device names */ -static const ALCchar alcDefaultName[] = "OpenAL Soft\0"; +constexpr ALCchar alcDefaultName[] = "OpenAL Soft\0"; -static al_string alcAllDevicesList; -static al_string alcCaptureDeviceList; +al_string alcAllDevicesList; +al_string alcCaptureDeviceList; /* Default is always the first in the list */ -static ALCchar *alcDefaultAllDevicesSpecifier; -static ALCchar *alcCaptureDefaultDeviceSpecifier; +std::string alcDefaultAllDevicesSpecifier; +std::string alcCaptureDefaultDeviceSpecifier; /* Default context extensions */ -static const ALchar alExtList[] = +constexpr ALchar alExtList[] = "AL_EXT_ALAW " "AL_EXT_BFORMAT " "AL_EXT_DOUBLE " @@ -742,12 +746,54 @@ static const ALchar alExtList[] = "AL_SOFT_source_resampler " "AL_SOFT_source_spatialize"; -static ATOMIC(ALCenum) LastNullDeviceError = ATOMIC_INIT_STATIC(ALC_NO_ERROR); +std::atomic LastNullDeviceError{ALC_NO_ERROR}; /* Thread-local current context */ -static altss_t LocalContext; +altss_t LocalContext; /* Process-wide current context */ -static ATOMIC(ALCcontext*) GlobalContext = ATOMIC_INIT_STATIC(nullptr); +std::atomic GlobalContext{nullptr}; + +/* Flag to trap ALC device errors */ +bool TrapALCError{false}; + +/* One-time configuration init control */ +std::once_flag alc_config_once{}; + +/* Default effect that applies to sources that don't have an effect on send 0 */ +ALeffect DefaultEffect; + +/* Flag to specify if alcSuspendContext/alcProcessContext should defer/process + * updates. + */ +bool SuspendDefers{true}; + + +/************************************************ + * ALC information + ************************************************/ +constexpr ALCchar alcNoDeviceExtList[] = + "ALC_ENUMERATE_ALL_EXT ALC_ENUMERATION_EXT ALC_EXT_CAPTURE " + "ALC_EXT_thread_local_context ALC_SOFT_loopback"; +constexpr ALCchar alcExtensionList[] = + "ALC_ENUMERATE_ALL_EXT ALC_ENUMERATION_EXT ALC_EXT_CAPTURE " + "ALC_EXT_DEDICATED ALC_EXT_disconnect ALC_EXT_EFX " + "ALC_EXT_thread_local_context ALC_SOFT_device_clock ALC_SOFT_HRTF " + "ALC_SOFT_loopback ALC_SOFT_output_limiter ALC_SOFT_pause_device"; +constexpr ALCint alcMajorVersion = 1; +constexpr ALCint alcMinorVersion = 1; + +constexpr ALCint alcEFXMajorVersion = 1; +constexpr ALCint alcEFXMinorVersion = 0; + + +/************************************************ + * Device lists + ************************************************/ +std::atomic DeviceList{nullptr}; + +std::recursive_mutex ListLock; + +} // namespace /* Mixing thread piority level */ ALint RTPrioLevel; @@ -759,56 +805,6 @@ enum LogLevel LogLevel = LogWarning; enum LogLevel LogLevel = LogError; #endif -/* Flag to trap ALC device errors */ -static ALCboolean TrapALCError = ALC_FALSE; - -/* One-time configuration init control */ -static alonce_flag alc_config_once = AL_ONCE_FLAG_INIT; - -/* Default effect that applies to sources that don't have an effect on send 0 */ -static ALeffect DefaultEffect; - -/* Flag to specify if alcSuspendContext/alcProcessContext should defer/process - * updates. - */ -static ALCboolean SuspendDefers = ALC_TRUE; - - -/************************************************ - * ALC information - ************************************************/ -static const ALCchar alcNoDeviceExtList[] = - "ALC_ENUMERATE_ALL_EXT ALC_ENUMERATION_EXT ALC_EXT_CAPTURE " - "ALC_EXT_thread_local_context ALC_SOFT_loopback"; -static const ALCchar alcExtensionList[] = - "ALC_ENUMERATE_ALL_EXT ALC_ENUMERATION_EXT ALC_EXT_CAPTURE " - "ALC_EXT_DEDICATED ALC_EXT_disconnect ALC_EXT_EFX " - "ALC_EXT_thread_local_context ALC_SOFT_device_clock ALC_SOFT_HRTF " - "ALC_SOFT_loopback ALC_SOFT_output_limiter ALC_SOFT_pause_device"; -static const ALCint alcMajorVersion = 1; -static const ALCint alcMinorVersion = 1; - -static const ALCint alcEFXMajorVersion = 1; -static const ALCint alcEFXMinorVersion = 0; - - -/************************************************ - * Device lists - ************************************************/ -static std::atomic DeviceList{nullptr}; - -static almtx_t ListLock; -static inline void LockLists(void) -{ - int ret = almtx_lock(&ListLock); - assert(ret == althrd_success); -} -static inline void UnlockLists(void) -{ - int ret = almtx_unlock(&ListLock); - assert(ret == althrd_success); -} - /************************************************ * Library initialization ************************************************/ @@ -899,9 +895,6 @@ static void alc_init(void) ret = altss_create(&LocalContext, ReleaseThreadCtx); assert(ret == althrd_success); - - ret = almtx_init(&ListLock, almtx_recursive); - assert(ret == althrd_success); } static void alc_initconfig(void) @@ -951,7 +944,7 @@ static void alc_initconfig(void) { if(strcasecmp(str, "ignore") == 0) { - SuspendDefers = ALC_FALSE; + SuspendDefers = false; TRACE("Selected context suspend behavior, \"ignore\"\n"); } else @@ -1023,7 +1016,7 @@ static void alc_initconfig(void) if(str && (strcasecmp(str, "true") == 0 || strtol(str, nullptr, 0) == 1)) { TrapALError = AL_TRUE; - TrapALCError = AL_TRUE; + TrapALCError = true; } else { @@ -1034,8 +1027,8 @@ static void alc_initconfig(void) str = getenv("ALSOFT_TRAP_ALC_ERROR"); if(str && (strcasecmp(str, "true") == 0 || strtol(str, nullptr, 0) == 1)) - TrapALCError = ALC_TRUE; - TrapALCError = GetConfigValueBool(nullptr, nullptr, "trap-alc-error", TrapALCError); + TrapALCError = true; + TrapALCError = !!GetConfigValueBool(nullptr, nullptr, "trap-alc-error", TrapALCError); } if(ConfigValueFloat(nullptr, "reverb", "boost", &valf)) @@ -1172,7 +1165,7 @@ static void alc_initconfig(void) if((str && str[0]) || ConfigValueStr(nullptr, nullptr, "default-reverb", &str)) LoadReverbPreset(str, &DefaultEffect); } -#define DO_INITCONFIG() alcall_once(&alc_config_once, alc_initconfig) +#define DO_INITCONFIG() std::call_once(alc_config_once, alc_initconfig) /************************************************ @@ -1183,10 +1176,8 @@ static void alc_cleanup(void) AL_STRING_DEINIT(alcAllDevicesList); AL_STRING_DEINIT(alcCaptureDeviceList); - free(alcDefaultAllDevicesSpecifier); - alcDefaultAllDevicesSpecifier = nullptr; - free(alcCaptureDefaultDeviceSpecifier); - alcCaptureDefaultDeviceSpecifier = nullptr; + alcDefaultAllDevicesSpecifier.clear(); + alcCaptureDefaultDeviceSpecifier.clear(); if(ALCdevice *dev{DeviceList.exchange(nullptr)}) { @@ -1206,7 +1197,6 @@ static void alc_deinit_safe(void) FreeHrtfs(); FreeALConfig(); - almtx_destroy(&ListLock); altss_delete(LocalContext); if(LogFile != stderr) @@ -1246,16 +1236,13 @@ static void ProbeDevices(al_string *list, struct BackendInfo *backendinfo, enum { DO_INITCONFIG(); - LockLists(); + std::lock_guard _{ListLock}; alstr_clear(list); - if(backendinfo->getFactory) { ALCbackendFactory *factory = backendinfo->getFactory(); V(factory,probe)(type, list); } - - UnlockLists(); } static void ProbeAllDevicesList(void) { ProbeDevices(&alcAllDevicesList, &PlaybackBackend, ALL_DEVICE_PROBE); } @@ -1634,7 +1621,7 @@ static void alcSetError(ALCdevice *device, ALCenum errorCode) if(device) ATOMIC_STORE_SEQ(&device->LastError, errorCode); else - ATOMIC_STORE_SEQ(&LastNullDeviceError, errorCode); + LastNullDeviceError.store(errorCode); } @@ -2311,8 +2298,8 @@ static ALCenum UpdateDeviceParams(ALCdevice *device, const ALCint *attrList) * auxiliary sends is changing. Active sources will have updates * respecified in UpdateAllSourceProps. */ - vprops = ATOMIC_EXCHANGE_PTR(&context->FreeVoiceProps, static_cast(nullptr), - almemory_order_acq_rel); + vprops = static_cast(ATOMIC_EXCHANGE_PTR(&context->FreeVoiceProps, + static_cast(nullptr), almemory_order_acq_rel)); while(vprops) { struct ALvoiceProps *next = ATOMIC_LOAD(&vprops->next, almemory_order_relaxed); @@ -2537,19 +2524,17 @@ void ALCdevice_DecRef(ALCdevice *device) */ static ALCboolean VerifyDevice(ALCdevice **device) { - LockLists(); + std::lock_guard _{ListLock}; ALCdevice *tmpDevice{DeviceList.load()}; while(tmpDevice) { if(tmpDevice == *device) { ALCdevice_IncRef(tmpDevice); - UnlockLists(); return ALC_TRUE; } tmpDevice = ATOMIC_LOAD(&tmpDevice->next, almemory_order_relaxed); } - UnlockLists(); *device = nullptr; return ALC_FALSE; @@ -2661,7 +2646,6 @@ static ALvoid InitContext(ALCcontext *Context) static void FreeContext(ALCcontext *context) { ALlistener *listener = context->Listener; - struct ALeffectslotArray *auxslots; struct ALeffectslotProps *eprops; struct ALlistenerProps *lprops; struct ALcontextProps *cprops; @@ -2694,9 +2678,8 @@ static void FreeContext(ALCcontext *context) context->DefaultSlot = nullptr; } - auxslots = ATOMIC_EXCHANGE_PTR(&context->ActiveAuxSlots, - static_cast(nullptr), almemory_order_relaxed); - al_free(auxslots); + al_free(ATOMIC_EXCHANGE_PTR(&context->ActiveAuxSlots, + static_cast(nullptr), almemory_order_relaxed)); ReleaseALSources(context); #define FREE_SOURCESUBLIST(x) al_free((x)->Sources) @@ -2795,7 +2778,7 @@ static bool ReleaseContext(ALCcontext *context, ALCdevice *device) } origctx = context; - if(ATOMIC_COMPARE_EXCHANGE_PTR_STRONG_SEQ(&GlobalContext, &origctx, static_cast(nullptr))) + if(GlobalContext.compare_exchange_strong(origctx, nullptr)) ALCcontext_DecRef(context); V0(device->Backend,lock)(); @@ -2857,7 +2840,7 @@ static void ReleaseThreadCtx(void *ptr) */ static ALCboolean VerifyContext(ALCcontext **context) { - LockLists(); + std::lock_guard _{ListLock}; ALCdevice *dev{DeviceList.load()}; while(dev) { @@ -2867,14 +2850,12 @@ static ALCboolean VerifyContext(ALCcontext **context) if(ctx == *context) { ALCcontext_IncRef(ctx); - UnlockLists(); return ALC_TRUE; } ctx = ATOMIC_LOAD(&ctx->next, almemory_order_relaxed); } dev = ATOMIC_LOAD(&dev->next, almemory_order_relaxed); } - UnlockLists(); *context = nullptr; return ALC_FALSE; @@ -2893,10 +2874,9 @@ ALCcontext *GetContextRef(void) ALCcontext_IncRef(context); else { - LockLists(); - context = ATOMIC_LOAD_SEQ(&GlobalContext); + std::lock_guard _{ListLock}; + context = GlobalContext.load(std::memory_order_acquire); if(context) ALCcontext_IncRef(context); - UnlockLists(); } return context; @@ -3007,7 +2987,7 @@ ALC_API ALCenum ALC_APIENTRY alcGetError(ALCdevice *device) ALCdevice_DecRef(device); } else - errorCode = ATOMIC_EXCHANGE_SEQ(&LastNullDeviceError, ALC_NO_ERROR); + errorCode = LastNullDeviceError.exchange(ALC_NO_ERROR); return errorCode; } @@ -3125,12 +3105,8 @@ ALC_API const ALCchar* ALC_APIENTRY alcGetString(ALCdevice *Device, ALCenum para VerifyDevice(&Device); - free(alcDefaultAllDevicesSpecifier); - alcDefaultAllDevicesSpecifier = strdup(alstr_get_cstr(alcAllDevicesList)); - if(!alcDefaultAllDevicesSpecifier) - alcSetError(Device, ALC_OUT_OF_MEMORY); - - value = alcDefaultAllDevicesSpecifier; + alcDefaultAllDevicesSpecifier = alstr_get_cstr(alcAllDevicesList); + value = alcDefaultAllDevicesSpecifier.c_str(); if(Device) ALCdevice_DecRef(Device); break; @@ -3140,12 +3116,8 @@ ALC_API const ALCchar* ALC_APIENTRY alcGetString(ALCdevice *Device, ALCenum para VerifyDevice(&Device); - free(alcCaptureDefaultDeviceSpecifier); - alcCaptureDefaultDeviceSpecifier = strdup(alstr_get_cstr(alcCaptureDeviceList)); - if(!alcCaptureDefaultDeviceSpecifier) - alcSetError(Device, ALC_OUT_OF_MEMORY); - - value = alcCaptureDefaultDeviceSpecifier; + alcCaptureDefaultDeviceSpecifier = alstr_get_cstr(alcCaptureDeviceList); + value = alcCaptureDefaultDeviceSpecifier.c_str(); if(Device) ALCdevice_DecRef(Device); break; @@ -3772,17 +3744,17 @@ ALC_API ALCcontext* ALC_APIENTRY alcCreateContext(ALCdevice *device, const ALCin * device is asynchronously destropyed, to ensure this new context is * properly cleaned up after being made. */ - LockLists(); + std::unique_lock listlock{ListLock}; if(!VerifyDevice(&device) || device->Type == Capture || !ATOMIC_LOAD(&device->Connected, almemory_order_relaxed)) { - UnlockLists(); + listlock.unlock(); alcSetError(device, ALC_INVALID_DEVICE); if(device) ALCdevice_DecRef(device); return nullptr; } almtx_lock(&device->BackendLock); - UnlockLists(); + listlock.unlock(); ATOMIC_STORE_SEQ(&device->LastError, ALC_NO_ERROR); @@ -3889,17 +3861,15 @@ ALC_API ALCcontext* ALC_APIENTRY alcCreateContext(ALCdevice *device, const ALCin */ ALC_API ALCvoid ALC_APIENTRY alcDestroyContext(ALCcontext *context) { - ALCdevice *Device; - - LockLists(); + std::unique_lock listlock{ListLock}; if(!VerifyContext(&context)) { - UnlockLists(); + listlock.unlock(); alcSetError(nullptr, ALC_INVALID_CONTEXT); return; } - Device = context->Device; + ALCdevice* Device{context->Device}; if(Device) { almtx_lock(&Device->BackendLock); @@ -3910,7 +3880,7 @@ ALC_API ALCvoid ALC_APIENTRY alcDestroyContext(ALCcontext *context) } almtx_unlock(&Device->BackendLock); } - UnlockLists(); + listlock.unlock(); ALCcontext_DecRef(context); } @@ -3923,7 +3893,7 @@ ALC_API ALCvoid ALC_APIENTRY alcDestroyContext(ALCcontext *context) ALC_API ALCcontext* ALC_APIENTRY alcGetCurrentContext(void) { ALCcontext *Context{static_cast(altss_get(LocalContext))}; - if(!Context) Context = ATOMIC_LOAD_SEQ(&GlobalContext); + if(!Context) Context = GlobalContext.load(); return Context; } @@ -3951,7 +3921,7 @@ ALC_API ALCboolean ALC_APIENTRY alcMakeContextCurrent(ALCcontext *context) return ALC_FALSE; } /* context's reference count is already incremented */ - context = ATOMIC_EXCHANGE_PTR_SEQ(&GlobalContext, context); + context = GlobalContext.exchange(context); if(context) ALCcontext_DecRef(context); if((context=static_cast(altss_get(LocalContext))) != nullptr) @@ -4206,7 +4176,7 @@ ALC_API ALCdevice* ALC_APIENTRY alcOpenDevice(const ALCchar *deviceName) */ ALC_API ALCboolean ALC_APIENTRY alcCloseDevice(ALCdevice *device) { - LockLists(); + std::unique_lock listlock{ListLock}; ALCdevice *iter{DeviceList.load()}; do { if(iter == device) @@ -4216,14 +4186,13 @@ ALC_API ALCboolean ALC_APIENTRY alcCloseDevice(ALCdevice *device) if(!iter || iter->Type == Capture) { alcSetError(iter, ALC_INVALID_DEVICE); - UnlockLists(); return ALC_FALSE; } almtx_lock(&device->BackendLock); ALCdevice *origdev{device}; ALCdevice *nextdev{ATOMIC_LOAD(&device->next, almemory_order_relaxed)}; - if(!ATOMIC_COMPARE_EXCHANGE_PTR_STRONG_SEQ(&DeviceList, &origdev, nextdev)) + if(!DeviceList.compare_exchange_strong(origdev, nextdev)) { ALCdevice *list; do { @@ -4231,7 +4200,7 @@ ALC_API ALCboolean ALC_APIENTRY alcCloseDevice(ALCdevice *device) origdev = device; } while(!ATOMIC_COMPARE_EXCHANGE_PTR_STRONG_SEQ(&list->next, &origdev, nextdev)); } - UnlockLists(); + listlock.unlock(); ALCcontext *ctx{ATOMIC_LOAD_SEQ(&device->ContextList)}; while(ctx != nullptr) @@ -4339,8 +4308,8 @@ ALC_API ALCdevice* ALC_APIENTRY alcCaptureOpenDevice(const ALCchar *deviceName, ALC_API ALCboolean ALC_APIENTRY alcCaptureCloseDevice(ALCdevice *device) { + std::unique_lock listlock{ListLock}; - LockLists(); ALCdevice *iter{DeviceList.load()}; do { if(iter == device) @@ -4350,7 +4319,6 @@ ALC_API ALCboolean ALC_APIENTRY alcCaptureCloseDevice(ALCdevice *device) if(!iter || iter->Type != Capture) { alcSetError(iter, ALC_INVALID_DEVICE); - UnlockLists(); return ALC_FALSE; } @@ -4364,7 +4332,7 @@ ALC_API ALCboolean ALC_APIENTRY alcCaptureCloseDevice(ALCdevice *device) origdev = device; } while(!ATOMIC_COMPARE_EXCHANGE_PTR_STRONG_SEQ(&list->next, &origdev, nextdev)); } - UnlockLists(); + listlock.unlock(); almtx_lock(&device->BackendLock); if((device->Flags&DEVICE_RUNNING)) @@ -4663,17 +4631,17 @@ ALC_API ALCboolean ALC_APIENTRY alcResetDeviceSOFT(ALCdevice *device, const ALCi { ALCenum err; - LockLists(); + std::unique_lock listlock{ListLock}; if(!VerifyDevice(&device) || device->Type == Capture || !ATOMIC_LOAD(&device->Connected, almemory_order_relaxed)) { - UnlockLists(); + listlock.unlock(); alcSetError(device, ALC_INVALID_DEVICE); if(device) ALCdevice_DecRef(device); return ALC_FALSE; } almtx_lock(&device->BackendLock); - UnlockLists(); + listlock.unlock(); err = UpdateDeviceParams(device, attribs); almtx_unlock(&device->BackendLock); diff --git a/OpenAL32/Include/alMain.h b/OpenAL32/Include/alMain.h index 666a8ade..b9ace4db 100644 --- a/OpenAL32/Include/alMain.h +++ b/OpenAL32/Include/alMain.h @@ -32,6 +32,15 @@ #include "threads.h" +#ifndef __cplusplus +#define COUNTOF(x) (sizeof(x) / sizeof(0[x])) +#else +template +constexpr inline size_t countof(const T(&)[N]) noexcept +{ return N; } +#define COUNTOF countof +#endif + #if defined(_WIN64) #define SZFMT "%I64u" #elif defined(_WIN32) @@ -197,8 +206,6 @@ static const union { #define IS_LITTLE_ENDIAN (EndianTest.b[0] == 1) #endif -#define COUNTOF(x) (sizeof(x) / sizeof(0[x])) - struct ll_ringbuffer; struct Hrtf;