mpdparser: Handle invalid external xml link for Period element

Section 5.3.3 in ISO/IEC 23009-1:2014 defines that invalid references
(e.g., invalide URI or cannot be resolved) specified by "@xlink:href" attribute
shall be removed. That means, we should play it without error,
and just ignore the corresponding element.
It's similar to "urn:mpeg:dash:resolve-to-zero:2013".

https://bugzilla.gnome.org/show_bug.cgi?id=774463
This commit is contained in:
Seungha Yang 2016-11-19 18:37:54 +09:00 committed by Sebastian Dröge
parent 5591f55db5
commit e454694f0a
2 changed files with 27 additions and 41 deletions

View file

@ -227,7 +227,7 @@ static GstUri *combine_urls (GstUri * base, GList * list, gchar ** query,
guint idx); guint idx);
static GList *gst_mpd_client_fetch_external_period (GstMpdClient * client, static GList *gst_mpd_client_fetch_external_period (GstMpdClient * client,
GstPeriodNode * period_node, gboolean * error); GstPeriodNode * period_node);
static GList *gst_mpd_client_fetch_external_adaptation_set (GstMpdClient * static GList *gst_mpd_client_fetch_external_adaptation_set (GstMpdClient *
client, GstPeriodNode * period, GstAdaptationSetNode * adapt_set, client, GstPeriodNode * period, GstAdaptationSetNode * adapt_set,
gboolean * error); gboolean * error);
@ -3637,13 +3637,8 @@ gst_mpd_client_fetch_on_load_external_resources (GstMpdClient * client)
if (period->xlink_href && period->actuate == GST_XLINK_ACTUATE_ON_LOAD) { if (period->xlink_href && period->actuate == GST_XLINK_ACTUATE_ON_LOAD) {
GList *new_periods, *prev, *next; GList *new_periods, *prev, *next;
gboolean error;
new_periods = new_periods = gst_mpd_client_fetch_external_period (client, period);
gst_mpd_client_fetch_external_period (client, period, &error);
if (!new_periods && error)
goto syntax_error;
prev = l->prev; prev = l->prev;
client->mpd_node->Periods = client->mpd_node->Periods =
@ -4191,21 +4186,19 @@ gst_mpd_client_setup_representation (GstMpdClient * client,
static GList * static GList *
gst_mpd_client_fetch_external_period (GstMpdClient * client, gst_mpd_client_fetch_external_period (GstMpdClient * client,
GstPeriodNode * period_node, gboolean * error) GstPeriodNode * period_node)
{ {
GstFragment *download; GstFragment *download;
GstAdapter *adapter; GstAdapter *adapter;
GstBuffer *period_buffer; GstBuffer *period_buffer;
GError *err = NULL; GError *err = NULL;
xmlDocPtr doc; xmlDocPtr doc = NULL;
GstUri *base_uri, *uri; GstUri *base_uri, *uri;
gchar *query = NULL; gchar *query = NULL;
gchar *uri_string, *wrapper; gchar *uri_string, *wrapper;
GList *new_periods = NULL; GList *new_periods = NULL;
const gchar *data; const gchar *data;
*error = FALSE;
/* ISO/IEC 23009-1:2014 5.5.3 4) /* ISO/IEC 23009-1:2014 5.5.3 4)
* Remove nodes that resolve to nothing when resolving * Remove nodes that resolve to nothing when resolving
*/ */
@ -4215,7 +4208,6 @@ gst_mpd_client_fetch_external_period (GstMpdClient * client,
} }
if (!client->downloader) { if (!client->downloader) {
*error = TRUE;
return NULL; return NULL;
} }
@ -4245,7 +4237,6 @@ gst_mpd_client_fetch_external_period (GstMpdClient * client,
GST_ERROR ("Failed to download external Period node at '%s': %s", GST_ERROR ("Failed to download external Period node at '%s': %s",
period_node->xlink_href, err->message); period_node->xlink_href, err->message);
g_clear_error (&err); g_clear_error (&err);
*error = TRUE;
return NULL; return NULL;
} }
@ -4273,6 +4264,11 @@ gst_mpd_client_fetch_external_period (GstMpdClient * client,
doc = doc =
xmlReadMemory (data, gst_adapter_available (adapter), "noname.xml", NULL, xmlReadMemory (data, gst_adapter_available (adapter), "noname.xml", NULL,
XML_PARSE_NONET); XML_PARSE_NONET);
gst_adapter_unmap (adapter);
gst_adapter_clear (adapter);
gst_object_unref (adapter);
if (doc) { if (doc) {
xmlNode *root_element = xmlDocGetRootElement (doc); xmlNode *root_element = xmlDocGetRootElement (doc);
xmlNode *iter; xmlNode *iter;
@ -4290,27 +4286,24 @@ gst_mpd_client_fetch_external_period (GstMpdClient * client,
} }
} }
} else { } else {
GST_ERROR ("Failed to parse period node XML"); goto error;
gst_adapter_unmap (adapter);
gst_adapter_clear (adapter);
gst_object_unref (adapter);
*error = TRUE;
return NULL;
} }
done:
if (doc)
xmlFreeDoc (doc); xmlFreeDoc (doc);
gst_adapter_unmap (adapter);
gst_adapter_clear (adapter);
gst_object_unref (adapter);
return new_periods; return new_periods;
error: error:
xmlFreeDoc (doc); GST_ERROR ("Failed to parse period node XML");
gst_adapter_unmap (adapter);
gst_adapter_clear (adapter); if (new_periods) {
gst_object_unref (adapter); g_list_free_full (new_periods,
*error = TRUE; (GDestroyNotify) gst_mpdparser_free_period_node);
return NULL; new_periods = NULL;
}
goto done;
} }
gboolean gboolean
@ -4374,13 +4367,9 @@ gst_mpd_client_setup_media_presentation (GstMpdClient * client,
/* Download external period */ /* Download external period */
if (period_node->xlink_href) { if (period_node->xlink_href) {
GList *new_periods; GList *new_periods;
gboolean error = FALSE;
GList *prev; GList *prev;
new_periods = new_periods = gst_mpd_client_fetch_external_period (client, period_node);
gst_mpd_client_fetch_external_period (client, period_node, &error);
if (!new_periods && error)
goto syntax_error;
prev = list->prev; prev = list->prev;
client->mpd_node->Periods = client->mpd_node->Periods =
@ -4449,15 +4438,10 @@ gst_mpd_client_setup_media_presentation (GstMpdClient * client,
next_period_node = next->data; next_period_node = next->data;
if (next_period_node->xlink_href) { if (next_period_node->xlink_href) {
gboolean next_error;
GList *new_periods; GList *new_periods;
new_periods = new_periods =
gst_mpd_client_fetch_external_period (client, next_period_node, gst_mpd_client_fetch_external_period (client, next_period_node);
&next_error);
if (!new_periods && next_error)
goto syntax_error;
client->mpd_node->Periods = client->mpd_node->Periods =
g_list_delete_link (client->mpd_node->Periods, next); g_list_delete_link (client->mpd_node->Periods, next);

View file

@ -5584,7 +5584,9 @@ GST_START_TEST (dash_mpdparser_xlink_period)
gst_filename_to_uri (XLINK_DOUBLE_PERIOD_FILENAME, NULL); gst_filename_to_uri (XLINK_DOUBLE_PERIOD_FILENAME, NULL);
/* constructs inital mpd using external xml uri */ /* constructs inital mpd using external xml uri */
/* For invalid URI, mpdparser should be ignore it */
xml_joined = g_strjoin ("", xml_frag_start, xml_joined = g_strjoin ("", xml_frag_start,
xml_uri_front, "http://404/ERROR/XML.period", xml_uri_rear,
xml_uri_front, (const char *) file_uri_single_period, xml_uri_rear, xml_uri_front, (const char *) file_uri_single_period, xml_uri_rear,
xml_uri_front, (const char *) file_uri_double_period, xml_uri_rear, xml_uri_front, (const char *) file_uri_double_period, xml_uri_rear,
xml_frag_end, NULL); xml_frag_end, NULL);
@ -5594,7 +5596,7 @@ GST_START_TEST (dash_mpdparser_xlink_period)
period_list = mpdclient->mpd_node->Periods; period_list = mpdclient->mpd_node->Periods;
/* only count periods on initial mpd (external xml does not parsed yet) */ /* only count periods on initial mpd (external xml does not parsed yet) */
assert_equals_int (g_list_length (period_list), 3); assert_equals_int (g_list_length (period_list), 4);
/* process the xml data */ /* process the xml data */
ret = gst_mpd_client_setup_media_presentation (mpdclient, GST_CLOCK_TIME_NONE, ret = gst_mpd_client_setup_media_presentation (mpdclient, GST_CLOCK_TIME_NONE,