2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "base.h"
|
|
|
|
|
2021-05-14 10:15:42 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <array>
|
2021-04-28 12:43:51 +08:00
|
|
|
#include <atomic>
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <windows.h>
|
|
|
|
#include <mmreg.h>
|
|
|
|
|
|
|
|
#include "albit.h"
|
2021-05-14 10:15:42 +08:00
|
|
|
#include "core/logging.h"
|
2023-05-31 23:57:33 +08:00
|
|
|
#include "aloptional.h"
|
2021-05-14 10:15:42 +08:00
|
|
|
#endif
|
|
|
|
|
2021-04-28 12:43:51 +08:00
|
|
|
#include "atomic.h"
|
2021-05-14 10:15:42 +08:00
|
|
|
#include "core/devformat.h"
|
2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
|
2023-02-04 15:03:54 +08:00
|
|
|
namespace al {
|
|
|
|
|
|
|
|
backend_exception::backend_exception(backend_error code, const char *msg, ...) : mErrorCode{code}
|
|
|
|
{
|
|
|
|
std::va_list args;
|
|
|
|
va_start(args, msg);
|
|
|
|
setMessage(msg, args);
|
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
backend_exception::~backend_exception() = default;
|
|
|
|
|
|
|
|
} // namespace al
|
|
|
|
|
|
|
|
|
2021-04-28 12:43:51 +08:00
|
|
|
bool BackendBase::reset()
|
|
|
|
{ throw al::backend_exception{al::backend_error::DeviceError, "Invalid BackendBase call"}; }
|
|
|
|
|
2023-05-31 23:57:33 +08:00
|
|
|
void BackendBase::captureSamples(al::byte*, uint)
|
2021-04-28 12:43:51 +08:00
|
|
|
{ }
|
|
|
|
|
|
|
|
uint BackendBase::availableSamples()
|
|
|
|
{ return 0; }
|
|
|
|
|
|
|
|
ClockLatency BackendBase::getClockLatency()
|
|
|
|
{
|
|
|
|
ClockLatency ret;
|
|
|
|
|
|
|
|
uint refcount;
|
|
|
|
do {
|
|
|
|
refcount = mDevice->waitForMix();
|
|
|
|
ret.ClockTime = GetDeviceClockTime(mDevice);
|
|
|
|
std::atomic_thread_fence(std::memory_order_acquire);
|
|
|
|
} while(refcount != ReadRef(mDevice->MixCount));
|
|
|
|
|
|
|
|
/* 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 = std::max(std::chrono::seconds{mDevice->BufferSize-mDevice->UpdateSize},
|
|
|
|
std::chrono::seconds::zero());
|
|
|
|
ret.Latency /= mDevice->Frequency;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BackendBase::setDefaultWFXChannelOrder()
|
|
|
|
{
|
2023-02-04 15:03:54 +08:00
|
|
|
mDevice->RealOut.ChannelIndex.fill(InvalidChannelIndex);
|
2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
switch(mDevice->FmtChans)
|
|
|
|
{
|
|
|
|
case DevFmtMono:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 0;
|
|
|
|
break;
|
|
|
|
case DevFmtStereo:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
break;
|
|
|
|
case DevFmtQuad:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackLeft] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackRight] = 3;
|
|
|
|
break;
|
|
|
|
case DevFmtX51:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 5;
|
|
|
|
break;
|
|
|
|
case DevFmtX61:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackCenter] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 6;
|
|
|
|
break;
|
|
|
|
case DevFmtX71:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackLeft] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackRight] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
break;
|
2023-02-04 15:03:54 +08:00
|
|
|
case DevFmtX714:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackLeft] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackRight] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopFrontLeft] = 8;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopFrontRight] = 9;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopBackLeft] = 10;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopBackRight] = 11;
|
|
|
|
break;
|
2022-07-14 23:17:11 +08:00
|
|
|
case DevFmtX3D71:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[Aux0] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[Aux1] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
break;
|
2021-04-28 12:43:51 +08:00
|
|
|
case DevFmtAmbi3D:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BackendBase::setDefaultChannelOrder()
|
|
|
|
{
|
2023-02-04 15:03:54 +08:00
|
|
|
mDevice->RealOut.ChannelIndex.fill(InvalidChannelIndex);
|
2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
switch(mDevice->FmtChans)
|
|
|
|
{
|
2022-04-25 12:02:45 +08:00
|
|
|
case DevFmtX51:
|
2021-04-28 12:43:51 +08:00
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
2022-04-25 12:02:45 +08:00
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 3;
|
2021-04-28 12:43:51 +08:00
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 5;
|
|
|
|
return;
|
|
|
|
case DevFmtX71:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackLeft] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackRight] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
return;
|
2023-02-04 15:03:54 +08:00
|
|
|
case DevFmtX714:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackLeft] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[BackRight] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopFrontLeft] = 8;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopFrontRight] = 9;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopBackLeft] = 10;
|
|
|
|
mDevice->RealOut.ChannelIndex[TopBackRight] = 11;
|
|
|
|
break;
|
2022-07-14 23:17:11 +08:00
|
|
|
case DevFmtX3D71:
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontLeft] = 0;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontRight] = 1;
|
|
|
|
mDevice->RealOut.ChannelIndex[Aux0] = 2;
|
|
|
|
mDevice->RealOut.ChannelIndex[Aux1] = 3;
|
|
|
|
mDevice->RealOut.ChannelIndex[FrontCenter] = 4;
|
|
|
|
mDevice->RealOut.ChannelIndex[LFE] = 5;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideLeft] = 6;
|
|
|
|
mDevice->RealOut.ChannelIndex[SideRight] = 7;
|
|
|
|
return;
|
2021-04-28 12:43:51 +08:00
|
|
|
|
|
|
|
/* Same as WFX order */
|
|
|
|
case DevFmtMono:
|
|
|
|
case DevFmtStereo:
|
|
|
|
case DevFmtQuad:
|
|
|
|
case DevFmtX61:
|
|
|
|
case DevFmtAmbi3D:
|
|
|
|
setDefaultWFXChannelOrder();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|