2007-11-13 18:02:18 -08:00
|
|
|
/**
|
|
|
|
* OpenAL cross platform audio library
|
|
|
|
* Copyright (C) 1999-2007 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., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
* Or go to http://www.gnu.org/copyleft/lgpl.html
|
|
|
|
*/
|
|
|
|
|
2008-01-16 14:09:04 -08:00
|
|
|
#include "config.h"
|
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <float.h>
|
2012-08-20 12:22:00 -07:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
#include "AL/al.h"
|
|
|
|
#include "AL/alc.h"
|
2012-08-20 12:22:00 -07:00
|
|
|
#include "alMain.h"
|
2007-11-13 18:02:18 -08:00
|
|
|
#include "alError.h"
|
|
|
|
#include "alSource.h"
|
2007-12-31 01:09:57 -08:00
|
|
|
#include "alBuffer.h"
|
|
|
|
#include "alThunk.h"
|
2008-01-16 14:01:24 -08:00
|
|
|
#include "alAuxEffectSlot.h"
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2014-05-11 03:52:22 -07:00
|
|
|
#include "threads.h"
|
|
|
|
|
2010-11-28 17:37:14 -08:00
|
|
|
|
2012-02-12 08:45:19 -08:00
|
|
|
enum Resampler DefaultResampler = LinearResampler;
|
|
|
|
const ALsizei ResamplerPadding[ResamplerMax] = {
|
2010-11-28 17:37:14 -08:00
|
|
|
0, /* Point */
|
|
|
|
1, /* Linear */
|
|
|
|
2, /* Cubic */
|
|
|
|
};
|
2012-02-12 08:45:19 -08:00
|
|
|
const ALsizei ResamplerPrePadding[ResamplerMax] = {
|
2010-11-28 17:37:14 -08:00
|
|
|
0, /* Point */
|
|
|
|
0, /* Linear */
|
|
|
|
1, /* Cubic */
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-11-04 13:44:46 -08:00
|
|
|
extern inline struct ALsource *LookupSource(ALCcontext *context, ALuint id);
|
|
|
|
extern inline struct ALsource *RemoveSource(ALCcontext *context, ALuint id);
|
|
|
|
|
2010-03-26 00:41:27 -07:00
|
|
|
static ALvoid InitSourceParams(ALsource *Source);
|
2012-08-21 16:01:11 -07:00
|
|
|
static ALint64 GetSourceOffset(const ALsource *Source);
|
|
|
|
static ALdouble GetSourceSecOffset(const ALsource *Source);
|
|
|
|
static ALvoid GetSourceOffsets(const ALsource *Source, ALenum name, ALdouble *offsets, ALdouble updateLen);
|
2011-10-03 10:07:50 -07:00
|
|
|
static ALint GetSampleOffset(ALsource *Source);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
typedef enum SrcFloatProp {
|
2012-12-05 18:23:21 -08:00
|
|
|
sfPitch = AL_PITCH,
|
|
|
|
sfGain = AL_GAIN,
|
|
|
|
sfMinGain = AL_MIN_GAIN,
|
|
|
|
sfMaxGain = AL_MAX_GAIN,
|
|
|
|
sfMaxDistance = AL_MAX_DISTANCE,
|
|
|
|
sfRolloffFactor = AL_ROLLOFF_FACTOR,
|
|
|
|
sfDopplerFactor = AL_DOPPLER_FACTOR,
|
|
|
|
sfConeOuterGain = AL_CONE_OUTER_GAIN,
|
|
|
|
sfSecOffset = AL_SEC_OFFSET,
|
|
|
|
sfSampleOffset = AL_SAMPLE_OFFSET,
|
|
|
|
sfByteOffset = AL_BYTE_OFFSET,
|
|
|
|
sfConeInnerAngle = AL_CONE_INNER_ANGLE,
|
|
|
|
sfConeOuterAngle = AL_CONE_OUTER_ANGLE,
|
|
|
|
sfRefDistance = AL_REFERENCE_DISTANCE,
|
|
|
|
|
|
|
|
sfPosition = AL_POSITION,
|
|
|
|
sfVelocity = AL_VELOCITY,
|
|
|
|
sfDirection = AL_DIRECTION,
|
|
|
|
|
|
|
|
sfSourceRelative = AL_SOURCE_RELATIVE,
|
|
|
|
sfLooping = AL_LOOPING,
|
|
|
|
sfBuffer = AL_BUFFER,
|
|
|
|
sfSourceState = AL_SOURCE_STATE,
|
|
|
|
sfBuffersQueued = AL_BUFFERS_QUEUED,
|
|
|
|
sfBuffersProcessed = AL_BUFFERS_PROCESSED,
|
|
|
|
sfSourceType = AL_SOURCE_TYPE,
|
2012-12-05 09:22:38 -08:00
|
|
|
|
|
|
|
/* ALC_EXT_EFX */
|
2012-12-05 18:23:21 -08:00
|
|
|
sfConeOuterGainHF = AL_CONE_OUTER_GAINHF,
|
|
|
|
sfAirAbsorptionFactor = AL_AIR_ABSORPTION_FACTOR,
|
|
|
|
sfRoomRolloffFactor = AL_ROOM_ROLLOFF_FACTOR,
|
|
|
|
sfDirectFilterGainHFAuto = AL_DIRECT_FILTER_GAINHF_AUTO,
|
|
|
|
sfAuxSendFilterGainAuto = AL_AUXILIARY_SEND_FILTER_GAIN_AUTO,
|
|
|
|
sfAuxSendFilterGainHFAuto = AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO,
|
2012-12-05 09:22:38 -08:00
|
|
|
|
2012-12-05 13:48:33 -08:00
|
|
|
/* AL_SOFT_direct_channels */
|
2012-12-05 18:23:21 -08:00
|
|
|
sfDirectChannelsSOFT = AL_DIRECT_CHANNELS_SOFT,
|
2012-12-05 13:48:33 -08:00
|
|
|
|
|
|
|
/* AL_EXT_source_distance_model */
|
2012-12-05 18:23:21 -08:00
|
|
|
sfDistanceModel = AL_DISTANCE_MODEL,
|
2012-12-05 13:48:33 -08:00
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
/* AL_SOFT_buffer_sub_data / AL_SOFT_buffer_samples */
|
2012-12-05 18:23:21 -08:00
|
|
|
sfSampleRWOffsetsSOFT = AL_SAMPLE_RW_OFFSETS_SOFT,
|
|
|
|
sfByteRWOffsetsSOFT = AL_BYTE_RW_OFFSETS_SOFT,
|
2012-12-05 09:22:38 -08:00
|
|
|
|
|
|
|
/* AL_SOFT_source_latency */
|
2012-12-05 18:23:21 -08:00
|
|
|
sfSecOffsetLatencySOFT = AL_SEC_OFFSET_LATENCY_SOFT,
|
2012-12-05 09:22:38 -08:00
|
|
|
} SrcFloatProp;
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
typedef enum SrcIntProp {
|
2012-12-05 18:23:21 -08:00
|
|
|
siMaxDistance = AL_MAX_DISTANCE,
|
|
|
|
siRolloffFactor = AL_ROLLOFF_FACTOR,
|
|
|
|
siRefDistance = AL_REFERENCE_DISTANCE,
|
|
|
|
siSourceRelative = AL_SOURCE_RELATIVE,
|
|
|
|
siConeInnerAngle = AL_CONE_INNER_ANGLE,
|
|
|
|
siConeOuterAngle = AL_CONE_OUTER_ANGLE,
|
|
|
|
siLooping = AL_LOOPING,
|
|
|
|
siBuffer = AL_BUFFER,
|
|
|
|
siSourceState = AL_SOURCE_STATE,
|
|
|
|
siBuffersQueued = AL_BUFFERS_QUEUED,
|
|
|
|
siBuffersProcessed = AL_BUFFERS_PROCESSED,
|
|
|
|
siSourceType = AL_SOURCE_TYPE,
|
|
|
|
siSecOffset = AL_SEC_OFFSET,
|
|
|
|
siSampleOffset = AL_SAMPLE_OFFSET,
|
|
|
|
siByteOffset = AL_BYTE_OFFSET,
|
|
|
|
siDopplerFactor = AL_DOPPLER_FACTOR,
|
|
|
|
siPosition = AL_POSITION,
|
|
|
|
siVelocity = AL_VELOCITY,
|
|
|
|
siDirection = AL_DIRECTION,
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
/* ALC_EXT_EFX */
|
2012-12-05 18:23:21 -08:00
|
|
|
siDirectFilterGainHFAuto = AL_DIRECT_FILTER_GAINHF_AUTO,
|
|
|
|
siAuxSendFilterGainAutio = AL_AUXILIARY_SEND_FILTER_GAIN_AUTO,
|
|
|
|
siAuxSendFilterGainHFAuto = AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO,
|
|
|
|
siDirectFilter = AL_DIRECT_FILTER,
|
|
|
|
siAuxSendFilter = AL_AUXILIARY_SEND_FILTER,
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
/* AL_SOFT_direct_channels */
|
2012-12-05 18:23:21 -08:00
|
|
|
siDirectChannelsSOFT = AL_DIRECT_CHANNELS_SOFT,
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
/* AL_EXT_source_distance_model */
|
2012-12-05 18:23:21 -08:00
|
|
|
siDistanceModel = AL_DISTANCE_MODEL,
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
/* AL_SOFT_buffer_sub_data / AL_SOFT_buffer_samples */
|
2012-12-05 18:23:21 -08:00
|
|
|
siSampleRWOffsetsSOFT = AL_SAMPLE_RW_OFFSETS_SOFT,
|
|
|
|
siByteRWOffsetsSOFT = AL_BYTE_RW_OFFSETS_SOFT,
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
/* AL_SOFT_source_latency */
|
2012-12-05 18:23:21 -08:00
|
|
|
siSampleOffsetLatencySOFT = AL_SAMPLE_OFFSET_LATENCY_SOFT,
|
2012-12-05 09:55:05 -08:00
|
|
|
} SrcIntProp;
|
|
|
|
|
2013-10-07 09:54:35 -07:00
|
|
|
static ALboolean SetSourcefv(ALsource *Source, ALCcontext *Context, SrcFloatProp prop, const ALfloat *values);
|
|
|
|
static ALboolean SetSourceiv(ALsource *Source, ALCcontext *Context, SrcIntProp prop, const ALint *values);
|
|
|
|
static ALboolean SetSourcei64v(ALsource *Source, ALCcontext *Context, SrcIntProp prop, const ALint64SOFT *values);
|
2012-08-28 22:16:55 -07:00
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
static ALboolean GetSourcedv(ALsource *Source, ALCcontext *Context, SrcFloatProp prop, ALdouble *values);
|
|
|
|
static ALboolean GetSourceiv(ALsource *Source, ALCcontext *Context, SrcIntProp prop, ALint *values);
|
|
|
|
static ALboolean GetSourcei64v(ALsource *Source, ALCcontext *Context, SrcIntProp prop, ALint64 *values);
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-12-05 20:51:25 -08:00
|
|
|
static ALint FloatValsByProp(ALenum prop)
|
|
|
|
{
|
|
|
|
if(prop != (ALenum)((SrcFloatProp)prop))
|
|
|
|
return 0;
|
|
|
|
switch((SrcFloatProp)prop)
|
|
|
|
{
|
|
|
|
case sfPitch:
|
|
|
|
case sfGain:
|
|
|
|
case sfMinGain:
|
|
|
|
case sfMaxGain:
|
|
|
|
case sfMaxDistance:
|
|
|
|
case sfRolloffFactor:
|
|
|
|
case sfDopplerFactor:
|
|
|
|
case sfConeOuterGain:
|
|
|
|
case sfSecOffset:
|
|
|
|
case sfSampleOffset:
|
|
|
|
case sfByteOffset:
|
|
|
|
case sfConeInnerAngle:
|
|
|
|
case sfConeOuterAngle:
|
|
|
|
case sfRefDistance:
|
|
|
|
case sfConeOuterGainHF:
|
|
|
|
case sfAirAbsorptionFactor:
|
|
|
|
case sfRoomRolloffFactor:
|
|
|
|
case sfDirectFilterGainHFAuto:
|
|
|
|
case sfAuxSendFilterGainAuto:
|
|
|
|
case sfAuxSendFilterGainHFAuto:
|
|
|
|
case sfDirectChannelsSOFT:
|
|
|
|
case sfDistanceModel:
|
|
|
|
case sfSourceRelative:
|
|
|
|
case sfLooping:
|
|
|
|
case sfBuffer:
|
|
|
|
case sfSourceState:
|
|
|
|
case sfBuffersQueued:
|
|
|
|
case sfBuffersProcessed:
|
|
|
|
case sfSourceType:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case sfSampleRWOffsetsSOFT:
|
|
|
|
case sfByteRWOffsetsSOFT:
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
case sfPosition:
|
|
|
|
case sfVelocity:
|
|
|
|
case sfDirection:
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
case sfSecOffsetLatencySOFT:
|
|
|
|
break; /* Double only */
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static ALint DoubleValsByProp(ALenum prop)
|
|
|
|
{
|
|
|
|
if(prop != (ALenum)((SrcFloatProp)prop))
|
|
|
|
return 0;
|
|
|
|
switch((SrcFloatProp)prop)
|
|
|
|
{
|
|
|
|
case sfPitch:
|
|
|
|
case sfGain:
|
|
|
|
case sfMinGain:
|
|
|
|
case sfMaxGain:
|
|
|
|
case sfMaxDistance:
|
|
|
|
case sfRolloffFactor:
|
|
|
|
case sfDopplerFactor:
|
|
|
|
case sfConeOuterGain:
|
|
|
|
case sfSecOffset:
|
|
|
|
case sfSampleOffset:
|
|
|
|
case sfByteOffset:
|
|
|
|
case sfConeInnerAngle:
|
|
|
|
case sfConeOuterAngle:
|
|
|
|
case sfRefDistance:
|
|
|
|
case sfConeOuterGainHF:
|
|
|
|
case sfAirAbsorptionFactor:
|
|
|
|
case sfRoomRolloffFactor:
|
|
|
|
case sfDirectFilterGainHFAuto:
|
|
|
|
case sfAuxSendFilterGainAuto:
|
|
|
|
case sfAuxSendFilterGainHFAuto:
|
|
|
|
case sfDirectChannelsSOFT:
|
|
|
|
case sfDistanceModel:
|
|
|
|
case sfSourceRelative:
|
|
|
|
case sfLooping:
|
|
|
|
case sfBuffer:
|
|
|
|
case sfSourceState:
|
|
|
|
case sfBuffersQueued:
|
|
|
|
case sfBuffersProcessed:
|
|
|
|
case sfSourceType:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case sfSampleRWOffsetsSOFT:
|
|
|
|
case sfByteRWOffsetsSOFT:
|
|
|
|
case sfSecOffsetLatencySOFT:
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
case sfPosition:
|
|
|
|
case sfVelocity:
|
|
|
|
case sfDirection:
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2012-08-28 22:16:55 -07:00
|
|
|
|
2012-12-05 19:58:01 -08:00
|
|
|
static ALint IntValsByProp(ALenum prop)
|
|
|
|
{
|
|
|
|
if(prop != (ALenum)((SrcIntProp)prop))
|
|
|
|
return 0;
|
|
|
|
switch((SrcIntProp)prop)
|
|
|
|
{
|
|
|
|
case siMaxDistance:
|
|
|
|
case siRolloffFactor:
|
|
|
|
case siRefDistance:
|
|
|
|
case siSourceRelative:
|
|
|
|
case siConeInnerAngle:
|
|
|
|
case siConeOuterAngle:
|
|
|
|
case siLooping:
|
|
|
|
case siBuffer:
|
|
|
|
case siSourceState:
|
|
|
|
case siBuffersQueued:
|
|
|
|
case siBuffersProcessed:
|
|
|
|
case siSourceType:
|
|
|
|
case siSecOffset:
|
|
|
|
case siSampleOffset:
|
|
|
|
case siByteOffset:
|
|
|
|
case siDopplerFactor:
|
|
|
|
case siDirectFilterGainHFAuto:
|
|
|
|
case siAuxSendFilterGainAutio:
|
|
|
|
case siAuxSendFilterGainHFAuto:
|
|
|
|
case siDirectFilter:
|
|
|
|
case siDirectChannelsSOFT:
|
|
|
|
case siDistanceModel:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case siSampleRWOffsetsSOFT:
|
|
|
|
case siByteRWOffsetsSOFT:
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
case siPosition:
|
|
|
|
case siVelocity:
|
|
|
|
case siDirection:
|
|
|
|
case siAuxSendFilter:
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
case siSampleOffsetLatencySOFT:
|
|
|
|
break; /* i64 only */
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static ALint Int64ValsByProp(ALenum prop)
|
|
|
|
{
|
|
|
|
if(prop != (ALenum)((SrcIntProp)prop))
|
|
|
|
return 0;
|
|
|
|
switch((SrcIntProp)prop)
|
|
|
|
{
|
|
|
|
case siMaxDistance:
|
|
|
|
case siRolloffFactor:
|
|
|
|
case siRefDistance:
|
|
|
|
case siSourceRelative:
|
|
|
|
case siConeInnerAngle:
|
|
|
|
case siConeOuterAngle:
|
|
|
|
case siLooping:
|
|
|
|
case siBuffer:
|
|
|
|
case siSourceState:
|
|
|
|
case siBuffersQueued:
|
|
|
|
case siBuffersProcessed:
|
|
|
|
case siSourceType:
|
|
|
|
case siSecOffset:
|
|
|
|
case siSampleOffset:
|
|
|
|
case siByteOffset:
|
|
|
|
case siDopplerFactor:
|
|
|
|
case siDirectFilterGainHFAuto:
|
|
|
|
case siAuxSendFilterGainAutio:
|
|
|
|
case siAuxSendFilterGainHFAuto:
|
|
|
|
case siDirectFilter:
|
|
|
|
case siDirectChannelsSOFT:
|
|
|
|
case siDistanceModel:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case siSampleRWOffsetsSOFT:
|
|
|
|
case siByteRWOffsetsSOFT:
|
|
|
|
case siSampleOffsetLatencySOFT:
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
case siPosition:
|
|
|
|
case siVelocity:
|
|
|
|
case siDirection:
|
|
|
|
case siAuxSendFilter:
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-28 22:16:55 -07:00
|
|
|
#define CHECKVAL(x) do { \
|
|
|
|
if(!(x)) \
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_VALUE, AL_FALSE); \
|
2012-08-28 22:16:55 -07:00
|
|
|
} while(0)
|
|
|
|
|
2013-10-07 09:54:35 -07:00
|
|
|
static ALboolean SetSourcefv(ALsource *Source, ALCcontext *Context, SrcFloatProp prop, const ALfloat *values)
|
2012-08-28 22:16:55 -07:00
|
|
|
{
|
2012-12-05 09:22:38 -08:00
|
|
|
ALint ival;
|
|
|
|
|
|
|
|
switch(prop)
|
2012-08-28 22:16:55 -07:00
|
|
|
{
|
|
|
|
case AL_PITCH:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
Source->Pitch = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 360.0f);
|
|
|
|
|
|
|
|
Source->InnerAngle = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 360.0f);
|
|
|
|
|
|
|
|
Source->OuterAngle = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_GAIN:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
Source->Gain = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
Source->MaxDistance = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
Source->RollOffFactor = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
Source->RefDistance = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_MIN_GAIN:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 1.0f);
|
|
|
|
|
|
|
|
Source->MinGain = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_MAX_GAIN:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 1.0f);
|
|
|
|
|
|
|
|
Source->MaxGain = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_CONE_OUTER_GAIN:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 1.0f);
|
|
|
|
|
|
|
|
Source->OuterGain = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_CONE_OUTER_GAINHF:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 1.0f);
|
|
|
|
|
|
|
|
Source->OuterGainHF = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_AIR_ABSORPTION_FACTOR:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 10.0f);
|
|
|
|
|
|
|
|
Source->AirAbsorptionFactor = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_ROOM_ROLLOFF_FACTOR:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 10.0f);
|
|
|
|
|
|
|
|
Source->RoomRolloffFactor = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DOPPLER_FACTOR:
|
|
|
|
CHECKVAL(*values >= 0.0f && *values <= 1.0f);
|
|
|
|
|
|
|
|
Source->DopplerFactor = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_SEC_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
|
|
|
case AL_BYTE_OFFSET:
|
|
|
|
CHECKVAL(*values >= 0.0f);
|
|
|
|
|
|
|
|
LockContext(Context);
|
2012-12-05 09:22:38 -08:00
|
|
|
Source->OffsetType = prop;
|
2012-08-28 22:16:55 -07:00
|
|
|
Source->Offset = *values;
|
|
|
|
|
|
|
|
if((Source->state == AL_PLAYING || Source->state == AL_PAUSED) &&
|
|
|
|
!Context->DeferUpdates)
|
|
|
|
{
|
|
|
|
if(ApplyOffset(Source) == AL_FALSE)
|
|
|
|
{
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_VALUE, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
|
2012-11-01 18:35:20 -07:00
|
|
|
case AL_SEC_OFFSET_LATENCY_SOFT:
|
|
|
|
/* Query only */
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-11-01 18:35:20 -07:00
|
|
|
|
|
|
|
|
2012-08-28 22:16:55 -07:00
|
|
|
case AL_POSITION:
|
|
|
|
CHECKVAL(isfinite(values[0]) && isfinite(values[1]) && isfinite(values[2]));
|
|
|
|
|
|
|
|
LockContext(Context);
|
|
|
|
Source->Position[0] = values[0];
|
|
|
|
Source->Position[1] = values[1];
|
|
|
|
Source->Position[2] = values[2];
|
|
|
|
UnlockContext(Context);
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_VELOCITY:
|
|
|
|
CHECKVAL(isfinite(values[0]) && isfinite(values[1]) && isfinite(values[2]));
|
|
|
|
|
|
|
|
LockContext(Context);
|
|
|
|
Source->Velocity[0] = values[0];
|
|
|
|
Source->Velocity[1] = values[1];
|
|
|
|
Source->Velocity[2] = values[2];
|
|
|
|
UnlockContext(Context);
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DIRECTION:
|
|
|
|
CHECKVAL(isfinite(values[0]) && isfinite(values[1]) && isfinite(values[2]));
|
|
|
|
|
|
|
|
LockContext(Context);
|
|
|
|
Source->Orientation[0] = values[0];
|
|
|
|
Source->Orientation[1] = values[1];
|
|
|
|
Source->Orientation[2] = values[2];
|
|
|
|
UnlockContext(Context);
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case sfSampleRWOffsetsSOFT:
|
|
|
|
case sfByteRWOffsetsSOFT:
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-12-05 09:22:38 -08:00
|
|
|
|
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case sfSourceRelative:
|
|
|
|
case sfLooping:
|
|
|
|
case sfSourceState:
|
|
|
|
case sfSourceType:
|
|
|
|
case sfDistanceModel:
|
|
|
|
case sfDirectFilterGainHFAuto:
|
|
|
|
case sfAuxSendFilterGainAuto:
|
|
|
|
case sfAuxSendFilterGainHFAuto:
|
|
|
|
case sfDirectChannelsSOFT:
|
2012-12-05 09:22:38 -08:00
|
|
|
ival = (ALint)values[0];
|
2013-06-05 01:52:49 -07:00
|
|
|
return SetSourceiv(Source, Context, (SrcIntProp)prop, &ival);
|
2012-12-05 09:22:38 -08:00
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case sfBuffer:
|
|
|
|
case sfBuffersQueued:
|
|
|
|
case sfBuffersProcessed:
|
2012-12-05 09:22:38 -08:00
|
|
|
ival = (ALint)((ALuint)values[0]);
|
2013-06-05 01:52:49 -07:00
|
|
|
return SetSourceiv(Source, Context, (SrcIntProp)prop, &ival);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:54:35 -07:00
|
|
|
static ALboolean SetSourceiv(ALsource *Source, ALCcontext *Context, SrcIntProp prop, const ALint *values)
|
2012-08-28 22:16:55 -07:00
|
|
|
{
|
|
|
|
ALCdevice *device = Context->Device;
|
|
|
|
ALbuffer *buffer = NULL;
|
|
|
|
ALfilter *filter = NULL;
|
|
|
|
ALeffectslot *slot = NULL;
|
|
|
|
ALbufferlistitem *oldlist;
|
|
|
|
ALfloat fvals[3];
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
switch(prop)
|
2012-08-28 22:16:55 -07:00
|
|
|
{
|
|
|
|
case AL_SOURCE_RELATIVE:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->HeadRelative = (ALboolean)*values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_LOOPING:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->Looping = (ALboolean)*values;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_BUFFER:
|
|
|
|
CHECKVAL(*values == 0 || (buffer=LookupBuffer(device, *values)) != NULL);
|
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteLock(&Source->queue_lock);
|
2012-08-28 22:16:55 -07:00
|
|
|
if(!(Source->state == AL_STOPPED || Source->state == AL_INITIAL))
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if(buffer != NULL)
|
|
|
|
{
|
|
|
|
ALbufferlistitem *BufferListItem;
|
|
|
|
|
|
|
|
/* Add the selected buffer to a one-item queue */
|
|
|
|
BufferListItem = malloc(sizeof(ALbufferlistitem));
|
|
|
|
BufferListItem->buffer = buffer;
|
|
|
|
BufferListItem->next = NULL;
|
|
|
|
BufferListItem->prev = NULL;
|
|
|
|
IncrementRef(&buffer->ref);
|
|
|
|
|
2014-05-10 03:33:41 -07:00
|
|
|
/* Source is now Static */
|
|
|
|
Source->SourceType = AL_STATIC;
|
2012-08-28 22:16:55 -07:00
|
|
|
oldlist = ExchangePtr((XchgPtr*)&Source->queue, BufferListItem);
|
|
|
|
|
|
|
|
ReadLock(&buffer->lock);
|
|
|
|
Source->NumChannels = ChannelsFromFmt(buffer->FmtChannels);
|
|
|
|
Source->SampleSize = BytesFromFmt(buffer->FmtType);
|
|
|
|
ReadUnlock(&buffer->lock);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Source is now Undetermined */
|
|
|
|
Source->SourceType = AL_UNDETERMINED;
|
|
|
|
oldlist = ExchangePtr((XchgPtr*)&Source->queue, NULL);
|
|
|
|
}
|
2014-05-10 03:21:40 -07:00
|
|
|
Source->current_buffer = Source->queue;
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&Source->queue_lock);
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
/* Delete all elements in the previous queue */
|
|
|
|
while(oldlist != NULL)
|
|
|
|
{
|
|
|
|
ALbufferlistitem *temp = oldlist;
|
|
|
|
oldlist = temp->next;
|
|
|
|
|
|
|
|
if(temp->buffer)
|
|
|
|
DecrementRef(&temp->buffer->ref);
|
|
|
|
free(temp);
|
|
|
|
}
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case siSourceState:
|
|
|
|
case siSourceType:
|
|
|
|
case siBuffersQueued:
|
|
|
|
case siBuffersProcessed:
|
2012-08-28 22:16:55 -07:00
|
|
|
/* Query only */
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_SEC_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
|
|
|
case AL_BYTE_OFFSET:
|
|
|
|
CHECKVAL(*values >= 0);
|
|
|
|
|
|
|
|
LockContext(Context);
|
2012-12-05 09:55:05 -08:00
|
|
|
Source->OffsetType = prop;
|
2012-08-28 22:16:55 -07:00
|
|
|
Source->Offset = *values;
|
|
|
|
|
|
|
|
if((Source->state == AL_PLAYING || Source->state == AL_PAUSED) &&
|
|
|
|
!Context->DeferUpdates)
|
|
|
|
{
|
|
|
|
if(ApplyOffset(Source) == AL_FALSE)
|
|
|
|
{
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_VALUE, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-12-05 09:55:05 -08:00
|
|
|
|
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case siSampleRWOffsetsSOFT:
|
|
|
|
case siByteRWOffsetsSOFT:
|
2012-12-05 09:55:05 -08:00
|
|
|
/* Query only */
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-12-05 09:55:05 -08:00
|
|
|
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DIRECT_FILTER:
|
|
|
|
CHECKVAL(*values == 0 || (filter=LookupFilter(device, *values)) != NULL);
|
|
|
|
|
|
|
|
LockContext(Context);
|
|
|
|
if(!filter)
|
|
|
|
{
|
2014-05-11 01:36:18 -07:00
|
|
|
Source->Direct.Gain = 1.0f;
|
|
|
|
Source->Direct.GainHF = 1.0f;
|
2014-05-14 01:24:18 -07:00
|
|
|
Source->Direct.HFReference = LOWPASSFREQREF;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Direct.GainLF = 1.0f;
|
|
|
|
Source->Direct.LFReference = HIGHPASSFREQREF;
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-05-11 01:36:18 -07:00
|
|
|
Source->Direct.Gain = filter->Gain;
|
|
|
|
Source->Direct.GainHF = filter->GainHF;
|
2014-05-14 01:24:18 -07:00
|
|
|
Source->Direct.HFReference = filter->HFReference;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Direct.GainLF = filter->GainLF;
|
|
|
|
Source->Direct.LFReference = filter->LFReference;
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
UnlockContext(Context);
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DIRECT_FILTER_GAINHF_AUTO:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->DryGainHFAuto = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->WetGainAuto = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->WetGainHFAuto = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DIRECT_CHANNELS_SOFT:
|
|
|
|
CHECKVAL(*values == AL_FALSE || *values == AL_TRUE);
|
|
|
|
|
|
|
|
Source->DirectChannels = *values;
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_DISTANCE_MODEL:
|
|
|
|
CHECKVAL(*values == AL_NONE ||
|
|
|
|
*values == AL_INVERSE_DISTANCE ||
|
|
|
|
*values == AL_INVERSE_DISTANCE_CLAMPED ||
|
|
|
|
*values == AL_LINEAR_DISTANCE ||
|
|
|
|
*values == AL_LINEAR_DISTANCE_CLAMPED ||
|
|
|
|
*values == AL_EXPONENT_DISTANCE ||
|
|
|
|
*values == AL_EXPONENT_DISTANCE_CLAMPED);
|
|
|
|
|
|
|
|
Source->DistanceModel = *values;
|
|
|
|
if(Context->SourceDistanceModel)
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
|
|
|
|
case AL_AUXILIARY_SEND_FILTER:
|
|
|
|
LockContext(Context);
|
|
|
|
if(!((ALuint)values[1] < device->NumAuxSends &&
|
|
|
|
(values[0] == 0 || (slot=LookupEffectSlot(Context, values[0])) != NULL) &&
|
|
|
|
(values[2] == 0 || (filter=LookupFilter(device, values[2])) != NULL)))
|
|
|
|
{
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_VALUE, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Add refcount on the new slot, and release the previous slot */
|
|
|
|
if(slot) IncrementRef(&slot->ref);
|
|
|
|
slot = ExchangePtr((XchgPtr*)&Source->Send[values[1]].Slot, slot);
|
|
|
|
if(slot) DecrementRef(&slot->ref);
|
|
|
|
|
|
|
|
if(!filter)
|
|
|
|
{
|
|
|
|
/* Disable filter */
|
|
|
|
Source->Send[values[1]].Gain = 1.0f;
|
|
|
|
Source->Send[values[1]].GainHF = 1.0f;
|
2014-05-14 01:24:18 -07:00
|
|
|
Source->Send[values[1]].HFReference = LOWPASSFREQREF;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Send[values[1]].GainLF = 1.0f;
|
|
|
|
Source->Send[values[1]].LFReference = HIGHPASSFREQREF;
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Source->Send[values[1]].Gain = filter->Gain;
|
|
|
|
Source->Send[values[1]].GainHF = filter->GainHF;
|
2014-05-14 01:24:18 -07:00
|
|
|
Source->Send[values[1]].HFReference = filter->HFReference;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Send[values[1]].GainLF = filter->GainLF;
|
|
|
|
Source->Send[values[1]].LFReference = filter->LFReference;
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
|
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
2012-12-05 18:23:21 -08:00
|
|
|
case siDopplerFactor:
|
2012-08-28 22:16:55 -07:00
|
|
|
fvals[0] = (ALfloat)*values;
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourcefv(Source, Context, (int)prop, fvals);
|
2012-08-28 22:16:55 -07:00
|
|
|
|
|
|
|
case AL_POSITION:
|
|
|
|
case AL_VELOCITY:
|
|
|
|
case AL_DIRECTION:
|
|
|
|
fvals[0] = (ALfloat)values[0];
|
|
|
|
fvals[1] = (ALfloat)values[1];
|
|
|
|
fvals[2] = (ALfloat)values[2];
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourcefv(Source, Context, (int)prop, fvals);
|
2012-12-06 09:03:48 -08:00
|
|
|
|
|
|
|
case siSampleOffsetLatencySOFT:
|
|
|
|
/* i64 only */
|
|
|
|
break;
|
2012-10-14 01:36:46 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-10-14 01:36:46 -07:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:54:35 -07:00
|
|
|
static ALboolean SetSourcei64v(ALsource *Source, ALCcontext *Context, SrcIntProp prop, const ALint64SOFT *values)
|
2012-10-14 01:36:46 -07:00
|
|
|
{
|
|
|
|
ALfloat fvals[3];
|
|
|
|
ALint ivals[3];
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
switch(prop)
|
2012-10-14 01:36:46 -07:00
|
|
|
{
|
2012-12-05 18:23:21 -08:00
|
|
|
case siSampleRWOffsetsSOFT:
|
|
|
|
case siByteRWOffsetsSOFT:
|
|
|
|
case siSampleOffsetLatencySOFT:
|
2012-11-01 18:35:20 -07:00
|
|
|
/* Query only */
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_OPERATION, AL_FALSE);
|
2012-11-01 18:35:20 -07:00
|
|
|
|
|
|
|
|
2012-10-14 01:36:46 -07:00
|
|
|
/* 1x int */
|
|
|
|
case AL_SOURCE_RELATIVE:
|
|
|
|
case AL_LOOPING:
|
|
|
|
case AL_SOURCE_STATE:
|
|
|
|
case AL_BYTE_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
2012-12-05 18:23:21 -08:00
|
|
|
case siSourceType:
|
|
|
|
case siBuffersQueued:
|
|
|
|
case siBuffersProcessed:
|
2012-10-14 01:36:46 -07:00
|
|
|
case AL_DIRECT_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_DIRECT_CHANNELS_SOFT:
|
|
|
|
case AL_DISTANCE_MODEL:
|
|
|
|
CHECKVAL(*values <= INT_MAX && *values >= INT_MIN);
|
|
|
|
|
2012-10-25 14:46:27 -07:00
|
|
|
ivals[0] = (ALint)*values;
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourceiv(Source, Context, (int)prop, ivals);
|
2012-10-14 01:36:46 -07:00
|
|
|
|
|
|
|
/* 1x uint */
|
|
|
|
case AL_BUFFER:
|
|
|
|
case AL_DIRECT_FILTER:
|
|
|
|
CHECKVAL(*values <= UINT_MAX && *values >= 0);
|
|
|
|
|
|
|
|
ivals[0] = (ALuint)*values;
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourceiv(Source, Context, (int)prop, ivals);
|
2012-10-14 01:36:46 -07:00
|
|
|
|
|
|
|
/* 3x uint */
|
|
|
|
case AL_AUXILIARY_SEND_FILTER:
|
|
|
|
CHECKVAL(values[0] <= UINT_MAX && values[0] >= 0 &&
|
|
|
|
values[1] <= UINT_MAX && values[1] >= 0 &&
|
|
|
|
values[2] <= UINT_MAX && values[2] >= 0);
|
|
|
|
|
|
|
|
ivals[0] = (ALuint)values[0];
|
|
|
|
ivals[1] = (ALuint)values[1];
|
|
|
|
ivals[2] = (ALuint)values[2];
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourceiv(Source, Context, (int)prop, ivals);
|
2012-10-14 01:36:46 -07:00
|
|
|
|
|
|
|
/* 1x float */
|
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
|
|
|
case AL_SEC_OFFSET:
|
2012-12-05 18:23:21 -08:00
|
|
|
case siDopplerFactor:
|
2012-10-14 01:36:46 -07:00
|
|
|
fvals[0] = (ALfloat)*values;
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourcefv(Source, Context, (int)prop, fvals);
|
2012-10-14 01:36:46 -07:00
|
|
|
|
|
|
|
/* 3x float */
|
|
|
|
case AL_POSITION:
|
|
|
|
case AL_VELOCITY:
|
|
|
|
case AL_DIRECTION:
|
|
|
|
fvals[0] = (ALfloat)values[0];
|
|
|
|
fvals[1] = (ALfloat)values[1];
|
|
|
|
fvals[2] = (ALfloat)values[2];
|
2012-12-05 09:55:05 -08:00
|
|
|
return SetSourcefv(Source, Context, (int)prop, fvals);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-08-28 22:16:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#undef CHECKVAL
|
|
|
|
|
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
static ALboolean GetSourcedv(ALsource *Source, ALCcontext *Context, SrcFloatProp prop, ALdouble *values)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
|
|
|
ALdouble offsets[2];
|
|
|
|
ALdouble updateLen;
|
2013-10-07 09:54:35 -07:00
|
|
|
ALint ivals[3];
|
|
|
|
ALboolean err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
switch(prop)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
2012-12-05 08:27:02 -08:00
|
|
|
case AL_GAIN:
|
|
|
|
*values = Source->Gain;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-12-05 08:27:02 -08:00
|
|
|
|
2012-10-21 11:36:27 -07:00
|
|
|
case AL_PITCH:
|
|
|
|
*values = Source->Pitch;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-10-21 11:36:27 -07:00
|
|
|
|
2012-08-20 14:16:58 -07:00
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
*values = Source->MaxDistance;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
*values = Source->RollOffFactor;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
|
|
|
*values = Source->RefDistance;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
*values = Source->InnerAngle;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
*values = Source->OuterAngle;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-10-21 11:36:27 -07:00
|
|
|
case AL_MIN_GAIN:
|
|
|
|
*values = Source->MinGain;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-10-21 11:36:27 -07:00
|
|
|
|
|
|
|
case AL_MAX_GAIN:
|
|
|
|
*values = Source->MaxGain;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-10-21 11:36:27 -07:00
|
|
|
|
|
|
|
case AL_CONE_OUTER_GAIN:
|
|
|
|
*values = Source->OuterGain;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-10-21 11:36:27 -07:00
|
|
|
|
2012-08-20 14:16:58 -07:00
|
|
|
case AL_SEC_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
|
|
|
case AL_BYTE_OFFSET:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2012-08-20 14:16:58 -07:00
|
|
|
LockContext(Context);
|
2014-05-10 05:07:13 -07:00
|
|
|
GetSourceOffsets(Source, prop, offsets, 0.0);
|
2012-08-20 14:16:58 -07:00
|
|
|
UnlockContext(Context);
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2012-08-20 14:16:58 -07:00
|
|
|
*values = offsets[0];
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-09-14 09:02:36 -07:00
|
|
|
case AL_CONE_OUTER_GAINHF:
|
|
|
|
*values = Source->OuterGainHF;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-09-14 09:02:36 -07:00
|
|
|
|
|
|
|
case AL_AIR_ABSORPTION_FACTOR:
|
|
|
|
*values = Source->AirAbsorptionFactor;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-09-14 09:02:36 -07:00
|
|
|
|
|
|
|
case AL_ROOM_ROLLOFF_FACTOR:
|
|
|
|
*values = Source->RoomRolloffFactor;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-09-14 09:02:36 -07:00
|
|
|
|
2012-08-20 14:16:58 -07:00
|
|
|
case AL_DOPPLER_FACTOR:
|
|
|
|
*values = Source->DopplerFactor;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SAMPLE_RW_OFFSETS_SOFT:
|
|
|
|
case AL_BYTE_RW_OFFSETS_SOFT:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2012-08-20 14:16:58 -07:00
|
|
|
LockContext(Context);
|
|
|
|
updateLen = (ALdouble)Context->Device->UpdateSize /
|
|
|
|
Context->Device->Frequency;
|
2012-12-05 09:22:38 -08:00
|
|
|
GetSourceOffsets(Source, prop, values, updateLen);
|
2012-08-20 14:16:58 -07:00
|
|
|
UnlockContext(Context);
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-08-20 15:57:27 -07:00
|
|
|
case AL_SEC_OFFSET_LATENCY_SOFT:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2012-08-20 15:57:27 -07:00
|
|
|
LockContext(Context);
|
|
|
|
values[0] = GetSourceSecOffset(Source);
|
2013-11-02 12:01:58 -07:00
|
|
|
values[1] = (ALdouble)ALCdevice_GetLatency(Context->Device) /
|
2012-08-20 15:57:27 -07:00
|
|
|
1000000000.0;
|
|
|
|
UnlockContext(Context);
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 15:57:27 -07:00
|
|
|
|
2012-08-20 14:16:58 -07:00
|
|
|
case AL_POSITION:
|
|
|
|
LockContext(Context);
|
|
|
|
values[0] = Source->Position[0];
|
|
|
|
values[1] = Source->Position[1];
|
|
|
|
values[2] = Source->Position[2];
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_VELOCITY:
|
|
|
|
LockContext(Context);
|
|
|
|
values[0] = Source->Velocity[0];
|
|
|
|
values[1] = Source->Velocity[1];
|
|
|
|
values[2] = Source->Velocity[2];
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_DIRECTION:
|
|
|
|
LockContext(Context);
|
|
|
|
values[0] = Source->Orientation[0];
|
|
|
|
values[1] = Source->Orientation[1];
|
|
|
|
values[2] = Source->Orientation[2];
|
|
|
|
UnlockContext(Context);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SOURCE_RELATIVE:
|
|
|
|
case AL_LOOPING:
|
|
|
|
case AL_BUFFER:
|
|
|
|
case AL_SOURCE_STATE:
|
|
|
|
case AL_BUFFERS_QUEUED:
|
|
|
|
case AL_BUFFERS_PROCESSED:
|
|
|
|
case AL_SOURCE_TYPE:
|
|
|
|
case AL_DIRECT_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_DIRECT_CHANNELS_SOFT:
|
|
|
|
case AL_DISTANCE_MODEL:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourceiv(Source, Context, (int)prop, ivals)) != AL_FALSE)
|
2012-12-05 09:22:38 -08:00
|
|
|
*values = (ALdouble)ivals[0];
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:22:38 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
static ALboolean GetSourceiv(ALsource *Source, ALCcontext *Context, SrcIntProp prop, ALint *values)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
|
|
|
ALbufferlistitem *BufferList;
|
|
|
|
ALdouble dvals[3];
|
2013-10-07 09:54:35 -07:00
|
|
|
ALboolean err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
switch(prop)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
|
|
|
case AL_SOURCE_RELATIVE:
|
|
|
|
*values = Source->HeadRelative;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_LOOPING:
|
|
|
|
*values = Source->Looping;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_BUFFER:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2014-05-10 03:21:40 -07:00
|
|
|
BufferList = (Source->SourceType == AL_STATIC) ? Source->queue :
|
|
|
|
Source->current_buffer;
|
|
|
|
*values = (BufferList && BufferList->buffer) ? BufferList->buffer->id : 0;
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SOURCE_STATE:
|
|
|
|
*values = Source->state;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_BUFFERS_QUEUED:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2014-05-10 03:33:41 -07:00
|
|
|
if(!(BufferList=Source->queue))
|
|
|
|
*values = 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ALsizei count = 0;
|
|
|
|
do {
|
|
|
|
++count;
|
|
|
|
} while((BufferList=BufferList->next) != NULL);
|
|
|
|
*values = count;
|
|
|
|
}
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_BUFFERS_PROCESSED:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2012-08-20 14:16:58 -07:00
|
|
|
if(Source->Looping || Source->SourceType != AL_STREAMING)
|
|
|
|
{
|
|
|
|
/* Buffers on a looping source are in a perpetual state of
|
|
|
|
* PENDING, so don't report any as PROCESSED */
|
|
|
|
*values = 0;
|
|
|
|
}
|
|
|
|
else
|
2014-05-10 03:21:40 -07:00
|
|
|
{
|
|
|
|
const ALbufferlistitem *BufferList = Source->queue;
|
|
|
|
ALsizei played = 0;
|
|
|
|
while(BufferList && BufferList != Source->current_buffer)
|
|
|
|
{
|
|
|
|
played++;
|
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
|
|
|
*values = played;
|
|
|
|
}
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SOURCE_TYPE:
|
|
|
|
*values = Source->SourceType;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_DIRECT_FILTER_GAINHF_AUTO:
|
|
|
|
*values = Source->DryGainHFAuto;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
|
|
|
|
*values = Source->WetGainAuto;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
|
|
|
|
*values = Source->WetGainHFAuto;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_DIRECT_CHANNELS_SOFT:
|
|
|
|
*values = Source->DirectChannels;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_DISTANCE_MODEL:
|
|
|
|
*values = Source->DistanceModel;
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
case AL_SEC_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
|
|
|
case AL_BYTE_OFFSET:
|
|
|
|
case AL_DOPPLER_FACTOR:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
*values = (ALint)dvals[0];
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SAMPLE_RW_OFFSETS_SOFT:
|
|
|
|
case AL_BYTE_RW_OFFSETS_SOFT:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
{
|
|
|
|
values[0] = (ALint)dvals[0];
|
|
|
|
values[1] = (ALint)dvals[1];
|
|
|
|
}
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_POSITION:
|
|
|
|
case AL_VELOCITY:
|
|
|
|
case AL_DIRECTION:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
{
|
|
|
|
values[0] = (ALint)dvals[0];
|
|
|
|
values[1] = (ALint)dvals[1];
|
|
|
|
values[2] = (ALint)dvals[2];
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case siSampleOffsetLatencySOFT:
|
2012-12-05 09:55:05 -08:00
|
|
|
/* i64 only */
|
2012-08-20 14:16:58 -07:00
|
|
|
break;
|
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case siDirectFilter:
|
|
|
|
case siAuxSendFilter:
|
2012-12-05 09:55:05 -08:00
|
|
|
/* ??? */
|
|
|
|
break;
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
static ALboolean GetSourcei64v(ALsource *Source, ALCcontext *Context, SrcIntProp prop, ALint64 *values)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
|
|
|
ALdouble dvals[3];
|
2013-10-07 09:54:35 -07:00
|
|
|
ALint ivals[3];
|
|
|
|
ALboolean err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
switch(prop)
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
|
|
|
case AL_SAMPLE_OFFSET_LATENCY_SOFT:
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2012-08-20 14:16:58 -07:00
|
|
|
LockContext(Context);
|
|
|
|
values[0] = GetSourceOffset(Source);
|
2013-11-02 12:01:58 -07:00
|
|
|
values[1] = ALCdevice_GetLatency(Context->Device);
|
2012-08-20 14:16:58 -07:00
|
|
|
UnlockContext(Context);
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2013-10-07 09:54:35 -07:00
|
|
|
return AL_TRUE;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_MAX_DISTANCE:
|
|
|
|
case AL_ROLLOFF_FACTOR:
|
|
|
|
case AL_REFERENCE_DISTANCE:
|
|
|
|
case AL_CONE_INNER_ANGLE:
|
|
|
|
case AL_CONE_OUTER_ANGLE:
|
|
|
|
case AL_SEC_OFFSET:
|
|
|
|
case AL_SAMPLE_OFFSET:
|
|
|
|
case AL_BYTE_OFFSET:
|
|
|
|
case AL_DOPPLER_FACTOR:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
*values = (ALint64)dvals[0];
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SAMPLE_RW_OFFSETS_SOFT:
|
|
|
|
case AL_BYTE_RW_OFFSETS_SOFT:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
{
|
|
|
|
values[0] = (ALint64)dvals[0];
|
|
|
|
values[1] = (ALint64)dvals[1];
|
|
|
|
}
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_POSITION:
|
|
|
|
case AL_VELOCITY:
|
|
|
|
case AL_DIRECTION:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourcedv(Source, Context, (int)prop, dvals)) != AL_FALSE)
|
2012-12-05 09:55:05 -08:00
|
|
|
{
|
|
|
|
values[0] = (ALint64)dvals[0];
|
|
|
|
values[1] = (ALint64)dvals[1];
|
|
|
|
values[2] = (ALint64)dvals[2];
|
|
|
|
}
|
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
case AL_SOURCE_RELATIVE:
|
|
|
|
case AL_LOOPING:
|
|
|
|
case AL_SOURCE_STATE:
|
|
|
|
case AL_BUFFERS_QUEUED:
|
|
|
|
case AL_BUFFERS_PROCESSED:
|
|
|
|
case AL_SOURCE_TYPE:
|
|
|
|
case AL_DIRECT_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAIN_AUTO:
|
|
|
|
case AL_AUXILIARY_SEND_FILTER_GAINHF_AUTO:
|
|
|
|
case AL_DIRECT_CHANNELS_SOFT:
|
|
|
|
case AL_DISTANCE_MODEL:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourceiv(Source, Context, (int)prop, ivals)) != AL_FALSE)
|
2012-12-07 18:43:13 -08:00
|
|
|
*values = ivals[0];
|
|
|
|
return err;
|
|
|
|
|
|
|
|
case siBuffer:
|
2012-12-05 18:23:21 -08:00
|
|
|
case siDirectFilter:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourceiv(Source, Context, (int)prop, ivals)) != AL_FALSE)
|
2012-12-07 18:43:13 -08:00
|
|
|
*values = ((ALuint*)ivals)[0];
|
|
|
|
return err;
|
|
|
|
|
2012-12-05 18:23:21 -08:00
|
|
|
case siAuxSendFilter:
|
2013-10-07 09:54:35 -07:00
|
|
|
if((err=GetSourceiv(Source, Context, (int)prop, ivals)) != AL_FALSE)
|
2012-12-07 18:43:13 -08:00
|
|
|
{
|
|
|
|
values[0] = ((ALuint*)ivals)[0];
|
|
|
|
values[1] = ((ALuint*)ivals)[1];
|
|
|
|
values[2] = ((ALuint*)ivals)[2];
|
|
|
|
}
|
2012-12-05 09:55:05 -08:00
|
|
|
return err;
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2012-12-05 09:55:05 -08:00
|
|
|
ERR("Unexpected property: 0x%04x\n", prop);
|
2013-10-07 09:54:35 -07:00
|
|
|
SET_ERROR_AND_RETURN_VALUE(Context, AL_INVALID_ENUM, AL_FALSE);
|
2012-08-20 14:16:58 -07:00
|
|
|
}
|
|
|
|
|
2010-05-01 19:59:41 -07:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alGenSources(ALsizei n, ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsizei cur = 0;
|
|
|
|
ALenum err;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
for(cur = 0;cur < n;cur++)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALsource *source = al_calloc(16, sizeof(ALsource));
|
|
|
|
if(!source)
|
2009-08-16 15:09:36 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
alDeleteSources(cur, sources);
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_OUT_OF_MEMORY, done);
|
|
|
|
}
|
|
|
|
InitSourceParams(source);
|
2010-09-21 15:12:08 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
err = NewThunkEntry(&source->id);
|
|
|
|
if(err == AL_NO_ERROR)
|
|
|
|
err = InsertUIntMapEntry(&context->SourceMap, source->id, source);
|
|
|
|
if(err != AL_NO_ERROR)
|
|
|
|
{
|
|
|
|
FreeThunkEntry(source->id);
|
|
|
|
memset(source, 0, sizeof(ALsource));
|
|
|
|
al_free(source);
|
2010-09-21 15:12:08 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
alDeleteSources(cur, sources);
|
|
|
|
SET_ERROR_AND_GOTO(context, err, done);
|
2009-08-16 15:09:36 -07:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
|
|
|
|
sources[cur] = source->id;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alDeleteSources(ALsizei n, const ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALbufferlistitem *BufferList;
|
|
|
|
ALsource *Source;
|
|
|
|
ALsizei i, j;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
2012-04-23 19:46:05 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
/* Check that all Sources are valid */
|
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
|
|
|
if(LookupSource(context, sources[i]) == NULL)
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
|
|
|
}
|
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
2014-03-18 19:56:25 -07:00
|
|
|
ALactivesource **srclist, **srclistend;
|
2012-04-23 19:46:05 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if((Source=RemoveSource(context, sources[i])) == NULL)
|
|
|
|
continue;
|
|
|
|
FreeThunkEntry(Source->id);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
LockContext(context);
|
|
|
|
srclist = context->ActiveSources;
|
|
|
|
srclistend = srclist + context->ActiveSourceCount;
|
|
|
|
while(srclist != srclistend)
|
2009-08-16 15:09:36 -07:00
|
|
|
{
|
2014-03-18 19:56:25 -07:00
|
|
|
if((*srclist)->Source == Source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2014-03-18 19:56:25 -07:00
|
|
|
ALactivesource *temp = *(--srclistend);
|
|
|
|
*srclistend = *srclist;
|
|
|
|
*srclist = temp;
|
|
|
|
--(context->ActiveSourceCount);
|
2013-10-07 09:24:50 -07:00
|
|
|
break;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
srclist++;
|
|
|
|
}
|
|
|
|
UnlockContext(context);
|
2010-11-06 14:07:30 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
while(Source->queue != NULL)
|
|
|
|
{
|
|
|
|
BufferList = Source->queue;
|
|
|
|
Source->queue = BufferList->next;
|
2010-09-21 16:54:33 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(BufferList->buffer != NULL)
|
|
|
|
DecrementRef(&BufferList->buffer->ref);
|
|
|
|
free(BufferList);
|
|
|
|
}
|
2010-09-21 16:54:33 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
for(j = 0;j < MAX_SENDS;++j)
|
|
|
|
{
|
|
|
|
if(Source->Send[j].Slot)
|
|
|
|
DecrementRef(&Source->Send[j].Slot->ref);
|
|
|
|
Source->Send[j].Slot = NULL;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
|
|
|
|
memset(Source, 0, sizeof(*Source));
|
|
|
|
al_free(Source);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALboolean AL_APIENTRY alIsSource(ALuint source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALboolean ret;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return AL_FALSE;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
ret = (LookupSource(context, source) ? AL_TRUE : AL_FALSE);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext_DecRef(context);
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
return ret;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcef(ALuint source, ALenum param, ALfloat value)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2011-09-11 00:47:31 -07:00
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(FloatValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourcefv(Source, Context, param, &value);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSource3f(ALuint source, ALenum param, ALfloat value1, ALfloat value2, ALfloat value3)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2011-09-11 00:47:31 -07:00
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(FloatValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALfloat fvals[3] = { value1, value2, value3 };
|
|
|
|
SetSourcefv(Source, Context, param, fvals);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcefv(ALuint source, ALenum param, const ALfloat *values)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2012-08-28 22:16:55 -07:00
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-08-28 22:16:55 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(FloatValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourcefv(Source, Context, param, values);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-13 00:56:39 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcedSOFT(ALuint source, ALenum param, ALdouble value)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(DoubleValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-10-13 00:56:39 -07:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALfloat fval = (ALfloat)value;
|
|
|
|
SetSourcefv(Source, Context, param, &fval);
|
2012-10-13 00:56:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API ALvoid AL_APIENTRY alSource3dSOFT(ALuint source, ALenum param, ALdouble value1, ALdouble value2, ALdouble value3)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(DoubleValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-10-13 00:56:39 -07:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALfloat fvals[3] = { (ALfloat)value1, (ALfloat)value2, (ALfloat)value3 };
|
|
|
|
SetSourcefv(Source, Context, param, fvals);
|
2012-10-13 00:56:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API ALvoid AL_APIENTRY alSourcedvSOFT(ALuint source, ALenum param, const ALdouble *values)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2012-12-05 20:51:25 -08:00
|
|
|
ALint count;
|
2012-10-13 00:56:39 -07:00
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!((count=DoubleValsByProp(param)) > 0 && count <= 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-10-13 00:56:39 -07:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALfloat fvals[3];
|
|
|
|
ALint i;
|
2012-11-01 18:35:20 -07:00
|
|
|
|
2012-12-05 20:51:25 -08:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
fvals[i] = (ALfloat)values[i];
|
|
|
|
SetSourcefv(Source, Context, param, fvals);
|
2012-10-13 00:56:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcei(ALuint source, ALenum param, ALint value)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-23 19:46:05 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2011-06-16 09:14:41 -07:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourceiv(Source, Context, param, &value);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API void AL_APIENTRY alSource3i(ALuint source, ALenum param, ALint value1, ALint value2, ALint value3)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-08-28 22:16:55 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2011-06-16 09:14:41 -07:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 19:58:01 -08:00
|
|
|
ALint ivals[3] = { value1, value2, value3 };
|
|
|
|
SetSourceiv(Source, Context, param, ivals);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API void AL_APIENTRY alSourceiv(ALuint source, ALenum param, const ALint *values)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2012-08-28 22:16:55 -07:00
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourceiv(Source, Context, param, values);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-13 00:56:39 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcei64SOFT(ALuint source, ALenum param, ALint64SOFT value)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourcei64v(Source, Context, param, &value);
|
2012-10-13 00:56:39 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API void AL_APIENTRY alSource3i64SOFT(ALuint source, ALenum param, ALint64SOFT value1, ALint64SOFT value2, ALint64SOFT value3)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-10-13 00:56:39 -07:00
|
|
|
{
|
2012-12-05 19:58:01 -08:00
|
|
|
ALint64SOFT i64vals[3] = { value1, value2, value3 };
|
|
|
|
SetSourcei64v(Source, Context, param, i64vals);
|
2012-10-13 00:56:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API void AL_APIENTRY alSourcei64vSOFT(ALuint source, ALenum param, const ALint64SOFT *values)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
SetSourcei64v(Source, Context, param, values);
|
2012-10-13 00:56:39 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alGetSourcef(ALuint source, ALenum param, ALfloat *value)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!value)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(FloatValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALdouble dval;
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourcedv(Source, Context, param, &dval))
|
2012-12-05 20:51:25 -08:00
|
|
|
*value = (ALfloat)dval;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alGetSource3f(ALuint source, ALenum param, ALfloat *value1, ALfloat *value2, ALfloat *value3)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2011-09-11 00:47:31 -07:00
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!(value1 && value2 && value3))
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(FloatValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALdouble dvals[3];
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourcedv(Source, Context, param, dvals))
|
2012-12-05 20:51:25 -08:00
|
|
|
{
|
|
|
|
*value1 = (ALfloat)dvals[0];
|
|
|
|
*value2 = (ALfloat)dvals[1];
|
|
|
|
*value3 = (ALfloat)dvals[2];
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alGetSourcefv(ALuint source, ALenum param, ALfloat *values)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2012-12-05 20:51:25 -08:00
|
|
|
ALint count;
|
2011-06-16 09:14:41 -07:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!((count=FloatValsByProp(param)) > 0 && count <= 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALdouble dvals[3];
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourcedv(Source, Context, param, dvals))
|
2012-12-05 20:51:25 -08:00
|
|
|
{
|
|
|
|
ALint i;
|
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
values[i] = (ALfloat)dvals[i];
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-20 15:26:35 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcedSOFT(ALuint source, ALenum param, ALdouble *value)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!value)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(DoubleValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourcedv(Source, Context, param, value);
|
2012-08-20 15:26:35 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API void AL_APIENTRY alGetSource3dSOFT(ALuint source, ALenum param, ALdouble *value1, ALdouble *value2, ALdouble *value3)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!(value1 && value2 && value3))
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(DoubleValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-08-20 15:26:35 -07:00
|
|
|
{
|
2012-12-05 20:51:25 -08:00
|
|
|
ALdouble dvals[3];
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourcedv(Source, Context, param, dvals))
|
2012-12-05 20:51:25 -08:00
|
|
|
{
|
|
|
|
*value1 = dvals[0];
|
|
|
|
*value2 = dvals[1];
|
|
|
|
*value3 = dvals[2];
|
|
|
|
}
|
2012-08-20 15:26:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API void AL_APIENTRY alGetSourcedvSOFT(ALuint source, ALenum param, ALdouble *values)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 20:51:25 -08:00
|
|
|
else if(!(DoubleValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourcedv(Source, Context, param, values);
|
2012-08-20 15:26:35 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alGetSourcei(ALuint source, ALenum param, ALint *value)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext *Context;
|
2010-03-26 00:41:27 -07:00
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!value)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourceiv(Source, Context, param, value);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSource3i(ALuint source, ALenum param, ALint *value1, ALint *value2, ALint *value3)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!(value1 && value2 && value3))
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-12-05 19:58:01 -08:00
|
|
|
ALint ivals[3];
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourceiv(Source, Context, param, ivals))
|
2012-12-05 19:58:01 -08:00
|
|
|
{
|
|
|
|
*value1 = ivals[0];
|
|
|
|
*value2 = ivals[1];
|
|
|
|
*value3 = ivals[2];
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourceiv(ALuint source, ALenum param, ALint *values)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
2011-06-16 09:14:41 -07:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(IntValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourceiv(Source, Context, param, values);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
ALCcontext_DecRef(Context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-20 14:50:43 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcei64SOFT(ALuint source, ALenum param, ALint64SOFT *value)
|
2012-08-18 11:02:54 -07:00
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!value)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) == 1))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourcei64v(Source, Context, param, value);
|
2012-08-20 14:16:58 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
AL_API void AL_APIENTRY alGetSource3i64SOFT(ALuint source, ALenum param, ALint64SOFT *value1, ALint64SOFT *value2, ALint64SOFT *value3)
|
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!(value1 && value2 && value3))
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) == 3))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
2012-08-20 14:16:58 -07:00
|
|
|
{
|
2012-12-05 19:58:01 -08:00
|
|
|
ALint64 i64vals[3];
|
2013-12-09 13:04:16 -08:00
|
|
|
if(GetSourcei64v(Source, Context, param, i64vals))
|
2012-12-05 19:58:01 -08:00
|
|
|
{
|
|
|
|
*value1 = i64vals[0];
|
|
|
|
*value2 = i64vals[1];
|
|
|
|
*value3 = i64vals[2];
|
|
|
|
}
|
2012-08-18 11:02:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
2012-08-20 12:22:00 -07:00
|
|
|
AL_API void AL_APIENTRY alGetSourcei64vSOFT(ALuint source, ALenum param, ALint64SOFT *values)
|
2012-08-18 11:02:54 -07:00
|
|
|
{
|
|
|
|
ALCcontext *Context;
|
|
|
|
ALsource *Source;
|
|
|
|
|
|
|
|
Context = GetContextRef();
|
|
|
|
if(!Context) return;
|
|
|
|
|
2012-08-29 00:25:01 -07:00
|
|
|
if((Source=LookupSource(Context, source)) == NULL)
|
|
|
|
alSetError(Context, AL_INVALID_NAME);
|
|
|
|
else if(!values)
|
|
|
|
alSetError(Context, AL_INVALID_VALUE);
|
2012-12-05 19:58:01 -08:00
|
|
|
else if(!(Int64ValsByProp(param) > 0))
|
|
|
|
alSetError(Context, AL_INVALID_ENUM);
|
|
|
|
else
|
|
|
|
GetSourcei64v(Source, Context, param, values);
|
2012-08-18 11:02:54 -07:00
|
|
|
|
|
|
|
ALCcontext_DecRef(Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcePlay(ALuint source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
|
|
|
alSourcePlayv(1, &source);
|
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcePlayv(ALsizei n, const ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
|
|
|
ALsizei i;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
for(i = 0;i < n;i++)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!LookupSource(context, sources[i]))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
LockContext(context);
|
|
|
|
while(n > context->MaxActiveSources-context->ActiveSourceCount)
|
|
|
|
{
|
2014-03-18 19:56:25 -07:00
|
|
|
ALactivesource **temp = NULL;
|
2013-10-07 09:24:50 -07:00
|
|
|
ALsizei newcount;
|
|
|
|
|
|
|
|
newcount = context->MaxActiveSources << 1;
|
|
|
|
if(newcount > 0)
|
|
|
|
temp = realloc(context->ActiveSources,
|
2014-03-18 19:56:25 -07:00
|
|
|
newcount * sizeof(context->ActiveSources[0]));
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!temp)
|
2010-06-06 00:17:50 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
UnlockContext(context);
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_OUT_OF_MEMORY, done);
|
2012-04-23 19:46:05 -07:00
|
|
|
}
|
2014-03-18 19:56:25 -07:00
|
|
|
for(i = context->MaxActiveSources;i < newcount;i++)
|
|
|
|
temp[i] = NULL;
|
2010-06-06 00:17:50 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context->ActiveSources = temp;
|
|
|
|
context->MaxActiveSources = newcount;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
|
|
|
source = LookupSource(context, sources[i]);
|
|
|
|
if(context->DeferUpdates) source->new_state = AL_PLAYING;
|
|
|
|
else SetSourceState(source, context, AL_PLAYING);
|
|
|
|
}
|
|
|
|
UnlockContext(context);
|
|
|
|
|
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcePause(ALuint source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
|
|
|
alSourcePausev(1, &source);
|
|
|
|
}
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourcePausev(ALsizei n, const ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
|
|
|
ALsizei i;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
for(i = 0;i < n;i++)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!LookupSource(context, sources[i]))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
LockContext(context);
|
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
|
|
|
source = LookupSource(context, sources[i]);
|
|
|
|
if(context->DeferUpdates) source->new_state = AL_PAUSED;
|
|
|
|
else SetSourceState(source, context, AL_PAUSED);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
UnlockContext(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceStop(ALuint source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
|
|
|
alSourceStopv(1, &source);
|
|
|
|
}
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceStopv(ALsizei n, const ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
|
|
|
ALsizei i;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
for(i = 0;i < n;i++)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!LookupSource(context, sources[i]))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
LockContext(context);
|
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
|
|
|
source = LookupSource(context, sources[i]);
|
|
|
|
source->new_state = AL_NONE;
|
|
|
|
SetSourceState(source, context, AL_STOPPED);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
UnlockContext(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceRewind(ALuint source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
|
|
|
alSourceRewindv(1, &source);
|
|
|
|
}
|
2010-03-19 14:34:18 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceRewindv(ALsizei n, const ALuint *sources)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
|
|
|
ALsizei i;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(n >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
for(i = 0;i < n;i++)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!LookupSource(context, sources[i]))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
LockContext(context);
|
|
|
|
for(i = 0;i < n;i++)
|
|
|
|
{
|
|
|
|
source = LookupSource(context, sources[i]);
|
|
|
|
source->new_state = AL_NONE;
|
|
|
|
SetSourceState(source, context, AL_INITIAL);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
UnlockContext(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceQueueBuffers(ALuint src, ALsizei nb, const ALuint *buffers)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCdevice *device;
|
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
|
|
|
ALsizei i;
|
2014-05-10 08:55:28 -07:00
|
|
|
ALbufferlistitem *BufferListStart;
|
2010-03-24 02:23:00 -07:00
|
|
|
ALbufferlistitem *BufferList;
|
2013-10-07 09:24:50 -07:00
|
|
|
ALbuffer *BufferFmt = NULL;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-23 19:46:05 -07:00
|
|
|
if(nb == 0)
|
2007-11-13 18:02:18 -08:00
|
|
|
return;
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
device = context->Device;
|
2010-03-16 18:54:36 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!(nb >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
if((source=LookupSource(context, src)) == NULL)
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
2009-10-22 09:31:26 -07:00
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteLock(&source->queue_lock);
|
2013-10-07 09:24:50 -07:00
|
|
|
if(source->SourceType == AL_STATIC)
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&source->queue_lock);
|
2013-10-07 09:24:50 -07:00
|
|
|
/* Can't queue on a Static Source */
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_OPERATION, done);
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
/* Check for a valid Buffer, for its frequency and format */
|
|
|
|
BufferList = source->queue;
|
|
|
|
while(BufferList)
|
|
|
|
{
|
|
|
|
if(BufferList->buffer)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
BufferFmt = BufferList->buffer;
|
|
|
|
break;
|
2010-03-24 02:23:00 -07:00
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2014-05-10 08:55:28 -07:00
|
|
|
BufferListStart = NULL;
|
|
|
|
BufferList = NULL;
|
2013-10-07 09:24:50 -07:00
|
|
|
for(i = 0;i < nb;i++)
|
|
|
|
{
|
|
|
|
ALbuffer *buffer = NULL;
|
|
|
|
if(buffers[i] && (buffer=LookupBuffer(device, buffers[i])) == NULL)
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&source->queue_lock);
|
2014-05-10 08:55:28 -07:00
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, buffer_error);
|
2013-10-07 09:24:50 -07:00
|
|
|
}
|
2012-04-23 19:46:05 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!BufferListStart)
|
2011-09-11 03:57:40 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
BufferListStart = malloc(sizeof(ALbufferlistitem));
|
|
|
|
BufferListStart->buffer = buffer;
|
|
|
|
BufferListStart->next = NULL;
|
|
|
|
BufferListStart->prev = NULL;
|
|
|
|
BufferList = BufferListStart;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
BufferList->next = malloc(sizeof(ALbufferlistitem));
|
|
|
|
BufferList->next->buffer = buffer;
|
|
|
|
BufferList->next->next = NULL;
|
|
|
|
BufferList->next->prev = BufferList;
|
2011-09-11 03:57:40 -07:00
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
2013-10-07 09:24:50 -07:00
|
|
|
if(!buffer) continue;
|
2012-04-21 05:53:27 -07:00
|
|
|
|
2014-05-10 08:55:28 -07:00
|
|
|
/* Hold a read lock on each buffer being queued while checking all
|
|
|
|
* provided buffers. This is done so other threads don't see an extra
|
|
|
|
* reference on some buffers if this operation ends up failing. */
|
2013-10-07 09:24:50 -07:00
|
|
|
ReadLock(&buffer->lock);
|
2014-05-10 08:55:28 -07:00
|
|
|
IncrementRef(&buffer->ref);
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
if(BufferFmt == NULL)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
BufferFmt = buffer;
|
2010-09-26 01:15:27 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
source->NumChannels = ChannelsFromFmt(buffer->FmtChannels);
|
|
|
|
source->SampleSize = BytesFromFmt(buffer->FmtType);
|
|
|
|
}
|
|
|
|
else if(BufferFmt->Frequency != buffer->Frequency ||
|
|
|
|
BufferFmt->OriginalChannels != buffer->OriginalChannels ||
|
|
|
|
BufferFmt->OriginalType != buffer->OriginalType)
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&source->queue_lock);
|
2014-05-10 08:55:28 -07:00
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_OPERATION, buffer_error);
|
|
|
|
|
|
|
|
buffer_error:
|
|
|
|
/* A buffer failed (invalid ID or format), so unlock and release
|
|
|
|
* each buffer we had. */
|
|
|
|
while(BufferList != NULL)
|
|
|
|
{
|
|
|
|
ALbufferlistitem *prev = BufferList->prev;
|
|
|
|
if((buffer=BufferList->buffer) != NULL)
|
|
|
|
{
|
|
|
|
DecrementRef(&buffer->ref);
|
|
|
|
ReadUnlock(&buffer->lock);
|
|
|
|
}
|
|
|
|
free(BufferList);
|
|
|
|
BufferList = prev;
|
|
|
|
}
|
|
|
|
goto done;
|
2010-03-24 02:23:00 -07:00
|
|
|
}
|
2014-05-10 08:55:28 -07:00
|
|
|
}
|
|
|
|
/* All buffers good, unlock them now. */
|
|
|
|
while(BufferList != NULL)
|
|
|
|
{
|
|
|
|
ALbuffer *buffer = BufferList->buffer;
|
|
|
|
if(buffer) ReadUnlock(&buffer->lock);
|
|
|
|
BufferList = BufferList->prev;
|
2013-10-07 09:24:50 -07:00
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
/* Source is now streaming */
|
|
|
|
source->SourceType = AL_STREAMING;
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2014-05-11 03:11:35 -07:00
|
|
|
if((BufferList=CompExchangePtr((XchgPtr*)&source->queue, NULL, BufferListStart)) != NULL)
|
2013-10-07 09:24:50 -07:00
|
|
|
{
|
2014-05-11 03:11:35 -07:00
|
|
|
/* Queue head is not NULL, append to the end of the queue */
|
2013-10-07 09:24:50 -07:00
|
|
|
while(BufferList->next != NULL)
|
|
|
|
BufferList = BufferList->next;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
BufferListStart->prev = BufferList;
|
|
|
|
BufferList->next = BufferListStart;
|
|
|
|
}
|
2014-05-10 08:55:28 -07:00
|
|
|
CompExchangePtr((XchgPtr*)&source->current_buffer, NULL, BufferListStart);
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&source->queue_lock);
|
2013-03-24 13:55:41 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
AL_API ALvoid AL_APIENTRY alSourceUnqueueBuffers(ALuint src, ALsizei nb, ALuint *buffers)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2013-10-07 09:24:50 -07:00
|
|
|
ALCcontext *context;
|
|
|
|
ALsource *source;
|
2010-03-24 02:23:00 -07:00
|
|
|
ALbufferlistitem *BufferList;
|
2014-05-11 03:11:35 -07:00
|
|
|
ALbufferlistitem *OldHead;
|
2014-05-10 03:21:40 -07:00
|
|
|
ALsizei i;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-23 19:46:05 -07:00
|
|
|
if(nb == 0)
|
2007-11-13 18:02:18 -08:00
|
|
|
return;
|
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
context = GetContextRef();
|
|
|
|
if(!context) return;
|
|
|
|
|
|
|
|
if(!(nb >= 0))
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
|
|
|
|
if((source=LookupSource(context, src)) == NULL)
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_NAME, done);
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteLock(&source->queue_lock);
|
|
|
|
/* Find the new buffer queue head */
|
2014-05-10 03:21:40 -07:00
|
|
|
BufferList = source->queue;
|
|
|
|
for(i = 0;i < nb && BufferList;i++)
|
|
|
|
{
|
|
|
|
if(BufferList == source->current_buffer)
|
|
|
|
break;
|
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
|
|
|
if(source->Looping || source->SourceType != AL_STREAMING || i != nb)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
WriteUnlock(&source->queue_lock);
|
2013-10-07 09:24:50 -07:00
|
|
|
/* Trying to unqueue pending buffers, or a buffer that wasn't queued. */
|
|
|
|
SET_ERROR_AND_GOTO(context, AL_INVALID_VALUE, done);
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
/* Swap it, and cut the new head from the old. */
|
2014-05-11 03:11:35 -07:00
|
|
|
OldHead = ExchangePtr((XchgPtr*)&source->queue, BufferList);
|
|
|
|
if(BufferList)
|
2013-10-07 09:24:50 -07:00
|
|
|
{
|
2014-05-11 03:52:22 -07:00
|
|
|
ALCdevice *device = context->Device;
|
2014-05-14 02:47:07 -07:00
|
|
|
uint count;
|
2014-05-11 03:52:22 -07:00
|
|
|
|
|
|
|
/* Cut the new head's link back to the old body. The mixer is robust
|
|
|
|
* enough to handle the link back going away. Once the active mix (if
|
|
|
|
* any) is complete, it's safe to finish cutting the old tail from the
|
|
|
|
* new head. */
|
|
|
|
BufferList = ExchangePtr((XchgPtr*)&BufferList->prev, NULL);
|
2014-05-14 02:47:07 -07:00
|
|
|
if(((count=ReadRef(&device->MixCount))&1) != 0)
|
2014-05-11 03:52:22 -07:00
|
|
|
{
|
2014-05-14 02:47:07 -07:00
|
|
|
while(count == ReadRef(&device->MixCount))
|
2014-05-11 03:52:22 -07:00
|
|
|
althrd_yield();
|
|
|
|
}
|
|
|
|
BufferList->next = NULL;
|
2014-05-10 05:07:13 -07:00
|
|
|
}
|
|
|
|
WriteUnlock(&source->queue_lock);
|
2009-08-16 15:09:36 -07:00
|
|
|
|
2014-05-11 03:11:35 -07:00
|
|
|
while(OldHead != NULL)
|
2014-05-10 05:07:13 -07:00
|
|
|
{
|
2014-05-11 03:11:35 -07:00
|
|
|
ALbufferlistitem *next = OldHead->next;
|
|
|
|
ALbuffer *buffer = OldHead->buffer;
|
2014-05-10 05:07:13 -07:00
|
|
|
|
2014-05-11 03:11:35 -07:00
|
|
|
if(!buffer)
|
|
|
|
*(buffers++) = 0;
|
2014-05-10 05:07:13 -07:00
|
|
|
else
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2014-05-11 03:11:35 -07:00
|
|
|
*(buffers++) = buffer->id;
|
|
|
|
DecrementRef(&buffer->ref);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2014-05-11 03:11:35 -07:00
|
|
|
free(OldHead);
|
|
|
|
OldHead = next;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2013-10-07 09:24:50 -07:00
|
|
|
done:
|
|
|
|
ALCcontext_DecRef(context);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-26 00:41:27 -07:00
|
|
|
static ALvoid InitSourceParams(ALsource *Source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2011-08-31 02:18:16 -07:00
|
|
|
ALuint i;
|
|
|
|
|
2014-05-10 05:07:13 -07:00
|
|
|
RWLockInit(&Source->queue_lock);
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
Source->InnerAngle = 360.0f;
|
|
|
|
Source->OuterAngle = 360.0f;
|
|
|
|
Source->Pitch = 1.0f;
|
|
|
|
Source->Position[0] = 0.0f;
|
|
|
|
Source->Position[1] = 0.0f;
|
|
|
|
Source->Position[2] = 0.0f;
|
|
|
|
Source->Orientation[0] = 0.0f;
|
|
|
|
Source->Orientation[1] = 0.0f;
|
|
|
|
Source->Orientation[2] = 0.0f;
|
|
|
|
Source->Velocity[0] = 0.0f;
|
|
|
|
Source->Velocity[1] = 0.0f;
|
|
|
|
Source->Velocity[2] = 0.0f;
|
|
|
|
Source->RefDistance = 1.0f;
|
|
|
|
Source->MaxDistance = FLT_MAX;
|
|
|
|
Source->RollOffFactor = 1.0f;
|
|
|
|
Source->Looping = AL_FALSE;
|
|
|
|
Source->Gain = 1.0f;
|
|
|
|
Source->MinGain = 0.0f;
|
|
|
|
Source->MaxGain = 1.0f;
|
|
|
|
Source->OuterGain = 0.0f;
|
2010-03-26 00:41:27 -07:00
|
|
|
Source->OuterGainHF = 1.0f;
|
|
|
|
|
|
|
|
Source->DryGainHFAuto = AL_TRUE;
|
|
|
|
Source->WetGainAuto = AL_TRUE;
|
|
|
|
Source->WetGainHFAuto = AL_TRUE;
|
|
|
|
Source->AirAbsorptionFactor = 0.0f;
|
|
|
|
Source->RoomRolloffFactor = 0.0f;
|
|
|
|
Source->DopplerFactor = 1.0f;
|
2012-02-09 23:35:17 -08:00
|
|
|
Source->DirectChannels = AL_FALSE;
|
2010-03-26 00:41:27 -07:00
|
|
|
|
2012-02-12 08:18:20 -08:00
|
|
|
Source->DistanceModel = DefaultDistanceModel;
|
2010-03-26 00:41:27 -07:00
|
|
|
|
|
|
|
Source->Resampler = DefaultResampler;
|
|
|
|
|
|
|
|
Source->state = AL_INITIAL;
|
2011-08-20 23:59:24 -07:00
|
|
|
Source->new_state = AL_NONE;
|
2012-04-19 21:46:29 -07:00
|
|
|
Source->SourceType = AL_UNDETERMINED;
|
2012-04-16 22:11:03 -07:00
|
|
|
Source->Offset = -1.0;
|
2010-03-26 00:41:27 -07:00
|
|
|
|
2014-05-10 06:32:39 -07:00
|
|
|
Source->queue = NULL;
|
|
|
|
Source->current_buffer = NULL;
|
|
|
|
|
2014-05-11 01:36:18 -07:00
|
|
|
Source->Direct.Gain = 1.0f;
|
|
|
|
Source->Direct.GainHF = 1.0f;
|
2014-05-11 10:09:52 -07:00
|
|
|
Source->Direct.HFReference = LOWPASSFREQREF;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Direct.GainLF = 1.0f;
|
|
|
|
Source->Direct.LFReference = HIGHPASSFREQREF;
|
2011-08-31 02:18:16 -07:00
|
|
|
for(i = 0;i < MAX_SENDS;i++)
|
|
|
|
{
|
2012-04-27 00:45:42 -07:00
|
|
|
Source->Send[i].Gain = 1.0f;
|
|
|
|
Source->Send[i].GainHF = 1.0f;
|
2014-05-11 10:09:52 -07:00
|
|
|
Source->Send[i].HFReference = LOWPASSFREQREF;
|
2014-05-17 07:54:25 -07:00
|
|
|
Source->Send[i].GainLF = 1.0f;
|
|
|
|
Source->Send[i].LFReference = HIGHPASSFREQREF;
|
2011-08-31 02:18:16 -07:00
|
|
|
}
|
|
|
|
|
2010-03-26 00:41:27 -07:00
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* SetSourceState
|
2011-08-20 06:31:10 -07:00
|
|
|
*
|
2012-04-21 05:53:27 -07:00
|
|
|
* Sets the source's new play state given its current state.
|
2011-08-20 06:31:10 -07:00
|
|
|
*/
|
|
|
|
ALvoid SetSourceState(ALsource *Source, ALCcontext *Context, ALenum state)
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadLock(&Source->queue_lock);
|
2011-08-20 06:31:10 -07:00
|
|
|
if(state == AL_PLAYING)
|
|
|
|
{
|
2014-03-23 16:11:21 -07:00
|
|
|
ALCdevice *device = Context->Device;
|
2011-08-20 06:31:10 -07:00
|
|
|
ALbufferlistitem *BufferList;
|
2014-03-19 13:14:11 -07:00
|
|
|
ALactivesource *src = NULL;
|
2011-08-20 06:31:10 -07:00
|
|
|
ALsizei j, k;
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Check that there is a queue containing at least one valid, non zero
|
|
|
|
* length Buffer. */
|
2011-08-20 06:31:10 -07:00
|
|
|
BufferList = Source->queue;
|
|
|
|
while(BufferList)
|
|
|
|
{
|
2014-05-10 03:21:40 -07:00
|
|
|
ALbuffer *buffer;
|
|
|
|
if((buffer=BufferList->buffer) != NULL && buffer->SampleLen > 0)
|
2011-08-20 06:31:10 -07:00
|
|
|
break;
|
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(Source->state != AL_PAUSED)
|
|
|
|
{
|
|
|
|
Source->state = AL_PLAYING;
|
|
|
|
Source->position = 0;
|
|
|
|
Source->position_fraction = 0;
|
2014-05-22 11:08:21 -07:00
|
|
|
Source->current_buffer = BufferList;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
Source->state = AL_PLAYING;
|
|
|
|
|
|
|
|
// Check if an Offset has been set
|
2012-04-16 22:11:03 -07:00
|
|
|
if(Source->Offset >= 0.0)
|
2011-08-20 06:31:10 -07:00
|
|
|
ApplyOffset(Source);
|
|
|
|
|
2011-10-15 10:33:30 -07:00
|
|
|
/* If there's nothing to play, or device is disconnected, go right to
|
|
|
|
* stopped */
|
2014-03-23 16:11:21 -07:00
|
|
|
if(!BufferList || !device->Connected)
|
2014-05-10 05:07:13 -07:00
|
|
|
goto do_stop;
|
2011-10-15 10:33:30 -07:00
|
|
|
|
2011-08-20 06:31:10 -07:00
|
|
|
for(j = 0;j < Context->ActiveSourceCount;j++)
|
|
|
|
{
|
2014-03-18 19:56:25 -07:00
|
|
|
if(Context->ActiveSources[j]->Source == Source)
|
2014-03-19 13:14:11 -07:00
|
|
|
{
|
|
|
|
src = Context->ActiveSources[j];
|
2011-08-20 06:31:10 -07:00
|
|
|
break;
|
2014-03-19 13:14:11 -07:00
|
|
|
}
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
2014-03-19 13:14:11 -07:00
|
|
|
if(src == NULL)
|
2014-03-18 19:56:25 -07:00
|
|
|
{
|
2014-03-19 13:14:11 -07:00
|
|
|
src = Context->ActiveSources[Context->ActiveSourceCount];
|
|
|
|
if(src == NULL)
|
|
|
|
{
|
|
|
|
src = al_malloc(16, sizeof(src[0]));
|
|
|
|
Context->ActiveSources[Context->ActiveSourceCount] = src;
|
|
|
|
}
|
|
|
|
memset(src, 0, sizeof(*src));
|
|
|
|
|
|
|
|
src->Source = Source;
|
|
|
|
if(BufferList->buffer->FmtChannels == FmtMono)
|
|
|
|
src->Update = CalcSourceParams;
|
|
|
|
else
|
|
|
|
src->Update = CalcNonAttnSourceParams;
|
2014-03-18 19:56:25 -07:00
|
|
|
Context->ActiveSourceCount++;
|
|
|
|
}
|
2014-03-23 02:45:50 -07:00
|
|
|
else
|
|
|
|
{
|
2014-03-23 16:11:21 -07:00
|
|
|
ALuint i;
|
|
|
|
|
2014-03-23 03:03:03 -07:00
|
|
|
src->Direct.Moving = AL_FALSE;
|
|
|
|
src->Direct.Counter = 0;
|
2014-03-23 02:45:50 -07:00
|
|
|
for(j = 0;j < MAX_INPUT_CHANNELS;j++)
|
|
|
|
{
|
|
|
|
for(k = 0;k < SRC_HISTORY_LENGTH;k++)
|
2014-05-03 18:59:26 -07:00
|
|
|
src->Direct.Mix.Hrtf.State[j].History[k] = 0.0f;
|
2014-03-23 02:45:50 -07:00
|
|
|
for(k = 0;k < HRIR_LENGTH;k++)
|
|
|
|
{
|
2014-05-03 18:59:26 -07:00
|
|
|
src->Direct.Mix.Hrtf.State[j].Values[k][0] = 0.0f;
|
|
|
|
src->Direct.Mix.Hrtf.State[j].Values[k][1] = 0.0f;
|
2014-03-23 02:45:50 -07:00
|
|
|
}
|
|
|
|
}
|
2014-03-23 16:11:21 -07:00
|
|
|
for(i = 0;i < device->NumAuxSends;i++)
|
|
|
|
{
|
|
|
|
src->Send[i].Counter = 0;
|
|
|
|
src->Send[i].Moving = AL_FALSE;
|
|
|
|
}
|
2014-03-23 02:45:50 -07:00
|
|
|
}
|
2014-03-19 13:14:11 -07:00
|
|
|
Source->NeedsUpdate = AL_TRUE;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
|
|
|
else if(state == AL_PAUSED)
|
|
|
|
{
|
|
|
|
if(Source->state == AL_PLAYING)
|
|
|
|
Source->state = AL_PAUSED;
|
|
|
|
}
|
|
|
|
else if(state == AL_STOPPED)
|
|
|
|
{
|
2014-05-10 05:07:13 -07:00
|
|
|
do_stop:
|
2011-08-20 06:31:10 -07:00
|
|
|
if(Source->state != AL_INITIAL)
|
|
|
|
{
|
|
|
|
Source->state = AL_STOPPED;
|
2014-05-10 03:21:40 -07:00
|
|
|
Source->current_buffer = NULL;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
2012-04-16 22:11:03 -07:00
|
|
|
Source->Offset = -1.0;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
|
|
|
else if(state == AL_INITIAL)
|
|
|
|
{
|
|
|
|
if(Source->state != AL_INITIAL)
|
|
|
|
{
|
|
|
|
Source->state = AL_INITIAL;
|
|
|
|
Source->position = 0;
|
|
|
|
Source->position_fraction = 0;
|
2014-05-10 03:21:40 -07:00
|
|
|
Source->current_buffer = Source->queue;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
2012-04-16 22:11:03 -07:00
|
|
|
Source->Offset = -1.0;
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
2014-05-10 05:07:13 -07:00
|
|
|
ReadUnlock(&Source->queue_lock);
|
2011-08-20 06:31:10 -07:00
|
|
|
}
|
|
|
|
|
2012-08-18 11:02:54 -07:00
|
|
|
/* GetSourceOffset
|
|
|
|
*
|
|
|
|
* Gets the current read offset for the given Source, in 32.32 fixed-point
|
|
|
|
* samples. The offset is relative to the start of the queue (not the start of
|
|
|
|
* the current buffer).
|
|
|
|
*/
|
2012-08-21 16:01:11 -07:00
|
|
|
static ALint64 GetSourceOffset(const ALsource *Source)
|
2012-08-18 11:02:54 -07:00
|
|
|
{
|
|
|
|
const ALbufferlistitem *BufferList;
|
|
|
|
ALuint64 readPos;
|
|
|
|
|
|
|
|
if(Source->state != AL_PLAYING && Source->state != AL_PAUSED)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* NOTE: This is the offset into the *current* buffer, so add the length of
|
|
|
|
* any played buffers */
|
|
|
|
readPos = (ALuint64)Source->position << 32;
|
|
|
|
readPos |= (ALuint64)Source->position_fraction << (32-FRACTIONBITS);
|
|
|
|
BufferList = Source->queue;
|
2014-05-10 03:21:40 -07:00
|
|
|
while(BufferList && BufferList != Source->current_buffer)
|
2012-08-18 11:02:54 -07:00
|
|
|
{
|
|
|
|
if(BufferList->buffer)
|
|
|
|
readPos += (ALuint64)BufferList->buffer->SampleLen << 32;
|
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
|
|
|
|
2013-10-06 04:21:03 -07:00
|
|
|
return (ALint64)minu64(readPos, U64(0x7fffffffffffffff));
|
2012-08-18 11:02:54 -07:00
|
|
|
}
|
|
|
|
|
2012-08-20 15:57:27 -07:00
|
|
|
/* GetSourceSecOffset
|
|
|
|
*
|
|
|
|
* Gets the current read offset for the given Source, in seconds. The offset is
|
|
|
|
* relative to the start of the queue (not the start of the current buffer).
|
|
|
|
*/
|
2012-08-21 16:01:11 -07:00
|
|
|
static ALdouble GetSourceSecOffset(const ALsource *Source)
|
2012-08-20 15:57:27 -07:00
|
|
|
{
|
|
|
|
const ALbufferlistitem *BufferList;
|
2014-05-10 07:56:41 -07:00
|
|
|
const ALbuffer *Buffer = NULL;
|
2012-08-20 15:57:27 -07:00
|
|
|
ALuint64 readPos;
|
|
|
|
|
2014-05-10 03:21:40 -07:00
|
|
|
if(Source->state != AL_PLAYING && Source->state != AL_PAUSED)
|
2012-08-20 15:57:27 -07:00
|
|
|
return 0.0;
|
|
|
|
|
|
|
|
/* NOTE: This is the offset into the *current* buffer, so add the length of
|
|
|
|
* any played buffers */
|
|
|
|
readPos = (ALuint64)Source->position << FRACTIONBITS;
|
|
|
|
readPos |= (ALuint64)Source->position_fraction;
|
|
|
|
BufferList = Source->queue;
|
2014-05-10 03:21:40 -07:00
|
|
|
while(BufferList && BufferList != Source->current_buffer)
|
2012-08-20 15:57:27 -07:00
|
|
|
{
|
2014-05-10 03:21:40 -07:00
|
|
|
const ALbuffer *buffer = BufferList->buffer;
|
|
|
|
if(buffer != NULL)
|
|
|
|
{
|
|
|
|
if(!Buffer) Buffer = buffer;
|
|
|
|
readPos += (ALuint64)buffer->SampleLen << FRACTIONBITS;
|
|
|
|
}
|
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(BufferList && !Buffer)
|
|
|
|
{
|
|
|
|
Buffer = BufferList->buffer;
|
2012-08-20 15:57:27 -07:00
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
2014-05-14 03:40:01 -07:00
|
|
|
assert(Buffer != NULL);
|
2012-08-20 15:57:27 -07:00
|
|
|
|
|
|
|
return (ALdouble)readPos / (ALdouble)FRACTIONONE / (ALdouble)Buffer->Frequency;
|
|
|
|
}
|
|
|
|
|
2012-04-23 19:46:05 -07:00
|
|
|
/* GetSourceOffsets
|
2012-04-21 05:53:27 -07:00
|
|
|
*
|
2012-04-23 19:46:05 -07:00
|
|
|
* Gets the current read and write offsets for the given Source, in the
|
|
|
|
* appropriate format (Bytes, Samples or Seconds). The offsets are relative to
|
|
|
|
* the start of the queue (not the start of the current buffer).
|
2012-04-21 05:53:27 -07:00
|
|
|
*/
|
2012-08-21 16:01:11 -07:00
|
|
|
static ALvoid GetSourceOffsets(const ALsource *Source, ALenum name, ALdouble *offset, ALdouble updateLen)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2010-12-09 19:47:08 -08:00
|
|
|
const ALbufferlistitem *BufferList;
|
2014-05-10 03:21:40 -07:00
|
|
|
const ALbuffer *Buffer = NULL;
|
|
|
|
ALboolean readFin = AL_FALSE;
|
2012-04-21 05:53:27 -07:00
|
|
|
ALuint readPos, writePos;
|
|
|
|
ALuint totalBufferLen;
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2014-05-10 03:21:40 -07:00
|
|
|
if(Source->state != AL_PLAYING && Source->state != AL_PAUSED)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2010-05-19 10:36:24 -07:00
|
|
|
offset[0] = 0.0;
|
|
|
|
offset[1] = 0.0;
|
2010-03-24 02:23:00 -07:00
|
|
|
return;
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-02-16 00:11:50 -08:00
|
|
|
if(updateLen > 0.0 && updateLen < 0.015)
|
|
|
|
updateLen = 0.015;
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* NOTE: This is the offset into the *current* buffer, so add the length of
|
|
|
|
* any played buffers */
|
2011-10-03 10:07:50 -07:00
|
|
|
totalBufferLen = 0;
|
2014-05-10 07:56:41 -07:00
|
|
|
readPos = Source->position;
|
2010-03-24 02:23:00 -07:00
|
|
|
BufferList = Source->queue;
|
2014-05-10 03:21:40 -07:00
|
|
|
while(BufferList != NULL)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2014-05-10 03:21:40 -07:00
|
|
|
const ALbuffer *buffer;
|
|
|
|
readFin = readFin || (BufferList == Source->current_buffer);
|
|
|
|
if((buffer=BufferList->buffer) != NULL)
|
2010-04-23 07:23:38 -07:00
|
|
|
{
|
2014-05-10 07:56:41 -07:00
|
|
|
if(!Buffer) Buffer = buffer;
|
2014-05-10 03:21:40 -07:00
|
|
|
totalBufferLen += buffer->SampleLen;
|
|
|
|
if(!readFin) readPos += buffer->SampleLen;
|
2010-04-23 07:23:38 -07:00
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
BufferList = BufferList->next;
|
|
|
|
}
|
2014-05-14 03:40:01 -07:00
|
|
|
assert(Buffer != NULL);
|
|
|
|
|
2010-03-24 02:23:00 -07:00
|
|
|
if(Source->state == AL_PLAYING)
|
2012-04-23 19:46:05 -07:00
|
|
|
writePos = readPos + (ALuint)(updateLen*Buffer->Frequency);
|
2010-03-24 02:23:00 -07:00
|
|
|
else
|
|
|
|
writePos = readPos;
|
|
|
|
|
2012-04-19 21:46:29 -07:00
|
|
|
if(Source->Looping)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2011-10-03 10:07:50 -07:00
|
|
|
readPos %= totalBufferLen;
|
|
|
|
writePos %= totalBufferLen;
|
2010-03-24 02:23:00 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Wrap positions back to 0 */
|
2011-10-03 10:07:50 -07:00
|
|
|
if(readPos >= totalBufferLen)
|
2010-09-09 18:27:14 -07:00
|
|
|
readPos = 0;
|
2011-10-03 10:07:50 -07:00
|
|
|
if(writePos >= totalBufferLen)
|
2010-09-09 18:27:14 -07:00
|
|
|
writePos = 0;
|
2010-03-24 02:23:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
switch(name)
|
|
|
|
{
|
2007-11-13 18:02:18 -08:00
|
|
|
case AL_SEC_OFFSET:
|
2011-10-03 10:07:50 -07:00
|
|
|
offset[0] = (ALdouble)readPos / Buffer->Frequency;
|
|
|
|
offset[1] = (ALdouble)writePos / Buffer->Frequency;
|
2007-11-13 18:02:18 -08:00
|
|
|
break;
|
2012-04-21 05:53:27 -07:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
case AL_SAMPLE_OFFSET:
|
2010-09-21 09:43:35 -07:00
|
|
|
case AL_SAMPLE_RW_OFFSETS_SOFT:
|
2011-10-03 10:07:50 -07:00
|
|
|
offset[0] = (ALdouble)readPos;
|
|
|
|
offset[1] = (ALdouble)writePos;
|
2007-11-13 18:02:18 -08:00
|
|
|
break;
|
2012-04-21 05:53:27 -07:00
|
|
|
|
2007-11-13 18:02:18 -08:00
|
|
|
case AL_BYTE_OFFSET:
|
2010-09-21 09:43:35 -07:00
|
|
|
case AL_BYTE_RW_OFFSETS_SOFT:
|
2011-10-03 10:07:50 -07:00
|
|
|
if(Buffer->OriginalType == UserFmtIMA4)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2014-03-03 17:05:08 -08:00
|
|
|
ALsizei align = (Buffer->OriginalAlign-1)/2 + 4;
|
|
|
|
ALuint BlockSize = align * ChannelsFromFmt(Buffer->FmtChannels);
|
|
|
|
ALuint FrameBlockSize = Buffer->OriginalAlign;
|
2010-11-27 00:51:21 -08:00
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Round down to nearest ADPCM block */
|
2010-11-27 00:51:21 -08:00
|
|
|
offset[0] = (ALdouble)(readPos / FrameBlockSize * BlockSize);
|
|
|
|
if(Source->state != AL_PLAYING)
|
|
|
|
offset[1] = offset[0];
|
|
|
|
else
|
2008-11-11 05:57:32 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Round up to nearest ADPCM block */
|
2010-11-27 00:51:21 -08:00
|
|
|
offset[1] = (ALdouble)((writePos+FrameBlockSize-1) /
|
|
|
|
FrameBlockSize * BlockSize);
|
2008-11-11 05:57:32 -08:00
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2014-03-04 22:44:30 -08:00
|
|
|
else if(Buffer->OriginalType == UserFmtMSADPCM)
|
|
|
|
{
|
|
|
|
ALsizei align = (Buffer->OriginalAlign-2)/2 + 7;
|
|
|
|
ALuint BlockSize = align * ChannelsFromFmt(Buffer->FmtChannels);
|
|
|
|
ALuint FrameBlockSize = Buffer->OriginalAlign;
|
|
|
|
|
|
|
|
/* Round down to nearest ADPCM block */
|
|
|
|
offset[0] = (ALdouble)(readPos / FrameBlockSize * BlockSize);
|
|
|
|
if(Source->state != AL_PLAYING)
|
|
|
|
offset[1] = offset[0];
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Round up to nearest ADPCM block */
|
|
|
|
offset[1] = (ALdouble)((writePos+FrameBlockSize-1) /
|
|
|
|
FrameBlockSize * BlockSize);
|
|
|
|
}
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
else
|
|
|
|
{
|
2011-10-03 10:07:50 -07:00
|
|
|
ALuint FrameSize = FrameSizeFromUserFmt(Buffer->OriginalChannels, Buffer->OriginalType);
|
|
|
|
offset[0] = (ALdouble)(readPos * FrameSize);
|
|
|
|
offset[1] = (ALdouble)(writePos * FrameSize);
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* ApplyOffset
|
|
|
|
*
|
|
|
|
* Apply the stored playback offset to the Source. This function will update
|
2012-04-23 19:46:05 -07:00
|
|
|
* the number of buffers "played" given the stored offset.
|
|
|
|
*/
|
2011-08-21 00:49:04 -07:00
|
|
|
ALboolean ApplyOffset(ALsource *Source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2014-05-10 03:21:40 -07:00
|
|
|
ALbufferlistitem *BufferList;
|
|
|
|
const ALbuffer *Buffer;
|
2011-10-03 10:07:50 -07:00
|
|
|
ALint bufferLen, totalBufferLen;
|
|
|
|
ALint offset;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Get sample frame offset */
|
2011-10-03 10:07:50 -07:00
|
|
|
offset = GetSampleOffset(Source);
|
|
|
|
if(offset == -1)
|
2010-01-12 02:22:38 -08:00
|
|
|
return AL_FALSE;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
totalBufferLen = 0;
|
|
|
|
BufferList = Source->queue;
|
2014-05-10 03:21:40 -07:00
|
|
|
while(BufferList && totalBufferLen <= offset)
|
2010-01-12 02:22:38 -08:00
|
|
|
{
|
2010-03-26 00:41:27 -07:00
|
|
|
Buffer = BufferList->buffer;
|
2011-10-03 10:07:50 -07:00
|
|
|
bufferLen = Buffer ? Buffer->SampleLen : 0;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2014-05-10 03:21:40 -07:00
|
|
|
if(bufferLen > offset-totalBufferLen)
|
2010-01-12 02:22:38 -08:00
|
|
|
{
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Offset is in this buffer */
|
2014-05-10 03:21:40 -07:00
|
|
|
Source->current_buffer = BufferList;
|
2010-01-12 02:22:38 -08:00
|
|
|
|
2011-10-03 10:07:50 -07:00
|
|
|
Source->position = offset - totalBufferLen;
|
2012-04-16 22:13:39 -07:00
|
|
|
Source->position_fraction = 0;
|
2010-05-11 11:59:41 -07:00
|
|
|
return AL_TRUE;
|
2010-01-12 02:22:38 -08:00
|
|
|
}
|
|
|
|
|
2011-10-03 10:07:50 -07:00
|
|
|
totalBufferLen += bufferLen;
|
2010-01-12 02:22:38 -08:00
|
|
|
|
2010-03-26 00:41:27 -07:00
|
|
|
BufferList = BufferList->next;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2012-04-21 05:53:27 -07:00
|
|
|
|
|
|
|
/* Offset is out of range of the queue */
|
2010-05-11 11:59:41 -07:00
|
|
|
return AL_FALSE;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* GetSampleOffset
|
|
|
|
*
|
|
|
|
* Returns the sample offset into the Source's queue (from the Sample, Byte or
|
|
|
|
* Second offset supplied by the application). This takes into account the fact
|
|
|
|
* that the buffer format may have been modifed since.
|
|
|
|
*/
|
2011-10-03 10:07:50 -07:00
|
|
|
static ALint GetSampleOffset(ALsource *Source)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2010-12-09 19:47:08 -08:00
|
|
|
const ALbuffer *Buffer = NULL;
|
|
|
|
const ALbufferlistitem *BufferList;
|
2011-10-03 10:07:50 -07:00
|
|
|
ALint Offset = -1;
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Find the first valid Buffer in the Queue */
|
2010-03-24 02:23:00 -07:00
|
|
|
BufferList = Source->queue;
|
|
|
|
while(BufferList)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2010-03-24 02:23:00 -07:00
|
|
|
if(BufferList->buffer)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2010-03-24 02:23:00 -07:00
|
|
|
Buffer = BufferList->buffer;
|
2007-11-13 18:02:18 -08:00
|
|
|
break;
|
|
|
|
}
|
2010-03-24 02:23:00 -07:00
|
|
|
BufferList = BufferList->next;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
|
|
|
|
2010-03-24 02:23:00 -07:00
|
|
|
if(!Buffer)
|
2007-11-13 18:02:18 -08:00
|
|
|
{
|
2012-04-16 22:11:03 -07:00
|
|
|
Source->Offset = -1.0;
|
2010-03-24 02:23:00 -07:00
|
|
|
return -1;
|
|
|
|
}
|
2007-11-13 18:02:18 -08:00
|
|
|
|
2012-04-16 22:11:03 -07:00
|
|
|
switch(Source->OffsetType)
|
2010-03-24 02:23:00 -07:00
|
|
|
{
|
2010-05-11 11:06:48 -07:00
|
|
|
case AL_BYTE_OFFSET:
|
2012-04-21 05:53:27 -07:00
|
|
|
/* Determine the ByteOffset (and ensure it is block aligned) */
|
2012-04-16 22:11:03 -07:00
|
|
|
Offset = (ALint)Source->Offset;
|
2010-12-03 22:33:41 -08:00
|
|
|
if(Buffer->OriginalType == UserFmtIMA4)
|
2010-11-29 19:27:33 -08:00
|
|
|
{
|
2014-03-03 17:05:08 -08:00
|
|
|
ALsizei align = (Buffer->OriginalAlign-1)/2 + 4;
|
|
|
|
Offset /= align * ChannelsFromUserFmt(Buffer->OriginalChannels);
|
|
|
|
Offset *= Buffer->OriginalAlign;
|
2010-11-29 19:27:33 -08:00
|
|
|
}
|
2014-03-04 22:44:30 -08:00
|
|
|
else if(Buffer->OriginalType == UserFmtMSADPCM)
|
|
|
|
{
|
|
|
|
ALsizei align = (Buffer->OriginalAlign-2)/2 + 7;
|
|
|
|
Offset /= align * ChannelsFromUserFmt(Buffer->OriginalChannels);
|
|
|
|
Offset *= Buffer->OriginalAlign;
|
|
|
|
}
|
2010-11-29 19:27:33 -08:00
|
|
|
else
|
2011-10-03 10:07:50 -07:00
|
|
|
Offset /= FrameSizeFromUserFmt(Buffer->OriginalChannels, Buffer->OriginalType);
|
2010-05-11 11:06:48 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_SAMPLE_OFFSET:
|
2012-04-16 22:11:03 -07:00
|
|
|
Offset = (ALint)Source->Offset;
|
2010-05-11 11:06:48 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case AL_SEC_OFFSET:
|
2012-04-16 22:11:03 -07:00
|
|
|
Offset = (ALint)(Source->Offset * Buffer->Frequency);
|
2010-05-11 11:06:48 -07:00
|
|
|
break;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2012-04-16 22:11:03 -07:00
|
|
|
Source->Offset = -1.0;
|
2010-03-24 02:23:00 -07:00
|
|
|
|
2011-10-03 10:07:50 -07:00
|
|
|
return Offset;
|
2007-11-13 18:02:18 -08:00
|
|
|
}
|
2008-01-16 13:27:15 -08:00
|
|
|
|
|
|
|
|
2012-04-21 05:53:27 -07:00
|
|
|
/* ReleaseALSources
|
|
|
|
*
|
|
|
|
* Destroys all sources in the source map.
|
|
|
|
*/
|
2008-01-16 13:27:15 -08:00
|
|
|
ALvoid ReleaseALSources(ALCcontext *Context)
|
|
|
|
{
|
2010-05-01 19:59:41 -07:00
|
|
|
ALsizei pos;
|
2009-10-24 07:09:44 -07:00
|
|
|
ALuint j;
|
2010-05-01 19:59:41 -07:00
|
|
|
for(pos = 0;pos < Context->SourceMap.size;pos++)
|
2008-01-16 13:27:15 -08:00
|
|
|
{
|
2010-05-01 19:59:41 -07:00
|
|
|
ALsource *temp = Context->SourceMap.array[pos].value;
|
|
|
|
Context->SourceMap.array[pos].value = NULL;
|
2009-10-24 07:09:44 -07:00
|
|
|
|
|
|
|
while(temp->queue != NULL)
|
|
|
|
{
|
2010-03-26 00:41:27 -07:00
|
|
|
ALbufferlistitem *BufferList = temp->queue;
|
2010-11-06 14:07:30 -07:00
|
|
|
temp->queue = BufferList->next;
|
|
|
|
|
2010-03-26 00:41:27 -07:00
|
|
|
if(BufferList->buffer != NULL)
|
2011-08-29 23:05:47 -07:00
|
|
|
DecrementRef(&BufferList->buffer->ref);
|
2010-03-26 00:41:27 -07:00
|
|
|
free(BufferList);
|
2009-10-24 07:09:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
for(j = 0;j < MAX_SENDS;++j)
|
|
|
|
{
|
|
|
|
if(temp->Send[j].Slot)
|
2011-08-29 23:05:47 -07:00
|
|
|
DecrementRef(&temp->Send[j].Slot->ref);
|
2010-11-06 14:07:30 -07:00
|
|
|
temp->Send[j].Slot = NULL;
|
2009-10-24 07:09:44 -07:00
|
|
|
}
|
2008-01-16 13:27:15 -08:00
|
|
|
|
2012-04-19 22:28:01 -07:00
|
|
|
FreeThunkEntry(temp->id);
|
2012-04-21 05:53:27 -07:00
|
|
|
memset(temp, 0, sizeof(*temp));
|
2012-08-15 05:54:13 -07:00
|
|
|
al_free(temp);
|
2008-01-16 13:27:15 -08:00
|
|
|
}
|
|
|
|
}
|