441a463737
chaining two soft volume knobs. Stop audio during pause to consume less CPU. Honour AOPLAY_FINAL_CHUNK flag which fixes playback of very short files. ok & tweaks stsp@, ok sthen@, dcoppa@, go ahead edd@ (maintainer)
344 lines
7.4 KiB
C
344 lines
7.4 KiB
C
/*
|
|
* Copyright (c) 2008 Alexandre Ratchov <alex@caoua.org>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <poll.h>
|
|
#include <errno.h>
|
|
#include <sndio.h>
|
|
|
|
#include "config.h"
|
|
#include "mp_msg.h"
|
|
#include "mixer.h"
|
|
#include "help_mp.h"
|
|
|
|
#include "libaf/af_format.h"
|
|
|
|
#include "audio_out.h"
|
|
#include "audio_out_internal.h"
|
|
|
|
static ao_info_t info = {
|
|
"sndio audio output",
|
|
"sndio",
|
|
"Alexandre Ratchov <alex@caoua.org>",
|
|
""
|
|
};
|
|
|
|
LIBAO_EXTERN(sndio)
|
|
|
|
static struct sio_hdl *hdl = NULL;
|
|
static struct sio_par par;
|
|
static int delay, vol, havevol;
|
|
#define SILENCE_NMAX 0x1000
|
|
static char silence[SILENCE_NMAX];
|
|
|
|
/*
|
|
* misc parameters (volume, etc...)
|
|
*/
|
|
static int control(int cmd, void *arg)
|
|
{
|
|
ao_control_vol_t *ctl = arg;
|
|
|
|
if (!havevol)
|
|
return CONTROL_FALSE;
|
|
switch (cmd) {
|
|
case AOCONTROL_GET_VOLUME:
|
|
ctl->left = ctl->right = vol * 100 / SIO_MAXVOL;
|
|
break;
|
|
case AOCONTROL_SET_VOLUME:
|
|
sio_setvol(hdl, ctl->left * SIO_MAXVOL / 100);
|
|
break;
|
|
default:
|
|
return CONTROL_UNKNOWN;
|
|
}
|
|
return CONTROL_OK;
|
|
}
|
|
|
|
/*
|
|
* call-back invoked to notify of the hardware position
|
|
*/
|
|
static void movecb(void *addr, int delta)
|
|
{
|
|
delay -= delta * (int)(par.bps * par.pchan);
|
|
}
|
|
|
|
/*
|
|
* call-back invoked to notify about volume changes
|
|
*/
|
|
static void volcb(void *addr, unsigned newvol)
|
|
{
|
|
vol = newvol;
|
|
}
|
|
|
|
/*
|
|
* open device and setup parameters
|
|
* return: 1 = success, 0 = failure
|
|
*/
|
|
static int init(int rate, int channels, int format, int flags)
|
|
{
|
|
int bpf;
|
|
int ac3 = 0;
|
|
|
|
hdl = sio_open(NULL, SIO_PLAY, 0);
|
|
if (hdl == NULL) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: can't open sndio\n");
|
|
return 0;
|
|
}
|
|
sio_initpar(&par);
|
|
switch (format) {
|
|
case AF_FORMAT_U8:
|
|
par.bits = 8;
|
|
par.sig = 0;
|
|
break;
|
|
case AF_FORMAT_S8:
|
|
par.bits = 8;
|
|
par.sig = 1;
|
|
break;
|
|
case AF_FORMAT_U16_LE:
|
|
par.bits = 16;
|
|
par.sig = 0;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_S16_LE:
|
|
par.bits = 16;
|
|
par.sig = 1;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_U16_BE:
|
|
par.bits = 16;
|
|
par.sig = 0;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_S16_BE:
|
|
par.bits = 16;
|
|
par.sig = 1;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_U24_LE:
|
|
par.bits = 24;
|
|
par.bps = 3;
|
|
par.sig = 0;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_S24_LE:
|
|
par.bits = 24;
|
|
par.bps = 3;
|
|
par.sig = 1;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_U24_BE:
|
|
par.bits = 24;
|
|
par.bps = 3;
|
|
par.sig = 0;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_S24_BE:
|
|
par.bits = 24;
|
|
par.bps = 3;
|
|
par.sig = 1;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_U32_LE:
|
|
par.bits = 32;
|
|
par.sig = 0;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_U32_BE:
|
|
par.bits = 32;
|
|
par.sig = 0;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_S32_LE:
|
|
par.bits = 32;
|
|
par.sig = 1;
|
|
par.le = 1;
|
|
break;
|
|
case AF_FORMAT_S32_BE:
|
|
par.bits = 32;
|
|
par.sig = 1;
|
|
par.le = 0;
|
|
break;
|
|
case AF_FORMAT_AC3_BE:
|
|
case AF_FORMAT_AC3_LE:
|
|
par.bits = 16;
|
|
par.sig = 1;
|
|
par.le = SIO_LE_NATIVE;
|
|
ac3 = 1;
|
|
break;
|
|
default:
|
|
mp_msg(MSGT_AO, MSGL_V, "ao2: unsupported format\n");
|
|
par.bits = 16;
|
|
par.sig = 1;
|
|
par.le = SIO_LE_NATIVE;
|
|
}
|
|
par.rate = rate;
|
|
par.pchan = channels;
|
|
par.appbufsz = par.rate * 250 / 1000; /* 250ms buffer */
|
|
par.round = par.rate * 10 / 1000; /* 10ms block size */
|
|
if (!sio_setpar(hdl, &par)) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: couldn't set params\n");
|
|
return 0;
|
|
}
|
|
if (!sio_getpar(hdl, &par)) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: couldn't get params\n");
|
|
return 0;
|
|
}
|
|
if (par.bits == 8 && par.bps == 1) {
|
|
format = par.sig ? AF_FORMAT_S8 : AF_FORMAT_U8;
|
|
} else if (par.bits == 16 && par.bps == 2) {
|
|
format = par.sig ?
|
|
(par.le ? AF_FORMAT_S16_LE : AF_FORMAT_S16_BE) :
|
|
(par.le ? AF_FORMAT_U16_LE : AF_FORMAT_U16_BE);
|
|
} else if ((par.bits == 24 || par.msb) && par.bps == 3) {
|
|
format = par.sig ?
|
|
(par.le ? AF_FORMAT_S24_LE : AF_FORMAT_S24_BE) :
|
|
(par.le ? AF_FORMAT_U24_LE : AF_FORMAT_U24_BE);
|
|
} else if ((par.bits == 32 || par.msb) && par.bps == 4) {
|
|
format = par.sig ?
|
|
(par.le ? AF_FORMAT_S32_LE : AF_FORMAT_S32_BE) :
|
|
(par.le ? AF_FORMAT_U32_LE : AF_FORMAT_U32_BE);
|
|
} else {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: couldn't set format\n");
|
|
return 0;
|
|
}
|
|
|
|
bpf = par.bps * par.pchan;
|
|
ao_data.channels = par.pchan;
|
|
ao_data.format = ac3 ? AF_FORMAT_AC3_NE : format;
|
|
ao_data.bps = bpf * par.rate;
|
|
ao_data.buffersize = par.appbufsz * bpf;
|
|
ao_data.outburst = par.round * bpf;
|
|
/* avoid resampling for close rates */
|
|
if ((par.rate >= rate * 0.97) && (par.rate <= rate * 1.03))
|
|
ao_data.samplerate = rate;
|
|
else
|
|
ao_data.samplerate = par.rate;
|
|
havevol = sio_onvol(hdl, volcb, NULL);
|
|
sio_onmove(hdl, movecb, NULL);
|
|
delay = 0;
|
|
if (!sio_start(hdl)) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: init: couldn't start\n");
|
|
}
|
|
if (ao_data.samplerate != rate) {
|
|
/* apparently mplayer rounds a little when resampling.
|
|
* anyway, it doesn't write quite a full buffer on the first
|
|
* write, which means libsndio never actually starts up
|
|
* because it's trying to fill the buffer. this is
|
|
* enough for everything I have come across.
|
|
*/
|
|
sio_write(hdl, silence, 8 * bpf);
|
|
delay += 8 * bpf;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* close device
|
|
*/
|
|
static void uninit(int immed)
|
|
{
|
|
if (hdl)
|
|
sio_close(hdl);
|
|
}
|
|
|
|
/*
|
|
* stop playing and empty buffers (for seeking/pause)
|
|
*/
|
|
static void reset(void)
|
|
{
|
|
if (!sio_stop(hdl)) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: reset: couldn't stop\n");
|
|
}
|
|
delay = 0;
|
|
if (!sio_start(hdl)) {
|
|
mp_msg(MSGT_AO, MSGL_ERR, "ao2: reset: couldn't start\n");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* how many bytes can be played without blocking
|
|
*/
|
|
static int get_space(void)
|
|
{
|
|
struct pollfd pfd;
|
|
int bufused, revents, n;
|
|
|
|
/*
|
|
* call poll() and sio_revents(), so the
|
|
* delay counter is updated
|
|
*/
|
|
n = sio_pollfd(hdl, &pfd, POLLOUT);
|
|
while (poll(&pfd, n, 0) < 0 && errno == EINTR)
|
|
; /* nothing */
|
|
revents = sio_revents(hdl, &pfd);
|
|
return par.bufsz * par.pchan * par.bps - delay;
|
|
}
|
|
|
|
/*
|
|
* play given number of bytes until sio_write() blocks
|
|
*/
|
|
static int play(void *data, int len, int flags)
|
|
{
|
|
int n;
|
|
|
|
n = sio_write(hdl, data, len);
|
|
delay += n;
|
|
if (flags & AOPLAY_FINAL_CHUNK)
|
|
reset();
|
|
return n;
|
|
}
|
|
|
|
/*
|
|
* return: delay in seconds between first and last sample in buffer
|
|
*/
|
|
static float get_delay(void)
|
|
{
|
|
return (float)delay / (par.bps * par.pchan * par.rate);
|
|
}
|
|
|
|
/*
|
|
* stop playing, keep buffers (for pause)
|
|
*/
|
|
static void audio_pause(void)
|
|
{
|
|
reset();
|
|
}
|
|
|
|
/*
|
|
* resume playing, after audio_pause()
|
|
*/
|
|
static void audio_resume(void)
|
|
{
|
|
int n, count, todo;
|
|
|
|
/*
|
|
* we want to start with buffers full, because mplayer uses
|
|
* get_space() pointer as clock, which would cause video to
|
|
* accelerate while buffers are filled. Remove this when not
|
|
* necessary anymore.
|
|
*/
|
|
todo = par.bufsz * par.pchan * par.bps;
|
|
while (todo > 0) {
|
|
count = todo;
|
|
if (count > SILENCE_NMAX)
|
|
count = SILENCE_NMAX;
|
|
n = sio_write(hdl, silence, count);
|
|
if (n == 0)
|
|
break;
|
|
todo -= n;
|
|
delay += n;
|
|
}
|
|
}
|