player_control: duplicate the song object

Make sure the player "owns" the next_song object, so nobody else can
free it.
This commit is contained in:
Max Kellermann 2012-08-09 22:19:39 +02:00
parent eb54337c40
commit e96779de48
5 changed files with 58 additions and 7 deletions

View File

@ -59,6 +59,9 @@ pc_new(unsigned buffer_chunks, unsigned int buffered_before_play)
void void
pc_free(struct player_control *pc) pc_free(struct player_control *pc)
{ {
if (pc->next_song != NULL)
song_free(pc->next_song);
g_cond_free(pc->cond); g_cond_free(pc->cond);
g_mutex_free(pc->mutex); g_mutex_free(pc->mutex);
g_free(pc); g_free(pc);
@ -284,6 +287,10 @@ pc_seek(struct player_control *pc, struct song *song, float seek_time)
assert(song != NULL); assert(song != NULL);
player_lock(pc); player_lock(pc);
if (pc->next_song != NULL)
song_free(pc->next_song);
pc->next_song = song; pc->next_song = song;
pc->seek_where = seek_time; pc->seek_where = seek_time;
player_command_locked(pc, PLAYER_COMMAND_SEEK); player_command_locked(pc, PLAYER_COMMAND_SEEK);

View File

@ -123,7 +123,15 @@ struct player_control {
struct audio_format audio_format; struct audio_format audio_format;
float total_time; float total_time;
float elapsed_time; float elapsed_time;
/**
* The next queued song.
*
* This is a duplicate, and must be freed when this attribute
* is cleared.
*/
struct song *next_song; struct song *next_song;
double seek_where; double seek_where;
float cross_fade_seconds; float cross_fade_seconds;
float mixramp_db; float mixramp_db;
@ -198,6 +206,10 @@ player_lock_signal(struct player_control *pc)
player_unlock(pc); player_unlock(pc);
} }
/**
* @param song the song to be queued; the given instance will be owned
* and freed by the player
*/
void void
pc_play(struct player_control *pc, struct song *song); pc_play(struct player_control *pc, struct song *song);
@ -261,12 +273,18 @@ pc_stop(struct player_control *pc);
void void
pc_update_audio(struct player_control *pc); pc_update_audio(struct player_control *pc);
/**
* @param song the song to be queued; the given instance will be owned
* and freed by the player
*/
void void
pc_enqueue_song(struct player_control *pc, struct song *song); pc_enqueue_song(struct player_control *pc, struct song *song);
/** /**
* Makes the player thread seek the specified song to a position. * Makes the player thread seek the specified song to a position.
* *
* @param song the song to be queued; the given instance will be owned
* and freed by the player
* @return true on success, false on failure (e.g. if MPD isn't * @return true on success, false on failure (e.g. if MPD isn't
* playing currently) * playing currently)
*/ */

View File

@ -239,12 +239,18 @@ player_wait_for_decoder(struct player *player)
if (error != NULL) { if (error != NULL) {
player_lock(pc); player_lock(pc);
pc_set_error(pc, PLAYER_ERROR_DECODER, error); pc_set_error(pc, PLAYER_ERROR_DECODER, error);
song_free(pc->next_song);
pc->next_song = NULL; pc->next_song = NULL;
player_unlock(pc); player_unlock(pc);
return false; return false;
} }
if (player->song != NULL)
song_free(player->song);
player->song = pc->next_song; player->song = pc->next_song;
player->elapsed_time = 0.0; player->elapsed_time = 0.0;
@ -486,6 +492,7 @@ static bool player_seek_decoder(struct player *player)
player->pipe = dc->pipe; player->pipe = dc->pipe;
} }
song_free(pc->next_song);
pc->next_song = NULL; pc->next_song = NULL;
player->queued = false; player->queued = false;
} }
@ -606,6 +613,7 @@ static void player_process_command(struct player *player)
player_lock(pc); player_lock(pc);
} }
song_free(pc->next_song);
pc->next_song = NULL; pc->next_song = NULL;
player->queued = false; player->queued = false;
player_command_finished_locked(pc); player_command_finished_locked(pc);
@ -886,6 +894,8 @@ static void do_play(struct player_control *pc, struct decoder_control *dc)
player_dc_start(&player, player.pipe); player_dc_start(&player, player.pipe);
if (!player_wait_for_decoder(&player)) { if (!player_wait_for_decoder(&player)) {
assert(player.song == NULL);
player_dc_stop(&player); player_dc_stop(&player);
player_command_finished(pc); player_command_finished(pc);
music_pipe_free(player.pipe); music_pipe_free(player.pipe);
@ -1048,10 +1058,14 @@ static void do_play(struct player_control *pc, struct decoder_control *dc)
if (player.cross_fade_tag != NULL) if (player.cross_fade_tag != NULL)
tag_free(player.cross_fade_tag); tag_free(player.cross_fade_tag);
if (player.song != NULL)
song_free(player.song);
player_lock(pc); player_lock(pc);
if (player.queued) { if (player.queued) {
assert(pc->next_song != NULL); assert(pc->next_song != NULL);
song_free(pc->next_song);
pc->next_song = NULL; pc->next_song = NULL;
} }
@ -1093,7 +1107,11 @@ player_task(gpointer arg)
/* fall through */ /* fall through */
case PLAYER_COMMAND_PAUSE: case PLAYER_COMMAND_PAUSE:
pc->next_song = NULL; if (pc->next_song != NULL) {
song_free(pc->next_song);
pc->next_song = NULL;
}
player_command_finished_locked(pc); player_command_finished_locked(pc);
break; break;
@ -1134,7 +1152,11 @@ player_task(gpointer arg)
return NULL; return NULL;
case PLAYER_COMMAND_CANCEL: case PLAYER_COMMAND_CANCEL:
pc->next_song = NULL; if (pc->next_song != NULL) {
song_free(pc->next_song);
pc->next_song = NULL;
}
player_command_finished_locked(pc); player_command_finished_locked(pc);
break; break;

View File

@ -78,14 +78,15 @@ static void
playlist_queue_song_order(struct playlist *playlist, struct player_control *pc, playlist_queue_song_order(struct playlist *playlist, struct player_control *pc,
unsigned order) unsigned order)
{ {
struct song *song;
char *uri; char *uri;
assert(queue_valid_order(&playlist->queue, order)); assert(queue_valid_order(&playlist->queue, order));
playlist->queued = order; playlist->queued = order;
song = queue_get_order(&playlist->queue, order); struct song *song =
song_dup_detached(queue_get_order(&playlist->queue, order));
uri = song_get_uri(song); uri = song_get_uri(song);
g_debug("queue song %i:\"%s\"", playlist->queued, uri); g_debug("queue song %i:\"%s\"", playlist->queued, uri);
g_free(uri); g_free(uri);
@ -191,13 +192,13 @@ void
playlist_play_order(struct playlist *playlist, struct player_control *pc, playlist_play_order(struct playlist *playlist, struct player_control *pc,
int orderNum) int orderNum)
{ {
struct song *song;
char *uri; char *uri;
playlist->playing = true; playlist->playing = true;
playlist->queued = -1; playlist->queued = -1;
song = queue_get_order(&playlist->queue, orderNum); struct song *song =
song_dup_detached(queue_get_order(&playlist->queue, orderNum));
uri = song_get_uri(song); uri = song_get_uri(song);
g_debug("play %i:\"%s\"", orderNum, uri); g_debug("play %i:\"%s\"", orderNum, uri);

View File

@ -25,6 +25,7 @@
#include "config.h" #include "config.h"
#include "playlist_internal.h" #include "playlist_internal.h"
#include "player_control.h" #include "player_control.h"
#include "song.h"
#include "idle.h" #include "idle.h"
#include <glib.h> #include <glib.h>
@ -239,7 +240,9 @@ playlist_seek_song(struct playlist *playlist, struct player_control *pc,
queued = NULL; queued = NULL;
} }
success = pc_seek(pc, queue_get_order(&playlist->queue, i), seek_time); struct song *the_song =
song_dup_detached(queue_get_order(&playlist->queue, i));
success = pc_seek(pc, the_song, seek_time);
if (!success) { if (!success) {
playlist_update_queued_song(playlist, pc, queued); playlist_update_queued_song(playlist, pc, queued);