Clean up float handling a bit.

ALSA now checks if floating point is supported to avoid one extra
conversion, also clean up the driver itself a bit.
This commit is contained in:
Themaister 2011-01-15 20:37:42 +01:00
parent e7e5363465
commit 5c9c4f42c3
6 changed files with 48 additions and 42 deletions

View File

@ -19,6 +19,7 @@
#include "driver.h" #include "driver.h"
#include <stdlib.h> #include <stdlib.h>
#include <alsa/asoundlib.h> #include <alsa/asoundlib.h>
#include "general.h"
#define TRY_ALSA(x) if ( x < 0 ) { \ #define TRY_ALSA(x) if ( x < 0 ) { \
goto error; \ goto error; \
@ -28,8 +29,26 @@ typedef struct alsa
{ {
snd_pcm_t *pcm; snd_pcm_t *pcm;
bool nonblock; bool nonblock;
bool has_float;
} alsa_t; } alsa_t;
static bool __alsa_use_float(void *data)
{
alsa_t *alsa = data;
return alsa->has_float;
}
static bool find_float_format(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
{
if (snd_pcm_hw_params_test_format(pcm, params, SND_PCM_FORMAT_FLOAT) == 0)
{
SSNES_LOG("ALSA: Using floating point format.\n");
return true;
}
SSNES_LOG("ALSA: Using signed 16-bit format.\n");
return false;
}
static void* __alsa_init(const char* device, int rate, int latency) static void* __alsa_init(const char* device, int rate, int latency)
{ {
alsa_t *alsa = calloc(1, sizeof(alsa_t)); alsa_t *alsa = calloc(1, sizeof(alsa_t));
@ -37,22 +56,19 @@ static void* __alsa_init(const char* device, int rate, int latency)
return NULL; return NULL;
snd_pcm_hw_params_t *params = NULL; snd_pcm_hw_params_t *params = NULL;
snd_pcm_sw_params_t *sw_params = NULL;
const char *alsa_dev = "default"; const char *alsa_dev = "default";
if ( device != NULL ) if ( device != NULL )
alsa_dev = device; alsa_dev = device;
//fprintf(stderr, "Opening device: %s\n", alsa_dev);
TRY_ALSA(snd_pcm_open(&alsa->pcm, alsa_dev, SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK)); TRY_ALSA(snd_pcm_open(&alsa->pcm, alsa_dev, SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK));
unsigned int latency_usec = latency * 1000; unsigned int latency_usec = latency * 1000;
snd_pcm_format_t format = SND_PCM_FORMAT_S16_LE;
unsigned int channels = 2;
TRY_ALSA(snd_pcm_hw_params_malloc(&params)); TRY_ALSA(snd_pcm_hw_params_malloc(&params));
alsa->has_float = find_float_format(alsa->pcm, params);
snd_pcm_format_t format = alsa->has_float ? SND_PCM_FORMAT_FLOAT : SND_PCM_FORMAT_S16;
unsigned int channels = 2;
TRY_ALSA(snd_pcm_hw_params_any(alsa->pcm, params)); TRY_ALSA(snd_pcm_hw_params_any(alsa->pcm, params));
TRY_ALSA(snd_pcm_hw_params_set_access(alsa->pcm, params, SND_PCM_ACCESS_RW_INTERLEAVED)); TRY_ALSA(snd_pcm_hw_params_set_access(alsa->pcm, params, SND_PCM_ACCESS_RW_INTERLEAVED));
@ -60,51 +76,30 @@ static void* __alsa_init(const char* device, int rate, int latency)
TRY_ALSA(snd_pcm_hw_params_set_channels(alsa->pcm, params, channels)); TRY_ALSA(snd_pcm_hw_params_set_channels(alsa->pcm, params, channels));
TRY_ALSA(snd_pcm_hw_params_set_rate(alsa->pcm, params, rate, 0)); TRY_ALSA(snd_pcm_hw_params_set_rate(alsa->pcm, params, rate, 0));
// We test if we can run the latencies we are allowed, if not, fallback to *_near. TRY_ALSA(snd_pcm_hw_params_set_buffer_time_near(alsa->pcm, params, &latency_usec, NULL));
unsigned periods = 4;
if ( snd_pcm_hw_params_set_buffer_time_max(alsa->pcm, params, &latency_usec, NULL) < 0) TRY_ALSA(snd_pcm_hw_params_set_periods_near(alsa->pcm, params, &periods, NULL));
{
latency_usec = latency * 1000;
TRY_ALSA(snd_pcm_hw_params_set_buffer_time_near(alsa->pcm, params, &latency_usec, NULL))
}
latency_usec = (latency < 32) ? 10000 : latency * 250;
if ( snd_pcm_hw_params_set_period_time_max(alsa->pcm, params, &latency_usec, NULL) )
{
latency_usec = (latency < 32) ? 10000 : latency * 250;
TRY_ALSA(snd_pcm_hw_params_set_period_time_near(alsa->pcm, params, &latency_usec, NULL));
}
TRY_ALSA(snd_pcm_hw_params(alsa->pcm, params)); TRY_ALSA(snd_pcm_hw_params(alsa->pcm, params));
snd_pcm_uframes_t buffer_size; snd_pcm_uframes_t buffer_size;
//snd_pcm_hw_params_get_period_size(params, &buffer_size, NULL); snd_pcm_hw_params_get_period_size(params, &buffer_size, NULL);
//fprintf(stderr, "ALSA Period size: %d frames\n", (int)alsa_sizes); SSNES_LOG("ALSA: Period size: %d frames\n", (int)buffer_size);
snd_pcm_hw_params_get_buffer_size(params, &buffer_size); snd_pcm_hw_params_get_buffer_size(params, &buffer_size);
//fprintf(stderr, "Buffer size: %d frames\n", (int)alsa_sizes); SSNES_LOG("ALSA: Buffer size: %d frames\n", (int)buffer_size);
if (snd_pcm_sw_params_malloc(&sw_params) < 0)
goto error;
TRY_ALSA(snd_pcm_sw_params_current(alsa->pcm, sw_params));
TRY_ALSA(snd_pcm_sw_params_set_start_threshold(alsa->pcm, sw_params, buffer_size/2));
TRY_ALSA(snd_pcm_sw_params(alsa->pcm, sw_params));
snd_pcm_sw_params_free(sw_params);
snd_pcm_hw_params_free(params); snd_pcm_hw_params_free(params);
return alsa; return alsa;
error: error:
if ( params != NULL ) SSNES_ERR("ALSA: Failed to initialize...\n");
if (params)
snd_pcm_hw_params_free(params); snd_pcm_hw_params_free(params);
if ( sw_params != NULL ) if (alsa)
snd_pcm_sw_params_free(sw_params);
if ( alsa != NULL )
{ {
if ( alsa->pcm != NULL ) if (alsa->pcm)
snd_pcm_close(alsa->pcm); snd_pcm_close(alsa->pcm);
free(alsa); free(alsa);
@ -119,7 +114,7 @@ static ssize_t __alsa_write(void* data, const void* buf, size_t size)
snd_pcm_sframes_t frames; snd_pcm_sframes_t frames;
snd_pcm_sframes_t written = 0; snd_pcm_sframes_t written = 0;
int rc; int rc;
size /= 4; // Frames to write size = snd_pcm_bytes_to_frames(alsa->pcm, size); // Frames to write
while (written < size) while (written < size)
{ {
@ -134,7 +129,7 @@ static ssize_t __alsa_write(void* data, const void* buf, size_t size)
} }
} }
frames = snd_pcm_writei(alsa->pcm, (const uint32_t*)buf + written, size - written); frames = snd_pcm_writei(alsa->pcm, (const char*)buf + written * 2 * (alsa->has_float ? sizeof(float) : sizeof(int16_t)), size - written);
if ( frames == -EPIPE || frames == -EINTR || frames == -ESTRPIPE ) if ( frames == -EPIPE || frames == -EINTR || frames == -ESTRPIPE )
{ {
@ -189,6 +184,7 @@ const audio_driver_t audio_alsa = {
.write = __alsa_write, .write = __alsa_write,
.stop = __alsa_stop, .stop = __alsa_stop,
.start = __alsa_start, .start = __alsa_start,
.use_float = __alsa_use_float,
.set_nonblock_state = __alsa_set_nonblock_state, .set_nonblock_state = __alsa_set_nonblock_state,
.free = __alsa_free, .free = __alsa_free,
.ident = "alsa" .ident = "alsa"

View File

@ -285,6 +285,12 @@ static void __jack_free(void *data)
free(jd); free(jd);
} }
static bool __jack_use_float(void *data)
{
(void)data;
return true;
}
const audio_driver_t audio_jack = { const audio_driver_t audio_jack = {
.init = __jack_init, .init = __jack_init,
.write = __jack_write, .write = __jack_write,
@ -292,7 +298,7 @@ const audio_driver_t audio_jack = {
.start = __jack_start, .start = __jack_start,
.set_nonblock_state = __jack_set_nonblock_state, .set_nonblock_state = __jack_set_nonblock_state,
.free = __jack_free, .free = __jack_free,
.float_samples = true, .use_float = __jack_use_float,
.ident = "jack" .ident = "jack"
}; };

View File

@ -149,6 +149,8 @@ void init_audio(void)
if ( driver.audio_data == NULL ) if ( driver.audio_data == NULL )
g_extern.audio_active = false; g_extern.audio_active = false;
if (g_extern.audio_active && driver.audio->use_float && driver.audio->use_float(driver.audio_data))
g_extern.audio_data.use_float = true;
if (!g_settings.audio.sync && g_extern.audio_active) if (!g_settings.audio.sync && g_extern.audio_active)
{ {

View File

@ -61,7 +61,7 @@ typedef struct audio_driver
bool (*start)(void* data); bool (*start)(void* data);
void (*set_nonblock_state)(void* data, bool toggle); // Should we care about blocking in audio thread? Fast forwarding. void (*set_nonblock_state)(void* data, bool toggle); // Should we care about blocking in audio thread? Fast forwarding.
void (*free)(void* data); void (*free)(void* data);
bool float_samples; // Defines if driver will take standard floating point samples, or int16_t samples. bool (*use_float)(void *data); // Defines if driver will take standard floating point samples, or int16_t samples.
const char *ident; const char *ident;
} audio_driver_t; } audio_driver_t;

View File

@ -125,6 +125,8 @@ struct global
size_t nonblock_chunk_size; size_t nonblock_chunk_size;
size_t block_chunk_size; size_t block_chunk_size;
bool use_float;
float *outsamples; float *outsamples;
int16_t *conv_outsamples; int16_t *conv_outsamples;
} audio_data; } audio_data;

View File

@ -176,7 +176,7 @@ static void audio_sample(uint16_t left, uint16_t right)
src_process(g_extern.source, &src_data); src_process(g_extern.source, &src_data);
if (driver.audio->float_samples) if (g_extern.audio_data.use_float)
{ {
if (driver.audio->write(driver.audio_data, g_extern.audio_data.outsamples, src_data.output_frames_gen * sizeof(float) * 2) < 0) if (driver.audio->write(driver.audio_data, g_extern.audio_data.outsamples, src_data.output_frames_gen * sizeof(float) * 2) < 0)
{ {