2023-11-18 06:21:51 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
#include "util/audio_stream.h"
|
|
|
|
|
2021-06-30 05:15:39 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/log.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
#include "common/windows_headers.h"
|
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <memory>
|
|
|
|
#include <wrl/client.h>
|
2021-06-30 05:15:39 +00:00
|
|
|
#include <xaudio2.h>
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2021-06-30 05:15:39 +00:00
|
|
|
Log_SetChannel(XAudio2AudioStream);
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
class XAudio2AudioStream final : public AudioStream, private IXAudio2VoiceCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
XAudio2AudioStream(u32 sample_rate, u32 channels, u32 buffer_ms, AudioStretchMode stretch);
|
|
|
|
~XAudio2AudioStream();
|
|
|
|
|
|
|
|
void SetPaused(bool paused) override;
|
|
|
|
void SetOutputVolume(u32 volume) override;
|
|
|
|
|
|
|
|
bool OpenDevice(u32 latency_ms);
|
|
|
|
void CloseDevice();
|
|
|
|
void EnqueueBuffer();
|
|
|
|
|
|
|
|
private:
|
|
|
|
enum : u32
|
|
|
|
{
|
|
|
|
NUM_BUFFERS = 2,
|
|
|
|
INTERNAL_BUFFER_SIZE = 512,
|
|
|
|
};
|
|
|
|
|
|
|
|
ALWAYS_INLINE bool IsOpen() const { return static_cast<bool>(m_xaudio); }
|
|
|
|
|
|
|
|
// Inherited via IXAudio2VoiceCallback
|
|
|
|
void __stdcall OnVoiceProcessingPassStart(UINT32 BytesRequired) override;
|
|
|
|
void __stdcall OnVoiceProcessingPassEnd(void) override;
|
|
|
|
void __stdcall OnStreamEnd(void) override;
|
|
|
|
void __stdcall OnBufferStart(void* pBufferContext) override;
|
|
|
|
void __stdcall OnBufferEnd(void* pBufferContext) override;
|
|
|
|
void __stdcall OnLoopEnd(void* pBufferContext) override;
|
|
|
|
void __stdcall OnVoiceError(void* pBufferContext, HRESULT Error) override;
|
|
|
|
|
|
|
|
Microsoft::WRL::ComPtr<IXAudio2> m_xaudio;
|
|
|
|
IXAudio2MasteringVoice* m_mastering_voice = nullptr;
|
|
|
|
IXAudio2SourceVoice* m_source_voice = nullptr;
|
|
|
|
|
|
|
|
std::array<std::unique_ptr<SampleType[]>, NUM_BUFFERS> m_enqueue_buffers;
|
|
|
|
u32 m_enqueue_buffer_size = 0;
|
|
|
|
u32 m_current_buffer = 0;
|
|
|
|
bool m_buffer_enqueued = false;
|
|
|
|
|
|
|
|
HMODULE m_xaudio2_library = {};
|
|
|
|
bool m_com_initialized_by_us = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2022-07-27 14:42:41 +00:00
|
|
|
XAudio2AudioStream::XAudio2AudioStream(u32 sample_rate, u32 channels, u32 buffer_ms, AudioStretchMode stretch)
|
|
|
|
: AudioStream(sample_rate, channels, buffer_ms, stretch)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
|
|
|
XAudio2AudioStream::~XAudio2AudioStream()
|
|
|
|
{
|
|
|
|
if (IsOpen())
|
2022-07-27 14:42:41 +00:00
|
|
|
CloseDevice();
|
2021-06-30 05:15:39 +00:00
|
|
|
|
|
|
|
if (m_xaudio2_library)
|
|
|
|
FreeLibrary(m_xaudio2_library);
|
|
|
|
|
|
|
|
if (m_com_initialized_by_us)
|
|
|
|
CoUninitialize();
|
|
|
|
}
|
|
|
|
|
2023-08-13 06:28:28 +00:00
|
|
|
std::unique_ptr<AudioStream> AudioStream::CreateXAudio2Stream(u32 sample_rate, u32 channels, u32 buffer_ms,
|
|
|
|
u32 latency_ms, AudioStretchMode stretch)
|
2022-07-27 14:42:41 +00:00
|
|
|
{
|
|
|
|
std::unique_ptr<XAudio2AudioStream> stream(
|
|
|
|
std::make_unique<XAudio2AudioStream>(sample_rate, channels, buffer_ms, stretch));
|
|
|
|
if (!stream->OpenDevice(latency_ms))
|
|
|
|
stream.reset();
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool XAudio2AudioStream::OpenDevice(u32 latency_ms)
|
2021-06-30 05:15:39 +00:00
|
|
|
{
|
2022-07-27 14:42:41 +00:00
|
|
|
DebugAssert(!IsOpen());
|
|
|
|
|
2021-06-30 05:15:39 +00:00
|
|
|
m_xaudio2_library = LoadLibraryW(XAUDIO2_DLL_W);
|
|
|
|
if (!m_xaudio2_library)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to load '%s', make sure you're using Windows 10", XAUDIO2_DLL_A);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
using PFNXAUDIO2CREATE =
|
|
|
|
HRESULT(STDAPICALLTYPE*)(IXAudio2 * *ppXAudio2, UINT32 Flags, XAUDIO2_PROCESSOR XAudio2Processor);
|
|
|
|
PFNXAUDIO2CREATE xaudio2_create =
|
|
|
|
reinterpret_cast<PFNXAUDIO2CREATE>(GetProcAddress(m_xaudio2_library, "XAudio2Create"));
|
|
|
|
if (!xaudio2_create)
|
|
|
|
return false;
|
|
|
|
|
2022-11-03 03:12:46 +00:00
|
|
|
HRESULT hr = CoInitializeEx(nullptr, COINIT_MULTITHREADED);
|
2021-06-30 05:15:39 +00:00
|
|
|
m_com_initialized_by_us = SUCCEEDED(hr);
|
|
|
|
if (FAILED(hr) && hr != RPC_E_CHANGED_MODE && hr != S_FALSE)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to initialize COM");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = xaudio2_create(m_xaudio.ReleaseAndGetAddressOf(), 0, XAUDIO2_DEFAULT_PROCESSOR);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("XAudio2Create() failed: %08X", hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-07-27 14:42:41 +00:00
|
|
|
hr = m_xaudio->CreateMasteringVoice(&m_mastering_voice, m_channels, m_sample_rate, 0, nullptr);
|
2021-06-30 05:15:39 +00:00
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("CreateMasteringVoice() failed: %08X", hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
WAVEFORMATEX wf = {};
|
|
|
|
wf.cbSize = sizeof(wf);
|
2022-07-27 14:42:41 +00:00
|
|
|
wf.nAvgBytesPerSec = m_sample_rate * m_channels * sizeof(s16);
|
2021-06-30 05:15:39 +00:00
|
|
|
wf.nBlockAlign = static_cast<WORD>(sizeof(s16) * m_channels);
|
|
|
|
wf.nChannels = static_cast<WORD>(m_channels);
|
2022-07-27 14:42:41 +00:00
|
|
|
wf.nSamplesPerSec = m_sample_rate;
|
2021-06-30 05:15:39 +00:00
|
|
|
wf.wBitsPerSample = sizeof(s16) * 8;
|
|
|
|
wf.wFormatTag = WAVE_FORMAT_PCM;
|
|
|
|
hr = m_xaudio->CreateSourceVoice(&m_source_voice, &wf, 0, 1.0f, this);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("CreateMasteringVoice() failed: %08X", hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = m_source_voice->SetFrequencyRatio(1.0f);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("SetFrequencyRatio() failed: %08X", hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-07-27 14:42:41 +00:00
|
|
|
m_enqueue_buffer_size = std::max<u32>(INTERNAL_BUFFER_SIZE, GetBufferSizeForMS(m_sample_rate, latency_ms));
|
|
|
|
Log_DevPrintf("Allocating %u buffers of %u frames", NUM_BUFFERS, m_enqueue_buffer_size);
|
2021-06-30 05:15:39 +00:00
|
|
|
for (u32 i = 0; i < NUM_BUFFERS; i++)
|
2022-07-27 14:42:41 +00:00
|
|
|
m_enqueue_buffers[i] = std::make_unique<SampleType[]>(m_enqueue_buffer_size * m_channels);
|
|
|
|
|
|
|
|
BaseInitialize();
|
|
|
|
m_volume = 100;
|
|
|
|
m_paused = false;
|
|
|
|
|
|
|
|
hr = m_source_voice->Start(0, 0);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Start() failed: %08X", hr);
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
2022-07-27 14:42:41 +00:00
|
|
|
EnqueueBuffer();
|
2021-06-30 05:15:39 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-07-27 14:42:41 +00:00
|
|
|
void XAudio2AudioStream::SetPaused(bool paused)
|
2021-06-30 05:15:39 +00:00
|
|
|
{
|
|
|
|
if (m_paused == paused)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (paused)
|
|
|
|
{
|
|
|
|
HRESULT hr = m_source_voice->Stop(0, 0);
|
|
|
|
if (FAILED(hr))
|
|
|
|
Log_ErrorPrintf("Stop() failed: %08X", hr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
HRESULT hr = m_source_voice->Start(0, 0);
|
|
|
|
if (FAILED(hr))
|
|
|
|
Log_ErrorPrintf("Start() failed: %08X", hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_paused = paused;
|
2022-07-27 14:42:41 +00:00
|
|
|
|
|
|
|
if (!m_buffer_enqueued)
|
|
|
|
EnqueueBuffer();
|
2021-06-30 05:15:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void XAudio2AudioStream::CloseDevice()
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
if (!m_paused)
|
|
|
|
{
|
|
|
|
hr = m_source_voice->Stop(0, 0);
|
|
|
|
if (FAILED(hr))
|
|
|
|
Log_ErrorPrintf("Stop() failed: %08X", hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_source_voice = nullptr;
|
|
|
|
m_mastering_voice = nullptr;
|
|
|
|
m_xaudio.Reset();
|
2022-07-27 14:42:41 +00:00
|
|
|
m_enqueue_buffers = {};
|
2021-06-30 05:15:39 +00:00
|
|
|
m_current_buffer = 0;
|
|
|
|
m_paused = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void XAudio2AudioStream::EnqueueBuffer()
|
|
|
|
{
|
2022-07-27 14:42:41 +00:00
|
|
|
SampleType* samples = m_enqueue_buffers[m_current_buffer].get();
|
|
|
|
ReadFrames(samples, m_enqueue_buffer_size);
|
2021-06-30 05:15:39 +00:00
|
|
|
|
|
|
|
const XAUDIO2_BUFFER buf = {
|
2022-07-27 14:42:41 +00:00
|
|
|
static_cast<UINT32>(0), // flags
|
|
|
|
static_cast<UINT32>(sizeof(s16) * m_channels * m_enqueue_buffer_size), // bytes
|
2023-09-03 04:30:26 +00:00
|
|
|
reinterpret_cast<const BYTE*>(samples), // data
|
|
|
|
0u,
|
|
|
|
0u,
|
|
|
|
0u,
|
|
|
|
0u,
|
|
|
|
0u,
|
|
|
|
nullptr,
|
2021-06-30 05:15:39 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
HRESULT hr = m_source_voice->SubmitSourceBuffer(&buf, nullptr);
|
|
|
|
if (FAILED(hr))
|
|
|
|
Log_ErrorPrintf("SubmitSourceBuffer() failed: %08X", hr);
|
|
|
|
|
|
|
|
m_current_buffer = (m_current_buffer + 1) % NUM_BUFFERS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void XAudio2AudioStream::SetOutputVolume(u32 volume)
|
|
|
|
{
|
2022-07-27 14:42:41 +00:00
|
|
|
HRESULT hr = m_mastering_voice->SetVolume(static_cast<float>(m_volume) / 100.0f);
|
2021-06-30 05:15:39 +00:00
|
|
|
if (FAILED(hr))
|
2022-07-27 14:42:41 +00:00
|
|
|
{
|
2021-06-30 05:15:39 +00:00
|
|
|
Log_ErrorPrintf("SetVolume() failed: %08X", hr);
|
2022-07-27 14:42:41 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_volume = volume;
|
2021-06-30 05:15:39 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnVoiceProcessingPassStart(UINT32 BytesRequired)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnVoiceProcessingPassEnd(void)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnStreamEnd(void)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnBufferStart(void* pBufferContext)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
|
|
|
void __stdcall XAudio2AudioStream::OnBufferEnd(void* pBufferContext)
|
|
|
|
{
|
|
|
|
EnqueueBuffer();
|
|
|
|
}
|
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnLoopEnd(void* pBufferContext)
|
|
|
|
{
|
|
|
|
}
|
2021-06-30 05:15:39 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
void __stdcall XAudio2AudioStream::OnVoiceError(void* pBufferContext, HRESULT Error)
|
|
|
|
{
|
|
|
|
}
|