2018-10-03 22:02:46 +08:00
|
|
|
// Copyright (c) 2016-2018 Easy2D - Nomango
|
|
|
|
|
//
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
|
//
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
|
//
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
#include "audio.h"
|
2018-11-16 15:53:39 +08:00
|
|
|
#include "base.hpp"
|
2018-11-08 21:39:26 +08:00
|
|
|
#include "modules.h"
|
2018-11-15 17:59:18 +08:00
|
|
|
#include "logs.h"
|
2018-11-08 21:39:26 +08:00
|
|
|
#include <mfapi.h>
|
|
|
|
|
#include <mfidl.h>
|
|
|
|
|
#include <mfreadwrite.h>
|
2018-11-16 17:19:03 +08:00
|
|
|
#include <assert.h>
|
2018-11-08 00:21:59 +08:00
|
|
|
namespace easy2d
|
2018-05-08 20:03:29 +08:00
|
|
|
{
|
2018-11-12 20:46:54 +08:00
|
|
|
//-------------------------------------------------------
|
|
|
|
|
// Voice
|
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
Voice::Voice()
|
|
|
|
|
: source_voice_(nullptr)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Voice::Voice(IXAudio2SourceVoice * source_voice)
|
|
|
|
|
: source_voice_(source_voice)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Voice::~Voice()
|
|
|
|
|
{
|
|
|
|
|
Destroy();
|
|
|
|
|
|
2018-11-15 14:35:19 +08:00
|
|
|
devices::Audio::Instance()->DeleteVoice(this);
|
2018-11-12 20:46:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::Play(const BYTE * wave_data, UINT32 data_size, UINT32 loop_count)
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
XAUDIO2_BUFFER buffer = { 0 };
|
|
|
|
|
buffer.pAudioData = wave_data;
|
|
|
|
|
buffer.Flags = XAUDIO2_END_OF_STREAM;
|
|
|
|
|
buffer.AudioBytes = data_size;
|
|
|
|
|
buffer.LoopCount = loop_count;
|
|
|
|
|
|
|
|
|
|
HRESULT hr = source_voice_->SubmitSourceBuffer(&buffer);
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = source_voice_->Start();
|
|
|
|
|
}
|
|
|
|
|
return hr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::Pause()
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
return source_voice_->Stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::Resume()
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
return source_voice_->Start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::Stop()
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
HRESULT hr = source_voice_->Stop();
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = source_voice_->ExitLoop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = source_voice_->FlushSourceBuffers();
|
|
|
|
|
}
|
|
|
|
|
return hr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::GetVolume(float * volume) const
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
if (volume == nullptr)
|
|
|
|
|
return E_POINTER;
|
|
|
|
|
|
|
|
|
|
source_voice_->GetVolume(volume);
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::SetVolume(float volume)
|
|
|
|
|
{
|
|
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
volume = std::min(std::max(volume, -224.f), 224.f);
|
|
|
|
|
return source_voice_->SetVolume(volume);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT Voice::GetBuffersQueued(UINT32 * queued) const
|
2018-05-10 14:03:54 +08:00
|
|
|
{
|
2018-11-12 20:46:54 +08:00
|
|
|
if (!source_voice_)
|
|
|
|
|
return E_UNEXPECTED;
|
|
|
|
|
|
|
|
|
|
if (queued == nullptr)
|
|
|
|
|
return E_POINTER;
|
|
|
|
|
|
|
|
|
|
XAUDIO2_VOICE_STATE state;
|
|
|
|
|
source_voice_->GetState(&state);
|
|
|
|
|
*queued = state.BuffersQueued;
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Voice::Destroy()
|
|
|
|
|
{
|
|
|
|
|
if (source_voice_)
|
|
|
|
|
{
|
|
|
|
|
source_voice_->Stop();
|
|
|
|
|
source_voice_->FlushSourceBuffers();
|
|
|
|
|
source_voice_->DestroyVoice();
|
|
|
|
|
source_voice_ = nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Voice::SetSourceVoice(IXAudio2SourceVoice * source_voice)
|
|
|
|
|
{
|
|
|
|
|
Destroy();
|
|
|
|
|
source_voice_ = source_voice;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace devices
|
|
|
|
|
{
|
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
// AudioDevice
|
|
|
|
|
//-------------------------------------------------------
|
2018-05-08 20:03:29 +08:00
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
AudioDevice::AudioDevice()
|
|
|
|
|
: x_audio2_(nullptr)
|
|
|
|
|
, mastering_voice_(nullptr)
|
2018-11-12 22:36:50 +08:00
|
|
|
, initialized(false)
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-14 01:34:41 +08:00
|
|
|
modules::Init();
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
2018-05-08 20:03:29 +08:00
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
AudioDevice::~AudioDevice()
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-15 17:59:18 +08:00
|
|
|
E2D_LOG("Destroying audio device");
|
|
|
|
|
|
2018-11-12 20:46:54 +08:00
|
|
|
ClearVoiceCache();
|
|
|
|
|
|
|
|
|
|
if (mastering_voice_)
|
|
|
|
|
{
|
|
|
|
|
mastering_voice_->DestroyVoice();
|
|
|
|
|
mastering_voice_ = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SafeRelease(x_audio2_);
|
|
|
|
|
|
|
|
|
|
modules::MediaFoundation.MFShutdown();
|
|
|
|
|
|
|
|
|
|
modules::Destroy();
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
2018-05-14 00:36:01 +08:00
|
|
|
|
2018-11-14 01:34:41 +08:00
|
|
|
void AudioDevice::Init(bool debug)
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-12 22:36:50 +08:00
|
|
|
if (initialized)
|
|
|
|
|
return;
|
|
|
|
|
|
2018-11-15 17:59:18 +08:00
|
|
|
E2D_LOG("Initing audio device");
|
|
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
ThrowIfFailed(
|
|
|
|
|
modules::MediaFoundation.MFStartup(MF_VERSION, MFSTARTUP_FULL)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
ThrowIfFailed(
|
|
|
|
|
modules::XAudio2.XAudio2Create(&x_audio2_, 0, XAUDIO2_DEFAULT_PROCESSOR)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
ThrowIfFailed(
|
|
|
|
|
x_audio2_->CreateMasteringVoice(&mastering_voice_)
|
|
|
|
|
);
|
2018-11-12 22:36:50 +08:00
|
|
|
|
|
|
|
|
initialized = true;
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
2018-11-08 21:39:26 +08:00
|
|
|
|
2018-11-12 20:46:54 +08:00
|
|
|
HRESULT AudioDevice::CreateVoice(Voice* voice, WAVEFORMATEX * wfx)
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-12 20:46:54 +08:00
|
|
|
HRESULT hr;
|
|
|
|
|
IXAudio2SourceVoice* source_voice;
|
|
|
|
|
|
|
|
|
|
if (!voice)
|
|
|
|
|
return E_POINTER;
|
|
|
|
|
|
|
|
|
|
hr = x_audio2_->CreateSourceVoice(&source_voice, wfx, 0, XAUDIO2_DEFAULT_FREQ_RATIO);
|
|
|
|
|
if (SUCCEEDED(hr))
|
2018-11-08 21:39:26 +08:00
|
|
|
{
|
2018-11-12 20:46:54 +08:00
|
|
|
voice->SetSourceVoice(source_voice);
|
|
|
|
|
voice_cache_.push_back(voice);
|
2018-11-08 21:39:26 +08:00
|
|
|
}
|
2018-11-12 20:46:54 +08:00
|
|
|
return hr;
|
|
|
|
|
}
|
2018-05-08 20:03:29 +08:00
|
|
|
|
2018-11-12 20:46:54 +08:00
|
|
|
void AudioDevice::DeleteVoice(Voice * voice)
|
|
|
|
|
{
|
|
|
|
|
for (auto iter = voice_cache_.begin(); iter != voice_cache_.end(); ++iter)
|
|
|
|
|
{
|
|
|
|
|
if (*iter == voice)
|
|
|
|
|
{
|
|
|
|
|
voice_cache_.erase(iter);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
2018-05-08 20:03:29 +08:00
|
|
|
|
2018-11-12 20:46:54 +08:00
|
|
|
void AudioDevice::ClearVoiceCache()
|
2018-05-10 14:03:54 +08:00
|
|
|
{
|
2018-11-12 20:46:54 +08:00
|
|
|
for (auto voice : voice_cache_)
|
|
|
|
|
{
|
|
|
|
|
voice->Destroy();
|
|
|
|
|
}
|
|
|
|
|
voice_cache_.clear();
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
|
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
void AudioDevice::Open()
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-08 21:39:26 +08:00
|
|
|
x_audio2_->StartEngine();
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|
2018-11-08 21:39:26 +08:00
|
|
|
|
|
|
|
|
void AudioDevice::Close()
|
2018-11-08 00:21:59 +08:00
|
|
|
{
|
2018-11-08 21:39:26 +08:00
|
|
|
x_audio2_->StopEngine();
|
2018-05-10 14:03:54 +08:00
|
|
|
}
|
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
}
|