mirror of
https://github.com/mpv-player/mpv
synced 2025-04-01 23:00:41 +00:00
audio/out: make draining a separate operation
Until now, this was always conflated with uninit. This was ugly, and also many AOs emulated this manually (or just ignored it). Make draining an explicit operation, so AOs which support it can provide it, and for all others generic code will emulate it. For ao_wasapi, we keep it simple and basically disable the internal draining implementation (maybe it should be restored later). Tested on Linux only.
This commit is contained in:
parent
2f03dc2599
commit
e16c91d07a
@ -235,11 +235,10 @@ done:
|
|||||||
return ao;
|
return ao;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Uninitialize and destroy the AO.
|
// Uninitialize and destroy the AO. Remaining audio must be dropped.
|
||||||
// cut_audio: if false, block until all remaining audio was played.
|
void ao_uninit(struct ao *ao)
|
||||||
void ao_uninit(struct ao *ao, bool cut_audio)
|
|
||||||
{
|
{
|
||||||
ao->api->uninit(ao, cut_audio);
|
ao->api->uninit(ao);
|
||||||
talloc_free(ao);
|
talloc_free(ao);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -315,13 +314,22 @@ void ao_resume(struct ao *ao)
|
|||||||
ao->api->resume(ao);
|
ao->api->resume(ao);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait until the audio buffer is drained. This can be used to emulate draining
|
// Be careful with locking
|
||||||
// if native functionality is not available.
|
|
||||||
// Only call this on uninit (otherwise, deadlock trouble ahead).
|
|
||||||
void ao_wait_drain(struct ao *ao)
|
void ao_wait_drain(struct ao *ao)
|
||||||
{
|
{
|
||||||
// This is probably not entirely accurate, but good enough.
|
// This is probably not entirely accurate, but good enough.
|
||||||
mp_sleep_us(ao_get_delay(ao) * 1000000);
|
mp_sleep_us(ao_get_delay(ao) * 1000000);
|
||||||
|
ao_reset(ao);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Block until the current audio buffer has played completely.
|
||||||
|
void ao_drain(struct ao *ao)
|
||||||
|
{
|
||||||
|
if (ao->api->drain) {
|
||||||
|
ao->api->drain(ao);
|
||||||
|
} else {
|
||||||
|
ao_wait_drain(ao);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ao_chmap_sel_adjust(struct ao *ao, const struct mp_chmap_sel *s,
|
bool ao_chmap_sel_adjust(struct ao *ao, const struct mp_chmap_sel *s,
|
||||||
|
@ -59,7 +59,7 @@ struct ao *ao_init_best(struct mpv_global *global,
|
|||||||
struct input_ctx *input_ctx,
|
struct input_ctx *input_ctx,
|
||||||
struct encode_lavc_context *encode_lavc_ctx,
|
struct encode_lavc_context *encode_lavc_ctx,
|
||||||
int samplerate, int format, struct mp_chmap channels);
|
int samplerate, int format, struct mp_chmap channels);
|
||||||
void ao_uninit(struct ao *ao, bool cut_audio);
|
void ao_uninit(struct ao *ao);
|
||||||
void ao_get_format(struct ao *ao, struct mp_audio *format);
|
void ao_get_format(struct ao *ao, struct mp_audio *format);
|
||||||
const char *ao_get_name(struct ao *ao);
|
const char *ao_get_name(struct ao *ao);
|
||||||
const char *ao_get_description(struct ao *ao);
|
const char *ao_get_description(struct ao *ao);
|
||||||
@ -71,5 +71,6 @@ int ao_get_space(struct ao *ao);
|
|||||||
void ao_reset(struct ao *ao);
|
void ao_reset(struct ao *ao);
|
||||||
void ao_pause(struct ao *ao);
|
void ao_pause(struct ao *ao);
|
||||||
void ao_resume(struct ao *ao);
|
void ao_resume(struct ao *ao);
|
||||||
|
void ao_drain(struct ao *ao);
|
||||||
|
|
||||||
#endif /* MPLAYER_AUDIO_OUT_H */
|
#endif /* MPLAYER_AUDIO_OUT_H */
|
||||||
|
@ -79,7 +79,7 @@ struct priv {
|
|||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
static float get_delay(struct ao *ao);
|
static float get_delay(struct ao *ao);
|
||||||
static void uninit(struct ao *ao, bool immed);
|
static void uninit(struct ao *ao);
|
||||||
|
|
||||||
/* to set/get/query special features/parameters */
|
/* to set/get/query special features/parameters */
|
||||||
static int control(struct ao *ao, enum aocontrol cmd, void *arg)
|
static int control(struct ao *ao, enum aocontrol cmd, void *arg)
|
||||||
@ -530,22 +530,19 @@ static int init(struct ao *ao)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
alsa_error:
|
alsa_error:
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
} // end init
|
} // end init
|
||||||
|
|
||||||
|
|
||||||
/* close audio device */
|
/* close audio device */
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
|
|
||||||
if (p->alsa) {
|
if (p->alsa) {
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!immed)
|
|
||||||
snd_pcm_drain(p->alsa);
|
|
||||||
|
|
||||||
err = snd_pcm_close(p->alsa);
|
err = snd_pcm_close(p->alsa);
|
||||||
CHECK_ALSA_ERROR("pcm close error");
|
CHECK_ALSA_ERROR("pcm close error");
|
||||||
|
|
||||||
@ -556,6 +553,12 @@ alsa_error:
|
|||||||
p->alsa = NULL;
|
p->alsa = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void drain(struct ao *ao)
|
||||||
|
{
|
||||||
|
struct priv *p = ao->priv;
|
||||||
|
snd_pcm_drain(p->alsa);
|
||||||
|
}
|
||||||
|
|
||||||
static void audio_pause(struct ao *ao)
|
static void audio_pause(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
@ -712,6 +715,7 @@ const struct ao_driver audio_out_alsa = {
|
|||||||
.pause = audio_pause,
|
.pause = audio_pause,
|
||||||
.resume = audio_resume,
|
.resume = audio_resume,
|
||||||
.reset = reset,
|
.reset = reset,
|
||||||
|
.drain = drain,
|
||||||
.priv_size = sizeof(struct priv),
|
.priv_size = sizeof(struct priv),
|
||||||
.priv_defaults = &(const struct priv) {
|
.priv_defaults = &(const struct priv) {
|
||||||
.cfg_block = 1,
|
.cfg_block = 1,
|
||||||
|
@ -626,14 +626,11 @@ static float get_delay(struct ao *ao)
|
|||||||
return mp_ring_buffered(p->buffer) / (float)ao->bps;
|
return mp_ring_buffered(p->buffer) / (float)ao->bps;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
OSStatus err = noErr;
|
OSStatus err = noErr;
|
||||||
|
|
||||||
if (!immed)
|
|
||||||
mp_sleep_us(get_delay(ao) * 1000000);
|
|
||||||
|
|
||||||
if (!p->is_digital) {
|
if (!p->is_digital) {
|
||||||
AudioOutputUnitStop(p->audio_unit);
|
AudioOutputUnitStop(p->audio_unit);
|
||||||
AudioUnitUninitialize(p->audio_unit);
|
AudioUnitUninitialize(p->audio_unit);
|
||||||
|
@ -546,10 +546,8 @@ static void audio_resume(struct ao *ao)
|
|||||||
\brief close audio device
|
\brief close audio device
|
||||||
\param immed stop playback immediately
|
\param immed stop playback immediately
|
||||||
*/
|
*/
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
if (!immed)
|
|
||||||
mp_sleep_us(get_delay(ao) * 1000000);
|
|
||||||
reset(ao);
|
reset(ao);
|
||||||
|
|
||||||
DestroyBuffer(ao);
|
DestroyBuffer(ao);
|
||||||
|
@ -207,13 +207,10 @@ err_chmap:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
|
|
||||||
if (!immed)
|
|
||||||
ao_wait_drain(ao);
|
|
||||||
|
|
||||||
jack_client_close(p->client);
|
jack_client_close(p->client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -181,7 +181,7 @@ fail:
|
|||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static int encode(struct ao *ao, double apts, void **data);
|
static int encode(struct ao *ao, double apts, void **data);
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *ac = ao->priv;
|
struct priv *ac = ao->priv;
|
||||||
struct encode_lavc_context *ectx = ao->encode_lavc_ctx;
|
struct encode_lavc_context *ectx = ao->encode_lavc_ctx;
|
||||||
@ -462,6 +462,11 @@ static int play(struct ao *ao, void **data, int samples, int flags)
|
|||||||
return bufpos;
|
return bufpos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void drain(struct ao *ao)
|
||||||
|
{
|
||||||
|
// pretend we support it, so generic code doesn't force a wait
|
||||||
|
}
|
||||||
|
|
||||||
const struct ao_driver audio_out_lavc = {
|
const struct ao_driver audio_out_lavc = {
|
||||||
.encode = true,
|
.encode = true,
|
||||||
.description = "audio encoding using libavcodec",
|
.description = "audio encoding using libavcodec",
|
||||||
@ -470,4 +475,5 @@ const struct ao_driver audio_out_lavc = {
|
|||||||
.uninit = uninit,
|
.uninit = uninit,
|
||||||
.get_space = get_space,
|
.get_space = get_space,
|
||||||
.play = play,
|
.play = play,
|
||||||
|
.drain = drain,
|
||||||
};
|
};
|
||||||
|
@ -98,10 +98,15 @@ static int init(struct ao *ao)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wait_drain(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
if (!cut_audio && !priv->paused)
|
drain(ao);
|
||||||
|
if (!priv->paused)
|
||||||
mp_sleep_us(1000000.0 * priv->buffered / ao->samplerate / priv->speed);
|
mp_sleep_us(1000000.0 * priv->buffered / ao->samplerate / priv->speed);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -185,6 +190,7 @@ const struct ao_driver audio_out_null = {
|
|||||||
.get_delay = get_delay,
|
.get_delay = get_delay,
|
||||||
.pause = pause,
|
.pause = pause,
|
||||||
.resume = resume,
|
.resume = resume,
|
||||||
|
.drain = wait_drain,
|
||||||
.priv_size = sizeof(struct priv),
|
.priv_size = sizeof(struct priv),
|
||||||
.priv_defaults = &(const struct priv) {
|
.priv_defaults = &(const struct priv) {
|
||||||
.bufferlen = 0.2,
|
.bufferlen = 0.2,
|
||||||
|
@ -178,11 +178,19 @@ err_out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
ALCcontext *ctx = alcGetCurrentContext();
|
ALCcontext *ctx = alcGetCurrentContext();
|
||||||
ALCdevice *dev = alcGetContextsDevice(ctx);
|
ALCdevice *dev = alcGetContextsDevice(ctx);
|
||||||
if (!immed) {
|
reset(ao);
|
||||||
|
alcMakeContextCurrent(NULL);
|
||||||
|
alcDestroyContext(ctx);
|
||||||
|
alcCloseDevice(dev);
|
||||||
|
ao_data = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void drain(struct ao *ao)
|
||||||
|
{
|
||||||
ALint state;
|
ALint state;
|
||||||
alGetSourcei(sources[0], AL_SOURCE_STATE, &state);
|
alGetSourcei(sources[0], AL_SOURCE_STATE, &state);
|
||||||
while (state == AL_PLAYING) {
|
while (state == AL_PLAYING) {
|
||||||
@ -190,12 +198,6 @@ static void uninit(struct ao *ao, bool immed)
|
|||||||
alGetSourcei(sources[0], AL_SOURCE_STATE, &state);
|
alGetSourcei(sources[0], AL_SOURCE_STATE, &state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
reset(ao);
|
|
||||||
alcMakeContextCurrent(NULL);
|
|
||||||
alcDestroyContext(ctx);
|
|
||||||
alcCloseDevice(dev);
|
|
||||||
ao_data = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void unqueue_buffers(void)
|
static void unqueue_buffers(void)
|
||||||
{
|
{
|
||||||
@ -298,6 +300,7 @@ const struct ao_driver audio_out_openal = {
|
|||||||
.pause = audio_pause,
|
.pause = audio_pause,
|
||||||
.resume = audio_resume,
|
.resume = audio_resume,
|
||||||
.reset = reset,
|
.reset = reset,
|
||||||
|
.drain = drain,
|
||||||
.priv_size = sizeof(struct priv),
|
.priv_size = sizeof(struct priv),
|
||||||
.options = (const struct m_option[]) {
|
.options = (const struct m_option[]) {
|
||||||
OPT_STRING_VALIDATE("device", cfg_device, 0, validate_device_opt),
|
OPT_STRING_VALIDATE("device", cfg_device, 0, validate_device_opt),
|
||||||
|
@ -427,24 +427,28 @@ ac3_retry:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
if (p->audio_fd == -1)
|
if (p->audio_fd == -1)
|
||||||
return;
|
return;
|
||||||
#ifdef SNDCTL_DSP_SYNC
|
|
||||||
// to get the buffer played
|
|
||||||
if (!immed)
|
|
||||||
ioctl(p->audio_fd, SNDCTL_DSP_SYNC, NULL);
|
|
||||||
#endif
|
|
||||||
#ifdef SNDCTL_DSP_RESET
|
#ifdef SNDCTL_DSP_RESET
|
||||||
if (immed)
|
|
||||||
ioctl(p->audio_fd, SNDCTL_DSP_RESET, NULL);
|
ioctl(p->audio_fd, SNDCTL_DSP_RESET, NULL);
|
||||||
#endif
|
#endif
|
||||||
close(p->audio_fd);
|
close(p->audio_fd);
|
||||||
p->audio_fd = -1;
|
p->audio_fd = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void drain(struct ao *ao)
|
||||||
|
{
|
||||||
|
#ifdef SNDCTL_DSP_SYNC
|
||||||
|
struct priv *p = ao->priv;
|
||||||
|
// to get the buffer played
|
||||||
|
if (p->audio_fd != -1)
|
||||||
|
ioctl(p->audio_fd, SNDCTL_DSP_SYNC, NULL);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
#ifndef SNDCTL_DSP_RESET
|
#ifndef SNDCTL_DSP_RESET
|
||||||
static void close_device(struct ao *ao)
|
static void close_device(struct ao *ao)
|
||||||
{
|
{
|
||||||
@ -601,6 +605,7 @@ const struct ao_driver audio_out_oss = {
|
|||||||
.pause = audio_pause,
|
.pause = audio_pause,
|
||||||
.resume = audio_resume,
|
.resume = audio_resume,
|
||||||
.reset = reset,
|
.reset = reset,
|
||||||
|
.drain = drain,
|
||||||
.priv_size = sizeof(struct priv),
|
.priv_size = sizeof(struct priv),
|
||||||
.priv_defaults = &(const struct priv) {
|
.priv_defaults = &(const struct priv) {
|
||||||
.audio_fd = -1,
|
.audio_fd = -1,
|
||||||
|
@ -166,7 +166,7 @@ static int init(struct ao *ao)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// close audio device
|
// close audio device
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
|
|
||||||
|
@ -146,16 +146,13 @@ static int stream_callback(const void *input,
|
|||||||
return paContinue;
|
return paContinue;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
|
|
||||||
if (priv->stream) {
|
if (priv->stream) {
|
||||||
if (!cut_audio && Pa_IsStreamActive(priv->stream) == 1) {
|
if (Pa_IsStreamActive(priv->stream) == 1)
|
||||||
ao_wait_drain(ao);
|
|
||||||
|
|
||||||
CHECK_PA_RET(Pa_StopStream(priv->stream));
|
CHECK_PA_RET(Pa_StopStream(priv->stream));
|
||||||
}
|
|
||||||
CHECK_PA_RET(Pa_CloseStream(priv->stream));
|
CHECK_PA_RET(Pa_CloseStream(priv->stream));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -220,7 +217,7 @@ static int init(struct ao *ao)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error_exit:
|
error_exit:
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -205,13 +205,18 @@ static bool select_chmap(struct ao *ao, pa_channel_map *dst)
|
|||||||
chmap_pa_from_mp(dst, &ao->channels);
|
chmap_pa_from_mp(dst, &ao->channels);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void drain(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
if (priv->stream && !cut_audio) {
|
if (priv->stream) {
|
||||||
pa_threaded_mainloop_lock(priv->mainloop);
|
pa_threaded_mainloop_lock(priv->mainloop);
|
||||||
waitop(priv, pa_stream_drain(priv->stream, success_cb, ao));
|
waitop(priv, pa_stream_drain(priv->stream, success_cb, ao));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void uninit(struct ao *ao)
|
||||||
|
{
|
||||||
|
struct priv *priv = ao->priv;
|
||||||
|
|
||||||
if (priv->mainloop)
|
if (priv->mainloop)
|
||||||
pa_threaded_mainloop_stop(priv->mainloop);
|
pa_threaded_mainloop_stop(priv->mainloop);
|
||||||
@ -366,7 +371,7 @@ fail:
|
|||||||
if (proplist)
|
if (proplist)
|
||||||
pa_proplist_free(proplist);
|
pa_proplist_free(proplist);
|
||||||
|
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -620,6 +625,7 @@ const struct ao_driver audio_out_pulse = {
|
|||||||
.get_delay = get_delay,
|
.get_delay = get_delay,
|
||||||
.pause = pause,
|
.pause = pause,
|
||||||
.resume = resume,
|
.resume = resume,
|
||||||
|
.drain = drain,
|
||||||
.priv_size = sizeof(struct priv),
|
.priv_size = sizeof(struct priv),
|
||||||
.priv_defaults = &(const struct priv) {
|
.priv_defaults = &(const struct priv) {
|
||||||
.cfg_buffer = 250,
|
.cfg_buffer = 250,
|
||||||
|
@ -128,15 +128,9 @@ static int init(struct ao *ao)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
/* The API does not provide a direct way to explicitly wait until
|
|
||||||
* the last byte has been played server-side as this cannot be
|
|
||||||
* guaranteed by backend drivers, so we approximate this behavior.
|
|
||||||
*/
|
|
||||||
if (!cut_audio)
|
|
||||||
mp_sleep_us(rsd_delay_ms(priv->rd) * 1000);
|
|
||||||
|
|
||||||
rsd_stop(priv->rd);
|
rsd_stop(priv->rd);
|
||||||
rsd_free(priv->rd);
|
rsd_free(priv->rd);
|
||||||
|
@ -78,7 +78,7 @@ static void audio_callback(void *userdata, Uint8 *stream, int len)
|
|||||||
SDL_UnlockMutex(priv->buffer_mutex);
|
SDL_UnlockMutex(priv->buffer_mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *priv = ao->priv;
|
struct priv *priv = ao->priv;
|
||||||
if (!priv)
|
if (!priv)
|
||||||
@ -134,14 +134,14 @@ static int init(struct ao *ao)
|
|||||||
if (SDL_InitSubSystem(SDL_INIT_AUDIO)) {
|
if (SDL_InitSubSystem(SDL_INIT_AUDIO)) {
|
||||||
if (!ao->probing)
|
if (!ao->probing)
|
||||||
MP_ERR(ao, "SDL_Init failed\n");
|
MP_ERR(ao, "SDL_Init failed\n");
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct mp_chmap_sel sel = {0};
|
struct mp_chmap_sel sel = {0};
|
||||||
mp_chmap_sel_add_waveext_def(&sel);
|
mp_chmap_sel_add_waveext_def(&sel);
|
||||||
if (!ao_chmap_sel_adjust(ao, &sel, &ao->channels)) {
|
if (!ao_chmap_sel_adjust(ao, &sel, &ao->channels)) {
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -186,7 +186,7 @@ static int init(struct ao *ao)
|
|||||||
if (SDL_OpenAudio(&desired, &obtained)) {
|
if (SDL_OpenAudio(&desired, &obtained)) {
|
||||||
if (!ao->probing)
|
if (!ao->probing)
|
||||||
MP_ERR(ao, "could not open audio: %s\n", SDL_GetError());
|
MP_ERR(ao, "could not open audio: %s\n", SDL_GetError());
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -217,12 +217,12 @@ static int init(struct ao *ao)
|
|||||||
default:
|
default:
|
||||||
if (!ao->probing)
|
if (!ao->probing)
|
||||||
MP_ERR(ao, "could not find matching format\n");
|
MP_ERR(ao, "could not find matching format\n");
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ao_chmap_sel_get_def(ao, &sel, &ao->channels, obtained.channels)) {
|
if (!ao_chmap_sel_get_def(ao, &sel, &ao->channels, obtained.channels)) {
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,13 +231,13 @@ static int init(struct ao *ao)
|
|||||||
priv->buffer_mutex = SDL_CreateMutex();
|
priv->buffer_mutex = SDL_CreateMutex();
|
||||||
if (!priv->buffer_mutex) {
|
if (!priv->buffer_mutex) {
|
||||||
MP_ERR(ao, "SDL_CreateMutex failed\n");
|
MP_ERR(ao, "SDL_CreateMutex failed\n");
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
priv->underrun_cond = SDL_CreateCond();
|
priv->underrun_cond = SDL_CreateCond();
|
||||||
if (!priv->underrun_cond) {
|
if (!priv->underrun_cond) {
|
||||||
MP_ERR(ao, "SDL_CreateCond failed\n");
|
MP_ERR(ao, "SDL_CreateCond failed\n");
|
||||||
uninit(ao, true);
|
uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -216,7 +216,7 @@ error:
|
|||||||
/*
|
/*
|
||||||
* close device
|
* close device
|
||||||
*/
|
*/
|
||||||
static void uninit(struct ao *ao, bool immed)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct priv *p = ao->priv;
|
struct priv *p = ao->priv;
|
||||||
|
|
||||||
|
@ -1218,11 +1218,11 @@ static int setup_buffers(struct wasapi_state *state)
|
|||||||
return !state->ringbuff;
|
return !state->ringbuff;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool block)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
MP_VERBOSE(ao, "uninit!\n");
|
MP_VERBOSE(ao, "uninit!\n");
|
||||||
struct wasapi_state *state = (struct wasapi_state *)ao->priv;
|
struct wasapi_state *state = (struct wasapi_state *)ao->priv;
|
||||||
state->immed = !block;
|
state->immed = 1;
|
||||||
SetEvent(state->hUninit);
|
SetEvent(state->hUninit);
|
||||||
/* wait up to 10 seconds */
|
/* wait up to 10 seconds */
|
||||||
if (WaitForSingleObject(state->threadLoop, 10000) == WAIT_TIMEOUT)
|
if (WaitForSingleObject(state->threadLoop, 10000) == WAIT_TIMEOUT)
|
||||||
|
@ -100,13 +100,14 @@ struct ao_driver {
|
|||||||
int (*init)(struct ao *ao);
|
int (*init)(struct ao *ao);
|
||||||
// Optional. See ao_control() etc. in ao.c
|
// Optional. See ao_control() etc. in ao.c
|
||||||
int (*control)(struct ao *ao, enum aocontrol cmd, void *arg);
|
int (*control)(struct ao *ao, enum aocontrol cmd, void *arg);
|
||||||
void (*uninit)(struct ao *ao, bool cut_audio);
|
void (*uninit)(struct ao *ao);
|
||||||
void (*reset)(struct ao*ao);
|
void (*reset)(struct ao*ao);
|
||||||
int (*get_space)(struct ao *ao);
|
int (*get_space)(struct ao *ao);
|
||||||
int (*play)(struct ao *ao, void **data, int samples, int flags);
|
int (*play)(struct ao *ao, void **data, int samples, int flags);
|
||||||
float (*get_delay)(struct ao *ao);
|
float (*get_delay)(struct ao *ao);
|
||||||
void (*pause)(struct ao *ao);
|
void (*pause)(struct ao *ao);
|
||||||
void (*resume)(struct ao *ao);
|
void (*resume)(struct ao *ao);
|
||||||
|
void (*drain)(struct ao *ao);
|
||||||
|
|
||||||
// For option parsing (see vo.h)
|
// For option parsing (see vo.h)
|
||||||
int priv_size;
|
int priv_size;
|
||||||
|
@ -190,9 +190,9 @@ static void resume(struct ao *ao)
|
|||||||
ao->driver->resume(ao);
|
ao->driver->resume(ao);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
ao->driver->uninit(ao, cut_audio);
|
ao->driver->uninit(ao);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int init(struct ao *ao)
|
static int init(struct ao *ao)
|
||||||
|
@ -106,6 +106,17 @@ static void resume(struct ao *ao)
|
|||||||
pthread_mutex_unlock(&p->lock);
|
pthread_mutex_unlock(&p->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void drain(struct ao *ao)
|
||||||
|
{
|
||||||
|
if (ao->driver->drain) {
|
||||||
|
struct ao_push_state *p = ao->api_priv;
|
||||||
|
pthread_mutex_lock(&p->lock);
|
||||||
|
ao->driver->drain(ao);
|
||||||
|
pthread_mutex_unlock(&p->lock);
|
||||||
|
} else {
|
||||||
|
ao_wait_drain(ao);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int get_space(struct ao *ao)
|
static int get_space(struct ao *ao)
|
||||||
{
|
{
|
||||||
@ -202,7 +213,7 @@ static void *playthread(void *arg)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void uninit(struct ao *ao, bool cut_audio)
|
static void uninit(struct ao *ao)
|
||||||
{
|
{
|
||||||
struct ao_push_state *p = ao->api_priv;
|
struct ao_push_state *p = ao->api_priv;
|
||||||
|
|
||||||
@ -213,7 +224,7 @@ static void uninit(struct ao *ao, bool cut_audio)
|
|||||||
|
|
||||||
pthread_join(p->thread, NULL);
|
pthread_join(p->thread, NULL);
|
||||||
|
|
||||||
ao->driver->uninit(ao, cut_audio);
|
ao->driver->uninit(ao);
|
||||||
|
|
||||||
pthread_cond_destroy(&p->wakeup);
|
pthread_cond_destroy(&p->wakeup);
|
||||||
pthread_mutex_destroy(&p->lock);
|
pthread_mutex_destroy(&p->lock);
|
||||||
@ -231,7 +242,7 @@ static int init(struct ao *ao)
|
|||||||
&ao->channels, ao->samplerate);
|
&ao->channels, ao->samplerate);
|
||||||
mp_audio_buffer_preallocate_min(p->buffer, ao->buffer);
|
mp_audio_buffer_preallocate_min(p->buffer, ao->buffer);
|
||||||
if (pthread_create(&p->thread, NULL, playthread, ao)) {
|
if (pthread_create(&p->thread, NULL, playthread, ao)) {
|
||||||
ao->driver->uninit(ao, true);
|
ao->driver->uninit(ao);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
@ -247,6 +258,7 @@ const struct ao_driver ao_api_push = {
|
|||||||
.get_delay = get_delay,
|
.get_delay = get_delay,
|
||||||
.pause = pause,
|
.pause = pause,
|
||||||
.resume = resume,
|
.resume = resume,
|
||||||
|
.drain = drain,
|
||||||
.priv_size = sizeof(struct ao_push_state),
|
.priv_size = sizeof(struct ao_push_state),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -173,10 +173,8 @@ void uninit_player(struct MPContext *mpctx, unsigned int mask)
|
|||||||
struct ao *ao = mpctx->ao;
|
struct ao *ao = mpctx->ao;
|
||||||
mpctx->initialized_flags &= ~INITIALIZED_AO;
|
mpctx->initialized_flags &= ~INITIALIZED_AO;
|
||||||
if (ao) {
|
if (ao) {
|
||||||
bool drain = false;
|
|
||||||
// Note: with gapless_audio, stop_play is not correctly set
|
// Note: with gapless_audio, stop_play is not correctly set
|
||||||
if (opts->gapless_audio || mpctx->stop_play == AT_END_OF_FILE) {
|
if (opts->gapless_audio || mpctx->stop_play == AT_END_OF_FILE) {
|
||||||
drain = true;
|
|
||||||
struct mp_audio data;
|
struct mp_audio data;
|
||||||
mp_audio_buffer_peek(mpctx->ao_buffer, &data);
|
mp_audio_buffer_peek(mpctx->ao_buffer, &data);
|
||||||
int samples = mpctx->ao_buffer_playable_samples;
|
int samples = mpctx->ao_buffer_playable_samples;
|
||||||
@ -187,8 +185,9 @@ void uninit_player(struct MPContext *mpctx, unsigned int mask)
|
|||||||
if (played < samples)
|
if (played < samples)
|
||||||
MP_WARN(mpctx, "Audio output truncated at end.\n");
|
MP_WARN(mpctx, "Audio output truncated at end.\n");
|
||||||
}
|
}
|
||||||
|
ao_drain(ao);
|
||||||
}
|
}
|
||||||
ao_uninit(ao, drain);
|
ao_uninit(ao);
|
||||||
}
|
}
|
||||||
mpctx->ao = NULL;
|
mpctx->ao = NULL;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user