2015-05-24 04:55:12 +00:00
|
|
|
// Copyright 2009 Dolphin Emulator Project
|
2015-05-17 23:08:10 +00:00
|
|
|
// Licensed under GPLv2+
|
2013-04-21 06:17:03 +00:00
|
|
|
// Refer to the license.txt file included.
|
2009-09-09 21:26:33 +00:00
|
|
|
|
2015-07-07 13:30:27 +00:00
|
|
|
#include <mutex>
|
|
|
|
|
2014-02-17 10:18:15 +00:00
|
|
|
#include "AudioCommon/AlsaSoundStream.h"
|
2014-09-08 01:06:58 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2015-09-26 21:13:07 +00:00
|
|
|
#include "Common/Logging/Log.h"
|
2016-06-24 08:43:46 +00:00
|
|
|
#include "Common/Thread.h"
|
2009-09-09 21:26:33 +00:00
|
|
|
|
2015-05-24 08:13:02 +00:00
|
|
|
AlsaSound::AlsaSound()
|
2016-06-24 08:43:46 +00:00
|
|
|
: m_thread_status(ALSAThreadStatus::STOPPED), handle(nullptr),
|
|
|
|
frames_to_deliver(FRAME_COUNT_MIN)
|
2009-09-09 21:26:33 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-10-21 23:23:40 +00:00
|
|
|
AlsaSound::~AlsaSound()
|
2009-09-09 21:26:33 +00:00
|
|
|
{
|
2017-10-21 23:23:40 +00:00
|
|
|
m_thread_status.store(ALSAThreadStatus::STOPPING);
|
|
|
|
|
2019-04-06 22:39:25 +00:00
|
|
|
// Immediately lock and unlock mutex to prevent cv race.
|
|
|
|
std::unique_lock<std::mutex>{cv_m};
|
|
|
|
|
2017-10-21 23:23:40 +00:00
|
|
|
// Give the opportunity to the audio thread
|
|
|
|
// to realize we are stopping the emulation
|
|
|
|
cv.notify_one();
|
|
|
|
thread.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AlsaSound::Init()
|
|
|
|
{
|
|
|
|
m_thread_status.store(ALSAThreadStatus::PAUSED);
|
2016-06-24 08:43:46 +00:00
|
|
|
if (!AlsaInit())
|
|
|
|
{
|
|
|
|
m_thread_status.store(ALSAThreadStatus::STOPPED);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread = std::thread(&AlsaSound::SoundLoop, this);
|
|
|
|
return true;
|
2009-09-09 21:26:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AlsaSound::Update()
|
|
|
|
{
|
2016-06-24 08:43:46 +00:00
|
|
|
// don't need to do anything here.
|
2009-09-09 21:26:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Called on audio thread.
|
|
|
|
void AlsaSound::SoundLoop()
|
|
|
|
{
|
2016-06-24 08:43:46 +00:00
|
|
|
Common::SetCurrentThreadName("Audio thread - alsa");
|
|
|
|
while (m_thread_status.load() != ALSAThreadStatus::STOPPING)
|
|
|
|
{
|
|
|
|
while (m_thread_status.load() == ALSAThreadStatus::RUNNING)
|
|
|
|
{
|
|
|
|
m_mixer->Mix(mix_buffer, frames_to_deliver);
|
|
|
|
int rc = snd_pcm_writei(handle, mix_buffer, frames_to_deliver);
|
|
|
|
if (rc == -EPIPE)
|
|
|
|
{
|
|
|
|
// Underrun
|
|
|
|
snd_pcm_prepare(handle);
|
|
|
|
}
|
|
|
|
else if (rc < 0)
|
|
|
|
{
|
|
|
|
ERROR_LOG(AUDIO, "writei fail: %s", snd_strerror(rc));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (m_thread_status.load() == ALSAThreadStatus::PAUSED)
|
|
|
|
{
|
|
|
|
snd_pcm_drop(handle); // Stop sound output
|
|
|
|
|
|
|
|
// Block until thread status changes.
|
|
|
|
std::unique_lock<std::mutex> lock(cv_m);
|
|
|
|
cv.wait(lock, [this] { return m_thread_status.load() != ALSAThreadStatus::PAUSED; });
|
|
|
|
|
|
|
|
snd_pcm_prepare(handle); // resume sound output
|
|
|
|
}
|
|
|
|
}
|
|
|
|
AlsaShutdown();
|
|
|
|
m_thread_status.store(ALSAThreadStatus::STOPPED);
|
2009-09-09 21:26:33 +00:00
|
|
|
}
|
|
|
|
|
2017-10-21 23:23:40 +00:00
|
|
|
bool AlsaSound::SetRunning(bool running)
|
2015-07-07 13:30:27 +00:00
|
|
|
{
|
2017-10-21 19:37:39 +00:00
|
|
|
m_thread_status.store(running ? ALSAThreadStatus::RUNNING : ALSAThreadStatus::PAUSED);
|
2019-04-06 22:39:25 +00:00
|
|
|
|
|
|
|
// Immediately lock and unlock mutex to prevent cv race.
|
|
|
|
std::unique_lock<std::mutex>{cv_m};
|
|
|
|
|
|
|
|
// Notify thread that status has changed
|
|
|
|
cv.notify_one();
|
2017-10-21 23:23:40 +00:00
|
|
|
return true;
|
2015-07-07 13:30:27 +00:00
|
|
|
}
|
|
|
|
|
2009-09-09 21:26:33 +00:00
|
|
|
bool AlsaSound::AlsaInit()
|
|
|
|
{
|
2016-06-24 08:43:46 +00:00
|
|
|
unsigned int sample_rate = m_mixer->GetSampleRate();
|
|
|
|
int err;
|
|
|
|
int dir;
|
|
|
|
snd_pcm_sw_params_t* swparams;
|
|
|
|
snd_pcm_hw_params_t* hwparams;
|
|
|
|
snd_pcm_uframes_t buffer_size, buffer_size_max;
|
|
|
|
unsigned int periods;
|
|
|
|
|
|
|
|
err = snd_pcm_open(&handle, "default", SND_PCM_STREAM_PLAYBACK, 0);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Audio open error: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
snd_pcm_hw_params_alloca(&hwparams);
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_any(handle, hwparams);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Broken configuration for this PCM: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_set_access(handle, hwparams, SND_PCM_ACCESS_RW_INTERLEAVED);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Access type not available: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_set_format(handle, hwparams, SND_PCM_FORMAT_S16_LE);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Sample format not available: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
dir = 0;
|
|
|
|
err = snd_pcm_hw_params_set_rate_near(handle, hwparams, &sample_rate, &dir);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Rate not available: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_set_channels(handle, hwparams, CHANNEL_COUNT);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Channels count not available: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
periods = BUFFER_SIZE_MAX / FRAME_COUNT_MIN;
|
|
|
|
err = snd_pcm_hw_params_set_periods_max(handle, hwparams, &periods, &dir);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Cannot set maximum periods per buffer: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer_size_max = BUFFER_SIZE_MAX;
|
|
|
|
err = snd_pcm_hw_params_set_buffer_size_max(handle, hwparams, &buffer_size_max);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Cannot set maximum buffer size: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params(handle, hwparams);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Unable to install hw params: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_get_buffer_size(hwparams, &buffer_size);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Cannot get buffer size: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_hw_params_get_periods_max(hwparams, &periods, &dir);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Cannot get periods: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// periods is the number of fragments alsa can wait for during one
|
|
|
|
// buffer_size
|
|
|
|
frames_to_deliver = buffer_size / periods;
|
|
|
|
// limit the minimum size. pulseaudio advertises a minimum of 32 samples.
|
|
|
|
if (frames_to_deliver < FRAME_COUNT_MIN)
|
|
|
|
frames_to_deliver = FRAME_COUNT_MIN;
|
|
|
|
// it is probably a bad idea to try to send more than one buffer of data
|
|
|
|
if ((unsigned int)frames_to_deliver > buffer_size)
|
|
|
|
frames_to_deliver = buffer_size;
|
2018-04-12 12:18:04 +00:00
|
|
|
NOTICE_LOG(AUDIO,
|
|
|
|
"ALSA gave us a %ld sample \"hardware\" buffer with %d periods. Will send %d "
|
|
|
|
"samples per fragments.",
|
2016-06-24 08:43:46 +00:00
|
|
|
buffer_size, periods, frames_to_deliver);
|
|
|
|
|
|
|
|
snd_pcm_sw_params_alloca(&swparams);
|
|
|
|
|
|
|
|
err = snd_pcm_sw_params_current(handle, swparams);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "cannot init sw params: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_sw_params_set_start_threshold(handle, swparams, 0U);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "cannot set start thresh: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_sw_params(handle, swparams);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "cannot set sw params: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_pcm_prepare(handle);
|
|
|
|
if (err < 0)
|
|
|
|
{
|
2016-11-02 01:19:00 +00:00
|
|
|
ERROR_LOG(AUDIO, "Unable to prepare: %s", snd_strerror(err));
|
2016-06-24 08:43:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
2016-11-02 01:19:00 +00:00
|
|
|
NOTICE_LOG(AUDIO, "ALSA successfully initialized.");
|
2016-06-24 08:43:46 +00:00
|
|
|
return true;
|
2009-09-09 21:26:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AlsaSound::AlsaShutdown()
|
|
|
|
{
|
2016-06-24 08:43:46 +00:00
|
|
|
if (handle != nullptr)
|
|
|
|
{
|
|
|
|
snd_pcm_drop(handle);
|
|
|
|
snd_pcm_close(handle);
|
|
|
|
handle = nullptr;
|
|
|
|
}
|
2009-09-09 21:26:33 +00:00
|
|
|
}
|