openal-soft/OpenAL32/alState.cpp

801 lines
21 KiB
C++
Raw Normal View History

2007-11-13 18:02:18 -08:00
/**
* OpenAL cross platform audio library
* Copyright (C) 1999-2000 by authors.
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
2007-11-13 18:02:18 -08:00
* Or go to http://www.gnu.org/copyleft/lgpl.html
*/
2008-01-16 14:09:04 -08:00
#include "config.h"
#include "version.h"
2007-11-13 18:02:18 -08:00
#include <stdlib.h>
2018-11-19 09:51:29 -08:00
#include <cmath>
2007-11-13 18:02:18 -08:00
#include "alMain.h"
#include "alcontext.h"
#include "alu.h"
2007-11-13 18:02:18 -08:00
#include "alError.h"
#include "backends/base.h"
2012-09-14 03:10:12 -07:00
2007-11-13 18:02:18 -08:00
2018-11-16 06:45:24 -08:00
namespace {
constexpr ALchar alVendor[] = "OpenAL Community";
constexpr ALchar alVersion[] = "1.1 ALSOFT " ALSOFT_VERSION;
constexpr ALchar alRenderer[] = "OpenAL Soft";
2007-11-13 18:02:18 -08:00
// Error Messages
2018-11-16 06:45:24 -08:00
constexpr ALchar alNoError[] = "No Error";
constexpr ALchar alErrInvalidName[] = "Invalid Name";
constexpr ALchar alErrInvalidEnum[] = "Invalid Enum";
constexpr ALchar alErrInvalidValue[] = "Invalid Value";
constexpr ALchar alErrInvalidOp[] = "Invalid Operation";
constexpr ALchar alErrOutOfMemory[] = "Out of Memory";
2007-11-13 18:02:18 -08:00
2017-04-21 04:15:08 -07:00
/* Resampler strings */
2018-11-16 06:45:24 -08:00
constexpr ALchar alPointResampler[] = "Nearest";
constexpr ALchar alLinearResampler[] = "Linear";
constexpr ALchar alCubicResampler[] = "Cubic";
constexpr ALchar alBSinc12Resampler[] = "11th order Sinc";
constexpr ALchar alBSinc24Resampler[] = "23rd order Sinc";
} // namespace
2017-04-21 04:15:08 -07:00
/* WARNING: Non-standard export! Not part of any extension, or exposed in the
* alcFunctions list.
*/
2018-11-16 06:45:24 -08:00
extern "C" AL_API const ALchar* AL_APIENTRY alsoft_get_version(void)
{
2018-11-16 06:45:24 -08:00
const char *spoof{getenv("ALSOFT_SPOOF_VERSION")};
if(spoof && spoof[0] != '\0') return spoof;
return ALSOFT_VERSION;
}
#define DO_UPDATEPROPS() do { \
if(!context->DeferUpdates.load(std::memory_order_acquire)) \
2018-11-16 06:45:24 -08:00
UpdateContextProps(context.get()); \
else \
context->PropsClean.clear(std::memory_order_release); \
} while(0)
2010-03-19 14:34:18 -07:00
AL_API ALvoid AL_APIENTRY alEnable(ALenum capability)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2013-10-07 08:50:17 -07:00
switch(capability)
2007-11-13 18:02:18 -08:00
{
2013-10-07 08:50:17 -07:00
case AL_SOURCE_DISTANCE_MODEL:
context->SourceDistanceModel = AL_TRUE;
DO_UPDATEPROPS();
2013-10-07 08:50:17 -07:00
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid enable property 0x%04x", capability);
2007-11-13 18:02:18 -08:00
}
}
2010-03-19 14:34:18 -07:00
AL_API ALvoid AL_APIENTRY alDisable(ALenum capability)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2013-10-07 08:50:17 -07:00
switch(capability)
2007-11-13 18:02:18 -08:00
{
2013-10-07 08:50:17 -07:00
case AL_SOURCE_DISTANCE_MODEL:
context->SourceDistanceModel = AL_FALSE;
DO_UPDATEPROPS();
2013-10-07 08:50:17 -07:00
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid disable property 0x%04x", capability);
2007-11-13 18:02:18 -08:00
}
}
2010-03-19 14:34:18 -07:00
AL_API ALboolean AL_APIENTRY alIsEnabled(ALenum capability)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return AL_FALSE;
2007-11-13 18:02:18 -08:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALboolean value{AL_FALSE};
2013-10-07 08:50:17 -07:00
switch(capability)
2007-11-13 18:02:18 -08:00
{
2013-10-07 08:50:17 -07:00
case AL_SOURCE_DISTANCE_MODEL:
value = context->SourceDistanceModel;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid is enabled property 0x%04x", capability);
2007-11-13 18:02:18 -08:00
}
return value;
}
2010-03-19 14:34:18 -07:00
AL_API ALboolean AL_APIENTRY alGetBoolean(ALenum pname)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return AL_FALSE;
2007-11-13 18:02:18 -08:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALboolean value{AL_FALSE};
2013-10-07 08:50:17 -07:00
switch(pname)
2009-08-16 15:09:36 -07:00
{
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_FACTOR:
if(context->DopplerFactor != 0.0f)
value = AL_TRUE;
break;
case AL_DOPPLER_VELOCITY:
if(context->DopplerVelocity != 0.0f)
value = AL_TRUE;
break;
case AL_DISTANCE_MODEL:
if(context->mDistanceModel == DistanceModel::Default)
2013-10-07 08:50:17 -07:00
value = AL_TRUE;
break;
case AL_SPEED_OF_SOUND:
if(context->SpeedOfSound != 0.0f)
value = AL_TRUE;
break;
case AL_DEFERRED_UPDATES_SOFT:
if(context->DeferUpdates.load(std::memory_order_acquire))
value = AL_TRUE;
2013-10-07 08:50:17 -07:00
break;
case AL_GAIN_LIMIT_SOFT:
if(GAIN_MIX_MAX/context->GainBoost != 0.0f)
value = AL_TRUE;
break;
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
/* Always non-0. */
value = AL_TRUE;
break;
case AL_DEFAULT_RESAMPLER_SOFT:
value = ResamplerDefault ? AL_TRUE : AL_FALSE;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid boolean property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
return value;
}
2010-03-19 14:34:18 -07:00
AL_API ALdouble AL_APIENTRY alGetDouble(ALenum pname)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return 0.0;
2009-08-16 15:09:36 -07:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALdouble value{0.0};
2013-10-07 08:50:17 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_FACTOR:
value = (ALdouble)context->DopplerFactor;
break;
2007-11-13 18:02:18 -08:00
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_VELOCITY:
value = (ALdouble)context->DopplerVelocity;
break;
2007-11-13 18:02:18 -08:00
2013-10-07 08:50:17 -07:00
case AL_DISTANCE_MODEL:
value = (ALdouble)context->mDistanceModel;
2013-10-07 08:50:17 -07:00
break;
2007-11-13 18:02:18 -08:00
2013-10-07 08:50:17 -07:00
case AL_SPEED_OF_SOUND:
value = (ALdouble)context->SpeedOfSound;
break;
2007-11-13 18:02:18 -08:00
2013-10-07 08:50:17 -07:00
case AL_DEFERRED_UPDATES_SOFT:
if(context->DeferUpdates.load(std::memory_order_acquire))
value = (ALdouble)AL_TRUE;
2013-10-07 08:50:17 -07:00
break;
case AL_GAIN_LIMIT_SOFT:
value = (ALdouble)GAIN_MIX_MAX/context->GainBoost;
break;
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
value = (ALdouble)(ResamplerMax + 1);
break;
case AL_DEFAULT_RESAMPLER_SOFT:
value = (ALdouble)ResamplerDefault;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid double property 0x%04x", pname);
2009-08-16 15:09:36 -07:00
}
return value;
}
2010-03-19 14:34:18 -07:00
AL_API ALfloat AL_APIENTRY alGetFloat(ALenum pname)
2009-08-16 15:09:36 -07:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return 0.0f;
2009-08-16 15:09:36 -07:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALfloat value{0.0f};
2013-10-07 08:50:17 -07:00
switch(pname)
2009-08-16 15:09:36 -07:00
{
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_FACTOR:
value = context->DopplerFactor;
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_VELOCITY:
value = context->DopplerVelocity;
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_DISTANCE_MODEL:
value = (ALfloat)context->mDistanceModel;
2013-10-07 08:50:17 -07:00
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_SPEED_OF_SOUND:
value = context->SpeedOfSound;
break;
2007-11-13 18:02:18 -08:00
2013-10-07 08:50:17 -07:00
case AL_DEFERRED_UPDATES_SOFT:
if(context->DeferUpdates.load(std::memory_order_acquire))
value = (ALfloat)AL_TRUE;
2013-10-07 08:50:17 -07:00
break;
case AL_GAIN_LIMIT_SOFT:
value = GAIN_MIX_MAX/context->GainBoost;
break;
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
value = (ALfloat)(ResamplerMax + 1);
break;
case AL_DEFAULT_RESAMPLER_SOFT:
value = (ALfloat)ResamplerDefault;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid float property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
2009-08-16 15:09:36 -07:00
return value;
}
2010-03-19 14:34:18 -07:00
AL_API ALint AL_APIENTRY alGetInteger(ALenum pname)
2009-08-16 15:09:36 -07:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return 0;
2009-08-16 15:09:36 -07:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALint value{0};
2013-10-07 08:50:17 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_FACTOR:
value = (ALint)context->DopplerFactor;
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_DOPPLER_VELOCITY:
value = (ALint)context->DopplerVelocity;
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_DISTANCE_MODEL:
value = (ALint)context->mDistanceModel;
2013-10-07 08:50:17 -07:00
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_SPEED_OF_SOUND:
value = (ALint)context->SpeedOfSound;
break;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
case AL_DEFERRED_UPDATES_SOFT:
if(context->DeferUpdates.load(std::memory_order_acquire))
value = (ALint)AL_TRUE;
2013-10-07 08:50:17 -07:00
break;
case AL_GAIN_LIMIT_SOFT:
value = (ALint)(GAIN_MIX_MAX/context->GainBoost);
break;
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
value = ResamplerMax + 1;
break;
case AL_DEFAULT_RESAMPLER_SOFT:
value = ResamplerDefault;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid integer property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
return value;
}
2018-11-16 06:45:24 -08:00
extern "C" AL_API ALint64SOFT AL_APIENTRY alGetInteger64SOFT(ALenum pname)
2013-11-28 01:53:05 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return 0;
2013-11-28 01:53:05 -08:00
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
ALint64SOFT value{0};
2013-11-28 01:53:05 -08:00
switch(pname)
{
case AL_DOPPLER_FACTOR:
value = (ALint64SOFT)context->DopplerFactor;
break;
case AL_DOPPLER_VELOCITY:
value = (ALint64SOFT)context->DopplerVelocity;
break;
case AL_DISTANCE_MODEL:
value = (ALint64SOFT)context->mDistanceModel;
2013-11-28 01:53:05 -08:00
break;
case AL_SPEED_OF_SOUND:
value = (ALint64SOFT)context->SpeedOfSound;
break;
case AL_DEFERRED_UPDATES_SOFT:
if(context->DeferUpdates.load(std::memory_order_acquire))
value = (ALint64SOFT)AL_TRUE;
2013-11-28 01:53:05 -08:00
break;
case AL_GAIN_LIMIT_SOFT:
value = (ALint64SOFT)(GAIN_MIX_MAX/context->GainBoost);
break;
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
value = (ALint64SOFT)(ResamplerMax + 1);
break;
case AL_DEFAULT_RESAMPLER_SOFT:
value = (ALint64SOFT)ResamplerDefault;
break;
2013-11-28 01:53:05 -08:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid integer64 property 0x%04x", pname);
2013-11-28 01:53:05 -08:00
}
return value;
}
AL_API void* AL_APIENTRY alGetPointerSOFT(ALenum pname)
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return nullptr;
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
2018-11-16 06:45:24 -08:00
void *value{nullptr};
switch(pname)
{
case AL_EVENT_CALLBACK_FUNCTION_SOFT:
2018-11-16 06:45:24 -08:00
value = reinterpret_cast<void*>(context->EventCb);
break;
case AL_EVENT_CALLBACK_USER_PARAM_SOFT:
value = context->EventParam;
break;
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid pointer property 0x%04x", pname);
}
return value;
}
2018-01-24 18:30:32 -08:00
AL_API ALvoid AL_APIENTRY alGetBooleanv(ALenum pname, ALboolean *values)
{
if(values)
{
switch(pname)
{
2018-01-24 18:30:32 -08:00
case AL_DOPPLER_FACTOR:
case AL_DOPPLER_VELOCITY:
case AL_DISTANCE_MODEL:
case AL_SPEED_OF_SOUND:
case AL_DEFERRED_UPDATES_SOFT:
case AL_GAIN_LIMIT_SOFT:
case AL_NUM_RESAMPLERS_SOFT:
case AL_DEFAULT_RESAMPLER_SOFT:
values[0] = alGetBoolean(pname);
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
{
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid boolean-vector property 0x%04x", pname);
}
}
2018-01-24 18:30:32 -08:00
AL_API ALvoid AL_APIENTRY alGetDoublev(ALenum pname, ALdouble *values)
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
case AL_DOPPLER_FACTOR:
case AL_DOPPLER_VELOCITY:
case AL_DISTANCE_MODEL:
case AL_SPEED_OF_SOUND:
case AL_DEFERRED_UPDATES_SOFT:
case AL_GAIN_LIMIT_SOFT:
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
case AL_DEFAULT_RESAMPLER_SOFT:
2018-01-24 18:30:32 -08:00
values[0] = alGetDouble(pname);
2011-07-11 00:51:18 -07:00
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
2011-07-11 00:51:18 -07:00
{
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid double-vector property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
}
2018-01-24 18:30:32 -08:00
AL_API ALvoid AL_APIENTRY alGetFloatv(ALenum pname, ALfloat *values)
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
case AL_DOPPLER_FACTOR:
case AL_DOPPLER_VELOCITY:
case AL_DISTANCE_MODEL:
case AL_SPEED_OF_SOUND:
case AL_DEFERRED_UPDATES_SOFT:
case AL_GAIN_LIMIT_SOFT:
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
case AL_DEFAULT_RESAMPLER_SOFT:
2018-01-24 18:30:32 -08:00
values[0] = alGetFloat(pname);
2011-07-11 00:51:18 -07:00
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
2011-07-11 00:51:18 -07:00
{
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid float-vector property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
}
2018-01-24 18:30:32 -08:00
AL_API ALvoid AL_APIENTRY alGetIntegerv(ALenum pname, ALint *values)
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
case AL_DOPPLER_FACTOR:
case AL_DOPPLER_VELOCITY:
case AL_DISTANCE_MODEL:
case AL_SPEED_OF_SOUND:
case AL_DEFERRED_UPDATES_SOFT:
case AL_GAIN_LIMIT_SOFT:
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
case AL_DEFAULT_RESAMPLER_SOFT:
2018-01-24 18:30:32 -08:00
values[0] = alGetInteger(pname);
2011-07-11 00:51:18 -07:00
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
2011-07-11 00:51:18 -07:00
{
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid integer-vector property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
}
2018-11-16 06:45:24 -08:00
extern "C" AL_API void AL_APIENTRY alGetInteger64vSOFT(ALenum pname, ALint64SOFT *values)
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
switch(pname)
2007-11-13 18:02:18 -08:00
{
2009-08-16 15:09:36 -07:00
case AL_DOPPLER_FACTOR:
case AL_DOPPLER_VELOCITY:
case AL_DISTANCE_MODEL:
case AL_SPEED_OF_SOUND:
case AL_DEFERRED_UPDATES_SOFT:
case AL_GAIN_LIMIT_SOFT:
2017-04-21 04:15:08 -07:00
case AL_NUM_RESAMPLERS_SOFT:
case AL_DEFAULT_RESAMPLER_SOFT:
2018-01-24 18:30:32 -08:00
values[0] = alGetInteger64SOFT(pname);
2011-07-11 00:51:18 -07:00
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
2011-07-11 00:51:18 -07:00
{
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid integer64-vector property 0x%04x", pname);
2007-11-13 18:02:18 -08:00
}
}
2018-01-24 18:30:32 -08:00
AL_API void AL_APIENTRY alGetPointervSOFT(ALenum pname, void **values)
2013-11-28 01:53:05 -08:00
{
if(values)
{
switch(pname)
{
2018-01-24 18:30:32 -08:00
case AL_EVENT_CALLBACK_FUNCTION_SOFT:
case AL_EVENT_CALLBACK_USER_PARAM_SOFT:
values[0] = alGetPointerSOFT(pname);
2013-11-28 01:53:05 -08:00
return;
}
}
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2013-11-28 01:53:05 -08:00
if(!values)
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "NULL pointer");
else switch(pname)
2013-11-28 01:53:05 -08:00
{
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid pointer-vector property 0x%04x", pname);
2013-11-28 01:53:05 -08:00
}
}
2010-03-19 14:34:18 -07:00
AL_API const ALchar* AL_APIENTRY alGetString(ALenum pname)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return nullptr;
2007-11-13 18:02:18 -08:00
2018-11-16 06:45:24 -08:00
const ALchar *value{nullptr};
2013-10-07 08:50:17 -07:00
switch(pname)
{
2013-10-07 08:50:17 -07:00
case AL_VENDOR:
value = alVendor;
break;
2013-10-07 08:50:17 -07:00
case AL_VERSION:
value = alVersion;
break;
2013-10-07 08:50:17 -07:00
case AL_RENDERER:
value = alRenderer;
break;
2013-10-07 08:50:17 -07:00
case AL_EXTENSIONS:
value = context->ExtensionList;
break;
2013-10-07 08:50:17 -07:00
case AL_NO_ERROR:
value = alNoError;
break;
2013-10-07 08:50:17 -07:00
case AL_INVALID_NAME:
value = alErrInvalidName;
break;
2013-10-07 08:50:17 -07:00
case AL_INVALID_ENUM:
value = alErrInvalidEnum;
break;
2013-10-07 08:50:17 -07:00
case AL_INVALID_VALUE:
value = alErrInvalidValue;
break;
2013-10-07 08:50:17 -07:00
case AL_INVALID_OPERATION:
value = alErrInvalidOp;
break;
2013-10-07 08:50:17 -07:00
case AL_OUT_OF_MEMORY:
value = alErrOutOfMemory;
break;
2013-10-07 08:50:17 -07:00
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid string property 0x%04x", pname);
}
2007-11-13 18:02:18 -08:00
return value;
}
2010-03-19 14:34:18 -07:00
AL_API ALvoid AL_APIENTRY alDopplerFactor(ALfloat value)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
2018-11-19 09:51:29 -08:00
if(!(value >= 0.0f && std::isfinite(value)))
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Doppler factor %f out of range", value);
else
{
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
context->DopplerFactor = value;
DO_UPDATEPROPS();
}
2007-11-13 18:02:18 -08:00
}
2010-03-19 14:34:18 -07:00
AL_API ALvoid AL_APIENTRY alDopplerVelocity(ALfloat value)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
2018-11-26 18:19:58 -08:00
if((context->EnabledEvts.load(std::memory_order_relaxed)&EventType_Deprecated))
{
2018-11-16 06:45:24 -08:00
static constexpr ALCchar msg[] =
"alDopplerVelocity is deprecated in AL1.1, use alSpeedOfSound";
2018-01-27 11:50:06 -08:00
const ALsizei msglen = (ALsizei)strlen(msg);
std::lock_guard<std::mutex> _{context->EventCbLock};
ALbitfieldSOFT enabledevts{context->EnabledEvts.load(std::memory_order_relaxed)};
2018-01-28 16:58:41 -08:00
if((enabledevts&EventType_Deprecated) && context->EventCb)
2018-01-27 11:50:06 -08:00
(*context->EventCb)(AL_EVENT_TYPE_DEPRECATED_SOFT, 0, 0, msglen, msg,
context->EventParam);
}
2018-11-19 09:51:29 -08:00
if(!(value >= 0.0f && std::isfinite(value)))
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Doppler velocity %f out of range", value);
else
{
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
context->DopplerVelocity = value;
DO_UPDATEPROPS();
}
2007-11-13 18:02:18 -08:00
}
AL_API ALvoid AL_APIENTRY alSpeedOfSound(ALfloat value)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2007-11-13 18:02:18 -08:00
2018-11-19 09:51:29 -08:00
if(!(value > 0.0f && std::isfinite(value)))
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Speed of sound %f out of range", value);
else
{
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
context->SpeedOfSound = value;
DO_UPDATEPROPS();
}
2007-11-13 18:02:18 -08:00
}
2010-03-19 14:34:18 -07:00
AL_API ALvoid AL_APIENTRY alDistanceModel(ALenum value)
2007-11-13 18:02:18 -08:00
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2009-08-16 15:09:36 -07:00
2013-10-07 08:50:17 -07:00
if(!(value == AL_INVERSE_DISTANCE || value == AL_INVERSE_DISTANCE_CLAMPED ||
value == AL_LINEAR_DISTANCE || value == AL_LINEAR_DISTANCE_CLAMPED ||
value == AL_EXPONENT_DISTANCE || value == AL_EXPONENT_DISTANCE_CLAMPED ||
value == AL_NONE))
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Distance model 0x%04x out of range", value);
else
{
2018-11-26 21:37:58 -08:00
std::lock_guard<std::mutex> _{context->PropLock};
context->mDistanceModel = static_cast<DistanceModel>(value);
if(!context->SourceDistanceModel)
DO_UPDATEPROPS();
}
2007-11-13 18:02:18 -08:00
}
AL_API ALvoid AL_APIENTRY alDeferUpdatesSOFT(void)
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2018-11-16 06:45:24 -08:00
ALCcontext_DeferUpdates(context.get());
}
AL_API ALvoid AL_APIENTRY alProcessUpdatesSOFT(void)
{
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return;
2018-11-16 06:45:24 -08:00
ALCcontext_ProcessUpdates(context.get());
}
2017-04-21 04:15:08 -07:00
AL_API const ALchar* AL_APIENTRY alGetStringiSOFT(ALenum pname, ALsizei index)
{
const char *ResamplerNames[] = {
alPointResampler, alLinearResampler,
alCubicResampler, alBSinc12Resampler,
alBSinc24Resampler,
2017-04-21 04:15:08 -07:00
};
static_assert(COUNTOF(ResamplerNames) == ResamplerMax+1, "Incorrect ResamplerNames list");
2018-11-16 06:45:24 -08:00
ContextRef context{GetContextRef()};
if(UNLIKELY(!context)) return nullptr;
2017-04-21 04:15:08 -07:00
2018-11-16 06:45:24 -08:00
const ALchar *value{nullptr};
2017-04-21 04:15:08 -07:00
switch(pname)
{
case AL_RESAMPLER_NAME_SOFT:
if(index < 0 || (size_t)index >= COUNTOF(ResamplerNames))
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Resampler name index %d out of range",
index);
else
value = ResamplerNames[index];
2017-04-21 04:15:08 -07:00
break;
default:
2018-11-16 06:45:24 -08:00
alSetError(context.get(), AL_INVALID_VALUE, "Invalid string indexed property");
2017-04-21 04:15:08 -07:00
}
return value;
}
void UpdateContextProps(ALCcontext *context)
{
/* Get an unused proprty container, or allocate a new one as needed. */
ALcontextProps *props{context->FreeContextProps.load(std::memory_order_acquire)};
if(!props)
2018-11-16 06:45:24 -08:00
props = static_cast<ALcontextProps*>(al_calloc(16, sizeof(*props)));
else
{
ALcontextProps *next;
do {
2018-11-19 03:21:58 -08:00
next = props->next.load(std::memory_order_relaxed);
} while(context->FreeContextProps.compare_exchange_weak(props, next,
std::memory_order_seq_cst, std::memory_order_acquire) == 0);
}
/* Copy in current property values. */
props->MetersPerUnit = context->MetersPerUnit;
props->DopplerFactor = context->DopplerFactor;
props->DopplerVelocity = context->DopplerVelocity;
props->SpeedOfSound = context->SpeedOfSound;
props->SourceDistanceModel = context->SourceDistanceModel;
props->mDistanceModel = context->mDistanceModel;
/* Set the new container for updating internal parameters. */
2018-11-19 03:21:58 -08:00
props = context->Update.exchange(props, std::memory_order_acq_rel);
if(props)
{
/* If there was an unused update container, put it back in the
* freelist.
*/
AtomicReplaceHead(context->FreeContextProps, props);
}
}