Add alsa sound output support

This commit is contained in:
Cjacker 2021-11-23 15:20:01 +08:00
parent 64bf636b5c
commit d65789edc4
2 changed files with 215 additions and 9 deletions

View File

@ -489,6 +489,25 @@ else
S9XDEFS="$S9XDEFS -DNOSOUND" S9XDEFS="$S9XDEFS -DNOSOUND"
fi fi
# Check if we can build with alsa support
AC_ARG_ENABLE([sound-alsa],
[AS_HELP_STRING([--enable-sound-alsa],
[enable Alsa if available (default: yes)])],
[], [enable_sound_alsa="yes"])
if test "x$enable_sound_alsa" = "xyes"; then
enable_sound_alsa="no"
AC_CHECK_HEADER([alsa/asoundlib.h],
[
enable_sound_alsa="yes"
enable_sound="yes"
S9XLIBS="$S9XLIBS -lasound"
#in case '--disable-sound' and 'enable-sound-alsa'
S9XDEFS=$(echo $S9XDEFS|sed 's/-DNOSOUND//g; s/-DUSE_THREADS//g')
S9XDEFS="$S9XDEFS -DALSA"
])
fi
# Output. # Output.
S9XFLGS="$CXXFLAGS $CPPFLAGS $LDFLAGS $S9XFLGS" S9XFLGS="$CXXFLAGS $CPPFLAGS $LDFLAGS $S9XFLGS"
@ -528,6 +547,7 @@ features:
Xvideo support....... $enable_xvideo Xvideo support....... $enable_xvideo
Xinerama support..... $enable_xinerama Xinerama support..... $enable_xinerama
sound support........ $enable_sound sound support........ $enable_sound
alsa support......... $enable_sound_alsa
screenshot support... $enable_screenshot screenshot support... $enable_screenshot
netplay support...... $enable_netplay netplay support...... $enable_netplay
gamepad support...... $enable_gamepad gamepad support...... $enable_gamepad

View File

