From 29742d23d3f55b64a36dd91158e1f9efe2620650 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Tue, 20 Sep 2011 21:14:56 +0200 Subject: [PATCH 1/4] configure.ac: fix --enable-id3 help string --- configure.ac | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/configure.ac b/configure.ac index 50e8d8c7a..020f15d35 100644 --- a/configure.ac +++ b/configure.ac @@ -199,7 +199,7 @@ AC_ARG_ENABLE(httpd-output, AC_ARG_ENABLE(id3, AS_HELP_STRING([--enable-id3], - [disable id3 support]),, + [enable id3 support]),, enable_id3=auto) AC_ARG_ENABLE(inotify, From b0b2c5b3e0abd27fa305769d1952b9295f0d1726 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Neusch=C3=A4fer?= Date: Mon, 19 Sep 2011 18:09:59 +0200 Subject: [PATCH 2/4] utils: uri.h: fix a typo: "schema" --- src/uri.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/uri.h b/src/uri.h index 422b959b0..00c63c039 100644 --- a/src/uri.h +++ b/src/uri.h @@ -25,7 +25,7 @@ #include /** - * Checks whether the specified URI has a schema in the form + * Checks whether the specified URI has a scheme in the form * "scheme://". */ G_GNUC_PURE From fb19210cfd3ec20d69f53ea258c36ef72e002edd Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Wed, 21 Sep 2011 22:59:11 +0200 Subject: [PATCH 3/4] decoder_internal.h: more API documentation --- src/decoder_internal.h | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/decoder_internal.h b/src/decoder_internal.h index 9e7e2037a..2347fdf4e 100644 --- a/src/decoder_internal.h +++ b/src/decoder_internal.h @@ -36,6 +36,11 @@ struct decoder { */ double timestamp; + /** + * This flag is set by decoder_seek_where(), and checked by + * decoder_command_finished(). It is used to clean up after + * seeking. + */ bool seeking; /** From 525a791987c66ed2f8b1ca9e5357836f536fdb8b Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Wed, 21 Sep 2011 23:17:34 +0200 Subject: [PATCH 4/4] decoder_api: emulate SEEK command for initial seek to CUE track When playing a CUE track, the player thread waited for the decoder to become ready, and then sent a SEEK command to the beginning of the CUE track. If that is near the start of the song file, and the track is short enough, the decoder could have finished decoding already at that point, and seeking fails. This commit makes this initial seek more robust: instead of letting the player thread deal with the difficult timings, let the decoder API emulate a SEEK command, and return it to the decoder plugin, as soon as the plugin finishes its initialization. --- NEWS | 1 + src/decoder_api.c | 63 ++++++++++++++++++++++++++++++++++++++---- src/decoder_internal.h | 14 ++++++++++ src/decoder_thread.c | 2 ++ src/player_thread.c | 10 ------- 5 files changed, 74 insertions(+), 16 deletions(-) diff --git a/NEWS b/NEWS index 2f1907cd7..f810c88e1 100644 --- a/NEWS +++ b/NEWS @@ -9,6 +9,7 @@ ver 0.16.5 (2010/??/??) * decoder: - ffmpeg: higher precision timestamps - ffmpeg: don't require key frame for seeking + - fix CUE track seeking * WIN32: close sockets properly diff --git a/src/decoder_api.c b/src/decoder_api.c index fe34ea34a..99c02db87 100644 --- a/src/decoder_api.c +++ b/src/decoder_api.c @@ -78,15 +78,40 @@ decoder_initialized(struct decoder *decoder, &af_string)); } -enum decoder_command decoder_get_command(G_GNUC_UNUSED struct decoder * decoder) +/** + * Returns the current decoder command. May return a "virtual" + * synthesized command, e.g. to seek to the beginning of the CUE + * track. + */ +G_GNUC_PURE +static enum decoder_command +decoder_get_virtual_command(struct decoder *decoder) { const struct decoder_control *dc = decoder->dc; - assert(dc->pipe != NULL); + if (decoder->initial_seek_running) + return DECODE_COMMAND_SEEK; + + if (decoder->initial_seek_pending) { + if (dc->command == DECODE_COMMAND_NONE) { + decoder->initial_seek_pending = false; + decoder->initial_seek_running = true; + return DECODE_COMMAND_SEEK; + } + + decoder->initial_seek_pending = false; + } + return dc->command; } +enum decoder_command +decoder_get_command(struct decoder *decoder) +{ + return decoder_get_virtual_command(decoder); +} + void decoder_command_finished(struct decoder *decoder) { @@ -94,11 +119,24 @@ decoder_command_finished(struct decoder *decoder) decoder_lock(dc); - assert(dc->command != DECODE_COMMAND_NONE); + assert(dc->command != DECODE_COMMAND_NONE || + decoder->initial_seek_running); assert(dc->command != DECODE_COMMAND_SEEK || + decoder->initial_seek_running || dc->seek_error || decoder->seeking); assert(dc->pipe != NULL); + if (decoder->initial_seek_running) { + assert(!decoder->seeking); + assert(decoder->chunk == NULL); + assert(music_pipe_empty(dc->pipe)); + + decoder->initial_seek_running = false; + decoder->timestamp = dc->song->start_ms / 1000.; + decoder_unlock(dc); + return; + } + if (decoder->seeking) { decoder->seeking = false; @@ -124,9 +162,13 @@ double decoder_seek_where(G_GNUC_UNUSED struct decoder * decoder) { const struct decoder_control *dc = decoder->dc; - assert(dc->command == DECODE_COMMAND_SEEK); assert(dc->pipe != NULL); + if (decoder->initial_seek_running) + return dc->song->start_ms / 1000.; + + assert(dc->command == DECODE_COMMAND_SEEK); + decoder->seeking = true; return dc->seek_where; @@ -136,9 +178,15 @@ void decoder_seek_error(struct decoder * decoder) { struct decoder_control *dc = decoder->dc; - assert(dc->command == DECODE_COMMAND_SEEK); assert(dc->pipe != NULL); + if (decoder->initial_seek_running) + /* d'oh, we can't seek to the sub-song start position, + what now? - no idea, ignoring the problem for now. */ + return; + + assert(dc->command == DECODE_COMMAND_SEEK); + dc->seek_error = true; decoder->seeking = false; @@ -270,7 +318,7 @@ decoder_data(struct decoder *decoder, assert(length % audio_format_frame_size(&dc->in_audio_format) == 0); decoder_lock(dc); - cmd = dc->command; + cmd = decoder_get_virtual_command(decoder); decoder_unlock(dc); if (cmd == DECODE_COMMAND_STOP || cmd == DECODE_COMMAND_SEEK || @@ -401,6 +449,9 @@ decoder_tag(G_GNUC_UNUSED struct decoder *decoder, struct input_stream *is, /* send only the decoder tag */ cmd = do_send_tag(decoder, is, tag); + if (cmd == DECODE_COMMAND_NONE) + cmd = decoder_get_virtual_command(decoder); + return cmd; } diff --git a/src/decoder_internal.h b/src/decoder_internal.h index 2347fdf4e..5818632e5 100644 --- a/src/decoder_internal.h +++ b/src/decoder_internal.h @@ -36,6 +36,20 @@ struct decoder { */ double timestamp; + /** + * Is the initial seek (to the start position of the sub-song) + * pending, or has it been performed already? + */ + bool initial_seek_pending; + + /** + * Is the initial seek currently running? During this time, + * the decoder command is SEEK. This flag is set by + * decoder_get_virtual_command(), when the virtual SEEK + * command is generated for the first time. + */ + bool initial_seek_running; + /** * This flag is set by decoder_seek_where(), and checked by * decoder_command_finished(). It is used to clean up after diff --git a/src/decoder_thread.c b/src/decoder_thread.c index 10a796967..ce849df47 100644 --- a/src/decoder_thread.c +++ b/src/decoder_thread.c @@ -369,6 +369,8 @@ decoder_run_song(struct decoder_control *dc, { struct decoder decoder = { .dc = dc, + .initial_seek_pending = song->start_ms > 0, + .initial_seek_running = false, }; int ret; diff --git a/src/player_thread.c b/src/player_thread.c index a89e59908..be08aa32d 100644 --- a/src/player_thread.c +++ b/src/player_thread.c @@ -881,16 +881,6 @@ static void do_play(struct decoder_control *dc) if (!player_check_decoder_startup(&player)) break; - /* seek to the beginning of the range */ - const struct song *song = decoder_current_song(dc); - if (song != NULL && song->start_ms > 0 && - /* we must not send a seek command until - the decoder is initialized - completely */ - !player.decoder_starting && - !dc_seek(dc, song->start_ms / 1000.0)) - player_dc_stop(&player); - player_lock(); continue; }