mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 02:15:31 +00:00
flacparse: fix broken reordering of flac metadata
Each FLAC metadata block starts with a flag denoting whether it is the last metadata block. The existing flacparse code moves any existing VORBISCOMMENT block to immediately follow the STREAMINFO block without changing any block's last-metadata-block flag. If no VORBISCOMMENT block exists, it created one with the last-metadata-block flag set to true. This results in gstflacdec sometimes giving bad headers to libflac when trying to play perfectly valid FLAC files depending on the file's metadata ordering. Depending on the contents of the other metadata blocks, current versions of libflac may or may not return FLAC__STREAM_DECODER_ERROR_STATUS_BAD_HEADER when given this broken metadata. This is most noticeable with files that have a large cover art image attached where VORBISCOMMENT is the very last metadata block with no PADDING afterwards. This patch changes that behavior so that: 1. For FLAC files that already have a VORBISCOMMENT block, the metadata order is preserved. 2. For FLAC files that do not have a VORBISCOMMENT block, the generated dummy VORBISCOMMENT is placed immediately after STREAMINFO and inherits the last-metadata-block flag from STREAMINFO. https://gitlab.freedesktop.org/gstreamer/gst-plugins-good/issues/484
This commit is contained in:
parent
830db205f6
commit
3287f1cb3f
1 changed files with 16 additions and 12 deletions
|
@ -183,7 +183,7 @@ static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||||
);
|
);
|
||||||
|
|
||||||
static GstBuffer *gst_flac_parse_generate_vorbiscomment (GstFlacParse *
|
static GstBuffer *gst_flac_parse_generate_vorbiscomment (GstFlacParse *
|
||||||
flacparse);
|
flacparse, gboolean is_last);
|
||||||
|
|
||||||
static inline void gst_flac_parse_reset_buffer_time_and_offset (GstBuffer *
|
static inline void gst_flac_parse_reset_buffer_time_and_offset (GstBuffer *
|
||||||
buffer);
|
buffer);
|
||||||
|
@ -1243,6 +1243,7 @@ gst_flac_parse_handle_headers (GstFlacParse * flacparse)
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
GList *l;
|
GList *l;
|
||||||
GstFlowReturn res = GST_FLOW_OK;
|
GstFlowReturn res = GST_FLOW_OK;
|
||||||
|
gboolean is_streaminfo_last = FALSE;
|
||||||
|
|
||||||
caps = gst_caps_new_simple ("audio/x-flac",
|
caps = gst_caps_new_simple ("audio/x-flac",
|
||||||
"channels", G_TYPE_INT, flacparse->channels,
|
"channels", G_TYPE_INT, flacparse->channels,
|
||||||
|
@ -1264,6 +1265,7 @@ gst_flac_parse_handle_headers (GstFlacParse * flacparse)
|
||||||
marker = header;
|
marker = header;
|
||||||
} else if (map.size > 1 && (map.data[0] & 0x7f) == 0) {
|
} else if (map.size > 1 && (map.data[0] & 0x7f) == 0) {
|
||||||
streaminfo = header;
|
streaminfo = header;
|
||||||
|
is_streaminfo_last = (map.data[0] & 0x80) != 0;
|
||||||
} else if (map.size > 1 && (map.data[0] & 0x7f) == 4) {
|
} else if (map.size > 1 && (map.data[0] & 0x7f) == 4) {
|
||||||
vorbiscomment = header;
|
vorbiscomment = header;
|
||||||
}
|
}
|
||||||
|
@ -1276,8 +1278,11 @@ gst_flac_parse_handle_headers (GstFlacParse * flacparse)
|
||||||
if (vorbiscomment == NULL && streaminfo != NULL) {
|
if (vorbiscomment == NULL && streaminfo != NULL) {
|
||||||
GST_DEBUG_OBJECT (flacparse,
|
GST_DEBUG_OBJECT (flacparse,
|
||||||
"missing vorbiscomment header; generating dummy");
|
"missing vorbiscomment header; generating dummy");
|
||||||
vorbiscomment = gst_flac_parse_generate_vorbiscomment (flacparse);
|
/* this vorbiscomment header is inserted after streaminfo and inherits its last-metadata-block flag */
|
||||||
flacparse->headers = g_list_insert (flacparse->headers, vorbiscomment,
|
vorbiscomment =
|
||||||
|
gst_flac_parse_generate_vorbiscomment (flacparse, is_streaminfo_last);
|
||||||
|
flacparse->headers =
|
||||||
|
g_list_insert (flacparse->headers, vorbiscomment,
|
||||||
g_list_index (flacparse->headers, streaminfo) + 1);
|
g_list_index (flacparse->headers, streaminfo) + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1312,6 +1317,8 @@ gst_flac_parse_handle_headers (GstFlacParse * flacparse)
|
||||||
writemap.data[8] = (num & 0x00FF) >> 0;
|
writemap.data[8] = (num & 0x00FF) >> 0;
|
||||||
memcpy (writemap.data + 9, "fLaC", 4);
|
memcpy (writemap.data + 9, "fLaC", 4);
|
||||||
memcpy (writemap.data + 13, sinfomap.data, sinfomap.size);
|
memcpy (writemap.data + 13, sinfomap.data, sinfomap.size);
|
||||||
|
/* clear the last-metadata-block flag because a VORBISCOMMENT always follows */
|
||||||
|
writemap.data[13] = 0x00; /* is_last = 0; type = 0; */
|
||||||
_value_array_append_buffer (&array, buf);
|
_value_array_append_buffer (&array, buf);
|
||||||
|
|
||||||
gst_buffer_unmap (streaminfo, &sinfomap);
|
gst_buffer_unmap (streaminfo, &sinfomap);
|
||||||
|
@ -1319,14 +1326,10 @@ gst_flac_parse_handle_headers (GstFlacParse * flacparse)
|
||||||
gst_buffer_unref (buf);
|
gst_buffer_unref (buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* add VORBISCOMMENT header */
|
/* add other headers, including VORBISCOMMENT */
|
||||||
_value_array_append_buffer (&array, vorbiscomment);
|
|
||||||
|
|
||||||
/* add other headers, if there are any */
|
|
||||||
for (l = flacparse->headers; l; l = l->next) {
|
for (l = flacparse->headers; l; l = l->next) {
|
||||||
if (GST_BUFFER_CAST (l->data) != marker &&
|
if (GST_BUFFER_CAST (l->data) != marker &&
|
||||||
GST_BUFFER_CAST (l->data) != streaminfo &&
|
GST_BUFFER_CAST (l->data) != streaminfo) {
|
||||||
GST_BUFFER_CAST (l->data) != vorbiscomment) {
|
|
||||||
_value_array_append_buffer (&array, GST_BUFFER_CAST (l->data));
|
_value_array_append_buffer (&array, GST_BUFFER_CAST (l->data));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1368,7 +1371,8 @@ push_headers:
|
||||||
|
|
||||||
/* empty vorbiscomment */
|
/* empty vorbiscomment */
|
||||||
static GstBuffer *
|
static GstBuffer *
|
||||||
gst_flac_parse_generate_vorbiscomment (GstFlacParse * flacparse)
|
gst_flac_parse_generate_vorbiscomment (GstFlacParse * flacparse,
|
||||||
|
gboolean is_last)
|
||||||
{
|
{
|
||||||
GstTagList *taglist = gst_tag_list_new_empty ();
|
GstTagList *taglist = gst_tag_list_new_empty ();
|
||||||
guchar header[4];
|
guchar header[4];
|
||||||
|
@ -1376,7 +1380,7 @@ gst_flac_parse_generate_vorbiscomment (GstFlacParse * flacparse)
|
||||||
GstBuffer *vorbiscomment;
|
GstBuffer *vorbiscomment;
|
||||||
GstMapInfo map;
|
GstMapInfo map;
|
||||||
|
|
||||||
header[0] = 0x84; /* is_last = 1; type = 4; */
|
header[0] = (is_last ? 0x80 : 0x00) | 0x04; /* is_last may vary; type = 4; */
|
||||||
|
|
||||||
vorbiscomment =
|
vorbiscomment =
|
||||||
gst_tag_list_to_vorbiscomment_buffer (taglist, header,
|
gst_tag_list_to_vorbiscomment_buffer (taglist, header,
|
||||||
|
@ -1475,7 +1479,7 @@ gst_flac_parse_generate_headers (GstFlacParse * flacparse)
|
||||||
flacparse->headers = g_list_append (flacparse->headers, streaminfo);
|
flacparse->headers = g_list_append (flacparse->headers, streaminfo);
|
||||||
|
|
||||||
flacparse->headers = g_list_append (flacparse->headers,
|
flacparse->headers = g_list_append (flacparse->headers,
|
||||||
gst_flac_parse_generate_vorbiscomment (flacparse));
|
gst_flac_parse_generate_vorbiscomment (flacparse, TRUE));
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue