2013-10-28 12:30:57 -07:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "alMain.h"
|
2017-02-18 16:55:48 -08:00
|
|
|
#include "alu.h"
|
2013-10-28 12:30:57 -07:00
|
|
|
|
|
|
|
#include "backends/base.h"
|
|
|
|
|
|
|
|
|
2016-05-28 00:43:14 -07:00
|
|
|
extern inline ALuint64 GetDeviceClockTime(ALCdevice *device);
|
2018-01-12 02:37:48 -08:00
|
|
|
extern inline void ALCdevice_Lock(ALCdevice *device);
|
|
|
|
extern inline void ALCdevice_Unlock(ALCdevice *device);
|
2016-05-28 00:43:14 -07:00
|
|
|
|
2013-10-28 12:30:57 -07:00
|
|
|
/* Base ALCbackend method implementations. */
|
|
|
|
void ALCbackend_Construct(ALCbackend *self, ALCdevice *device)
|
|
|
|
{
|
2016-05-30 02:10:06 -07:00
|
|
|
int ret = almtx_init(&self->mMutex, almtx_recursive);
|
2014-04-16 01:39:11 -07:00
|
|
|
assert(ret == althrd_success);
|
2016-05-30 02:10:06 -07:00
|
|
|
self->mDevice = device;
|
2013-10-28 12:30:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void ALCbackend_Destruct(ALCbackend *self)
|
|
|
|
{
|
2014-04-16 01:39:11 -07:00
|
|
|
almtx_destroy(&self->mMutex);
|
2013-10-28 12:30:57 -07:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:07:13 -07:00
|
|
|
ALCboolean ALCbackend_reset(ALCbackend* UNUSED(self))
|
|
|
|
{
|
|
|
|
return ALC_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ALCenum ALCbackend_captureSamples(ALCbackend* UNUSED(self), void* UNUSED(buffer), ALCuint UNUSED(samples))
|
|
|
|
{
|
|
|
|
return ALC_INVALID_DEVICE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ALCuint ALCbackend_availableSamples(ALCbackend* UNUSED(self))
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-28 00:43:14 -07:00
|
|
|
ClockLatency ALCbackend_getClockLatency(ALCbackend *self)
|
2013-10-28 12:30:57 -07:00
|
|
|
{
|
2016-05-28 00:43:14 -07:00
|
|
|
ALCdevice *device = self->mDevice;
|
2017-02-28 03:50:42 -08:00
|
|
|
ALuint refcount;
|
2016-05-28 00:43:14 -07:00
|
|
|
ClockLatency ret;
|
|
|
|
|
2017-02-28 03:50:42 -08:00
|
|
|
do {
|
|
|
|
while(((refcount=ATOMIC_LOAD(&device->MixCount, almemory_order_acquire))&1))
|
|
|
|
althrd_yield();
|
|
|
|
ret.ClockTime = GetDeviceClockTime(device);
|
|
|
|
ATOMIC_THREAD_FENCE(almemory_order_acquire);
|
|
|
|
} while(refcount != ATOMIC_LOAD(&device->MixCount, almemory_order_relaxed));
|
|
|
|
|
2017-02-18 16:55:48 -08:00
|
|
|
/* NOTE: The device will generally have about all but one periods filled at
|
|
|
|
* any given time during playback. Without a more accurate measurement from
|
|
|
|
* the output, this is an okay approximation.
|
|
|
|
*/
|
|
|
|
ret.Latency = device->UpdateSize * DEVICE_CLOCK_RES / device->Frequency *
|
|
|
|
maxu(device->NumUpdates-1, 1);
|
2016-05-28 00:43:14 -07:00
|
|
|
|
|
|
|
return ret;
|
2013-10-28 12:30:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void ALCbackend_lock(ALCbackend *self)
|
|
|
|
{
|
2014-04-16 01:39:11 -07:00
|
|
|
int ret = almtx_lock(&self->mMutex);
|
|
|
|
assert(ret == althrd_success);
|
2013-10-28 12:30:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void ALCbackend_unlock(ALCbackend *self)
|
|
|
|
{
|
2014-04-16 01:39:11 -07:00
|
|
|
int ret = almtx_unlock(&self->mMutex);
|
|
|
|
assert(ret == althrd_success);
|
2013-10-28 12:30:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-02 15:42:45 -07:00
|
|
|
/* Base ALCbackendFactory method implementations. */
|
|
|
|
void ALCbackendFactory_deinit(ALCbackendFactory* UNUSED(self))
|
|
|
|
{
|
|
|
|
}
|