• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GStreamer
2  * Copyright (C) <2016> Vivia Nikolaidou <vivia@toolsonair.com>
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public
15  * License along with this library; if not, write to the
16  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  */
19 #ifdef HAVE_CONFIG_H
20 #include "config.h"
21 #endif
22 
23 #include <stdio.h>
24 #include "gstvideotimecode.h"
25 
26 static void
27 gst_video_time_code_gvalue_to_string (const GValue * tc_val, GValue * str_val);
28 static void
29 gst_video_time_code_gvalue_from_string (const GValue * str_val,
30     GValue * tc_val);
31 static gboolean gst_video_time_code_deserialize (GValue * dest,
32     const gchar * tc_str);
33 static gchar *gst_video_time_code_serialize (const GValue * val);
34 
35 static void
_init(GType type)36 _init (GType type)
37 {
38   static GstValueTable table =
39       { 0, (GstValueCompareFunc) gst_video_time_code_compare,
40     (GstValueSerializeFunc) gst_video_time_code_serialize,
41     (GstValueDeserializeFunc) gst_video_time_code_deserialize
42   };
43 
44   table.type = type;
45   gst_value_register (&table);
46   g_value_register_transform_func (type, G_TYPE_STRING,
47       (GValueTransform) gst_video_time_code_gvalue_to_string);
48   g_value_register_transform_func (G_TYPE_STRING, type,
49       (GValueTransform) gst_video_time_code_gvalue_from_string);
50 }
51 
52 G_DEFINE_BOXED_TYPE_WITH_CODE (GstVideoTimeCode, gst_video_time_code,
53     (GBoxedCopyFunc) gst_video_time_code_copy,
54     (GBoxedFreeFunc) gst_video_time_code_free, _init (g_define_type_id));
55 
56 /**
57  * gst_video_time_code_is_valid:
58  * @tc: #GstVideoTimeCode to check
59  *
60  * Returns: whether @tc is a valid timecode (supported frame rate,
61  * hours/minutes/seconds/frames not overflowing)
62  *
63  * Since: 1.10
64  */
65 gboolean
gst_video_time_code_is_valid(const GstVideoTimeCode * tc)66 gst_video_time_code_is_valid (const GstVideoTimeCode * tc)
67 {
68   guint fr;
69 
70   g_return_val_if_fail (tc != NULL, FALSE);
71 
72   if (tc->config.fps_n == 0 || tc->config.fps_d == 0)
73     return FALSE;
74 
75   if (tc->hours >= 24)
76     return FALSE;
77   if (tc->minutes >= 60)
78     return FALSE;
79   if (tc->seconds >= 60)
80     return FALSE;
81 
82   /* We can't have more frames than rounded up frames per second */
83   fr = (tc->config.fps_n + (tc->config.fps_d >> 1)) / tc->config.fps_d;
84   if (tc->frames >= fr && (tc->config.fps_n != 0 || tc->config.fps_d != 1))
85     return FALSE;
86 
87   /* We either need a specific X/1001 framerate or otherwise an integer
88    * framerate */
89   if (tc->config.fps_d == 1001) {
90     if (tc->config.fps_n != 30000 && tc->config.fps_n != 60000 &&
91         tc->config.fps_n != 24000)
92       return FALSE;
93   } else if (tc->config.fps_n % tc->config.fps_d != 0) {
94     return FALSE;
95   }
96 
97   /* We only support 30000/1001 and 60000/1001 as drop-frame framerates.
98    * 24000/1001 is *not* a drop-frame framerate! */
99   if (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME) {
100     if (tc->config.fps_d != 1001 || (tc->config.fps_n != 30000
101             && tc->config.fps_n != 60000))
102       return FALSE;
103   }
104 
105   /* Drop-frame framerates require skipping over the first two
106    * timecodes every minutes except for every tenth minute in case
107    * of 30000/1001 and the first four timecodes for 60000/1001 */
108   if ((tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME) &&
109       tc->minutes % 10 && tc->seconds == 0 && tc->frames < fr / 15) {
110     return FALSE;
111   }
112 
113   return TRUE;
114 }
115 
116 /**
117  * gst_video_time_code_to_string:
118  * @tc: A #GstVideoTimeCode to convert
119  *
120  * Returns: the SMPTE ST 2059-1:2015 string representation of @tc. That will
121  * take the form hh:mm:ss:ff. The last separator (between seconds and frames)
122  * may vary:
123  *
124  * ';' for drop-frame, non-interlaced content and for drop-frame interlaced
125  * field 2
126  * ',' for drop-frame interlaced field 1
127  * ':' for non-drop-frame, non-interlaced content and for non-drop-frame
128  * interlaced field 2
129  * '.' for non-drop-frame interlaced field 1
130  *
131  * Since: 1.10
132  */
133 gchar *
gst_video_time_code_to_string(const GstVideoTimeCode * tc)134 gst_video_time_code_to_string (const GstVideoTimeCode * tc)
135 {
136   gchar *ret;
137   gboolean top_dot_present;
138   gchar sep;
139 
140   /* Top dot is present for non-interlaced content, and for field 2 in
141    * interlaced content */
142   top_dot_present =
143       !((tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_INTERLACED) != 0
144       && tc->field_count == 1);
145 
146   if (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME)
147     sep = top_dot_present ? ';' : ',';
148   else
149     sep = top_dot_present ? ':' : '.';
150 
151   ret =
152       g_strdup_printf ("%02d:%02d:%02d%c%02d", tc->hours, tc->minutes,
153       tc->seconds, sep, tc->frames);
154 
155   return ret;
156 }
157 
158 /**
159  * gst_video_time_code_to_date_time:
160  * @tc: A valid #GstVideoTimeCode to convert
161  *
162  * The @tc.config->latest_daily_jam is required to be non-NULL.
163  *
164  * Returns: (nullable): the #GDateTime representation of @tc or %NULL if @tc
165  *   has no daily jam.
166  *
167  * Since: 1.10
168  */
169 GDateTime *
gst_video_time_code_to_date_time(const GstVideoTimeCode * tc)170 gst_video_time_code_to_date_time (const GstVideoTimeCode * tc)
171 {
172   GDateTime *ret;
173   GDateTime *ret2;
174   gdouble add_us;
175 
176   g_return_val_if_fail (gst_video_time_code_is_valid (tc), NULL);
177 
178   if (tc->config.latest_daily_jam == NULL) {
179     gchar *tc_str = gst_video_time_code_to_string (tc);
180     GST_WARNING
181         ("Asked to convert time code %s to GDateTime, but its latest daily jam is NULL",
182         tc_str);
183     g_free (tc_str);
184     return NULL;
185   }
186 
187   ret = g_date_time_ref (tc->config.latest_daily_jam);
188 
189   gst_util_fraction_to_double (tc->frames * tc->config.fps_d, tc->config.fps_n,
190       &add_us);
191   if ((tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_INTERLACED)
192       && tc->field_count == 1) {
193     gdouble sub_us;
194 
195     gst_util_fraction_to_double (tc->config.fps_d, 2 * tc->config.fps_n,
196         &sub_us);
197     add_us -= sub_us;
198   }
199 
200   ret2 = g_date_time_add_seconds (ret, add_us + tc->seconds);
201   g_date_time_unref (ret);
202   ret = g_date_time_add_minutes (ret2, tc->minutes);
203   g_date_time_unref (ret2);
204   ret2 = g_date_time_add_hours (ret, tc->hours);
205   g_date_time_unref (ret);
206 
207   return ret2;
208 }
209 
210 /**
211  * gst_video_time_code_init_from_date_time:
212  * @tc: an uninitialized #GstVideoTimeCode
213  * @fps_n: Numerator of the frame rate
214  * @fps_d: Denominator of the frame rate
215  * @dt: #GDateTime to convert
216  * @flags: #GstVideoTimeCodeFlags
217  * @field_count: Interlaced video field count
218  *
219  * The resulting config->latest_daily_jam is set to midnight, and timecode is
220  * set to the given time.
221  *
222  * Will assert on invalid parameters, use gst_video_time_code_init_from_date_time_full()
223  * for being able to handle invalid parameters.
224  *
225  * Since: 1.12
226  */
227 void
gst_video_time_code_init_from_date_time(GstVideoTimeCode * tc,guint fps_n,guint fps_d,GDateTime * dt,GstVideoTimeCodeFlags flags,guint field_count)228 gst_video_time_code_init_from_date_time (GstVideoTimeCode * tc,
229     guint fps_n, guint fps_d,
230     GDateTime * dt, GstVideoTimeCodeFlags flags, guint field_count)
231 {
232   if (!gst_video_time_code_init_from_date_time_full (tc, fps_n, fps_d, dt,
233           flags, field_count))
234     g_return_if_fail (gst_video_time_code_is_valid (tc));
235 }
236 
237 /**
238  * gst_video_time_code_init_from_date_time_full:
239  * @tc: a #GstVideoTimeCode
240  * @fps_n: Numerator of the frame rate
241  * @fps_d: Denominator of the frame rate
242  * @dt: #GDateTime to convert
243  * @flags: #GstVideoTimeCodeFlags
244  * @field_count: Interlaced video field count
245  *
246  * The resulting config->latest_daily_jam is set to
247  * midnight, and timecode is set to the given time.
248  *
249  * Returns: %TRUE if @tc could be correctly initialized to a valid timecode
250  *
251  * Since: 1.16
252  */
253 gboolean
gst_video_time_code_init_from_date_time_full(GstVideoTimeCode * tc,guint fps_n,guint fps_d,GDateTime * dt,GstVideoTimeCodeFlags flags,guint field_count)254 gst_video_time_code_init_from_date_time_full (GstVideoTimeCode * tc,
255     guint fps_n, guint fps_d,
256     GDateTime * dt, GstVideoTimeCodeFlags flags, guint field_count)
257 {
258   GDateTime *jam;
259   guint64 frames;
260   gboolean add_a_frame = FALSE;
261 
262   g_return_val_if_fail (tc != NULL, FALSE);
263   g_return_val_if_fail (dt != NULL, FALSE);
264   g_return_val_if_fail (fps_n != 0 && fps_d != 0, FALSE);
265 
266   gst_video_time_code_clear (tc);
267 
268   jam = g_date_time_new_local (g_date_time_get_year (dt),
269       g_date_time_get_month (dt), g_date_time_get_day_of_month (dt), 0, 0, 0.0);
270 
271   /* Note: This might be inaccurate for 1 frame
272    * in case we have a drop frame timecode */
273   frames =
274       gst_util_uint64_scale_round (g_date_time_get_microsecond (dt) *
275       G_GINT64_CONSTANT (1000), fps_n, fps_d * GST_SECOND);
276   if (G_UNLIKELY (((frames == fps_n) && (fps_d == 1)) ||
277           ((frames == fps_n / 1000) && (fps_d == 1001)))) {
278     /* Avoid invalid timecodes */
279     frames--;
280     add_a_frame = TRUE;
281   }
282 
283   gst_video_time_code_init (tc, fps_n, fps_d, jam, flags,
284       g_date_time_get_hour (dt), g_date_time_get_minute (dt),
285       g_date_time_get_second (dt), frames, field_count);
286 
287   if (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME) {
288     guint df = (tc->config.fps_n + (tc->config.fps_d >> 1)) /
289         (15 * tc->config.fps_d);
290     if (tc->minutes % 10 && tc->seconds == 0 && tc->frames < df) {
291       tc->frames = df;
292     }
293   }
294   if (add_a_frame)
295     gst_video_time_code_increment_frame (tc);
296 
297   g_date_time_unref (jam);
298 
299   return gst_video_time_code_is_valid (tc);
300 }
301 
302 /**
303  * gst_video_time_code_nsec_since_daily_jam:
304  * @tc: a valid #GstVideoTimeCode
305  *
306  * Returns: how many nsec have passed since the daily jam of @tc.
307  *
308  * Since: 1.10
309  */
310 guint64
gst_video_time_code_nsec_since_daily_jam(const GstVideoTimeCode * tc)311 gst_video_time_code_nsec_since_daily_jam (const GstVideoTimeCode * tc)
312 {
313   guint64 frames, nsec;
314 
315   g_return_val_if_fail (gst_video_time_code_is_valid (tc), -1);
316 
317   frames = gst_video_time_code_frames_since_daily_jam (tc);
318   nsec =
319       gst_util_uint64_scale (frames, GST_SECOND * tc->config.fps_d,
320       tc->config.fps_n);
321 
322   return nsec;
323 }
324 
325 /**
326  * gst_video_time_code_frames_since_daily_jam:
327  * @tc: a valid #GstVideoTimeCode
328  *
329  * Returns: how many frames have passed since the daily jam of @tc.
330  *
331  * Since: 1.10
332  */
333 guint64
gst_video_time_code_frames_since_daily_jam(const GstVideoTimeCode * tc)334 gst_video_time_code_frames_since_daily_jam (const GstVideoTimeCode * tc)
335 {
336   guint ff_nom;
337   gdouble ff;
338 
339   g_return_val_if_fail (gst_video_time_code_is_valid (tc), -1);
340 
341   gst_util_fraction_to_double (tc->config.fps_n, tc->config.fps_d, &ff);
342   if (tc->config.fps_d == 1001) {
343     ff_nom = tc->config.fps_n / 1000;
344   } else {
345     ff_nom = ff;
346   }
347   if (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME) {
348     /* these need to be truncated to integer: side effect, code looks cleaner
349      * */
350     guint ff_minutes = 60 * ff;
351     guint ff_hours = 3600 * ff;
352     /* for 30000/1001 we drop the first 2 frames per minute, for 60000/1001 we
353      * drop the first 4 : so we use this number */
354     guint dropframe_multiplier;
355 
356     if (tc->config.fps_n == 30000) {
357       dropframe_multiplier = 2;
358     } else if (tc->config.fps_n == 60000) {
359       dropframe_multiplier = 4;
360     } else {
361       /* already checked by gst_video_time_code_is_valid() */
362       g_assert_not_reached ();
363     }
364 
365     return tc->frames + (ff_nom * tc->seconds) +
366         (ff_minutes * tc->minutes) +
367         dropframe_multiplier * ((gint) (tc->minutes / 10)) +
368         (ff_hours * tc->hours);
369   } else {
370     return tc->frames + (ff_nom * (tc->seconds + (60 * (tc->minutes +
371                     (60 * tc->hours)))));
372   }
373 
374 }
375 
376 /**
377  * gst_video_time_code_increment_frame:
378  * @tc: a valid #GstVideoTimeCode
379  *
380  * Adds one frame to @tc.
381  *
382  * Since: 1.10
383  */
384 void
gst_video_time_code_increment_frame(GstVideoTimeCode * tc)385 gst_video_time_code_increment_frame (GstVideoTimeCode * tc)
386 {
387   gst_video_time_code_add_frames (tc, 1);
388 }
389 
390 /**
391  * gst_video_time_code_add_frames:
392  * @tc: a valid #GstVideoTimeCode
393  * @frames: How many frames to add or subtract
394  *
395  * Adds or subtracts @frames amount of frames to @tc. tc needs to
396  * contain valid data, as verified by gst_video_time_code_is_valid().
397  *
398  * Since: 1.10
399  */
400 void
gst_video_time_code_add_frames(GstVideoTimeCode * tc,gint64 frames)401 gst_video_time_code_add_frames (GstVideoTimeCode * tc, gint64 frames)
402 {
403   guint64 framecount;
404   guint64 h_notmod24;
405   guint64 h_new, min_new, sec_new, frames_new;
406   gdouble ff;
407   guint ff_nom;
408   /* This allows for better readability than putting G_GUINT64_CONSTANT(60)
409    * into a long calculation line */
410   const guint64 sixty = 60;
411   /* formulas found in SMPTE ST 2059-1:2015 section 9.4.3
412    * and adapted for 60/1.001 as well as 30/1.001 */
413 
414   g_return_if_fail (gst_video_time_code_is_valid (tc));
415 
416   gst_util_fraction_to_double (tc->config.fps_n, tc->config.fps_d, &ff);
417   if (tc->config.fps_d == 1001) {
418     ff_nom = tc->config.fps_n / 1000;
419   } else {
420     ff_nom = ff;
421   }
422 
423   if (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME) {
424     /* these need to be truncated to integer: side effect, code looks cleaner
425      * */
426     guint ff_minutes = 60 * ff;
427     guint ff_hours = 3600 * ff;
428     /* a bunch of intermediate variables, to avoid monster code with possible
429      * integer overflows */
430     guint64 min_new_tmp1, min_new_tmp2, min_new_tmp3, min_new_denom;
431     /* for 30000/1001 we drop the first 2 frames per minute, for 60000/1001 we
432      * drop the first 4 : so we use this number */
433     guint dropframe_multiplier;
434 
435     if (tc->config.fps_n == 30000) {
436       dropframe_multiplier = 2;
437     } else if (tc->config.fps_n == 60000) {
438       dropframe_multiplier = 4;
439     } else {
440       /* already checked by gst_video_time_code_is_valid() */
441       g_assert_not_reached ();
442     }
443 
444     framecount =
445         frames + tc->frames + (ff_nom * tc->seconds) +
446         (ff_minutes * tc->minutes) +
447         dropframe_multiplier * ((gint) (tc->minutes / 10)) +
448         (ff_hours * tc->hours);
449     h_notmod24 = gst_util_uint64_scale_int (framecount, 1, ff_hours);
450 
451     min_new_denom = sixty * ff_nom;
452     min_new_tmp1 = (framecount - (h_notmod24 * ff_hours)) / min_new_denom;
453     min_new_tmp2 = framecount + dropframe_multiplier * min_new_tmp1;
454     min_new_tmp1 =
455         (framecount - (h_notmod24 * ff_hours)) / (sixty * 10 * ff_nom);
456     min_new_tmp3 =
457         dropframe_multiplier * min_new_tmp1 + (h_notmod24 * ff_hours);
458     min_new =
459         gst_util_uint64_scale_int (min_new_tmp2 - min_new_tmp3, 1,
460         min_new_denom);
461 
462     sec_new =
463         (guint64) ((framecount - (ff_minutes * min_new) -
464             dropframe_multiplier * ((gint) (min_new / 10)) -
465             (ff_hours * h_notmod24)) / ff_nom);
466 
467     frames_new =
468         framecount - (ff_nom * sec_new) - (ff_minutes * min_new) -
469         (dropframe_multiplier * ((gint) (min_new / 10))) -
470         (ff_hours * h_notmod24);
471   } else {
472     framecount =
473         frames + tc->frames + (ff_nom * (tc->seconds + (sixty * (tc->minutes +
474                     (sixty * tc->hours)))));
475     h_notmod24 =
476         gst_util_uint64_scale_int (framecount, 1, ff_nom * sixty * sixty);
477     min_new =
478         gst_util_uint64_scale_int ((framecount -
479             (ff_nom * sixty * sixty * h_notmod24)), 1, (ff_nom * sixty));
480     sec_new =
481         gst_util_uint64_scale_int ((framecount - (ff_nom * sixty * (min_new +
482                     (sixty * h_notmod24)))), 1, ff_nom);
483     frames_new =
484         framecount - (ff_nom * (sec_new + sixty * (min_new +
485                 (sixty * h_notmod24))));
486     if (frames_new > ff_nom)
487       frames_new = 0;
488   }
489 
490   h_new = h_notmod24 % 24;
491 
492   /* The calculations above should always give correct results */
493   g_assert (min_new < 60);
494   g_assert (sec_new < 60);
495   g_assert (frames_new < ff_nom);
496 
497   tc->hours = h_new;
498   tc->minutes = min_new;
499   tc->seconds = sec_new;
500   tc->frames = frames_new;
501 }
502 
503 /**
504  * gst_video_time_code_compare:
505  * @tc1: a valid #GstVideoTimeCode
506  * @tc2: another valid #GstVideoTimeCode
507  *
508  * Compares @tc1 and @tc2. If both have latest daily jam information, it is
509  * taken into account. Otherwise, it is assumed that the daily jam of both
510  * @tc1 and @tc2 was at the same time. Both time codes must be valid.
511  *
512  * Returns: 1 if @tc1 is after @tc2, -1 if @tc1 is before @tc2, 0 otherwise.
513  *
514  * Since: 1.10
515  */
516 gint
gst_video_time_code_compare(const GstVideoTimeCode * tc1,const GstVideoTimeCode * tc2)517 gst_video_time_code_compare (const GstVideoTimeCode * tc1,
518     const GstVideoTimeCode * tc2)
519 {
520   g_return_val_if_fail (gst_video_time_code_is_valid (tc1), -1);
521   g_return_val_if_fail (gst_video_time_code_is_valid (tc2), -1);
522 
523   if (tc1->config.latest_daily_jam == NULL
524       || tc2->config.latest_daily_jam == NULL) {
525     guint64 nsec1, nsec2;
526 #ifndef GST_DISABLE_GST_DEBUG
527     gchar *str1, *str2;
528 
529     str1 = gst_video_time_code_to_string (tc1);
530     str2 = gst_video_time_code_to_string (tc2);
531     GST_INFO
532         ("Comparing time codes %s and %s, but at least one of them has no "
533         "latest daily jam information. Assuming they started together",
534         str1, str2);
535     g_free (str1);
536     g_free (str2);
537 #endif
538     if (tc1->hours > tc2->hours) {
539       return 1;
540     } else if (tc1->hours < tc2->hours) {
541       return -1;
542     }
543     if (tc1->minutes > tc2->minutes) {
544       return 1;
545     } else if (tc1->minutes < tc2->minutes) {
546       return -1;
547     }
548     if (tc1->seconds > tc2->seconds) {
549       return 1;
550     } else if (tc1->seconds < tc2->seconds) {
551       return -1;
552     }
553 
554     nsec1 =
555         gst_util_uint64_scale (GST_SECOND,
556         tc1->frames * tc1->config.fps_n, tc1->config.fps_d);
557     nsec2 =
558         gst_util_uint64_scale (GST_SECOND,
559         tc2->frames * tc2->config.fps_n, tc2->config.fps_d);
560     if (nsec1 > nsec2) {
561       return 1;
562     } else if (nsec1 < nsec2) {
563       return -1;
564     }
565     if (tc1->config.flags & GST_VIDEO_TIME_CODE_FLAGS_INTERLACED) {
566       if (tc1->field_count > tc2->field_count)
567         return 1;
568       else if (tc1->field_count < tc2->field_count)
569         return -1;
570     }
571     return 0;
572   } else {
573     GDateTime *dt1, *dt2;
574     gint ret;
575 
576     dt1 = gst_video_time_code_to_date_time (tc1);
577     dt2 = gst_video_time_code_to_date_time (tc2);
578 
579     ret = g_date_time_compare (dt1, dt2);
580 
581     g_date_time_unref (dt1);
582     g_date_time_unref (dt2);
583 
584     return ret;
585   }
586 }
587 
588 /**
589  * gst_video_time_code_new:
590  * @fps_n: Numerator of the frame rate
591  * @fps_d: Denominator of the frame rate
592  * @latest_daily_jam: The latest daily jam of the #GstVideoTimeCode
593  * @flags: #GstVideoTimeCodeFlags
594  * @hours: the hours field of #GstVideoTimeCode
595  * @minutes: the minutes field of #GstVideoTimeCode
596  * @seconds: the seconds field of #GstVideoTimeCode
597  * @frames: the frames field of #GstVideoTimeCode
598  * @field_count: Interlaced video field count
599  *
600  * @field_count is 0 for progressive, 1 or 2 for interlaced.
601  * @latest_daiy_jam reference is stolen from caller.
602  *
603  * Returns: a new #GstVideoTimeCode with the given values.
604  * The values are not checked for being in a valid range. To see if your
605  * timecode actually has valid content, use gst_video_time_code_is_valid().
606  *
607  * Since: 1.10
608  */
609 GstVideoTimeCode *
gst_video_time_code_new(guint fps_n,guint fps_d,GDateTime * latest_daily_jam,GstVideoTimeCodeFlags flags,guint hours,guint minutes,guint seconds,guint frames,guint field_count)610 gst_video_time_code_new (guint fps_n, guint fps_d, GDateTime * latest_daily_jam,
611     GstVideoTimeCodeFlags flags, guint hours, guint minutes, guint seconds,
612     guint frames, guint field_count)
613 {
614   GstVideoTimeCode *tc;
615 
616   tc = g_new0 (GstVideoTimeCode, 1);
617   gst_video_time_code_init (tc, fps_n, fps_d, latest_daily_jam, flags, hours,
618       minutes, seconds, frames, field_count);
619   return tc;
620 }
621 
622 /**
623  * gst_video_time_code_new_empty:
624  *
625  * Returns: a new empty, invalid #GstVideoTimeCode
626  *
627  * Since: 1.10
628  */
629 GstVideoTimeCode *
gst_video_time_code_new_empty(void)630 gst_video_time_code_new_empty (void)
631 {
632   GstVideoTimeCode *tc;
633 
634   tc = g_new0 (GstVideoTimeCode, 1);
635   gst_video_time_code_clear (tc);
636   return tc;
637 }
638 
639 static void
gst_video_time_code_gvalue_from_string(const GValue * str_val,GValue * tc_val)640 gst_video_time_code_gvalue_from_string (const GValue * str_val, GValue * tc_val)
641 {
642   const gchar *tc_str = g_value_get_string (str_val);
643   GstVideoTimeCode *tc;
644 
645   tc = gst_video_time_code_new_from_string (tc_str);
646   g_value_take_boxed (tc_val, tc);
647 }
648 
649 static void
gst_video_time_code_gvalue_to_string(const GValue * tc_val,GValue * str_val)650 gst_video_time_code_gvalue_to_string (const GValue * tc_val, GValue * str_val)
651 {
652   const GstVideoTimeCode *tc = g_value_get_boxed (tc_val);
653   gchar *tc_str;
654 
655   tc_str = gst_video_time_code_to_string (tc);
656   g_value_take_string (str_val, tc_str);
657 }
658 
659 static gchar *
gst_video_time_code_serialize(const GValue * val)660 gst_video_time_code_serialize (const GValue * val)
661 {
662   GstVideoTimeCode *tc = g_value_get_boxed (val);
663   return gst_video_time_code_to_string (tc);
664 }
665 
666 static gboolean
gst_video_time_code_deserialize(GValue * dest,const gchar * tc_str)667 gst_video_time_code_deserialize (GValue * dest, const gchar * tc_str)
668 {
669   GstVideoTimeCode *tc = gst_video_time_code_new_from_string (tc_str);
670 
671   if (tc == NULL) {
672     return FALSE;
673   }
674 
675   g_value_take_boxed (dest, tc);
676   return TRUE;
677 }
678 
679 /**
680  * gst_video_time_code_new_from_string:
681  * @tc_str: The string that represents the #GstVideoTimeCode
682  *
683  * Returns: (nullable): a new #GstVideoTimeCode from the given string or %NULL
684  *   if the string could not be passed.
685  *
686  * Since: 1.12
687  */
688 GstVideoTimeCode *
gst_video_time_code_new_from_string(const gchar * tc_str)689 gst_video_time_code_new_from_string (const gchar * tc_str)
690 {
691   GstVideoTimeCode *tc;
692   guint hours, minutes, seconds, frames;
693 
694   if (sscanf (tc_str, "%02u:%02u:%02u:%02u", &hours, &minutes, &seconds,
695           &frames)
696       == 4
697       || sscanf (tc_str, "%02u:%02u:%02u.%02u", &hours, &minutes, &seconds,
698           &frames)
699       == 4) {
700     tc = gst_video_time_code_new (0, 1, NULL, GST_VIDEO_TIME_CODE_FLAGS_NONE,
701         hours, minutes, seconds, frames, 0);
702 
703     return tc;
704   } else if (sscanf (tc_str, "%02u:%02u:%02u;%02u", &hours, &minutes, &seconds,
705           &frames)
706       == 4 || sscanf (tc_str, "%02u:%02u:%02u,%02u", &hours, &minutes, &seconds,
707           &frames)
708       == 4) {
709     tc = gst_video_time_code_new (0, 1, NULL,
710         GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME, hours, minutes, seconds, frames,
711         0);
712 
713     return tc;
714   } else {
715     GST_ERROR ("Warning: Could not parse timecode %s. "
716         "Please input a timecode in the form 00:00:00:00", tc_str);
717     return NULL;
718   }
719 }
720 
721 /**
722  * gst_video_time_code_new_from_date_time:
723  * @fps_n: Numerator of the frame rate
724  * @fps_d: Denominator of the frame rate
725  * @dt: #GDateTime to convert
726  * @flags: #GstVideoTimeCodeFlags
727  * @field_count: Interlaced video field count
728  *
729  * The resulting config->latest_daily_jam is set to
730  * midnight, and timecode is set to the given time.
731  *
732  * This might return a completely invalid timecode, use
733  * gst_video_time_code_new_from_date_time_full() to ensure
734  * that you would get %NULL instead in that case.
735  *
736  * Returns: the #GstVideoTimeCode representation of @dt.
737  *
738  * Since: 1.12
739  */
740 GstVideoTimeCode *
gst_video_time_code_new_from_date_time(guint fps_n,guint fps_d,GDateTime * dt,GstVideoTimeCodeFlags flags,guint field_count)741 gst_video_time_code_new_from_date_time (guint fps_n, guint fps_d,
742     GDateTime * dt, GstVideoTimeCodeFlags flags, guint field_count)
743 {
744   GstVideoTimeCode *tc;
745   tc = gst_video_time_code_new_empty ();
746   gst_video_time_code_init_from_date_time_full (tc, fps_n, fps_d, dt, flags,
747       field_count);
748   return tc;
749 }
750 
751 /**
752  * gst_video_time_code_new_from_date_time_full:
753  * @fps_n: Numerator of the frame rate
754  * @fps_d: Denominator of the frame rate
755  * @dt: #GDateTime to convert
756  * @flags: #GstVideoTimeCodeFlags
757  * @field_count: Interlaced video field count
758  *
759  * The resulting config->latest_daily_jam is set to
760  * midnight, and timecode is set to the given time.
761  *
762  * Returns: the #GstVideoTimeCode representation of @dt, or %NULL if
763  *   no valid timecode could be created.
764  *
765  * Since: 1.16
766  */
767 GstVideoTimeCode *
gst_video_time_code_new_from_date_time_full(guint fps_n,guint fps_d,GDateTime * dt,GstVideoTimeCodeFlags flags,guint field_count)768 gst_video_time_code_new_from_date_time_full (guint fps_n, guint fps_d,
769     GDateTime * dt, GstVideoTimeCodeFlags flags, guint field_count)
770 {
771   GstVideoTimeCode *tc;
772   tc = gst_video_time_code_new_empty ();
773   if (!gst_video_time_code_init_from_date_time_full (tc, fps_n, fps_d, dt,
774           flags, field_count)) {
775     gst_video_time_code_free (tc);
776     return NULL;
777   }
778   return tc;
779 }
780 
781 /**
782  * gst_video_time_code_init:
783  * @tc: a #GstVideoTimeCode
784  * @fps_n: Numerator of the frame rate
785  * @fps_d: Denominator of the frame rate
786  * @latest_daily_jam: The latest daily jam of the #GstVideoTimeCode
787  * @flags: #GstVideoTimeCodeFlags
788  * @hours: the hours field of #GstVideoTimeCode
789  * @minutes: the minutes field of #GstVideoTimeCode
790  * @seconds: the seconds field of #GstVideoTimeCode
791  * @frames: the frames field of #GstVideoTimeCode
792  * @field_count: Interlaced video field count
793  *
794  * @field_count is 0 for progressive, 1 or 2 for interlaced.
795  * @latest_daiy_jam reference is stolen from caller.
796  *
797  * Initializes @tc with the given values.
798  * The values are not checked for being in a valid range. To see if your
799  * timecode actually has valid content, use gst_video_time_code_is_valid().
800  *
801  * Since: 1.10
802  */
803 void
gst_video_time_code_init(GstVideoTimeCode * tc,guint fps_n,guint fps_d,GDateTime * latest_daily_jam,GstVideoTimeCodeFlags flags,guint hours,guint minutes,guint seconds,guint frames,guint field_count)804 gst_video_time_code_init (GstVideoTimeCode * tc, guint fps_n, guint fps_d,
805     GDateTime * latest_daily_jam, GstVideoTimeCodeFlags flags, guint hours,
806     guint minutes, guint seconds, guint frames, guint field_count)
807 {
808   tc->hours = hours;
809   tc->minutes = minutes;
810   tc->seconds = seconds;
811   tc->frames = frames;
812   tc->field_count = field_count;
813   tc->config.fps_n = fps_n;
814   tc->config.fps_d = fps_d;
815   if (latest_daily_jam != NULL)
816     tc->config.latest_daily_jam = g_date_time_ref (latest_daily_jam);
817   else
818     tc->config.latest_daily_jam = NULL;
819   tc->config.flags = flags;
820 }
821 
822 /**
823  * gst_video_time_code_clear:
824  * @tc: a #GstVideoTimeCode
825  *
826  * Initializes @tc with empty/zero/NULL values and frees any memory
827  * it might currently use.
828  *
829  * Since: 1.10
830  */
831 void
gst_video_time_code_clear(GstVideoTimeCode * tc)832 gst_video_time_code_clear (GstVideoTimeCode * tc)
833 {
834   tc->hours = 0;
835   tc->minutes = 0;
836   tc->seconds = 0;
837   tc->frames = 0;
838   tc->field_count = 0;
839   tc->config.fps_n = 0;
840   tc->config.fps_d = 1;
841   if (tc->config.latest_daily_jam != NULL)
842     g_date_time_unref (tc->config.latest_daily_jam);
843   tc->config.latest_daily_jam = NULL;
844   tc->config.flags = 0;
845 }
846 
847 /**
848  * gst_video_time_code_copy:
849  * @tc: a #GstVideoTimeCode
850  *
851  * Returns: a new #GstVideoTimeCode with the same values as @tc.
852  *
853  * Since: 1.10
854  */
855 GstVideoTimeCode *
gst_video_time_code_copy(const GstVideoTimeCode * tc)856 gst_video_time_code_copy (const GstVideoTimeCode * tc)
857 {
858   return gst_video_time_code_new (tc->config.fps_n, tc->config.fps_d,
859       tc->config.latest_daily_jam, tc->config.flags, tc->hours, tc->minutes,
860       tc->seconds, tc->frames, tc->field_count);
861 }
862 
863 /**
864  * gst_video_time_code_free:
865  * @tc: a #GstVideoTimeCode
866  *
867  * Frees @tc.
868  *
869  * Since: 1.10
870  */
871 void
gst_video_time_code_free(GstVideoTimeCode * tc)872 gst_video_time_code_free (GstVideoTimeCode * tc)
873 {
874   if (tc->config.latest_daily_jam != NULL)
875     g_date_time_unref (tc->config.latest_daily_jam);
876 
877   g_free (tc);
878 }
879 
880 /**
881  * gst_video_time_code_add_interval:
882  * @tc: The #GstVideoTimeCode where the diff should be added. This
883  * must contain valid timecode values.
884  * @tc_inter: The #GstVideoTimeCodeInterval to add to @tc.
885  * The interval must contain valid values, except that for drop-frame
886  * timecode, it may also contain timecodes which would normally
887  * be dropped. These are then corrected to the next reasonable timecode.
888  *
889  * This makes a component-wise addition of @tc_inter to @tc. For example,
890  * adding ("01:02:03:04", "00:01:00:00") will return "01:03:03:04".
891  * When it comes to drop-frame timecodes,
892  * adding ("00:00:00;00", "00:01:00:00") will return "00:01:00;02"
893  * because of drop-frame oddities. However,
894  * adding ("00:09:00;02", "00:01:00:00") will return "00:10:00;00"
895  * because this time we can have an exact minute.
896  *
897  * Returns: (nullable): A new #GstVideoTimeCode with @tc_inter added or %NULL
898  *   if the interval can't be added.
899  *
900  * Since: 1.12
901  */
902 GstVideoTimeCode *
gst_video_time_code_add_interval(const GstVideoTimeCode * tc,const GstVideoTimeCodeInterval * tc_inter)903 gst_video_time_code_add_interval (const GstVideoTimeCode * tc,
904     const GstVideoTimeCodeInterval * tc_inter)
905 {
906   GstVideoTimeCode *ret;
907   guint frames_to_add;
908   guint df;
909   gboolean needs_correction;
910 
911   g_return_val_if_fail (gst_video_time_code_is_valid (tc), NULL);
912 
913   ret = gst_video_time_code_new (tc->config.fps_n, tc->config.fps_d,
914       tc->config.latest_daily_jam, tc->config.flags, tc_inter->hours,
915       tc_inter->minutes, tc_inter->seconds, tc_inter->frames, 0);
916 
917   df = (tc->config.fps_n + (tc->config.fps_d >> 1)) / (tc->config.fps_d * 15);
918 
919   /* Drop-frame compensation: Create a valid timecode from the
920    * interval */
921   needs_correction = (tc->config.flags & GST_VIDEO_TIME_CODE_FLAGS_DROP_FRAME)
922       && ret->minutes % 10 && ret->seconds == 0 && ret->frames < df;
923   if (needs_correction) {
924     ret->minutes--;
925     ret->seconds = 59;
926     ret->frames = df * 14;
927   }
928 
929   if (!gst_video_time_code_is_valid (ret)) {
930     GST_ERROR ("Unsupported time code interval");
931     gst_video_time_code_free (ret);
932     return NULL;
933   }
934 
935   frames_to_add = gst_video_time_code_frames_since_daily_jam (tc);
936 
937   /* Drop-frame compensation: 00:01:00;00 is falsely interpreted as
938    * 00:00:59;28 */
939   if (needs_correction) {
940     /* User wants us to split at invalid timecodes */
941     if (tc->minutes % 10 == 0 && tc->frames <= df) {
942       /* Apply compensation every 10th minute: before adding the frames,
943        * but only if we are before the "invalid frame" mark */
944       frames_to_add += df;
945       needs_correction = FALSE;
946     }
947   }
948   gst_video_time_code_add_frames (ret, frames_to_add);
949   if (needs_correction && ret->minutes % 10 == 0 && tc->frames > df) {
950     gst_video_time_code_add_frames (ret, df);
951   }
952 
953   return ret;
954 }
955 
956 G_DEFINE_BOXED_TYPE (GstVideoTimeCodeInterval, gst_video_time_code_interval,
957     (GBoxedCopyFunc) gst_video_time_code_interval_copy,
958     (GBoxedFreeFunc) gst_video_time_code_interval_free);
959 
960 /**
961  * gst_video_time_code_interval_new:
962  * @hours: the hours field of #GstVideoTimeCodeInterval
963  * @minutes: the minutes field of #GstVideoTimeCodeInterval
964  * @seconds: the seconds field of #GstVideoTimeCodeInterval
965  * @frames: the frames field of #GstVideoTimeCodeInterval
966  *
967  * Returns: a new #GstVideoTimeCodeInterval with the given values.
968  *
969  * Since: 1.12
970  */
971 GstVideoTimeCodeInterval *
gst_video_time_code_interval_new(guint hours,guint minutes,guint seconds,guint frames)972 gst_video_time_code_interval_new (guint hours, guint minutes, guint seconds,
973     guint frames)
974 {
975   GstVideoTimeCodeInterval *tc;
976 
977   tc = g_new0 (GstVideoTimeCodeInterval, 1);
978   gst_video_time_code_interval_init (tc, hours, minutes, seconds, frames);
979   return tc;
980 }
981 
982 /**
983  * gst_video_time_code_interval_new_from_string:
984  * @tc_inter_str: The string that represents the #GstVideoTimeCodeInterval
985  *
986  * @tc_inter_str must only have ":" as separators.
987  *
988  * Returns: (nullable): a new #GstVideoTimeCodeInterval from the given string
989  *   or %NULL if the string could not be passed.
990  *
991  * Since: 1.12
992  */
993 GstVideoTimeCodeInterval *
gst_video_time_code_interval_new_from_string(const gchar * tc_inter_str)994 gst_video_time_code_interval_new_from_string (const gchar * tc_inter_str)
995 {
996   GstVideoTimeCodeInterval *tc;
997   guint hours, minutes, seconds, frames;
998 
999   if (sscanf (tc_inter_str, "%02u:%02u:%02u:%02u", &hours, &minutes, &seconds,
1000           &frames)
1001       == 4
1002       || sscanf (tc_inter_str, "%02u:%02u:%02u;%02u", &hours, &minutes,
1003           &seconds, &frames)
1004       == 4
1005       || sscanf (tc_inter_str, "%02u:%02u:%02u.%02u", &hours, &minutes,
1006           &seconds, &frames)
1007       == 4
1008       || sscanf (tc_inter_str, "%02u:%02u:%02u,%02u", &hours, &minutes,
1009           &seconds, &frames)
1010       == 4) {
1011     tc = gst_video_time_code_interval_new (hours, minutes, seconds, frames);
1012 
1013     return tc;
1014   } else {
1015     GST_ERROR ("Warning: Could not parse timecode %s. "
1016         "Please input a timecode in the form 00:00:00:00", tc_inter_str);
1017     return NULL;
1018   }
1019 
1020 }
1021 
1022 /**
1023  * gst_video_time_code_interval_init:
1024  * @tc: a #GstVideoTimeCodeInterval
1025  * @hours: the hours field of #GstVideoTimeCodeInterval
1026  * @minutes: the minutes field of #GstVideoTimeCodeInterval
1027  * @seconds: the seconds field of #GstVideoTimeCodeInterval
1028  * @frames: the frames field of #GstVideoTimeCodeInterval
1029  *
1030  * Initializes @tc with the given values.
1031  *
1032  * Since: 1.12
1033  */
1034 void
gst_video_time_code_interval_init(GstVideoTimeCodeInterval * tc,guint hours,guint minutes,guint seconds,guint frames)1035 gst_video_time_code_interval_init (GstVideoTimeCodeInterval * tc, guint hours,
1036     guint minutes, guint seconds, guint frames)
1037 {
1038   tc->hours = hours;
1039   tc->minutes = minutes;
1040   tc->seconds = seconds;
1041   tc->frames = frames;
1042 }
1043 
1044 /**
1045  * gst_video_time_code_interval_clear:
1046  * @tc: a #GstVideoTimeCodeInterval
1047  *
1048  * Initializes @tc with empty/zero/NULL values.
1049  *
1050  * Since: 1.12
1051  */
1052 void
gst_video_time_code_interval_clear(GstVideoTimeCodeInterval * tc)1053 gst_video_time_code_interval_clear (GstVideoTimeCodeInterval * tc)
1054 {
1055   tc->hours = 0;
1056   tc->minutes = 0;
1057   tc->seconds = 0;
1058   tc->frames = 0;
1059 }
1060 
1061 /**
1062  * gst_video_time_code_interval_copy:
1063  * @tc: a #GstVideoTimeCodeInterval
1064  *
1065  * Returns: a new #GstVideoTimeCodeInterval with the same values as @tc.
1066  *
1067  * Since: 1.12
1068  */
1069 GstVideoTimeCodeInterval *
gst_video_time_code_interval_copy(const GstVideoTimeCodeInterval * tc)1070 gst_video_time_code_interval_copy (const GstVideoTimeCodeInterval * tc)
1071 {
1072   return gst_video_time_code_interval_new (tc->hours, tc->minutes,
1073       tc->seconds, tc->frames);
1074 }
1075 
1076 /**
1077  * gst_video_time_code_interval_free:
1078  * @tc: a #GstVideoTimeCodeInterval
1079  *
1080  * Frees @tc.
1081  *
1082  * Since: 1.12
1083  */
1084 void
gst_video_time_code_interval_free(GstVideoTimeCodeInterval * tc)1085 gst_video_time_code_interval_free (GstVideoTimeCodeInterval * tc)
1086 {
1087   g_free (tc);
1088 }
1089