mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-10 11:29:55 +00:00
329 lines
8.5 KiB
C
329 lines
8.5 KiB
C
/**
|
|
* Gstreamer
|
|
*
|
|
* Copyright (c) 2012, Collabora Ltd.
|
|
* Author: Thibault Saunier <thibault.saunier@collabora.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "media-descriptor.h"
|
|
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GstMediaDescriptor, gst_media_descriptor,
|
|
G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE (GST_TYPE_VALIDATE_REPORTER, NULL));
|
|
|
|
#define GST_MEDIA_DESCRIPTOR_GET_PRIVATE(o)\
|
|
(G_TYPE_INSTANCE_GET_PRIVATE ((o), GST_TYPE_MEDIA_DESCRIPTOR, GstMediaDescriptorPrivate))
|
|
|
|
static inline void
|
|
free_tagnode (TagNode * tagnode)
|
|
{
|
|
g_free (tagnode->str_open);
|
|
g_free (tagnode->str_close);
|
|
if (tagnode->taglist)
|
|
gst_tag_list_unref (tagnode->taglist);
|
|
|
|
g_slice_free (TagNode, tagnode);
|
|
}
|
|
|
|
static inline void
|
|
free_tagsnode (TagsNode * tagsnode)
|
|
{
|
|
g_free (tagsnode->str_open);
|
|
g_free (tagsnode->str_close);
|
|
g_list_free_full (tagsnode->tags, (GDestroyNotify) free_tagnode);
|
|
g_slice_free (TagsNode, tagsnode);
|
|
}
|
|
|
|
static inline void
|
|
free_framenode (FrameNode * framenode)
|
|
{
|
|
g_free (framenode->str_open);
|
|
g_free (framenode->str_close);
|
|
|
|
if (framenode->buf)
|
|
gst_buffer_unref (framenode->buf);
|
|
|
|
g_slice_free (FrameNode, framenode);
|
|
}
|
|
|
|
static inline void
|
|
free_streamnode (StreamNode * streamnode)
|
|
{
|
|
if (streamnode->caps)
|
|
gst_caps_unref (streamnode->caps);
|
|
|
|
g_list_free_full (streamnode->frames, (GDestroyNotify) free_framenode);
|
|
|
|
if (streamnode->pad)
|
|
gst_object_unref (streamnode->pad);
|
|
|
|
if (streamnode->tags)
|
|
free_tagsnode (streamnode->tags);
|
|
|
|
if (streamnode->padname)
|
|
g_free (streamnode->padname);
|
|
|
|
if (streamnode->id)
|
|
g_free (streamnode->id);
|
|
|
|
g_free (streamnode->str_open);
|
|
g_free (streamnode->str_close);
|
|
g_slice_free (StreamNode, streamnode);
|
|
}
|
|
|
|
void
|
|
free_filenode (FileNode * filenode)
|
|
{
|
|
g_list_free_full (filenode->streams, (GDestroyNotify) free_streamnode);
|
|
if (filenode->tags)
|
|
free_tagsnode (filenode->tags);
|
|
|
|
if (filenode->uri)
|
|
g_free (filenode->uri);
|
|
|
|
g_free (filenode->str_open);
|
|
g_free (filenode->str_close);
|
|
|
|
g_slice_free (FileNode, filenode);
|
|
}
|
|
|
|
gboolean
|
|
tag_node_compare (TagNode * tnode, const GstTagList * tlist)
|
|
{
|
|
if (gst_structure_is_equal (GST_STRUCTURE (tlist),
|
|
GST_STRUCTURE (tnode->taglist)) == FALSE) {
|
|
return FALSE;
|
|
}
|
|
|
|
tnode->found = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
struct _GstMediaDescriptorPrivate
|
|
{
|
|
gpointer dummy;
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_RUNNER,
|
|
PROP_LAST
|
|
};
|
|
|
|
|
|
static void
|
|
gst_media_descriptor_dispose (GstMediaDescriptor * self)
|
|
{
|
|
G_OBJECT_CLASS (gst_media_descriptor_parent_class)->dispose (G_OBJECT (self));
|
|
}
|
|
|
|
static void
|
|
gst_media_descriptor_finalize (GstMediaDescriptor * self)
|
|
{
|
|
if (self->filenode)
|
|
free_filenode (self->filenode);
|
|
|
|
G_OBJECT_CLASS (gst_media_descriptor_parent_class)->finalize (G_OBJECT
|
|
(self));
|
|
}
|
|
|
|
static void
|
|
gst_media_descriptor_init (GstMediaDescriptor * self)
|
|
{
|
|
self->filenode = g_slice_new0 (FileNode);
|
|
}
|
|
|
|
static void
|
|
_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
switch (prop_id) {
|
|
case PROP_RUNNER:
|
|
/* we assume the runner is valid as long as this scenario is,
|
|
* no ref taken */
|
|
gst_validate_reporter_set_runner (GST_VALIDATE_REPORTER (object),
|
|
g_value_get_object (value));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
switch (prop_id) {
|
|
case PROP_RUNNER:
|
|
/* we assume the runner is valid as long as this scenario is,
|
|
* no ref taken */
|
|
g_value_set_object (value,
|
|
gst_validate_reporter_get_runner (GST_VALIDATE_REPORTER (object)));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_media_descriptor_class_init (GstMediaDescriptorClass * self_class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (self_class);
|
|
|
|
g_type_class_add_private (self_class, sizeof (GstMediaDescriptorPrivate));
|
|
object_class->dispose =
|
|
(void (*)(GObject * object)) gst_media_descriptor_dispose;
|
|
object_class->finalize =
|
|
(void (*)(GObject * object)) gst_media_descriptor_finalize;
|
|
|
|
object_class->get_property = _get_property;
|
|
object_class->set_property = _set_property;
|
|
|
|
g_object_class_install_property (object_class, PROP_RUNNER,
|
|
g_param_spec_object ("validate-runner", "VALIDATE Runner",
|
|
"The Validate runner to " "report errors to",
|
|
GST_TYPE_VALIDATE_RUNNER,
|
|
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE));
|
|
}
|
|
|
|
static gint
|
|
compare_tags (GstMediaDescriptor * ref, StreamNode * rstream,
|
|
StreamNode * cstream)
|
|
{
|
|
gboolean found;
|
|
TagNode *rtag, *ctag;
|
|
GList *rtag_list, *ctag_list;
|
|
TagsNode *rtags, *ctags;
|
|
|
|
rtags = rstream->tags;
|
|
if (rtags == NULL)
|
|
return 1;
|
|
|
|
ctags = cstream->tags;
|
|
for (rtag_list = rtags->tags; rtag_list; rtag_list = rtag_list->next) {
|
|
rtag = rtag_list->data;
|
|
found = FALSE;
|
|
for (ctag_list = ctags->tags; ctag_list; ctag_list = ctag_list->next) {
|
|
ctag = ctag_list->data;
|
|
if (gst_tag_list_is_equal (rtag->taglist, ctag->taglist)) {
|
|
found = TRUE;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (found == FALSE) {
|
|
gchar *rtaglist = gst_tag_list_to_string (rtag->taglist);
|
|
|
|
GST_VALIDATE_REPORT (ref, FILE_TAG_DETECTION_INCORRECT,
|
|
"Reference descriptor for stream %s has tags %s"
|
|
" but no equivalent taglist was found on the compared stream",
|
|
rstream->id, rtaglist);
|
|
g_free (rtaglist);
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Return -1 if not found 1 if OK 0 if an error occured */
|
|
static gint
|
|
comparse_stream (GstMediaDescriptor * ref, StreamNode * rstream,
|
|
StreamNode * cstream)
|
|
{
|
|
if (g_strcmp0 (rstream->id, cstream->id) == 0) {
|
|
if (!gst_caps_is_equal (rstream->caps, cstream->caps)) {
|
|
gchar *rcaps = gst_caps_to_string (rstream->caps),
|
|
*ccaps = gst_caps_to_string (cstream->caps);
|
|
GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
|
|
"Reference descriptor for stream %s has caps: %s"
|
|
" but compared stream %s has caps: %s",
|
|
rstream->id, rcaps, cstream->id, ccaps);
|
|
g_free (rcaps);
|
|
g_free (ccaps);
|
|
return 0;
|
|
}
|
|
|
|
compare_tags (ref, rstream, cstream);
|
|
|
|
return 1;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
gboolean
|
|
gst_media_descriptors_compare (GstMediaDescriptor * ref,
|
|
GstMediaDescriptor * compared)
|
|
{
|
|
GList *rstream_list;
|
|
FileNode *rfilenode = ref->filenode, *cfilenode = compared->filenode;
|
|
|
|
if (rfilenode->duration != cfilenode->duration) {
|
|
GST_VALIDATE_REPORT (ref, FILE_DURATION_INCORRECT,
|
|
"Duration %" GST_TIME_FORMAT " is different from the reference %"
|
|
GST_TIME_FORMAT, GST_TIME_ARGS (cfilenode->duration),
|
|
GST_TIME_ARGS (rfilenode->duration));
|
|
}
|
|
|
|
if (rfilenode->seekable != cfilenode->seekable) {
|
|
GST_VALIDATE_REPORT (ref, FILE_SEEKABLE_INCORRECT,
|
|
"File known as %s but is reported %s now",
|
|
rfilenode->seekable ? "seekable" : "not seekable",
|
|
cfilenode->seekable ? "seekable" : "not seekable");
|
|
}
|
|
|
|
if (g_list_length (rfilenode->streams) != g_list_length (cfilenode->streams)) {
|
|
GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
|
|
"Reference descriptor has %i streams != compared which has %i streams",
|
|
g_list_length (rfilenode->streams), g_list_length (cfilenode->streams));
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
for (rstream_list = rfilenode->streams; rstream_list;
|
|
rstream_list = rstream_list->next) {
|
|
GList *cstream_list;
|
|
gint sfound = -1;
|
|
|
|
for (cstream_list = cfilenode->streams; cstream_list;
|
|
cstream_list = cstream_list->next) {
|
|
|
|
sfound = comparse_stream (ref, rstream_list->data, cstream_list->data);
|
|
if (sfound == 0) {
|
|
return FALSE;
|
|
} else if (sfound == 1) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (sfound == FALSE) {
|
|
GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
|
|
"Could not find stream %s in the compared descriptor",
|
|
((StreamNode *) rstream_list->data)->id);
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|