alsa-tools/ac3dec/output.c
2001-12-30 09:37:54 +00:00

203 lines
5.5 KiB
C

/*
* Copyright (c) by Jaroslav Kysela <perex@suse.cz>
*
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include <stdio.h>
#include <errno.h>
#include <alsa/asoundlib.h>
typedef signed short sint_16;
typedef unsigned int uint_32;
#include "output.h"
static output_t out_config;
static snd_pcm_t *pcm;
/*
* open the audio device for writing to
*/
int output_open(output_t *output)
{
const char *pcm_name = output->pcm_name;
char devstr[128];
snd_pcm_hw_params_t *params;
snd_pcm_sframes_t buffer_time;
snd_pcm_sframes_t period_time, tmp;
int err, step;
snd_pcm_hw_params_alloca(&params);
out_config = *output;
/*
* Open the device driver
*/
if (pcm_name == NULL) {
switch (output->channels) {
case 1:
case 2:
sprintf(devstr, "default");
if (output->spdif != SPDIF_NONE) {
unsigned char s[4];
if (output->spdif == SPDIF_PRO) {
s[0] = (IEC958_AES0_PROFESSIONAL |
IEC958_AES0_NONAUDIO |
IEC958_AES0_PRO_EMPHASIS_NONE |
IEC958_AES0_PRO_FS_48000);
s[1] = (IEC958_AES1_PRO_MODE_NOTID |
IEC958_AES1_PRO_USERBITS_NOTID);
s[2] = IEC958_AES2_PRO_WORDLEN_NOTID;
s[3] = 0;
} else {
s[0] = IEC958_AES0_CON_EMPHASIS_NONE;
if (output->spdif == SPDIF_CON)
s[0] |= IEC958_AES0_NONAUDIO;
s[1] = (IEC958_AES1_CON_ORIGINAL |
IEC958_AES1_CON_PCM_CODER);
s[2] = 0;
s[3] = IEC958_AES3_CON_FS_48000;
}
sprintf(devstr, "iec958:AES0=0x%x,AES1=0x%x,AES2=0x%x,AES3=0x%x", s[0], s[1], s[2], s[3]);
if (out_config.card)
sprintf(devstr + strlen(devstr), ",CARD=%s", out_config.card);
}
break;
case 4:
strcpy(devstr, "surround40");
if (out_config.card)
sprintf(devstr + strlen(devstr), ",CARD=%s", out_config.card);
break;
case 6:
strcpy(devstr, "surround51");
if (out_config.card)
sprintf(devstr + strlen(devstr), ",CARD=%s", out_config.card);
break;
default:
fprintf(stderr, "%d channels are not supported\n", output->channels);
return -EINVAL;
}
pcm_name = devstr;
}
if (!output->quiet)
fprintf(stdout, "Using PCM device '%s'\n", pcm_name);
if ((err = snd_pcm_open(&pcm, pcm_name, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
fprintf(stderr, "snd_pcm_open: %s\n", snd_strerror(err));
return err;
}
err = snd_pcm_hw_params_any(pcm, params);
if (err < 0) {
fprintf(stderr, "Broken configuration for this PCM: no configurations available");
goto __close;
}
/* set interleaved access */
err = snd_pcm_hw_params_set_access(pcm, params,
SND_PCM_ACCESS_RW_INTERLEAVED);
if (err < 0) {
fprintf(stderr, "Access type not available");
goto __close;
}
err = snd_pcm_hw_params_set_format(pcm, params, output->bits == 16 ? SND_PCM_FORMAT_S16_LE : SND_PCM_FORMAT_U8);
if (err < 0) {
fprintf(stderr, "Sample format non available");
goto __close;
}
err = snd_pcm_hw_params_set_channels(pcm, params, output->channels);
if (err < 0) {
fprintf(stderr, "Channels count non available");
goto __close;
}
err = snd_pcm_hw_params_set_rate_near(pcm, params, output->rate, 0);
if (err < 0) {
fprintf(stderr, "Rate not available");
goto __close;
}
buffer_time = snd_pcm_hw_params_set_buffer_time_near(pcm, params,
500000, 0);
if (buffer_time < 0) {
fprintf(stderr, "Buffer time not available");
goto __close;
}
step = 2;
period_time = 10000 * 2;
do {
period_time /= 2;
tmp = snd_pcm_hw_params_set_period_time_near(pcm, params,
period_time, 0);
if (tmp == period_time) {
period_time /= 3;
tmp = snd_pcm_hw_params_set_period_time_near(pcm, params,
period_time, 0);
if (tmp == period_time)
period_time = 10000 * 2;
}
if (period_time < 0) {
fprintf(stderr, "Period time not available");
goto __close;
}
} while (buffer_time == period_time && period_time > 10000);
if (buffer_time == period_time) {
fprintf(stderr, "Buffer time and period time match, could not use\n");
goto __close;
}
if ((err = snd_pcm_hw_params(pcm, params)) < 0) {
fprintf(stderr, "PCM hw_params failed: %s\n", snd_strerror(err));
goto __close;
}
return 0;
__close:
snd_pcm_close(pcm);
pcm = NULL;
return err;
}
/*
* play the sample to the already opened file descriptor
*/
int output_play(sint_16* output_samples, uint_32 num_frames)
{
snd_pcm_sframes_t res = 0;
do {
if (res == -EPIPE) /* underrun */
res = snd_pcm_prepare(pcm);
else if (res == -ESTRPIPE) { /* suspend */
while ((res = snd_pcm_resume(pcm)) == -EAGAIN)
sleep(1);
if (res < 0)
res = snd_pcm_prepare(pcm);
}
res = res < 0 ? res : snd_pcm_writei(pcm, (void *)output_samples, num_frames);
if (res > 0) {
output_samples += out_config.channels * res;
num_frames -= res;
}
} while (res == -EPIPE || num_frames > 0);
if (res < 0)
fprintf(stderr, "writei returned error: %s\n", snd_strerror(res));
return res < 0 ? (int)res : 0;
}
void
output_close(void)
{
snd_pcm_close(pcm);
}