Merging gst-devtools
[platform/upstream/gstreamer.git] / validate / gst / validate / media-descriptor.c
1 /**
2  * Gstreamer
3  *
4  * Copyright (c) 2012, Collabora Ltd.
5  * Author: Thibault Saunier <thibault.saunier@collabora.com>
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Library General Public
9  * License as published by the Free Software Foundation; either
10  * version 2 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Library General Public License for more details.
16  *
17  * You should have received a copy of the GNU Library General Public
18  * License along with this library; if not, write to the
19  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
20  * Boston, MA 02110-1301, USA.
21  */
22
23 #include <string.h>
24 #include "media-descriptor.h"
25
26 struct _GstValidateMediaDescriptorPrivate
27 {
28   gpointer dummy;
29 };
30
31 G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GstValidateMediaDescriptor,
32     gst_validate_media_descriptor, GST_TYPE_OBJECT,
33     G_ADD_PRIVATE (GstValidateMediaDescriptor)
34     G_IMPLEMENT_INTERFACE (GST_TYPE_VALIDATE_REPORTER, NULL));
35
36 static inline void
37 free_tagnode (GstValidateMediaTagNode * tagnode)
38 {
39   g_free (tagnode->str_open);
40   g_free (tagnode->str_close);
41   if (tagnode->taglist)
42     gst_tag_list_unref (tagnode->taglist);
43
44   g_slice_free (GstValidateMediaTagNode, tagnode);
45 }
46
47 static inline void
48 free_tagsnode (GstValidateMediaTagsNode * tagsnode)
49 {
50   g_free (tagsnode->str_open);
51   g_free (tagsnode->str_close);
52   g_list_free_full (tagsnode->tags, (GDestroyNotify) free_tagnode);
53   g_slice_free (GstValidateMediaTagsNode, tagsnode);
54 }
55
56 static inline void
57 free_framenode (GstValidateMediaFrameNode * framenode)
58 {
59   g_free (framenode->str_open);
60   g_free (framenode->str_close);
61
62   if (framenode->buf)
63     gst_buffer_unref (framenode->buf);
64
65   g_slice_free (GstValidateMediaFrameNode, framenode);
66 }
67
68 static inline void
69 free_segmentnode (GstValidateSegmentNode * segmentnode)
70 {
71   g_free (segmentnode->str_open);
72   g_free (segmentnode->str_close);
73
74   g_slice_free (GstValidateSegmentNode, segmentnode);
75 }
76
77 static inline void
78 free_streamnode (GstValidateMediaStreamNode * streamnode)
79 {
80   if (streamnode->caps)
81     gst_caps_unref (streamnode->caps);
82
83   g_list_free_full (streamnode->frames, (GDestroyNotify) free_framenode);
84   g_list_free_full (streamnode->segments, (GDestroyNotify) free_segmentnode);
85
86   if (streamnode->pad)
87     gst_object_unref (streamnode->pad);
88
89   if (streamnode->tags)
90     free_tagsnode (streamnode->tags);
91
92   g_free (streamnode->padname);
93   g_free (streamnode->id);
94   g_free (streamnode->str_open);
95   g_free (streamnode->str_close);
96   g_slice_free (GstValidateMediaStreamNode, streamnode);
97 }
98
99 void
100 gst_validate_filenode_free (GstValidateMediaFileNode * filenode)
101 {
102   g_list_free_full (filenode->streams, (GDestroyNotify) free_streamnode);
103   if (filenode->tags)
104     free_tagsnode (filenode->tags);
105
106   g_free (filenode->uri);
107
108   if (filenode->caps)
109     gst_caps_unref (filenode->caps);
110
111   g_free (filenode->str_open);
112   g_free (filenode->str_close);
113
114   g_slice_free (GstValidateMediaFileNode, filenode);
115 }
116
117 gboolean
118     gst_validate_tag_node_compare
119     (GstValidateMediaTagNode * tnode, const GstTagList * tlist)
120 {
121   if (gst_structure_is_equal (GST_STRUCTURE (tlist),
122           GST_STRUCTURE (tnode->taglist)) == FALSE) {
123     return FALSE;
124   }
125
126   tnode->found = TRUE;
127
128   return TRUE;
129 }
130
131 enum
132 {
133   PROP_0,
134   PROP_RUNNER,
135   PROP_LAST
136 };
137
138
139 static void
140 gst_validate_media_descriptor_dispose (GstValidateMediaDescriptor * self)
141 {
142   G_OBJECT_CLASS (gst_validate_media_descriptor_parent_class)->dispose (G_OBJECT
143       (self));
144 }
145
146 static void
147 gst_validate_media_descriptor_finalize (GstValidateMediaDescriptor * self)
148 {
149   if (self->filenode)
150     gst_validate_filenode_free (self->filenode);
151
152   G_OBJECT_CLASS (gst_validate_media_descriptor_parent_class)->finalize
153       (G_OBJECT (self));
154 }
155
156 static void
157 gst_validate_media_descriptor_init (GstValidateMediaDescriptor * self)
158 {
159   self->filenode = g_slice_new0 (GstValidateMediaFileNode);
160 }
161
162 static void
163 _set_property (GObject * object, guint prop_id,
164     const GValue * value, GParamSpec * pspec)
165 {
166   switch (prop_id) {
167     case PROP_RUNNER:
168       /* we assume the runner is valid as long as this scenario is,
169        * no ref taken */
170       gst_validate_reporter_set_runner (GST_VALIDATE_REPORTER (object),
171           g_value_get_object (value));
172       break;
173     default:
174       break;
175   }
176 }
177
178 static void
179 _get_property (GObject * object, guint prop_id,
180     GValue * value, GParamSpec * pspec)
181 {
182   switch (prop_id) {
183     case PROP_RUNNER:
184       /* we assume the runner is valid as long as this scenario is,
185        * no ref taken */
186       g_value_take_object (value,
187           gst_validate_reporter_get_runner (GST_VALIDATE_REPORTER (object)));
188       break;
189     default:
190       break;
191   }
192 }
193
194 static void
195 gst_validate_media_descriptor_class_init (GstValidateMediaDescriptorClass *
196     self_class)
197 {
198   GObjectClass *object_class = G_OBJECT_CLASS (self_class);
199
200   object_class->dispose =
201       (void (*)(GObject * object)) gst_validate_media_descriptor_dispose;
202   object_class->finalize =
203       (void (*)(GObject * object)) gst_validate_media_descriptor_finalize;
204
205   object_class->get_property = _get_property;
206   object_class->set_property = _set_property;
207
208   g_object_class_install_property (object_class, PROP_RUNNER,
209       g_param_spec_object ("validate-runner", "VALIDATE Runner",
210           "The Validate runner to report errors to",
211           GST_TYPE_VALIDATE_RUNNER,
212           G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE));
213 }
214
215 static gint
216 compare_tags (GstValidateMediaDescriptor * ref,
217     GstValidateMediaStreamNode * rstream, GstValidateMediaStreamNode * cstream)
218 {
219   gboolean found;
220   GstValidateMediaTagNode *rtag, *ctag;
221   GList *rtag_list, *ctag_list;
222   GstValidateMediaTagsNode *rtags, *ctags;
223
224   rtags = rstream->tags;
225   ctags = cstream->tags;
226   if (!rtags && !ctags)
227     return 1;
228   else if (!rtags && ctags) {
229     GList *taglist;
230     GString *all_tags = g_string_new (NULL);
231
232     for (taglist = ctags->tags; taglist; taglist = taglist->next) {
233       gchar *stags = gst_tag_list_to_string (((GstValidateMediaTagNode *)
234               taglist->data)->taglist);
235
236       g_string_append_printf (all_tags, "%s\n", stags);
237       g_free (stags);
238     }
239
240     GST_VALIDATE_REPORT (ref, FILE_TAG_DETECTION_INCORRECT,
241         "Reference descriptor for stream %s has NO tags"
242         " but tags found: %s", rstream->id, all_tags->str);
243
244     g_string_free (all_tags, TRUE);
245
246     return 0;
247   } else if (rtags && !ctags) {
248     GList *taglist;
249     GString *all_tags = g_string_new (NULL);
250
251     for (taglist = rtags->tags; taglist; taglist = taglist->next) {
252       gchar *stags = gst_tag_list_to_string (((GstValidateMediaTagNode *)
253               taglist->data)->taglist);
254
255       g_string_append_printf (all_tags, "%s\n", stags);
256       g_free (stags);
257     }
258
259     GST_VALIDATE_REPORT (ref, FILE_TAG_DETECTION_INCORRECT,
260         "Reference descriptor for stream %s has tags:\n %s\n"
261         " but NO tags found on the stream", rstream->id, all_tags->str);
262
263     g_string_free (all_tags, TRUE);
264     return 0;
265   }
266
267   for (rtag_list = rtags->tags; rtag_list; rtag_list = rtag_list->next) {
268     rtag = rtag_list->data;
269     found = FALSE;
270     for (ctag_list = ctags->tags; ctag_list; ctag_list = ctag_list->next) {
271       ctag = ctag_list->data;
272       if (gst_tag_list_is_equal (rtag->taglist, ctag->taglist)) {
273         found = TRUE;
274
275         break;
276       }
277     }
278
279     if (found == FALSE) {
280       gchar *rtaglist = gst_tag_list_to_string (rtag->taglist);
281
282       GST_VALIDATE_REPORT (ref, FILE_TAG_DETECTION_INCORRECT,
283           "Reference descriptor for stream %s has tags %s"
284           " but no equivalent taglist was found on the compared stream",
285           rstream->id, rtaglist);
286       g_free (rtaglist);
287
288       return 0;
289     }
290   }
291
292   return 1;
293 }
294
295 /* Workaround false warning caused by differnet file path */
296 static gboolean
297 stream_id_is_equal (const gchar * uri, const gchar * rid, const gchar * cid)
298 {
299   GChecksum *cs;
300   const gchar *stream_id;
301
302   /* Simple case it's the same */
303   if (g_strcmp0 (rid, cid) == 0)
304     return TRUE;
305
306   /* If it's not from file or from our local http server, it should have been the same */
307   if (!g_str_has_prefix (uri, "file://")
308       && !g_str_has_prefix (uri, "imagesequence:/")
309       && !g_str_has_prefix (uri, "http://127.0.0.1"))
310     return FALSE;
311
312   /* taken from basesrc, compute the reference stream-id */
313   cs = g_checksum_new (G_CHECKSUM_SHA256);
314   g_checksum_update (cs, (const guchar *) uri, strlen (uri));
315
316   stream_id = g_checksum_get_string (cs);
317
318   /* If the reference stream_id is the URI SHA256, that means we have a single
319    * stream file (no demuxing), just assume it's the same id */
320   if (g_strcmp0 (rid, stream_id) == 0) {
321     g_checksum_free (cs);
322     return TRUE;
323   }
324
325   /* It should always be prefixed with the SHA256, otherwise it likely means
326    * that basesrc is no longer using a SHA256 checksum on the URI, and this
327    * workaround will need to be fixed */
328   if (!g_str_has_prefix (rid, stream_id)) {
329     g_checksum_free (cs);
330     return FALSE;
331   }
332   g_checksum_free (cs);
333
334   /* we strip the IDS to the delimitor, and then compare */
335   rid = strchr (rid, '/');
336   cid = strchr (cid, '/');
337
338   if (rid == NULL || cid == NULL)
339     return FALSE;
340
341   if (g_strcmp0 (rid, cid) == 0)
342     return TRUE;
343
344   return FALSE;
345 }
346
347 static gboolean
348 compare_segments (GstValidateMediaDescriptor * ref,
349     gint i,
350     GstValidateMediaStreamNode * rstream,
351     GstValidateSegmentNode * rsegment, GstValidateSegmentNode * csegment)
352 {
353   if (rsegment->next_frame_id != csegment->next_frame_id) {
354     GST_VALIDATE_REPORT (ref, FILE_SEGMENT_INCORRECT,
355         "Segment %" GST_SEGMENT_FORMAT
356         " didn't come before the same frame ID, expected to come before %d, came before %d",
357         &rsegment->segment, rsegment->next_frame_id, csegment->next_frame_id);
358     return FALSE;
359   }
360 #define CHECK_SEGMENT_FIELD(fieldname, format) \
361   if (rsegment->segment.fieldname != csegment->segment.fieldname) { \
362     GST_ERROR ("Expected: %" GST_SEGMENT_FORMAT " got: %" GST_SEGMENT_FORMAT, \
363       &rsegment->segment, &csegment->segment); \
364     GST_VALIDATE_REPORT (ref, FILE_SEGMENT_INCORRECT, \
365         "Stream %s segment %d has " #fieldname \
366         " mismatch, Expected " format " got: " format , \
367         rstream->id, i, rsegment->segment.fieldname, \
368         csegment->segment.fieldname); \
369       return FALSE; \
370   }
371
372   CHECK_SEGMENT_FIELD (flags, "%d");
373   CHECK_SEGMENT_FIELD (rate, "%f");
374   CHECK_SEGMENT_FIELD (applied_rate, "%f");
375   CHECK_SEGMENT_FIELD (base, "%" G_GUINT64_FORMAT);
376   CHECK_SEGMENT_FIELD (offset, "%" G_GUINT64_FORMAT);
377   CHECK_SEGMENT_FIELD (start, "%" G_GUINT64_FORMAT);
378   CHECK_SEGMENT_FIELD (stop, "%" G_GUINT64_FORMAT);
379   CHECK_SEGMENT_FIELD (time, "%" G_GUINT64_FORMAT);
380   /* We do not compare segment position since it's a field for usage only within the element */
381   /* CHECK_SEGMENT_FIELD (position, "%" G_GUINT64_FORMAT); */
382   CHECK_SEGMENT_FIELD (duration, "%" G_GUINT64_FORMAT);
383
384   return TRUE;
385 }
386
387 static void
388 append_segment_diff (GString * diff, char diffsign, GList * segments)
389 {
390   GList *tmp;
391
392   for (tmp = segments; tmp; tmp = tmp->next) {
393     gchar *ssegment =
394         gst_info_strdup_printf ("%c %" GST_SEGMENT_FORMAT "\n", diffsign,
395         &((GstValidateSegmentNode *) tmp->data)->segment);
396     g_string_append (diff, ssegment);
397     g_free (ssegment);
398
399   }
400 }
401
402 static gboolean
403 compare_segment_list (GstValidateMediaDescriptor * ref,
404     GstValidateMediaStreamNode * rstream, GstValidateMediaStreamNode * cstream)
405 {
406   gint i;
407   GList *rsegments, *csegments;
408
409   /* Keep compatibility with media stream files that do not have segments */
410   if (rstream->segments
411       && g_list_length (rstream->segments) !=
412       g_list_length (cstream->segments)) {
413     GString *diff = g_string_new (NULL);
414
415     append_segment_diff (diff, '-', rstream->segments);
416     append_segment_diff (diff, '+', cstream->segments);
417     GST_VALIDATE_REPORT (ref, FILE_SEGMENT_INCORRECT,
418         "Stream reference has %i segments, compared one has %i segments\n%s",
419         g_list_length (rstream->segments), g_list_length (cstream->segments),
420         diff->str);
421     g_string_free (diff, TRUE);
422   }
423
424   for (i = 0, rsegments = rstream->segments, csegments = cstream->segments;
425       rsegments;
426       rsegments = rsegments->next, csegments = csegments->next, i++) {
427     GstValidateSegmentNode *rsegment, *csegment;
428
429     if (csegments == NULL) {
430       /* The list was checked to be of the same size */
431       g_assert_not_reached ();
432       return FALSE;
433     }
434
435     rsegment = rsegments->data;
436     csegment = csegments->data;
437
438     if (!compare_segments (ref, i, rstream, rsegment, csegment))
439       return FALSE;
440   }
441
442   return TRUE;
443 }
444
445 static gboolean
446 compare_frames (GstValidateMediaDescriptor * ref,
447     GstValidateMediaStreamNode *
448     rstream, GstValidateMediaFrameNode * rframe,
449     GstValidateMediaFrameNode * cframe)
450 {
451   if (rframe->id != cframe->id) {
452     GST_VALIDATE_REPORT (ref, FILE_FRAMES_INCORRECT,
453         "Stream frame %s ids mismatch: %" G_GUINT64_FORMAT " != %"
454         G_GUINT64_FORMAT, rstream->id, rframe->id, cframe->id);
455     return FALSE;
456   }
457 #define CHECK_FRAME_FIELD(fieldname, format, unknown_value) \
458   if (rframe->fieldname != unknown_value && rframe->fieldname != cframe->fieldname) { \
459     GST_VALIDATE_REPORT (ref, FILE_FRAMES_INCORRECT, \
460         "Stream %s frames with id %" G_GUINT64_FORMAT " have " #fieldname \
461         " mismatch. Expected " format ", got " format, rstream->id, \
462         rframe->id, rframe->fieldname, cframe->fieldname); \
463     return FALSE; \
464   }
465
466   CHECK_FRAME_FIELD (pts, "%" G_GUINT64_FORMAT, GST_VALIDATE_UNKNOWN_UINT64);
467   CHECK_FRAME_FIELD (dts, "%" G_GUINT64_FORMAT, GST_VALIDATE_UNKNOWN_UINT64);
468   CHECK_FRAME_FIELD (duration, "%" G_GUINT64_FORMAT,
469       GST_VALIDATE_UNKNOWN_UINT64);
470   CHECK_FRAME_FIELD (running_time, "%" G_GUINT64_FORMAT,
471       GST_VALIDATE_UNKNOWN_UINT64);
472   CHECK_FRAME_FIELD (is_keyframe, "%d", GST_VALIDATE_UNKNOWN_BOOL);
473
474   return TRUE;
475 }
476
477 static gboolean
478 compare_frames_list (GstValidateMediaDescriptor * ref,
479     GstValidateMediaStreamNode * rstream, GstValidateMediaStreamNode * cstream)
480 {
481   GList *rframes, *cframes;
482
483   if (g_list_length (rstream->frames) != g_list_length (cstream->frames)) {
484     GST_VALIDATE_REPORT (ref, FILE_FRAMES_INCORRECT,
485         "Stream reference has %i frames, compared one has %i frames",
486         g_list_length (rstream->frames), g_list_length (cstream->frames));
487     return FALSE;
488   }
489
490   for (rframes = rstream->frames, cframes = cstream->frames; rframes;
491       rframes = g_list_next (rframes), cframes = g_list_next (cframes)) {
492     GstValidateMediaFrameNode *rframe, *cframe;
493
494     if (cframes == NULL) {
495       /* The list was checked to be of the same size */
496       g_assert_not_reached ();
497       return FALSE;
498     }
499
500     rframe = rframes->data;
501     cframe = cframes->data;
502
503     if (!compare_frames (ref, rstream, rframe, cframe)) {
504       return FALSE;
505     }
506   }
507
508   return TRUE;
509 }
510
511 static GstCaps *
512 caps_cleanup_parsing_fields (GstCaps * caps)
513 {
514   gint i;
515   GstCaps *res = gst_caps_copy (caps);
516
517   for (i = 0; i < gst_caps_get_size (res); i++) {
518     GstStructure *s = gst_caps_get_structure (res, i);
519
520     gst_structure_remove_fields (s, "stream-format", "codec_data", "parsed",
521         "frames", "alignment", NULL);
522   }
523
524   return res;
525 }
526
527 /*  Return TRUE if found FALSE otherwise */
528 static gboolean
529 compare_streams (GstValidateMediaDescriptor * ref,
530     GstValidateMediaStreamNode * rstream, GstValidateMediaStreamNode * cstream)
531 {
532   GstCaps *rcaps, *ccaps;
533
534   if (!stream_id_is_equal (ref->filenode->uri, rstream->id, cstream->id))
535     return FALSE;
536
537   rcaps = caps_cleanup_parsing_fields (rstream->caps);
538   ccaps = caps_cleanup_parsing_fields (cstream->caps);
539
540   if (!gst_caps_is_equal (rcaps, ccaps)) {
541     gchar *rcaps_str = gst_caps_to_string (rcaps),
542         *ccaps_str = gst_caps_to_string (ccaps);
543     GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
544         "Reference descriptor for stream %s has caps: %s"
545         " but compared stream %s has caps: %s",
546         rstream->id, rcaps_str, cstream->id, ccaps_str);
547     g_free (rcaps_str);
548     g_free (ccaps_str);
549   }
550
551   gst_caps_unref (rcaps);
552   gst_caps_unref (ccaps);
553   /* We ignore the return value on purpose as this is not critical */
554   compare_tags (ref, rstream, cstream);
555
556   compare_segment_list (ref, rstream, cstream);
557   compare_frames_list (ref, rstream, cstream);
558
559   return TRUE;
560 }
561
562 gboolean
563 gst_validate_media_descriptors_compare (GstValidateMediaDescriptor * ref,
564     GstValidateMediaDescriptor * compared)
565 {
566   GList *rstream_list;
567   GstValidateMediaFileNode
568       * rfilenode = ref->filenode, *cfilenode = compared->filenode;
569
570   if (rfilenode->duration != cfilenode->duration) {
571     GST_VALIDATE_REPORT (ref, FILE_DURATION_INCORRECT,
572         "Duration %" GST_TIME_FORMAT " is different from the reference %"
573         GST_TIME_FORMAT, GST_TIME_ARGS (cfilenode->duration),
574         GST_TIME_ARGS (rfilenode->duration));
575   }
576
577   if (rfilenode->seekable != cfilenode->seekable) {
578     GST_VALIDATE_REPORT (ref, FILE_SEEKABLE_INCORRECT,
579         "File known as %s but is reported %s now",
580         rfilenode->seekable ? "seekable" : "not seekable",
581         cfilenode->seekable ? "seekable" : "not seekable");
582   }
583
584   if (g_list_length (rfilenode->streams) != g_list_length (cfilenode->streams)) {
585     GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
586         "Reference descriptor has %i streams != compared which has %i streams",
587         g_list_length (rfilenode->streams), g_list_length (cfilenode->streams));
588
589     return FALSE;
590   }
591
592
593   for (rstream_list = rfilenode->streams; rstream_list;
594       rstream_list = rstream_list->next) {
595     GList *cstream_list;
596     gboolean sfound = FALSE;
597
598     for (cstream_list = cfilenode->streams; cstream_list;
599         cstream_list = cstream_list->next) {
600
601       sfound = compare_streams (ref, rstream_list->data, cstream_list->data);
602       if (sfound)
603         break;
604     }
605
606     if (!sfound) {
607       GST_VALIDATE_REPORT (ref, FILE_PROFILE_INCORRECT,
608           "Could not find stream %s in the compared descriptor",
609           ((GstValidateMediaStreamNode *) rstream_list->data)->id);
610     }
611   }
612
613   return TRUE;
614 }
615
616 gboolean
617 gst_validate_media_descriptor_detects_frames (GstValidateMediaDescriptor * self)
618 {
619   g_return_val_if_fail (GST_IS_VALIDATE_MEDIA_DESCRIPTOR (self), FALSE);
620   g_return_val_if_fail (self->filenode, FALSE);
621
622   return self->filenode->frame_detection;
623 }
624
625 /**
626  * gst_validate_media_descriptor_get_buffers: (skip):
627  */
628 gboolean
629 gst_validate_media_descriptor_get_buffers (GstValidateMediaDescriptor * self,
630     GstPad * pad, GCompareFunc compare_func, GList ** bufs)
631 {
632   GList *tmpstream, *tmpframe;
633   gboolean check = (pad == NULL), ret = FALSE;
634   GstCaps *pad_caps = gst_pad_get_current_caps (pad);
635
636   g_return_val_if_fail (GST_IS_VALIDATE_MEDIA_DESCRIPTOR (self), FALSE);
637   g_return_val_if_fail (self->filenode, FALSE);
638
639   for (tmpstream = self->filenode->streams;
640       tmpstream; tmpstream = tmpstream->next) {
641     GstValidateMediaStreamNode
642         * streamnode = (GstValidateMediaStreamNode *) tmpstream->data;
643
644     if (pad && streamnode->pad == pad)
645       check = TRUE;
646
647     if (!streamnode->pad && gst_caps_is_subset (pad_caps, streamnode->caps)) {
648       check = TRUE;
649     }
650
651     if (check) {
652       ret = TRUE;
653       for (tmpframe = streamnode->frames; tmpframe; tmpframe = tmpframe->next) {
654         if (compare_func)
655           *bufs =
656               g_list_insert_sorted (*bufs,
657               gst_buffer_ref ((
658                       (GstValidateMediaFrameNode
659                           *) tmpframe->data)->buf), compare_func);
660         else
661           *bufs =
662               g_list_prepend (*bufs,
663               gst_buffer_ref ((
664                       (GstValidateMediaFrameNode *) tmpframe->data)->buf));
665       }
666
667       if (pad != NULL)
668         goto done;
669     }
670   }
671
672
673 done:
674
675   if (compare_func == NULL)
676     *bufs = g_list_reverse (*bufs);
677
678   gst_caps_unref (pad_caps);
679   return ret;
680 }
681
682 gboolean
683 gst_validate_media_descriptor_has_frame_info (GstValidateMediaDescriptor * self)
684 {
685   GList *tmpstream;
686
687   for (tmpstream = self->filenode->streams;
688       tmpstream; tmpstream = tmpstream->next) {
689     GstValidateMediaStreamNode
690         * streamnode = (GstValidateMediaStreamNode *) tmpstream->data;
691
692     if (g_list_length (streamnode->frames))
693       return TRUE;
694   }
695
696   return FALSE;
697 }
698
699 GstClockTime
700 gst_validate_media_descriptor_get_duration (GstValidateMediaDescriptor * self)
701 {
702   g_return_val_if_fail (GST_IS_VALIDATE_MEDIA_DESCRIPTOR (self), FALSE);
703   g_return_val_if_fail (self->filenode, FALSE);
704
705   return self->filenode->duration;
706 }
707
708 gboolean
709 gst_validate_media_descriptor_get_seekable (GstValidateMediaDescriptor * self)
710 {
711   g_return_val_if_fail (GST_IS_VALIDATE_MEDIA_DESCRIPTOR (self), FALSE);
712   g_return_val_if_fail (self->filenode, FALSE);
713
714   return self->filenode->seekable;
715 }
716
717 /**
718  * gst_validate_media_descriptor_get_pads: (skip):
719  */
720 GList *
721 gst_validate_media_descriptor_get_pads (GstValidateMediaDescriptor * self)
722 {
723   GList *ret = NULL, *tmp;
724
725   for (tmp = self->filenode->streams; tmp; tmp = tmp->next) {
726     GstValidateMediaStreamNode
727         * snode = (GstValidateMediaStreamNode *) tmp->data;
728     ret = g_list_append (ret, gst_pad_new (snode->padname, GST_PAD_UNKNOWN));
729   }
730
731   return ret;
732 }