// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "util/audio_stream.h" #include "common/assert.h" #include "common/log.h" #include "common/windows_headers.h" #include #include #include #include #include Log_SetChannel(XAudio2AudioStream); 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(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 m_xaudio; IXAudio2MasteringVoice* m_mastering_voice = nullptr; IXAudio2SourceVoice* m_source_voice = nullptr; std::array, 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 XAudio2AudioStream::XAudio2AudioStream(u32 sample_rate, u32 channels, u32 buffer_ms, AudioStretchMode stretch) : AudioStream(sample_rate, channels, buffer_ms, stretch) { } XAudio2AudioStream::~XAudio2AudioStream() { if (IsOpen()) CloseDevice(); if (m_xaudio2_library) FreeLibrary(m_xaudio2_library); if (m_com_initialized_by_us) CoUninitialize(); } std::unique_ptr AudioStream::CreateXAudio2Stream(u32 sample_rate, u32 channels, u32 buffer_ms, u32 latency_ms, AudioStretchMode stretch) { std::unique_ptr stream( std::make_unique(sample_rate, channels, buffer_ms, stretch)); if (!stream->OpenDevice(latency_ms)) stream.reset(); return stream; } bool XAudio2AudioStream::OpenDevice(u32 latency_ms) { DebugAssert(!IsOpen()); 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(GetProcAddress(m_xaudio2_library, "XAudio2Create")); if (!xaudio2_create) return false; HRESULT hr = CoInitializeEx(nullptr, COINIT_MULTITHREADED); 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; } hr = m_xaudio->CreateMasteringVoice(&m_mastering_voice, m_channels, m_sample_rate, 0, nullptr); if (FAILED(hr)) { Log_ErrorPrintf("CreateMasteringVoice() failed: %08X", hr); return false; } WAVEFORMATEX wf = {}; wf.cbSize = sizeof(wf); wf.nAvgBytesPerSec = m_sample_rate * m_channels * sizeof(s16); wf.nBlockAlign = static_cast(sizeof(s16) * m_channels); wf.nChannels = static_cast(m_channels); wf.nSamplesPerSec = m_sample_rate; 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; } m_enqueue_buffer_size = std::max(INTERNAL_BUFFER_SIZE, GetBufferSizeForMS(m_sample_rate, latency_ms)); Log_DevPrintf("Allocating %u buffers of %u frames", NUM_BUFFERS, m_enqueue_buffer_size); for (u32 i = 0; i < NUM_BUFFERS; i++) m_enqueue_buffers[i] = std::make_unique(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; } EnqueueBuffer(); return true; } void XAudio2AudioStream::SetPaused(bool paused) { 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; if (!m_buffer_enqueued) EnqueueBuffer(); } 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(); m_enqueue_buffers = {}; m_current_buffer = 0; m_paused = true; } void XAudio2AudioStream::EnqueueBuffer() { SampleType* samples = m_enqueue_buffers[m_current_buffer].get(); ReadFrames(samples, m_enqueue_buffer_size); const XAUDIO2_BUFFER buf = { static_cast(0), // flags static_cast(sizeof(s16) * m_channels * m_enqueue_buffer_size), // bytes reinterpret_cast(samples), // data 0u, 0u, 0u, 0u, 0u, nullptr, }; 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) { HRESULT hr = m_mastering_voice->SetVolume(static_cast(m_volume) / 100.0f); if (FAILED(hr)) { Log_ErrorPrintf("SetVolume() failed: %08X", hr); return; } m_volume = volume; } void __stdcall XAudio2AudioStream::OnVoiceProcessingPassStart(UINT32 BytesRequired) { } void __stdcall XAudio2AudioStream::OnVoiceProcessingPassEnd(void) { } void __stdcall XAudio2AudioStream::OnStreamEnd(void) { } void __stdcall XAudio2AudioStream::OnBufferStart(void* pBufferContext) { } void __stdcall XAudio2AudioStream::OnBufferEnd(void* pBufferContext) { EnqueueBuffer(); } void __stdcall XAudio2AudioStream::OnLoopEnd(void* pBufferContext) { } void __stdcall XAudio2AudioStream::OnVoiceError(void* pBufferContext, HRESULT Error) { }