mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-27 22:46:02 -05:00
[player] Try to consolidate metadata handling + use input interface
This commit is contained in:
parent
7b6a7b65b3
commit
0b9b008a1a
253
src/player.c
253
src/player.c
@ -123,28 +123,18 @@ struct spk_enum
|
|||||||
void *arg;
|
void *arg;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct icy_artwork
|
|
||||||
{
|
|
||||||
uint32_t id;
|
|
||||||
char *artwork_url;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct player_metadata
|
|
||||||
{
|
|
||||||
int id;
|
|
||||||
uint64_t rtptime;
|
|
||||||
uint64_t offset;
|
|
||||||
int startup;
|
|
||||||
|
|
||||||
struct output_metadata *omd;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct speaker_set_param
|
struct speaker_set_param
|
||||||
{
|
{
|
||||||
uint64_t *device_ids;
|
uint64_t *device_ids;
|
||||||
int intval;
|
int intval;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct metadata_param
|
||||||
|
{
|
||||||
|
struct input_metadata *input;
|
||||||
|
struct output_metadata *output;
|
||||||
|
};
|
||||||
|
|
||||||
union player_arg
|
union player_arg
|
||||||
{
|
{
|
||||||
struct volume_param vol_param;
|
struct volume_param vol_param;
|
||||||
@ -153,13 +143,12 @@ union player_arg
|
|||||||
struct output_device *device;
|
struct output_device *device;
|
||||||
struct player_status *status;
|
struct player_status *status;
|
||||||
struct player_source *ps;
|
struct player_source *ps;
|
||||||
struct player_metadata *pmd;
|
struct metadata_param metadata_param;
|
||||||
uint32_t *id_ptr;
|
uint32_t *id_ptr;
|
||||||
struct speaker_set_param speaker_set_param;
|
struct speaker_set_param speaker_set_param;
|
||||||
enum repeat_mode mode;
|
enum repeat_mode mode;
|
||||||
uint32_t id;
|
uint32_t id;
|
||||||
int intval;
|
int intval;
|
||||||
struct icy_artwork icy;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct event_base *evbase_player;
|
struct event_base *evbase_player;
|
||||||
@ -447,9 +436,10 @@ static void
|
|||||||
playback_suspend(void);
|
playback_suspend(void);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
player_metadata_send(struct player_metadata *pmd);
|
player_metadata_send(struct input_metadata *imd, struct output_metadata *omd);
|
||||||
|
|
||||||
/* Callback from the worker thread (async operation as it may block) */
|
|
||||||
|
// Callback from the worker thread (async operation as it may block)
|
||||||
static void
|
static void
|
||||||
playcount_inc_cb(void *arg)
|
playcount_inc_cb(void *arg)
|
||||||
{
|
{
|
||||||
@ -459,7 +449,7 @@ playcount_inc_cb(void *arg)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef LASTFM
|
#ifdef LASTFM
|
||||||
/* Callback from the worker thread (async operation as it may block) */
|
// Callback from the worker thread (async operation as it may block)
|
||||||
static void
|
static void
|
||||||
scrobble_cb(void *arg)
|
scrobble_cb(void *arg)
|
||||||
{
|
{
|
||||||
@ -469,108 +459,72 @@ scrobble_cb(void *arg)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Callback from the worker thread
|
// Callback from the worker thread. Here the heavy lifting is done: updating the
|
||||||
* This prepares metadata in the worker thread, since especially the artwork
|
// db_queue_item, retrieving artwork (through outputs_metadata_prepare) and
|
||||||
* retrieval may take some time. outputs_metadata_prepare() must be thread safe.
|
// when done, telling the player to send the metadata to the clients
|
||||||
* The sending must be done in the player thread.
|
|
||||||
*/
|
|
||||||
static void
|
static void
|
||||||
metadata_prepare_cb(void *arg)
|
metadata_update_cb(void *arg)
|
||||||
{
|
{
|
||||||
struct player_metadata *pmd = arg;
|
struct input_metadata *metadata = arg;
|
||||||
|
struct output_metadata *o_metadata;
|
||||||
|
struct db_queue_item *queue_item;
|
||||||
|
int ret;
|
||||||
|
|
||||||
pmd->omd = outputs_metadata_prepare(pmd->id);
|
queue_item = db_queue_fetch_byitemid(metadata->item_id);
|
||||||
|
if (!queue_item)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Bug! Input metadata item_id does not match anything in queue\n");
|
||||||
|
goto out_free_metadata;
|
||||||
|
}
|
||||||
|
|
||||||
if (pmd->omd)
|
// Since we won't be using the metadata struct values for anything else than
|
||||||
player_metadata_send(pmd);
|
// this we just swap pointers
|
||||||
|
if (metadata->artist)
|
||||||
|
swap_pointers(&queue_item->artist, &metadata->artist);
|
||||||
|
if (metadata->title)
|
||||||
|
swap_pointers(&queue_item->title, &metadata->title);
|
||||||
|
if (metadata->album)
|
||||||
|
swap_pointers(&queue_item->album, &metadata->album);
|
||||||
|
if (metadata->artwork_url)
|
||||||
|
swap_pointers(&queue_item->artwork_url, &metadata->artwork_url);
|
||||||
|
|
||||||
outputs_metadata_free(pmd->omd);
|
ret = db_queue_update_item(queue_item);
|
||||||
|
if (ret < 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Database error while updating queue with new metadata\n");
|
||||||
|
goto out_free_queueitem;
|
||||||
|
}
|
||||||
|
|
||||||
|
o_metadata = outputs_metadata_prepare(metadata->item_id);
|
||||||
|
|
||||||
|
// Actual sending must be done by player, since the worker does not own the outputs
|
||||||
|
player_metadata_send(metadata, o_metadata);
|
||||||
|
|
||||||
|
outputs_metadata_free(o_metadata);
|
||||||
|
|
||||||
|
out_free_queueitem:
|
||||||
|
free_queue_item(queue_item, 0);
|
||||||
|
|
||||||
|
out_free_metadata:
|
||||||
|
input_metadata_free(metadata, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Callback from the worker thread (async operation as it may block) */
|
// Gets the metadata, but since the actual update requires db writes and
|
||||||
static void
|
// possibly retrieving artwork we let the worker do the next step
|
||||||
update_icy_cb(void *arg)
|
void
|
||||||
{
|
|
||||||
struct http_icy_metadata *metadata = arg;
|
|
||||||
|
|
||||||
db_queue_update_icymetadata(metadata->id, metadata->artist, metadata->title);
|
|
||||||
|
|
||||||
http_icy_metadata_free(metadata, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Metadata */
|
|
||||||
static void
|
|
||||||
metadata_prune(uint64_t pos)
|
|
||||||
{
|
|
||||||
outputs_metadata_prune(pos);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
metadata_purge(void)
|
|
||||||
{
|
|
||||||
outputs_metadata_purge();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
metadata_trigger(int startup)
|
metadata_trigger(int startup)
|
||||||
{
|
{
|
||||||
struct player_metadata pmd;
|
struct input_metadata metadata;
|
||||||
|
int ret;
|
||||||
|
|
||||||
memset(&pmd, 0, sizeof(struct player_metadata));
|
ret = input_metadata_get(&metadata, cur_streaming, startup);
|
||||||
|
if (ret < 0)
|
||||||
pmd.id = cur_streaming->item_id;
|
|
||||||
pmd.startup = startup;
|
|
||||||
|
|
||||||
if (cur_streaming->stream_start && cur_streaming->output_start)
|
|
||||||
{
|
|
||||||
pmd.offset = cur_streaming->output_start - cur_streaming->stream_start;
|
|
||||||
pmd.rtptime = cur_streaming->stream_start;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
DPRINTF(E_LOG, L_PLAYER, "PTOH! Unhandled song boundary case in metadata_trigger()\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Defer the actual work of preparing the metadata to the worker thread */
|
|
||||||
worker_execute(metadata_prepare_cb, &pmd, sizeof(struct player_metadata), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Checks if there is new HTTP ICY metadata, and if so sends updates to clients */
|
|
||||||
void
|
|
||||||
metadata_check_icy(void)
|
|
||||||
{
|
|
||||||
struct http_icy_metadata *metadata;
|
|
||||||
int changed;
|
|
||||||
|
|
||||||
metadata = transcode_metadata(cur_streaming->xcode, &changed);
|
|
||||||
if (!metadata)
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (!changed || !metadata->title)
|
worker_execute(metadata_update_cb, &metadata, sizeof(metadata), 0);
|
||||||
goto no_update;
|
|
||||||
|
|
||||||
if (metadata->title[0] == '\0')
|
|
||||||
goto no_update;
|
|
||||||
|
|
||||||
metadata->id = cur_streaming->item_id;
|
|
||||||
|
|
||||||
/* Defer the database update to the worker thread */
|
|
||||||
worker_execute(update_icy_cb, metadata, sizeof(struct http_icy_metadata), 0);
|
|
||||||
|
|
||||||
/* Triggers preparing and sending output metadata */
|
|
||||||
metadata_trigger(0);
|
|
||||||
|
|
||||||
/* Only free the struct, the content must be preserved for update_icy_cb */
|
|
||||||
free(metadata);
|
|
||||||
|
|
||||||
status_update(player_state);
|
|
||||||
|
|
||||||
return;
|
|
||||||
|
|
||||||
no_update:
|
|
||||||
http_icy_metadata_free(metadata, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
struct player_history *
|
struct player_history *
|
||||||
player_history_get(void)
|
player_history_get(void)
|
||||||
{
|
{
|
||||||
@ -950,7 +904,7 @@ source_check(void)
|
|||||||
|
|
||||||
status_update(PLAY_PLAYING);
|
status_update(PLAY_PLAYING);
|
||||||
|
|
||||||
metadata_prune(pos);
|
outputs_metadata_prune(pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
return pos;
|
return pos;
|
||||||
@ -1101,6 +1055,10 @@ source_read(uint8_t *buf, int len)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
else if (flags & INPUT_FLAG_METADATA)
|
||||||
|
{
|
||||||
|
metadata_trigger(0);
|
||||||
|
}
|
||||||
|
|
||||||
// We pad the output buffer with silence if we don't have enough data for a
|
// We pad the output buffer with silence if we don't have enough data for a
|
||||||
// full packet and there is no more data coming up (no more tracks in queue)
|
// full packet and there is no more data coming up (no more tracks in queue)
|
||||||
@ -1429,18 +1387,16 @@ static enum command_state
|
|||||||
metadata_send(void *arg, int *retval)
|
metadata_send(void *arg, int *retval)
|
||||||
{
|
{
|
||||||
union player_arg *cmdarg;
|
union player_arg *cmdarg;
|
||||||
struct player_metadata *pmd;
|
struct input_metadata *imd;
|
||||||
|
struct output_metadata *omd;
|
||||||
|
|
||||||
cmdarg = arg;
|
cmdarg = arg;
|
||||||
pmd = cmdarg->pmd;
|
imd = cmdarg->metadata_param.input;
|
||||||
|
omd = cmdarg->metadata_param.output;
|
||||||
|
|
||||||
/* Do the setting of rtptime which was deferred in metadata_trigger because we
|
outputs_metadata_send(omd, imd->rtptime, imd->offset, imd->startup);
|
||||||
* wanted to wait until we had the actual last_rtptime
|
|
||||||
*/
|
|
||||||
if ((pmd->rtptime == 0) && (pmd->startup))
|
|
||||||
pmd->rtptime = last_rtptime + AIRTUNES_V2_PACKET_SAMPLES;
|
|
||||||
|
|
||||||
outputs_metadata_send(pmd->omd, pmd->rtptime, pmd->offset, pmd->startup);
|
status_update(player_state);
|
||||||
|
|
||||||
*retval = 0;
|
*retval = 0;
|
||||||
return COMMAND_END;
|
return COMMAND_END;
|
||||||
@ -1724,7 +1680,7 @@ playback_abort(void)
|
|||||||
|
|
||||||
status_update(PLAY_STOPPED);
|
status_update(PLAY_STOPPED);
|
||||||
|
|
||||||
metadata_purge();
|
outputs_metadata_purge();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Internal routine for waiting when input buffer underruns */
|
/* Internal routine for waiting when input buffer underruns */
|
||||||
@ -1854,37 +1810,6 @@ now_playing(void *arg, int *retval)
|
|||||||
return COMMAND_END;
|
return COMMAND_END;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum command_state
|
|
||||||
artwork_url_get(void *arg, int *retval)
|
|
||||||
{
|
|
||||||
union player_arg *cmdarg = arg;
|
|
||||||
struct player_source *ps;
|
|
||||||
|
|
||||||
cmdarg->icy.artwork_url = NULL;
|
|
||||||
|
|
||||||
if (cur_playing)
|
|
||||||
ps = cur_playing;
|
|
||||||
else if (cur_streaming)
|
|
||||||
ps = cur_streaming;
|
|
||||||
else
|
|
||||||
{
|
|
||||||
*retval = -1;
|
|
||||||
return COMMAND_END;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check that we are playing a viable stream, and that it has the requested id */
|
|
||||||
if (!ps->xcode || ps->data_kind != DATA_KIND_HTTP || ps->id != cmdarg->icy.id)
|
|
||||||
{
|
|
||||||
*retval = -1;
|
|
||||||
return COMMAND_END;
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdarg->icy.artwork_url = transcode_metadata_artwork_url(ps->xcode);
|
|
||||||
|
|
||||||
*retval = 0;
|
|
||||||
return COMMAND_END;
|
|
||||||
}
|
|
||||||
|
|
||||||
static enum command_state
|
static enum command_state
|
||||||
playback_stop(void *arg, int *retval)
|
playback_stop(void *arg, int *retval)
|
||||||
{
|
{
|
||||||
@ -1908,7 +1833,7 @@ playback_stop(void *arg, int *retval)
|
|||||||
|
|
||||||
status_update(PLAY_STOPPED);
|
status_update(PLAY_STOPPED);
|
||||||
|
|
||||||
metadata_purge();
|
outputs_metadata_purge();
|
||||||
|
|
||||||
/* We're async if we need to flush devices */
|
/* We're async if we need to flush devices */
|
||||||
if (*retval > 0)
|
if (*retval > 0)
|
||||||
@ -2343,7 +2268,7 @@ playback_pause(void *arg, int *retval)
|
|||||||
|
|
||||||
source_pause(pos);
|
source_pause(pos);
|
||||||
|
|
||||||
metadata_purge();
|
outputs_metadata_purge();
|
||||||
|
|
||||||
/* We're async if we need to flush devices */
|
/* We're async if we need to flush devices */
|
||||||
if (*retval > 0)
|
if (*retval > 0)
|
||||||
@ -2794,25 +2719,6 @@ player_now_playing(uint32_t *id)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *
|
|
||||||
player_get_icy_artwork_url(uint32_t id)
|
|
||||||
{
|
|
||||||
union player_arg cmdarg;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
cmdarg.icy.id = id;
|
|
||||||
|
|
||||||
if (pthread_self() != tid_player)
|
|
||||||
ret = commands_exec_sync(cmdbase, artwork_url_get, NULL, &cmdarg);
|
|
||||||
else
|
|
||||||
artwork_url_get(&cmdarg, &ret);
|
|
||||||
|
|
||||||
if (ret < 0)
|
|
||||||
return NULL;
|
|
||||||
else
|
|
||||||
return cmdarg.icy.artwork_url;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Starts/resumes playback
|
* Starts/resumes playback
|
||||||
*
|
*
|
||||||
@ -3057,11 +2963,12 @@ player_device_remove(void *device)
|
|||||||
|
|
||||||
/* Thread: worker */
|
/* Thread: worker */
|
||||||
static void
|
static void
|
||||||
player_metadata_send(struct player_metadata *pmd)
|
player_metadata_send(struct input_metadata *imd, struct output_metadata *omd)
|
||||||
{
|
{
|
||||||
union player_arg cmdarg;
|
union player_arg cmdarg;
|
||||||
|
|
||||||
cmdarg.pmd = pmd;
|
cmdarg.metadata_param.input = imd;
|
||||||
|
cmdarg.metadata_param.output = omd;
|
||||||
|
|
||||||
commands_exec_sync(cmdbase, metadata_send, NULL, &cmdarg);
|
commands_exec_sync(cmdbase, metadata_send, NULL, &cmdarg);
|
||||||
}
|
}
|
||||||
|
@ -80,9 +80,6 @@ player_get_status(struct player_status *status);
|
|||||||
int
|
int
|
||||||
player_now_playing(uint32_t *id);
|
player_now_playing(uint32_t *id);
|
||||||
|
|
||||||
char *
|
|
||||||
player_get_icy_artwork_url(uint32_t id);
|
|
||||||
|
|
||||||
void
|
void
|
||||||
player_speaker_enumerate(spk_enum_cb cb, void *arg);
|
player_speaker_enumerate(spk_enum_cb cb, void *arg);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user