@ -27,10 +27,16 @@
#ifdef HAVE_SYS_IOCTL_H #ifdef HAVE_SYS_IOCTL_H
#include <sys/ioctl.h> #include <sys/ioctl.h>
#endif #endif
#ifndef NOSOUND #ifndef NOSOUND
#ifndef ALSA
#include <sys/soundcard.h> #include <sys/soundcard.h>
#include <sys/mman.h> #include <sys/mman.h>
#else
#include <alsa/asoundlib.h>
#endif #endif
#endif
#ifdef JOYSTICK_SUPPORT #ifdef JOYSTICK_SUPPORT
#include <linux/joystick.h> #include <linux/joystick.h>
#endif #endif
@ -114,8 +120,13 @@ struct SUnixSettings
struct SoundStatus struct SoundStatus
{ {
#ifndef ALSA
int sound_fd; int sound_fd;
uint32 fragment_size; uint32 fragment_size;
#else
snd_pcm_t *pcm_handle;
int output_buffer_size;
#endif
}; };
@ -165,7 +176,7 @@ static long log2 (long num)
namespace { namespace {
#if ! defined(NOSOUND) #if ! defined(NOSOUND) && ! defined(ALSA)
class S9xAudioOutput class S9xAudioOutput
{ {
public: public:
@ -348,14 +359,17 @@ void S9xExtraUsage (void)
S9xMessage(S9X_INFO, S9X_USAGE, ""); S9xMessage(S9X_INFO, S9X_USAGE, "");
#endif #endif
#ifdef USE_THREADS #ifndef NOSOUND
#ifdef USE_THREADS && ! defined(ALSA)
S9xMessage(S9X_INFO, S9X_USAGE, "-threadsound Use a separate thread to output sound"); S9xMessage(S9X_INFO, S9X_USAGE, "-threadsound Use a separate thread to output sound");
#endif #endif
S9xMessage(S9X_INFO, S9X_USAGE, "-buffersize Sound generating buffer size in millisecond"); S9xMessage(S9X_INFO, S9X_USAGE, "-buffersize Sound generating buffer size in millisecond");
#ifndef ALSA
S9xMessage(S9X_INFO, S9X_USAGE, "-fragmentsize Sound playback buffer fragment size in bytes"); S9xMessage(S9X_INFO, S9X_USAGE, "-fragmentsize Sound playback buffer fragment size in bytes");
#endif
S9xMessage(S9X_INFO, S9X_USAGE, "-sounddev <string> Specify sound device"); S9xMessage(S9X_INFO, S9X_USAGE, "-sounddev <string> Specify sound device");
S9xMessage(S9X_INFO, S9X_USAGE, ""); S9xMessage(S9X_INFO, S9X_USAGE, "");
#endif
S9xMessage(S9X_INFO, S9X_USAGE, "-loadsnapshot Load snapshot file at start"); S9xMessage(S9X_INFO, S9X_USAGE, "-loadsnapshot Load snapshot file at start");
S9xMessage(S9X_INFO, S9X_USAGE, "-playmovie <filename> Start emulator playing the .smv file"); S9xMessage(S9X_INFO, S9X_USAGE, "-playmovie <filename> Start emulator playing the .smv file");
S9xMessage(S9X_INFO, S9X_USAGE, "-recordmovie <filename> Start emulator recording the .smv file"); S9xMessage(S9X_INFO, S9X_USAGE, "-recordmovie <filename> Start emulator recording the .smv file");
@ -588,8 +602,11 @@ void S9xParsePortConfig (ConfigFile &conf, int pass)
#endif #endif
unixSettings.SoundBufferSize = conf.GetUInt ("Unix::SoundBufferSize", 100); unixSettings.SoundBufferSize = conf.GetUInt ("Unix::SoundBufferSize", 100);
unixSettings.SoundFragmentSize = conf.GetUInt ("Unix::SoundFragmentSize", 2048); unixSettings.SoundFragmentSize = conf.GetUInt ("Unix::SoundFragmentSize", 2048);
#ifndef ALSA
sound_device = conf.GetStringDup("Unix::SoundDevice", "/dev/dsp"); sound_device = conf.GetStringDup("Unix::SoundDevice", "/dev/dsp");
#else
sound_device = conf.GetStringDup("Unix::SoundDevice", "default");
#endif
keymaps.clear(); keymaps.clear();
if (!conf.GetBool("Unix::ClearAllControls", false)) if (!conf.GetBool("Unix::ClearAllControls", false))
{ {
@ -1335,9 +1352,25 @@ void S9xSamplesAvailable(void *data)
static uint8 *sound_buffer = NULL; static uint8 *sound_buffer = NULL;
static int sound_buffer_size = 0; static int sound_buffer_size = 0;
#ifdef ALSA
snd_pcm_sframes_t frames_written, frames;
frames = snd_pcm_avail(so.pcm_handle);
if (frames < 0)
{
frames = snd_pcm_recover(so.pcm_handle, frames, 1);
return;
}
#endif
if (Settings.DynamicRateControl) if (Settings.DynamicRateControl)
{ {
#ifndef ALSA
S9xUpdateDynamicRate(s_AudioOutput->GetFreeBufferSize(), so.fragment_size * 4); S9xUpdateDynamicRate(s_AudioOutput->GetFreeBufferSize(), so.fragment_size * 4);
#else
S9xUpdateDynamicRate(snd_pcm_frames_to_bytes(so.pcm_handle, frames),
so.output_buffer_size);
#endif
} }
samples_to_write = S9xGetSampleCount(); samples_to_write = S9xGetSampleCount();
@ -1345,21 +1378,79 @@ void S9xSamplesAvailable(void *data)
if (samples_to_write < 0) if (samples_to_write < 0)
return; return;
#ifdef ALSA
if (Settings.DynamicRateControl && !Settings.SoundSync)
{
// Using rate control, we should always keep the emulator's sound buffers empty to
// maintain an accurate measurement.
if (frames < samples_to_write/2)
{
S9xClearSamples();
return;
}
}
if (Settings.SoundSync && !Settings.TurboMode && !Settings.Mute)
{
snd_pcm_nonblock(so.pcm_handle, 0);
frames = samples_to_write/2;
} else {
snd_pcm_nonblock(so.pcm_handle, 1);
frames = MIN(frames, samples_to_write/2);
}
int bytes = snd_pcm_frames_to_bytes(so.pcm_handle, frames);
if (bytes <= 0)
return;
if (sound_buffer_size < bytes || sound_buffer == NULL)
{
sound_buffer = (uint8 *)realloc(sound_buffer, bytes);
sound_buffer_size = bytes;
}
#else //OSS
if (sound_buffer_size < samples_to_write * 2) if (sound_buffer_size < samples_to_write * 2)
{ {
sound_buffer = (uint8 *)realloc(sound_buffer, samples_to_write * 2); sound_buffer = (uint8 *)realloc(sound_buffer, samples_to_write * 2);
sound_buffer_size = samples_to_write * 2; sound_buffer_size = samples_to_write * 2;
} }
#endif
S9xMixSamples(sound_buffer, samples_to_write); S9xMixSamples(sound_buffer, samples_to_write);
#ifndef ALSA
s_AudioOutput->Write(sound_buffer, samples_to_write * 2); s_AudioOutput->Write(sound_buffer, samples_to_write * 2);
#endif #else
frames_written = 0;
while (frames_written < frames) {
int result;
result = snd_pcm_writei(so.pcm_handle,
sound_buffer +
snd_pcm_frames_to_bytes(so.pcm_handle, frames_written),
frames - frames_written);
if (result < 0)
{
result = snd_pcm_recover(so.pcm_handle, result, 1);
if (result < 0)
{
break;
}
}
else
{
frames_written += result;
}
}
#endif //ALSA
#endif //NOSOUND
} }
bool8 S9xOpenSoundDevice (void) bool8 S9xOpenSoundDevice (void)
{ {
#ifndef NOSOUND #ifndef NOSOUND
#ifndef ALSA
int J, K; int J, K;
so.sound_fd = open(sound_device, O_WRONLY | O_NONBLOCK); so.sound_fd = open(sound_device, O_WRONLY | O_NONBLOCK);
@ -1397,8 +1488,103 @@ bool8 S9xOpenSoundDevice (void)
so.fragment_size = J; so.fragment_size = J;
printf("fragment size: %d\n", J); printf("fragment size: %d\n", J);
#endif
#else
int err;
unsigned int periods = 8;
unsigned int buffer_size = unixSettings.SoundBufferSize * 1000;
snd_pcm_sw_params_t *sw_params;
snd_pcm_hw_params_t *hw_params;
snd_pcm_uframes_t alsa_buffer_size, alsa_period_size;
unsigned int min = 0;
unsigned int max = 0;
unsigned int rate = Settings.SoundPlaybackRate;
err = snd_pcm_open(&so.pcm_handle, sound_device, SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK);
if (err < 0) {
printf("Failed: %s\n", snd_strerror(err));
return (FALSE);
}
snd_pcm_hw_params_alloca(&hw_params);
snd_pcm_hw_params_any(so.pcm_handle, hw_params);
snd_pcm_hw_params_set_format(so.pcm_handle, hw_params, SND_PCM_FORMAT_S16);
snd_pcm_hw_params_set_access(so.pcm_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
snd_pcm_hw_params_set_rate_resample(so.pcm_handle, hw_params, 0);
snd_pcm_hw_params_set_channels(so.pcm_handle, hw_params, 2);
snd_pcm_hw_params_get_rate_min(hw_params, &min, NULL);
snd_pcm_hw_params_get_rate_max(hw_params, &max, NULL);
fprintf(stderr, "Alsa available rates: %d to %d\n", min, max);
if (rate > max && rate < min)
{
fprintf(stderr, "Rate %d not available. Using %d instead.\n", rate, max);
rate = max;
}
snd_pcm_hw_params_set_rate_near(so.pcm_handle, hw_params, &rate, NULL);
snd_pcm_hw_params_get_buffer_time_min(hw_params, &min, NULL);
snd_pcm_hw_params_get_buffer_time_max(hw_params, &max, NULL);
fprintf(stderr, "Alsa available buffer sizes: %dms to %dms\n", min / 1000, max / 1000);
if (buffer_size < min && buffer_size > max)
{
fprintf(stderr, "Buffer size %dms not available. Using %d instead.\n", buffer_size / 1000, (min + max) / 2000);
buffer_size = (min + max) / 2;
}
snd_pcm_hw_params_set_buffer_time_near(so.pcm_handle, hw_params, &buffer_size, NULL);
snd_pcm_hw_params_get_periods_min(hw_params, &min, NULL);
snd_pcm_hw_params_get_periods_max(hw_params, &max, NULL);
fprintf(stderr, "Alsa period ranges: %d to %d blocks\n", min, max);
if (periods > max)
periods = max;
snd_pcm_hw_params_set_periods_near(so.pcm_handle, hw_params, &periods, NULL);
err = snd_pcm_hw_params(so.pcm_handle, hw_params);
if (err < 0)
{
fprintf(stderr, "Alsa error: unable to install hw params\n");
snd_pcm_drain(so.pcm_handle);
snd_pcm_close(so.pcm_handle);
so.pcm_handle = NULL;
return (FALSE);
}
snd_pcm_sw_params_alloca(&sw_params);
snd_pcm_sw_params_current(so.pcm_handle, sw_params);
snd_pcm_get_params(so.pcm_handle, &alsa_buffer_size, &alsa_period_size);
/* Don't start until we're [nearly] full */
snd_pcm_sw_params_set_start_threshold(so.pcm_handle, sw_params, (alsa_buffer_size / 2));
err = snd_pcm_sw_params(so.pcm_handle, sw_params);
if (err < 0) {
fprintf(stderr, "Alsa error: unable to install sw params\n");
snd_pcm_drain(so.pcm_handle);
snd_pcm_close(so.pcm_handle);
so.pcm_handle = NULL;
return (FALSE);
}
err = snd_pcm_prepare(so.pcm_handle);
if (err < 0) {
fprintf(stderr, "Alsa error: unable to prepare audio: %s\n", snd_strerror(err));
snd_pcm_drain(so.pcm_handle);
snd_pcm_close(so.pcm_handle);
so.pcm_handle = NULL;
return (FALSE);
}
so.output_buffer_size = snd_pcm_frames_to_bytes(so.pcm_handle, alsa_buffer_size);
#endif //ALSA
#endif //NOSOUND
S9xSetSamplesAvailableCallback(S9xSamplesAvailable, NULL); S9xSetSamplesAvailableCallback(S9xSamplesAvailable, NULL);
return (TRUE); return (TRUE);
@ -1417,7 +1603,7 @@ void S9xExit (void)
S9xNPDisconnect(); S9xNPDisconnect();
#endif #endif
#ifndef NOSOUND #if !defined(NOSOUND) && !defined(ALSA)
delete s_AudioOutput; delete s_AudioOutput;
#endif #endif