openal-soft/al/listener.cpp

453 lines
13 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 "listener.h"
2019-07-28 21:29:59 -07:00
2018-11-19 09:51:29 -08:00
#include <cmath>
2019-07-28 21:29:59 -07:00
#include <mutex>
2019-08-05 22:40:19 -07:00
#include "AL/al.h"
#include "AL/alc.h"
2019-07-28 21:29:59 -07:00
#include "AL/efx.h"
2018-11-19 09:51:29 -08:00
#include "alc/context.h"
2019-07-28 21:29:59 -07:00
#include "almalloc.h"
#include "atomic.h"
2020-12-21 18:00:43 -08:00
#include "core/except.h"
2019-07-28 21:29:59 -07:00
#include "opthelpers.h"
2007-11-13 18:02:18 -08:00
#define DO_UPDATEPROPS() do { \
2019-07-30 09:05:54 -07:00
if(!context->mDeferUpdates.load(std::memory_order_acquire)) \
2018-11-20 11:17:54 -08:00
UpdateListenerProps(context.get()); \
else \
listener.mPropsDirty.set(std::memory_order_release); \
} while(0)
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alListenerf(ALenum param, ALfloat value)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2007-11-13 18:02:18 -08:00
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
2013-10-07 09:04:52 -07:00
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_GAIN:
2018-11-19 09:51:29 -08:00
if(!(value >= 0.0f && std::isfinite(value)))
SETERR_RETURN(context, AL_INVALID_VALUE,, "Listener gain out of range");
2018-11-20 11:17:54 -08:00
listener.Gain = value;
DO_UPDATEPROPS();
2013-10-07 09:04:52 -07:00
break;
2007-11-13 18:02:18 -08:00
2013-10-07 09:04:52 -07:00
case AL_METERS_PER_UNIT:
2017-08-21 00:27:52 -07:00
if(!(value >= AL_MIN_METERS_PER_UNIT && value <= AL_MAX_METERS_PER_UNIT))
SETERR_RETURN(context, AL_INVALID_VALUE,, "Listener meters per unit out of range");
listener.mMetersPerUnit = value;
DO_UPDATEPROPS();
break;
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener float property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alListener3f(ALenum param, ALfloat value1, ALfloat value2, ALfloat value3)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2007-11-13 18:02:18 -08:00
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
2013-10-07 09:04:52 -07:00
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
2018-11-19 09:51:29 -08:00
if(!(std::isfinite(value1) && std::isfinite(value2) && std::isfinite(value3)))
SETERR_RETURN(context, AL_INVALID_VALUE,, "Listener position out of range");
2018-11-20 11:17:54 -08:00
listener.Position[0] = value1;
listener.Position[1] = value2;
listener.Position[2] = value3;
DO_UPDATEPROPS();
2013-10-07 09:04:52 -07:00
break;
case AL_VELOCITY:
2018-11-19 09:51:29 -08:00
if(!(std::isfinite(value1) && std::isfinite(value2) && std::isfinite(value3)))
SETERR_RETURN(context, AL_INVALID_VALUE,, "Listener velocity out of range");
2018-11-20 11:17:54 -08:00
listener.Velocity[0] = value1;
listener.Velocity[1] = value2;
listener.Velocity[2] = value3;
DO_UPDATEPROPS();
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener 3-float property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alListenerfv(ALenum param, const ALfloat *values)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_GAIN:
case AL_METERS_PER_UNIT:
alListenerf(param, values[0]);
return;
case AL_POSITION:
case AL_VELOCITY:
alListener3f(param, values[0], values[1], values[2]);
return;
}
}
2007-11-13 18:02:18 -08:00
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!values) SETERR_RETURN(context, AL_INVALID_VALUE,, "NULL pointer");
2013-10-07 09:04:52 -07:00
switch(param)
{
2013-10-07 09:04:52 -07:00
case AL_ORIENTATION:
2018-11-19 09:51:29 -08:00
if(!(std::isfinite(values[0]) && std::isfinite(values[1]) && std::isfinite(values[2]) &&
std::isfinite(values[3]) && std::isfinite(values[4]) && std::isfinite(values[5])))
SETERR_RETURN(context, AL_INVALID_VALUE,, "Listener orientation out of range");
2013-10-07 09:04:52 -07:00
/* AT then UP */
listener.OrientAt[0] = values[0];
listener.OrientAt[1] = values[1];
listener.OrientAt[2] = values[2];
listener.OrientUp[0] = values[3];
listener.OrientUp[1] = values[4];
listener.OrientUp[2] = values[5];
DO_UPDATEPROPS();
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener float-vector property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alListeneri(ALenum param, ALint /*value*/)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
std::lock_guard<std::mutex> _{context->mPropLock};
2013-10-07 09:04:52 -07:00
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener integer property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
AL_API void AL_APIENTRY alListener3i(ALenum param, ALint value1, ALint value2, ALint value3)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
case AL_VELOCITY:
alListener3f(param, static_cast<ALfloat>(value1), static_cast<ALfloat>(value2), static_cast<ALfloat>(value3));
2013-10-07 09:04:52 -07:00
return;
}
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2007-11-13 18:02:18 -08:00
2019-07-30 09:05:54 -07:00
std::lock_guard<std::mutex> _{context->mPropLock};
2013-10-07 09:04:52 -07:00
switch(param)
{
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener 3-integer property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
AL_API void AL_APIENTRY alListeneriv(ALenum param, const ALint *values)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
if(values)
2007-11-13 18:02:18 -08:00
{
ALfloat fvals[6];
switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
case AL_VELOCITY:
alListener3f(param, static_cast<ALfloat>(values[0]), static_cast<ALfloat>(values[1]), static_cast<ALfloat>(values[2]));
2013-10-07 09:04:52 -07:00
return;
case AL_ORIENTATION:
fvals[0] = static_cast<ALfloat>(values[0]);
fvals[1] = static_cast<ALfloat>(values[1]);
fvals[2] = static_cast<ALfloat>(values[2]);
fvals[3] = static_cast<ALfloat>(values[3]);
fvals[4] = static_cast<ALfloat>(values[4]);
fvals[5] = static_cast<ALfloat>(values[5]);
2013-10-07 09:04:52 -07:00
alListenerfv(param, fvals);
return;
}
}
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2007-11-13 18:02:18 -08:00
2019-07-30 09:05:54 -07:00
std::lock_guard<std::mutex> _{context->mPropLock};
if(!values)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
{
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener integer-vector property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alGetListenerf(ALenum param, ALfloat *value)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!value)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_GAIN:
2018-11-20 11:17:54 -08:00
*value = listener.Gain;
2013-10-07 09:04:52 -07:00
break;
2007-11-13 18:02:18 -08:00
2013-10-07 09:04:52 -07:00
case AL_METERS_PER_UNIT:
*value = listener.mMetersPerUnit;
2013-10-07 09:04:52 -07:00
break;
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener float property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alGetListener3f(ALenum param, ALfloat *value1, ALfloat *value2, ALfloat *value3)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!value1 || !value2 || !value3)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
2018-11-20 11:17:54 -08:00
*value1 = listener.Position[0];
*value2 = listener.Position[1];
*value3 = listener.Position[2];
2013-10-07 09:04:52 -07:00
break;
case AL_VELOCITY:
2018-11-20 11:17:54 -08:00
*value1 = listener.Velocity[0];
*value2 = listener.Velocity[1];
*value3 = listener.Velocity[2];
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener 3-float property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alGetListenerfv(ALenum param, ALfloat *values)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
switch(param)
{
2013-10-07 09:04:52 -07:00
case AL_GAIN:
case AL_METERS_PER_UNIT:
alGetListenerf(param, values);
return;
case AL_POSITION:
case AL_VELOCITY:
alGetListener3f(param, values+0, values+1, values+2);
return;
}
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!values)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_ORIENTATION:
// AT then UP
values[0] = listener.OrientAt[0];
values[1] = listener.OrientAt[1];
values[2] = listener.OrientAt[2];
values[3] = listener.OrientUp[0];
values[4] = listener.OrientUp[1];
values[5] = listener.OrientUp[2];
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener float-vector property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
2020-04-28 14:48:12 -07:00
AL_API void AL_APIENTRY alGetListeneri(ALenum param, ALint *value)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
std::lock_guard<std::mutex> _{context->mPropLock};
if(!value)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
default:
context->setError(AL_INVALID_ENUM, "Invalid listener integer property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
AL_API void AL_APIENTRY alGetListener3i(ALenum param, ALint *value1, ALint *value2, ALint *value3)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!value1 || !value2 || !value3)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
*value1 = static_cast<ALint>(listener.Position[0]);
*value2 = static_cast<ALint>(listener.Position[1]);
*value3 = static_cast<ALint>(listener.Position[2]);
2013-10-07 09:04:52 -07:00
break;
case AL_VELOCITY:
*value1 = static_cast<ALint>(listener.Velocity[0]);
*value2 = static_cast<ALint>(listener.Velocity[1]);
*value3 = static_cast<ALint>(listener.Velocity[2]);
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener 3-integer property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
2007-11-13 18:02:18 -08:00
AL_API void AL_APIENTRY alGetListeneriv(ALenum param, ALint* values)
START_API_FUNC
2007-11-13 18:02:18 -08:00
{
switch(param)
{
2013-10-07 09:04:52 -07:00
case AL_POSITION:
case AL_VELOCITY:
alGetListener3i(param, values+0, values+1, values+2);
return;
}
2018-11-20 11:17:54 -08:00
ContextRef context{GetContextRef()};
if UNLIKELY(!context) return;
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
std::lock_guard<std::mutex> _{context->mPropLock};
if(!values)
context->setError(AL_INVALID_VALUE, "NULL pointer");
else switch(param)
2007-11-13 18:02:18 -08:00
{
2013-10-07 09:04:52 -07:00
case AL_ORIENTATION:
// AT then UP
values[0] = static_cast<ALint>(listener.OrientAt[0]);
values[1] = static_cast<ALint>(listener.OrientAt[1]);
values[2] = static_cast<ALint>(listener.OrientAt[2]);
values[3] = static_cast<ALint>(listener.OrientUp[0]);
values[4] = static_cast<ALint>(listener.OrientUp[1]);
values[5] = static_cast<ALint>(listener.OrientUp[2]);
2013-10-07 09:04:52 -07:00
break;
default:
context->setError(AL_INVALID_ENUM, "Invalid listener integer-vector property");
2007-11-13 18:02:18 -08:00
}
}
END_API_FUNC
void UpdateListenerProps(ALCcontext *context)
{
/* Get an unused proprty container, or allocate a new one as needed. */
ListenerProps *props{context->mFreeListenerProps.load(std::memory_order_acquire)};
if(!props)
props = new ListenerProps{};
else
{
ListenerProps *next;
do {
2018-11-19 03:21:58 -08:00
next = props->next.load(std::memory_order_relaxed);
2019-07-30 09:05:54 -07:00
} while(context->mFreeListenerProps.compare_exchange_weak(props, next,
2018-11-19 03:21:58 -08:00
std::memory_order_seq_cst, std::memory_order_acquire) == 0);
}
/* Copy in current property values. */
2019-07-30 09:05:54 -07:00
ALlistener &listener = context->mListener;
props->Position = listener.Position;
props->Velocity = listener.Velocity;
props->OrientAt = listener.OrientAt;
props->OrientUp = listener.OrientUp;
2018-11-20 11:17:54 -08:00
props->Gain = listener.Gain;
props->MetersPerUnit = listener.mMetersPerUnit;
/* Set the new container for updating internal parameters. */
props = context->mParams.ListenerUpdate.exchange(props, std::memory_order_acq_rel);
if(props)
{
/* If there was an unused update container, put it back in the
* freelist.
*/
2019-07-30 09:05:54 -07:00
AtomicReplaceHead(context->mFreeListenerProps, props);
}
}