gstreamer/validate/gst/validate/media-descriptor.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;
}