Blame gst/gstclock.c

Packit Service 963350
/* GStreamer
Packit Service 963350
 * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
Packit Service 963350
 *                    2000 Wim Taymans <wtay@chello.be>
Packit Service 963350
 *                    2004 Wim Taymans <wim@fluendo.com>
Packit Service 963350
 *
Packit Service 963350
 * gstclock.c: Clock subsystem for maintaining time sync
Packit Service 963350
 *
Packit Service 963350
 * This library is free software; you can redistribute it and/or
Packit Service 963350
 * modify it under the terms of the GNU Library General Public
Packit Service 963350
 * License as published by the Free Software Foundation; either
Packit Service 963350
 * version 2 of the License, or (at your option) any later version.
Packit Service 963350
 *
Packit Service 963350
 * This library is distributed in the hope that it will be useful,
Packit Service 963350
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 963350
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 963350
 * Library General Public License for more details.
Packit Service 963350
 *
Packit Service 963350
 * You should have received a copy of the GNU Library General Public
Packit Service 963350
 * License along with this library; if not, write to the
Packit Service 963350
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit Service 963350
 * Boston, MA 02110-1301, USA.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * SECTION:gstclock
Packit Service 963350
 * @title: GstClock
Packit Service 963350
 * @short_description: Abstract class for global clocks
Packit Service 963350
 * @see_also: #GstSystemClock, #GstPipeline
Packit Service 963350
 *
Packit Service 963350
 * GStreamer uses a global clock to synchronize the plugins in a pipeline.
Packit Service 963350
 * Different clock implementations are possible by implementing this abstract
Packit Service 963350
 * base class or, more conveniently, by subclassing #GstSystemClock.
Packit Service 963350
 *
Packit Service 963350
 * The #GstClock returns a monotonically increasing time with the method
Packit Service 963350
 * gst_clock_get_time(). Its accuracy and base time depend on the specific
Packit Service 963350
 * clock implementation but time is always expressed in nanoseconds. Since the
Packit Service 963350
 * baseline of the clock is undefined, the clock time returned is not
Packit Service 963350
 * meaningful in itself, what matters are the deltas between two clock times.
Packit Service 963350
 * The time returned by a clock is called the absolute time.
Packit Service 963350
 *
Packit Service 963350
 * The pipeline uses the clock to calculate the running time. Usually all
Packit Service 963350
 * renderers synchronize to the global clock using the buffer timestamps, the
Packit Service 963350
 * newsegment events and the element's base time, see #GstPipeline.
Packit Service 963350
 *
Packit Service 963350
 * A clock implementation can support periodic and single shot clock
Packit Service 963350
 * notifications both synchronous and asynchronous.
Packit Service 963350
 *
Packit Service 963350
 * One first needs to create a #GstClockID for the periodic or single shot
Packit Service 963350
 * notification using gst_clock_new_single_shot_id() or
Packit Service 963350
 * gst_clock_new_periodic_id().
Packit Service 963350
 *
Packit Service 963350
 * To perform a blocking wait for the specific time of the #GstClockID use the
Packit Service 963350
 * gst_clock_id_wait(). To receive a callback when the specific time is reached
Packit Service 963350
 * in the clock use gst_clock_id_wait_async(). Both these calls can be
Packit Service 963350
 * interrupted with the gst_clock_id_unschedule() call. If the blocking wait is
Packit Service 963350
 * unscheduled a return value of #GST_CLOCK_UNSCHEDULED is returned.
Packit Service 963350
 *
Packit Service 963350
 * Periodic callbacks scheduled async will be repeatedly called automatically
Packit Service 963350
 * until it is unscheduled. To schedule a sync periodic callback,
Packit Service 963350
 * gst_clock_id_wait() should be called repeatedly.
Packit Service 963350
 *
Packit Service 963350
 * The async callbacks can happen from any thread, either provided by the core
Packit Service 963350
 * or from a streaming thread. The application should be prepared for this.
Packit Service 963350
 *
Packit Service 963350
 * A #GstClockID that has been unscheduled cannot be used again for any wait
Packit Service 963350
 * operation, a new #GstClockID should be created and the old unscheduled one
Packit Service 963350
 * should be destroyed with gst_clock_id_unref().
Packit Service 963350
 *
Packit Service 963350
 * It is possible to perform a blocking wait on the same #GstClockID from
Packit Service 963350
 * multiple threads. However, registering the same #GstClockID for multiple
Packit Service 963350
 * async notifications is not possible, the callback will only be called for
Packit Service 963350
 * the thread registering the entry last.
Packit Service 963350
 *
Packit Service 963350
 * None of the wait operations unref the #GstClockID, the owner is responsible
Packit Service 963350
 * for unreffing the ids itself. This holds for both periodic and single shot
Packit Service 963350
 * notifications. The reason being that the owner of the #GstClockID has to
Packit Service 963350
 * keep a handle to the #GstClockID to unblock the wait on FLUSHING events or
Packit Service 963350
 * state changes and if the entry would be unreffed automatically, the handle
Packit Service 963350
 * might become invalid without any notification.
Packit Service 963350
 *
Packit Service 963350
 * These clock operations do not operate on the running time, so the callbacks
Packit Service 963350
 * will also occur when not in PLAYING state as if the clock just keeps on
Packit Service 963350
 * running. Some clocks however do not progress when the element that provided
Packit Service 963350
 * the clock is not PLAYING.
Packit Service 963350
 *
Packit Service 963350
 * When a clock has the #GST_CLOCK_FLAG_CAN_SET_MASTER flag set, it can be
Packit Service 963350
 * slaved to another #GstClock with the gst_clock_set_master(). The clock will
Packit Service 963350
 * then automatically be synchronized to this master clock by repeatedly
Packit Service 963350
 * sampling the master clock and the slave clock and recalibrating the slave
Packit Service 963350
 * clock with gst_clock_set_calibration(). This feature is mostly useful for
Packit Service 963350
 * plugins that have an internal clock but must operate with another clock
Packit Service 963350
 * selected by the #GstPipeline.  They can track the offset and rate difference
Packit Service 963350
 * of their internal clock relative to the master clock by using the
Packit Service 963350
 * gst_clock_get_calibration() function.
Packit Service 963350
 *
Packit Service 963350
 * The master/slave synchronisation can be tuned with the #GstClock:timeout,
Packit Service 963350
 * #GstClock:window-size and #GstClock:window-threshold properties.
Packit Service 963350
 * The #GstClock:timeout property defines the interval to sample the master
Packit Service 963350
 * clock and run the calibration functions. #GstClock:window-size defines the
Packit Service 963350
 * number of samples to use when calibrating and #GstClock:window-threshold
Packit Service 963350
 * defines the minimum number of samples before the calibration is performed.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#include "gst_private.h"
Packit Service 963350
#include <time.h>
Packit Service 963350
Packit Service 963350
#include "gstclock.h"
Packit Service 963350
#include "gstinfo.h"
Packit Service 963350
#include "gstutils.h"
Packit Service 963350
#include "glib-compat-private.h"
Packit Service 963350
Packit Service 963350
/* #define DEBUGGING_ENABLED */
Packit Service 963350
Packit Service 963350
#define DEFAULT_WINDOW_SIZE             32
Packit Service 963350
#define DEFAULT_WINDOW_THRESHOLD        4
Packit Service 963350
#define DEFAULT_TIMEOUT                 GST_SECOND / 10
Packit Service 963350
Packit Service 963350
enum
Packit Service 963350
{
Packit Service 963350
  PROP_0,
Packit Service 963350
  PROP_WINDOW_SIZE,
Packit Service 963350
  PROP_WINDOW_THRESHOLD,
Packit Service 963350
  PROP_TIMEOUT
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
enum
Packit Service 963350
{
Packit Service 963350
  SIGNAL_SYNCED,
Packit Service 963350
  SIGNAL_LAST
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
#define GST_CLOCK_SLAVE_LOCK(clock)     g_mutex_lock (&GST_CLOCK_CAST (clock)->priv->slave_lock)
Packit Service 963350
#define GST_CLOCK_SLAVE_UNLOCK(clock)   g_mutex_unlock (&GST_CLOCK_CAST (clock)->priv->slave_lock)
Packit Service 963350
Packit Service 963350
struct _GstClockPrivate
Packit Service 963350
{
Packit Service 963350
  GMutex slave_lock;            /* order: SLAVE_LOCK, OBJECT_LOCK */
Packit Service 963350
Packit Service 963350
  GCond sync_cond;
Packit Service 963350
Packit Service 963350
  /* with LOCK */
Packit Service 963350
  GstClockTime internal_calibration;
Packit Service 963350
  GstClockTime external_calibration;
Packit Service 963350
  GstClockTime rate_numerator;
Packit Service 963350
  GstClockTime rate_denominator;
Packit Service 963350
  GstClockTime last_time;
Packit Service 963350
Packit Service 963350
  /* with LOCK */
Packit Service 963350
  GstClockTime resolution;
Packit Service 963350
Packit Service 963350
  /* for master/slave clocks */
Packit Service 963350
  GstClock *master;
Packit Service 963350
Packit Service 963350
  /* with SLAVE_LOCK */
Packit Service 963350
  gboolean filling;
Packit Service 963350
  gint window_size;
Packit Service 963350
  gint window_threshold;
Packit Service 963350
  gint time_index;
Packit Service 963350
  GstClockTime timeout;
Packit Service 963350
  GstClockTime *times;
Packit Service 963350
  GstClockTime *times_temp;
Packit Service 963350
  GstClockID clockid;
Packit Service 963350
Packit Service 963350
  gint pre_count;
Packit Service 963350
  gint post_count;
Packit Service 963350
Packit Service 963350
  gboolean synced;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
/* seqlocks */
Packit Service 963350
#define read_seqbegin(clock)                                   \
Packit Service 963350
  g_atomic_int_get (&clock->priv->post_count);
Packit Service 963350
Packit Service 963350
static inline gboolean
Packit Service 963350
read_seqretry (GstClock * clock, gint seq)
Packit Service 963350
{
Packit Service 963350
  /* no retry if the seqnum did not change */
Packit Service 963350
  if (G_LIKELY (seq == g_atomic_int_get (&clock->priv->pre_count)))
Packit Service 963350
    return FALSE;
Packit Service 963350
Packit Service 963350
  /* wait for the writer to finish and retry */
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
#define write_seqlock(clock)                      \
Packit Service 963350
G_STMT_START {                                    \
Packit Service 963350
  GST_OBJECT_LOCK (clock);                        \
Packit Service 963350
  g_atomic_int_inc (&clock->priv->pre_count);     \
Packit Service 963350
} G_STMT_END;
Packit Service 963350
Packit Service 963350
#define write_sequnlock(clock)                    \
Packit Service 963350
G_STMT_START {                                    \
Packit Service 963350
  g_atomic_int_inc (&clock->priv->post_count);    \
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);                      \
Packit Service 963350
} G_STMT_END;
Packit Service 963350
Packit Service 963350
#ifndef GST_DISABLE_GST_DEBUG
Packit Service 963350
static const gchar *
Packit Service 963350
gst_clock_return_get_name (GstClockReturn ret)
Packit Service 963350
{
Packit Service 963350
  switch (ret) {
Packit Service 963350
    case GST_CLOCK_OK:
Packit Service 963350
      return "ok";
Packit Service 963350
    case GST_CLOCK_EARLY:
Packit Service 963350
      return "early";
Packit Service 963350
    case GST_CLOCK_UNSCHEDULED:
Packit Service 963350
      return "unscheduled";
Packit Service 963350
    case GST_CLOCK_BUSY:
Packit Service 963350
      return "busy";
Packit Service 963350
    case GST_CLOCK_BADTIME:
Packit Service 963350
      return "bad-time";
Packit Service 963350
    case GST_CLOCK_ERROR:
Packit Service 963350
      return "error";
Packit Service 963350
    case GST_CLOCK_UNSUPPORTED:
Packit Service 963350
      return "unsupported";
Packit Service 963350
    case GST_CLOCK_DONE:
Packit Service 963350
      return "done";
Packit Service 963350
    default:
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return "unknown";
Packit Service 963350
}
Packit Service 963350
#endif /* GST_DISABLE_GST_DEBUG */
Packit Service 963350
Packit Service 963350
static void gst_clock_dispose (GObject * object);
Packit Service 963350
static void gst_clock_finalize (GObject * object);
Packit Service 963350
Packit Service 963350
static void gst_clock_set_property (GObject * object, guint prop_id,
Packit Service 963350
    const GValue * value, GParamSpec * pspec);
Packit Service 963350
static void gst_clock_get_property (GObject * object, guint prop_id,
Packit Service 963350
    GValue * value, GParamSpec * pspec);
Packit Service 963350
Packit Service 963350
static guint gst_clock_signals[SIGNAL_LAST] = { 0 };
Packit Service 963350
Packit Service 963350
static GstClockID
Packit Service 963350
gst_clock_entry_new (GstClock * clock, GstClockTime time,
Packit Service 963350
    GstClockTime interval, GstClockEntryType type)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry;
Packit Service 963350
Packit Service 963350
  entry = g_slice_new (GstClockEntry);
Packit Service 963350
Packit Service 963350
  /* FIXME: add tracer hook for struct allocations such as clock entries */
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "created entry %p, time %" GST_TIME_FORMAT, entry, GST_TIME_ARGS (time));
Packit Service 963350
Packit Service 963350
  entry->refcount = 1;
Packit Service 963350
  entry->clock = clock;
Packit Service 963350
  entry->type = type;
Packit Service 963350
  entry->time = time;
Packit Service 963350
  entry->interval = interval;
Packit Service 963350
  entry->status = GST_CLOCK_OK;
Packit Service 963350
  entry->func = NULL;
Packit Service 963350
  entry->user_data = NULL;
Packit Service 963350
  entry->destroy_data = NULL;
Packit Service 963350
  entry->unscheduled = FALSE;
Packit Service 963350
  entry->woken_up = FALSE;
Packit Service 963350
Packit Service 963350
  return (GstClockID) entry;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* WARNING : Does not modify the refcount
Packit Service 963350
 * WARNING : Do not use if a pending clock operation is happening on that entry */
Packit Service 963350
static gboolean
Packit Service 963350
gst_clock_entry_reinit (GstClock * clock, GstClockEntry * entry,
Packit Service 963350
    GstClockTime time, GstClockTime interval, GstClockEntryType type)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (entry->status != GST_CLOCK_BUSY, FALSE);
Packit Service 963350
  g_return_val_if_fail (entry->clock == clock, FALSE);
Packit Service 963350
Packit Service 963350
  entry->type = type;
Packit Service 963350
  entry->time = time;
Packit Service 963350
  entry->interval = interval;
Packit Service 963350
  entry->status = GST_CLOCK_OK;
Packit Service 963350
  entry->unscheduled = FALSE;
Packit Service 963350
  entry->woken_up = FALSE;
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_single_shot_id_reinit:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @id: a #GstClockID
Packit Service 963350
 * @time: The requested time.
Packit Service 963350
 *
Packit Service 963350
 * Reinitializes the provided single shot @id to the provided time. Does not
Packit Service 963350
 * modify the reference count.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the GstClockID could be reinitialized to the provided
Packit Service 963350
 * @time, else %FALSE.
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_single_shot_id_reinit (GstClock * clock, GstClockID id,
Packit Service 963350
    GstClockTime time)
Packit Service 963350
{
Packit Service 963350
  return gst_clock_entry_reinit (clock, (GstClockEntry *) id, time,
Packit Service 963350
      GST_CLOCK_TIME_NONE, GST_CLOCK_ENTRY_SINGLE);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_periodic_id_reinit:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @id: a #GstClockID
Packit Service 963350
 * @start_time: the requested start time
Packit Service 963350
 * @interval: the requested interval
Packit Service 963350
 *
Packit Service 963350
 * Reinitializes the provided periodic @id to the provided start time and
Packit Service 963350
 * interval. Does not modify the reference count.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the GstClockID could be reinitialized to the provided
Packit Service 963350
 * @time, else %FALSE.
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_periodic_id_reinit (GstClock * clock, GstClockID id,
Packit Service 963350
    GstClockTime start_time, GstClockTime interval)
Packit Service 963350
{
Packit Service 963350
  return gst_clock_entry_reinit (clock, (GstClockEntry *) id, start_time,
Packit Service 963350
      interval, GST_CLOCK_ENTRY_PERIODIC);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_ref:
Packit Service 963350
 * @id: The #GstClockID to ref
Packit Service 963350
 *
Packit Service 963350
 * Increase the refcount of given @id.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): The same #GstClockID with increased refcount.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockID
Packit Service 963350
gst_clock_id_ref (GstClockID id)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (id != NULL, NULL);
Packit Service 963350
Packit Service 963350
  g_atomic_int_inc (&((GstClockEntry *) id)->refcount);
Packit Service 963350
Packit Service 963350
  return id;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
_gst_clock_id_free (GstClockID id)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry;
Packit Service 963350
  g_return_if_fail (id != NULL);
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG (GST_CAT_CLOCK, "freed entry %p", id);
Packit Service 963350
  entry = (GstClockEntry *) id;
Packit Service 963350
  if (entry->destroy_data)
Packit Service 963350
    entry->destroy_data (entry->user_data);
Packit Service 963350
Packit Service 963350
  /* FIXME: add tracer hook for struct allocations such as clock entries */
Packit Service 963350
Packit Service 963350
  g_slice_free (GstClockEntry, id);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_unref:
Packit Service 963350
 * @id: (transfer full): The #GstClockID to unref
Packit Service 963350
 *
Packit Service 963350
 * Unref given @id. When the refcount reaches 0 the
Packit Service 963350
 * #GstClockID will be freed.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_id_unref (GstClockID id)
Packit Service 963350
{
Packit Service 963350
  gint zero;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (id != NULL);
Packit Service 963350
Packit Service 963350
  zero = g_atomic_int_dec_and_test (&((GstClockEntry *) id)->refcount);
Packit Service 963350
  /* if we ended up with the refcount at zero, free the id */
Packit Service 963350
  if (zero) {
Packit Service 963350
    _gst_clock_id_free (id);
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_new_single_shot_id:
Packit Service 963350
 * @clock: The #GstClockID to get a single shot notification from
Packit Service 963350
 * @time: the requested time
Packit Service 963350
 *
Packit Service 963350
 * Get a #GstClockID from @clock to trigger a single shot
Packit Service 963350
 * notification at the requested time. The single shot id should be
Packit Service 963350
 * unreffed after usage.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_clock_id_unref
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a #GstClockID that can be used to request the
Packit Service 963350
 *     time notification.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockID
Packit Service 963350
gst_clock_new_single_shot_id (GstClock * clock, GstClockTime time)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), NULL);
Packit Service 963350
Packit Service 963350
  return gst_clock_entry_new (clock,
Packit Service 963350
      time, GST_CLOCK_TIME_NONE, GST_CLOCK_ENTRY_SINGLE);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_new_periodic_id:
Packit Service 963350
 * @clock: The #GstClockID to get a periodic notification id from
Packit Service 963350
 * @start_time: the requested start time
Packit Service 963350
 * @interval: the requested interval
Packit Service 963350
 *
Packit Service 963350
 * Get an ID from @clock to trigger a periodic notification.
Packit Service 963350
 * The periodic notifications will start at time @start_time and
Packit Service 963350
 * will then be fired with the given @interval. @id should be unreffed
Packit Service 963350
 * after usage.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_clock_id_unref
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a #GstClockID that can be used to request the
Packit Service 963350
 *     time notification.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockID
Packit Service 963350
gst_clock_new_periodic_id (GstClock * clock, GstClockTime start_time,
Packit Service 963350
    GstClockTime interval)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), NULL);
Packit Service 963350
  g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (start_time), NULL);
Packit Service 963350
  g_return_val_if_fail (interval != 0, NULL);
Packit Service 963350
  g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (interval), NULL);
Packit Service 963350
Packit Service 963350
  return gst_clock_entry_new (clock,
Packit Service 963350
      start_time, interval, GST_CLOCK_ENTRY_PERIODIC);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_compare_func:
Packit Service 963350
 * @id1: A #GstClockID
Packit Service 963350
 * @id2: A #GstClockID to compare with
Packit Service 963350
 *
Packit Service 963350
 * Compares the two #GstClockID instances. This function can be used
Packit Service 963350
 * as a GCompareFunc when sorting ids.
Packit Service 963350
 *
Packit Service 963350
 * Returns: negative value if a < b; zero if a = b; positive value if a > b
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
gint
Packit Service 963350
gst_clock_id_compare_func (gconstpointer id1, gconstpointer id2)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry1, *entry2;
Packit Service 963350
Packit Service 963350
  entry1 = (GstClockEntry *) id1;
Packit Service 963350
  entry2 = (GstClockEntry *) id2;
Packit Service 963350
Packit Service 963350
  if (GST_CLOCK_ENTRY_TIME (entry1) > GST_CLOCK_ENTRY_TIME (entry2)) {
Packit Service 963350
    return 1;
Packit Service 963350
  }
Packit Service 963350
  if (GST_CLOCK_ENTRY_TIME (entry1) < GST_CLOCK_ENTRY_TIME (entry2)) {
Packit Service 963350
    return -1;
Packit Service 963350
  }
Packit Service 963350
  return 0;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_get_time:
Packit Service 963350
 * @id: The #GstClockID to query
Packit Service 963350
 *
Packit Service 963350
 * Get the time of the clock ID
Packit Service 963350
 *
Packit Service 963350
 * Returns: the time of the given clock id.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_id_get_time (GstClockID id)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (id != NULL, GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  return GST_CLOCK_ENTRY_TIME ((GstClockEntry *) id);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_wait:
Packit Service 963350
 * @id: The #GstClockID to wait on
Packit Service 963350
 * @jitter: (out) (allow-none): a pointer that will contain the jitter,
Packit Service 963350
 *     can be %NULL.
Packit Service 963350
 *
Packit Service 963350
 * Perform a blocking wait on @id.
Packit Service 963350
 * @id should have been created with gst_clock_new_single_shot_id()
Packit Service 963350
 * or gst_clock_new_periodic_id() and should not have been unscheduled
Packit Service 963350
 * with a call to gst_clock_id_unschedule().
Packit Service 963350
 *
Packit Service 963350
 * If the @jitter argument is not %NULL and this function returns #GST_CLOCK_OK
Packit Service 963350
 * or #GST_CLOCK_EARLY, it will contain the difference
Packit Service 963350
 * against the clock and the time of @id when this method was
Packit Service 963350
 * called.
Packit Service 963350
 * Positive values indicate how late @id was relative to the clock
Packit Service 963350
 * (in which case this function will return #GST_CLOCK_EARLY).
Packit Service 963350
 * Negative values indicate how much time was spent waiting on the clock
Packit Service 963350
 * before this function returned.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the result of the blocking wait. #GST_CLOCK_EARLY will be returned
Packit Service 963350
 * if the current clock time is past the time of @id, #GST_CLOCK_OK if
Packit Service 963350
 * @id was scheduled in time. #GST_CLOCK_UNSCHEDULED if @id was
Packit Service 963350
 * unscheduled with gst_clock_id_unschedule().
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockReturn
Packit Service 963350
gst_clock_id_wait (GstClockID id, GstClockTimeDiff * jitter)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry;
Packit Service 963350
  GstClock *clock;
Packit Service 963350
  GstClockReturn res;
Packit Service 963350
  GstClockTime requested;
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (id != NULL, GST_CLOCK_ERROR);
Packit Service 963350
Packit Service 963350
  entry = (GstClockEntry *) id;
Packit Service 963350
  requested = GST_CLOCK_ENTRY_TIME (entry);
Packit Service 963350
Packit Service 963350
  clock = GST_CLOCK_ENTRY_CLOCK (entry);
Packit Service 963350
Packit Service 963350
  /* can't sync on invalid times */
Packit Service 963350
  if (G_UNLIKELY (!GST_CLOCK_TIME_IS_VALID (requested)))
Packit Service 963350
    goto invalid_time;
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "waiting on clock entry %p", id);
Packit Service 963350
Packit Service 963350
  /* if we have a wait_jitter function, use that */
Packit Service 963350
  if (G_UNLIKELY (cclass->wait == NULL))
Packit Service 963350
    goto not_supported;
Packit Service 963350
Packit Service 963350
  res = cclass->wait (clock, entry, jitter);
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "done waiting entry %p, res: %d (%s)", id, res,
Packit Service 963350
      gst_clock_return_get_name (res));
Packit Service 963350
Packit Service 963350
  if (entry->type == GST_CLOCK_ENTRY_PERIODIC)
Packit Service 963350
    entry->time = requested + entry->interval;
Packit Service 963350
Packit Service 963350
  return res;
Packit Service 963350
Packit Service 963350
  /* ERRORS */
Packit Service 963350
invalid_time:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "invalid time requested, returning _BADTIME");
Packit Service 963350
    return GST_CLOCK_BADTIME;
Packit Service 963350
  }
Packit Service 963350
not_supported:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "clock wait is not supported");
Packit Service 963350
    return GST_CLOCK_UNSUPPORTED;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_wait_async:
Packit Service 963350
 * @id: a #GstClockID to wait on
Packit Service 963350
 * @func: The callback function
Packit Service 963350
 * @user_data: User data passed in the callback
Packit Service 963350
 * @destroy_data: #GDestroyNotify for user_data
Packit Service 963350
 *
Packit Service 963350
 * Register a callback on the given #GstClockID @id with the given
Packit Service 963350
 * function and user_data. When passing a #GstClockID with an invalid
Packit Service 963350
 * time to this function, the callback will be called immediately
Packit Service 963350
 * with  a time set to GST_CLOCK_TIME_NONE. The callback will
Packit Service 963350
 * be called when the time of @id has been reached.
Packit Service 963350
 *
Packit Service 963350
 * The callback @func can be invoked from any thread, either provided by the
Packit Service 963350
 * core or from a streaming thread. The application should be prepared for this.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the result of the non blocking wait.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockReturn
Packit Service 963350
gst_clock_id_wait_async (GstClockID id,
Packit Service 963350
    GstClockCallback func, gpointer user_data, GDestroyNotify destroy_data)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry;
Packit Service 963350
  GstClock *clock;
Packit Service 963350
  GstClockReturn res;
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
  GstClockTime requested;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (id != NULL, GST_CLOCK_ERROR);
Packit Service 963350
  g_return_val_if_fail (func != NULL, GST_CLOCK_ERROR);
Packit Service 963350
Packit Service 963350
  entry = (GstClockEntry *) id;
Packit Service 963350
  requested = GST_CLOCK_ENTRY_TIME (entry);
Packit Service 963350
  clock = GST_CLOCK_ENTRY_CLOCK (entry);
Packit Service 963350
Packit Service 963350
  /* can't sync on invalid times */
Packit Service 963350
  if (G_UNLIKELY (!GST_CLOCK_TIME_IS_VALID (requested)))
Packit Service 963350
    goto invalid_time;
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
Packit Service 963350
  if (G_UNLIKELY (cclass->wait_async == NULL))
Packit Service 963350
    goto not_supported;
Packit Service 963350
Packit Service 963350
  entry->func = func;
Packit Service 963350
  entry->user_data = user_data;
Packit Service 963350
  entry->destroy_data = destroy_data;
Packit Service 963350
Packit Service 963350
  res = cclass->wait_async (clock, entry);
Packit Service 963350
Packit Service 963350
  return res;
Packit Service 963350
Packit Service 963350
  /* ERRORS */
Packit Service 963350
invalid_time:
Packit Service 963350
  {
Packit Service 963350
    (func) (clock, GST_CLOCK_TIME_NONE, id, user_data);
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "invalid time requested, returning _BADTIME");
Packit Service 963350
    return GST_CLOCK_BADTIME;
Packit Service 963350
  }
Packit Service 963350
not_supported:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "clock wait is not supported");
Packit Service 963350
    return GST_CLOCK_UNSUPPORTED;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_id_unschedule:
Packit Service 963350
 * @id: The id to unschedule
Packit Service 963350
 *
Packit Service 963350
 * Cancel an outstanding request with @id. This can either
Packit Service 963350
 * be an outstanding async notification or a pending sync notification.
Packit Service 963350
 * After this call, @id cannot be used anymore to receive sync or
Packit Service 963350
 * async notifications, you need to create a new #GstClockID.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_id_unschedule (GstClockID id)
Packit Service 963350
{
Packit Service 963350
  GstClockEntry *entry;
Packit Service 963350
  GstClock *clock;
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (id != NULL);
Packit Service 963350
Packit Service 963350
  entry = (GstClockEntry *) id;
Packit Service 963350
  clock = entry->clock;
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
Packit Service 963350
  if (G_LIKELY (cclass->unschedule))
Packit Service 963350
    cclass->unschedule (clock, entry);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
Packit Service 963350
/*
Packit Service 963350
 * GstClock abstract base class implementation
Packit Service 963350
 */
Packit Service 963350
#define gst_clock_parent_class parent_class
Packit Service 963350
G_DEFINE_ABSTRACT_TYPE (GstClock, gst_clock, GST_TYPE_OBJECT);
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_class_init (GstClockClass * klass)
Packit Service 963350
{
Packit Service 963350
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit Service 963350
Packit Service 963350
  gobject_class->dispose = gst_clock_dispose;
Packit Service 963350
  gobject_class->finalize = gst_clock_finalize;
Packit Service 963350
  gobject_class->set_property = gst_clock_set_property;
Packit Service 963350
  gobject_class->get_property = gst_clock_get_property;
Packit Service 963350
Packit Service 963350
  g_object_class_install_property (gobject_class, PROP_WINDOW_SIZE,
Packit Service 963350
      g_param_spec_int ("window-size", "Window size",
Packit Service 963350
          "The size of the window used to calculate rate and offset", 2, 1024,
Packit Service 963350
          DEFAULT_WINDOW_SIZE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit Service 963350
  g_object_class_install_property (gobject_class, PROP_WINDOW_THRESHOLD,
Packit Service 963350
      g_param_spec_int ("window-threshold", "Window threshold",
Packit Service 963350
          "The threshold to start calculating rate and offset", 2, 1024,
Packit Service 963350
          DEFAULT_WINDOW_THRESHOLD,
Packit Service 963350
          G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit Service 963350
  g_object_class_install_property (gobject_class, PROP_TIMEOUT,
Packit Service 963350
      g_param_spec_uint64 ("timeout", "Timeout",
Packit Service 963350
          "The amount of time, in nanoseconds, to sample master and slave clocks",
Packit Service 963350
          0, G_MAXUINT64, DEFAULT_TIMEOUT,
Packit Service 963350
          G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit Service 963350
Packit Service 963350
  /**
Packit Service 963350
   * GstClock::synced:
Packit Service 963350
   * @clock: the clock
Packit Service 963350
   * @synced: if the clock is synced now
Packit Service 963350
   *
Packit Service 963350
   * Signaled on clocks with GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC set once
Packit Service 963350
   * the clock is synchronized, or when it completely lost synchronization.
Packit Service 963350
   * This signal will not be emitted on clocks without the flag.
Packit Service 963350
   *
Packit Service 963350
   * This signal will be emitted from an arbitrary thread, most likely not
Packit Service 963350
   * the application's main thread.
Packit Service 963350
   *
Packit Service 963350
   * Since: 1.6
Packit Service 963350
   */
Packit Service 963350
  gst_clock_signals[SIGNAL_SYNCED] =
Packit Service 963350
      g_signal_new ("synced", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
Packit Service 963350
      0, NULL, NULL,
Packit Service 963350
      g_cclosure_marshal_generic, G_TYPE_NONE, 1, G_TYPE_BOOLEAN);
Packit Service 963350
Packit Service 963350
  g_type_class_add_private (klass, sizeof (GstClockPrivate));
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_init (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  clock->priv = priv =
Packit Service 963350
      G_TYPE_INSTANCE_GET_PRIVATE (clock, GST_TYPE_CLOCK, GstClockPrivate);
Packit Service 963350
Packit Service 963350
  priv->last_time = 0;
Packit Service 963350
Packit Service 963350
  priv->internal_calibration = 0;
Packit Service 963350
  priv->external_calibration = 0;
Packit Service 963350
  priv->rate_numerator = 1;
Packit Service 963350
  priv->rate_denominator = 1;
Packit Service 963350
Packit Service 963350
  g_mutex_init (&priv->slave_lock);
Packit Service 963350
  g_cond_init (&priv->sync_cond);
Packit Service 963350
  priv->window_size = DEFAULT_WINDOW_SIZE;
Packit Service 963350
  priv->window_threshold = DEFAULT_WINDOW_THRESHOLD;
Packit Service 963350
  priv->filling = TRUE;
Packit Service 963350
  priv->time_index = 0;
Packit Service 963350
  priv->timeout = DEFAULT_TIMEOUT;
Packit Service 963350
  priv->times = g_new0 (GstClockTime, 4 * priv->window_size);
Packit Service 963350
  priv->times_temp = priv->times + 2 * priv->window_size;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_dispose (GObject * object)
Packit Service 963350
{
Packit Service 963350
  GstClock *clock = GST_CLOCK (object);
Packit Service 963350
  GstClock **master_p;
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  master_p = &clock->priv->master;
Packit Service 963350
  gst_object_replace ((GstObject **) master_p, NULL);
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  G_OBJECT_CLASS (parent_class)->dispose (object);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_finalize (GObject * object)
Packit Service 963350
{
Packit Service 963350
  GstClock *clock = GST_CLOCK (object);
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
  if (clock->priv->clockid) {
Packit Service 963350
    gst_clock_id_unschedule (clock->priv->clockid);
Packit Service 963350
    gst_clock_id_unref (clock->priv->clockid);
Packit Service 963350
    clock->priv->clockid = NULL;
Packit Service 963350
  }
Packit Service 963350
  g_free (clock->priv->times);
Packit Service 963350
  clock->priv->times = NULL;
Packit Service 963350
  clock->priv->times_temp = NULL;
Packit Service 963350
  GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  g_mutex_clear (&clock->priv->slave_lock);
Packit Service 963350
  g_cond_clear (&clock->priv->sync_cond);
Packit Service 963350
Packit Service 963350
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_set_resolution:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @resolution: The resolution to set
Packit Service 963350
 *
Packit Service 963350
 * Set the accuracy of the clock. Some clocks have the possibility to operate
Packit Service 963350
 * with different accuracy at the expense of more resource usage. There is
Packit Service 963350
 * normally no need to change the default resolution of a clock. The resolution
Packit Service 963350
 * of a clock can only be changed if the clock has the
Packit Service 963350
 * #GST_CLOCK_FLAG_CAN_SET_RESOLUTION flag set.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the new resolution of the clock.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_set_resolution (GstClock * clock, GstClockTime resolution)
Packit Service 963350
{
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), 0);
Packit Service 963350
  g_return_val_if_fail (resolution != 0, 0);
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  if (cclass->change_resolution)
Packit Service 963350
    priv->resolution =
Packit Service 963350
        cclass->change_resolution (clock, priv->resolution, resolution);
Packit Service 963350
Packit Service 963350
  return priv->resolution;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_resolution:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 *
Packit Service 963350
 * Get the accuracy of the clock. The accuracy of the clock is the granularity
Packit Service 963350
 * of the values returned by gst_clock_get_time().
Packit Service 963350
 *
Packit Service 963350
 * Returns: the resolution of the clock in units of #GstClockTime.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_get_resolution (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), 0);
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
Packit Service 963350
  if (cclass->get_resolution)
Packit Service 963350
    return cclass->get_resolution (clock);
Packit Service 963350
Packit Service 963350
  return 1;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* FIXME 2.0: Remove clock parameter below */
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_adjust_with_calibration:
Packit Service 963350
 * @clock: (allow-none): a #GstClock to use
Packit Service 963350
 * @internal_target: a clock time
Packit Service 963350
 * @cinternal: a reference internal time
Packit Service 963350
 * @cexternal: a reference external time
Packit Service 963350
 * @cnum: the numerator of the rate of the clock relative to its
Packit Service 963350
 *        internal time
Packit Service 963350
 * @cdenom: the denominator of the rate of the clock
Packit Service 963350
 *
Packit Service 963350
 * Converts the given @internal_target clock time to the external time,
Packit Service 963350
 * using the passed calibration parameters. This function performs the
Packit Service 963350
 * same calculation as gst_clock_adjust_unlocked() when called using the
Packit Service 963350
 * current calibration parameters, but doesn't ensure a monotonically
Packit Service 963350
 * increasing result as gst_clock_adjust_unlocked() does.
Packit Service 963350
 *
Packit Service 963350
 * Note: The @clock parameter is unused and can be NULL
Packit Service 963350
 *
Packit Service 963350
 * Returns: the converted time of the clock.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.6
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_adjust_with_calibration (GstClock * clock,
Packit Service 963350
    GstClockTime internal_target, GstClockTime cinternal,
Packit Service 963350
    GstClockTime cexternal, GstClockTime cnum, GstClockTime cdenom)
Packit Service 963350
{
Packit Service 963350
  GstClockTime ret;
Packit Service 963350
Packit Service 963350
  /* avoid divide by 0 */
Packit Service 963350
  if (G_UNLIKELY (cdenom == 0))
Packit Service 963350
    cnum = cdenom = 1;
Packit Service 963350
Packit Service 963350
  /* The formula is (internal - cinternal) * cnum / cdenom + cexternal
Packit Service 963350
   *
Packit Service 963350
   * Since we do math on unsigned 64-bit ints we have to special case for
Packit Service 963350
   * internal < cinternal to get the sign right. this case is not very common,
Packit Service 963350
   * though.
Packit Service 963350
   */
Packit Service 963350
  if (G_LIKELY (internal_target >= cinternal)) {
Packit Service 963350
    ret = internal_target - cinternal;
Packit Service 963350
    ret = gst_util_uint64_scale (ret, cnum, cdenom);
Packit Service 963350
    ret += cexternal;
Packit Service 963350
  } else {
Packit Service 963350
    ret = cinternal - internal_target;
Packit Service 963350
    ret = gst_util_uint64_scale (ret, cnum, cdenom);
Packit Service 963350
    /* clamp to 0 */
Packit Service 963350
    if (G_LIKELY (cexternal > ret))
Packit Service 963350
      ret = cexternal - ret;
Packit Service 963350
    else
Packit Service 963350
      ret = 0;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_adjust_unlocked:
Packit Service 963350
 * @clock: a #GstClock to use
Packit Service 963350
 * @internal: a clock time
Packit Service 963350
 *
Packit Service 963350
 * Converts the given @internal clock time to the external time, adjusting for the
Packit Service 963350
 * rate and reference time set with gst_clock_set_calibration() and making sure
Packit Service 963350
 * that the returned time is increasing. This function should be called with the
Packit Service 963350
 * clock's OBJECT_LOCK held and is mainly used by clock subclasses.
Packit Service 963350
 *
Packit Service 963350
 * This function is the reverse of gst_clock_unadjust_unlocked().
Packit Service 963350
 *
Packit Service 963350
 * Returns: the converted time of the clock.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_adjust_unlocked (GstClock * clock, GstClockTime internal)
Packit Service 963350
{
Packit Service 963350
  GstClockTime ret, cinternal, cexternal, cnum, cdenom;
Packit Service 963350
  GstClockPrivate *priv = clock->priv;
Packit Service 963350
Packit Service 963350
  /* get calibration values for readability */
Packit Service 963350
  cinternal = priv->internal_calibration;
Packit Service 963350
  cexternal = priv->external_calibration;
Packit Service 963350
  cnum = priv->rate_numerator;
Packit Service 963350
  cdenom = priv->rate_denominator;
Packit Service 963350
Packit Service 963350
  ret =
Packit Service 963350
      gst_clock_adjust_with_calibration (clock, internal, cinternal, cexternal,
Packit Service 963350
      cnum, cdenom);
Packit Service 963350
Packit Service 963350
  /* make sure the time is increasing */
Packit Service 963350
  priv->last_time = MAX (ret, priv->last_time);
Packit Service 963350
Packit Service 963350
  return priv->last_time;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* FIXME 2.0: Remove clock parameter below */
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_unadjust_with_calibration:
Packit Service 963350
 * @clock: (allow-none): a #GstClock to use
Packit Service 963350
 * @external_target: a clock time
Packit Service 963350
 * @cinternal: a reference internal time
Packit Service 963350
 * @cexternal: a reference external time
Packit Service 963350
 * @cnum: the numerator of the rate of the clock relative to its
Packit Service 963350
 *        internal time
Packit Service 963350
 * @cdenom: the denominator of the rate of the clock
Packit Service 963350
 *
Packit Service 963350
 * Converts the given @external_target clock time to the internal time,
Packit Service 963350
 * using the passed calibration parameters. This function performs the
Packit Service 963350
 * same calculation as gst_clock_unadjust_unlocked() when called using the
Packit Service 963350
 * current calibration parameters.
Packit Service 963350
 *
Packit Service 963350
 * Note: The @clock parameter is unused and can be NULL
Packit Service 963350
 *
Packit Service 963350
 * Returns: the converted time of the clock.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.8
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_unadjust_with_calibration (GstClock * clock,
Packit Service 963350
    GstClockTime external_target, GstClockTime cinternal,
Packit Service 963350
    GstClockTime cexternal, GstClockTime cnum, GstClockTime cdenom)
Packit Service 963350
{
Packit Service 963350
  GstClockTime ret;
Packit Service 963350
Packit Service 963350
  /* avoid divide by 0 */
Packit Service 963350
  if (G_UNLIKELY (cnum == 0))
Packit Service 963350
    cnum = cdenom = 1;
Packit Service 963350
Packit Service 963350
  /* The formula is (external - cexternal) * cdenom / cnum + cinternal */
Packit Service 963350
  if (G_LIKELY (external_target >= cexternal)) {
Packit Service 963350
    ret = external_target - cexternal;
Packit Service 963350
    ret = gst_util_uint64_scale (ret, cdenom, cnum);
Packit Service 963350
    ret += cinternal;
Packit Service 963350
  } else {
Packit Service 963350
    ret = cexternal - external_target;
Packit Service 963350
    ret = gst_util_uint64_scale (ret, cdenom, cnum);
Packit Service 963350
    if (G_LIKELY (cinternal > ret))
Packit Service 963350
      ret = cinternal - ret;
Packit Service 963350
    else
Packit Service 963350
      ret = 0;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_unadjust_unlocked:
Packit Service 963350
 * @clock: a #GstClock to use
Packit Service 963350
 * @external: an external clock time
Packit Service 963350
 *
Packit Service 963350
 * Converts the given @external clock time to the internal time of @clock,
Packit Service 963350
 * using the rate and reference time set with gst_clock_set_calibration().
Packit Service 963350
 * This function should be called with the clock's OBJECT_LOCK held and
Packit Service 963350
 * is mainly used by clock subclasses.
Packit Service 963350
 *
Packit Service 963350
 * This function is the reverse of gst_clock_adjust_unlocked().
Packit Service 963350
 *
Packit Service 963350
 * Returns: the internal time of the clock corresponding to @external.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_unadjust_unlocked (GstClock * clock, GstClockTime external)
Packit Service 963350
{
Packit Service 963350
  GstClockTime cinternal, cexternal, cnum, cdenom;
Packit Service 963350
  GstClockPrivate *priv = clock->priv;
Packit Service 963350
Packit Service 963350
  /* get calibration values for readability */
Packit Service 963350
  cinternal = priv->internal_calibration;
Packit Service 963350
  cexternal = priv->external_calibration;
Packit Service 963350
  cnum = priv->rate_numerator;
Packit Service 963350
  cdenom = priv->rate_denominator;
Packit Service 963350
Packit Service 963350
  return gst_clock_unadjust_with_calibration (clock, external, cinternal,
Packit Service 963350
      cexternal, cnum, cdenom);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_internal_time:
Packit Service 963350
 * @clock: a #GstClock to query
Packit Service 963350
 *
Packit Service 963350
 * Gets the current internal time of the given clock. The time is returned
Packit Service 963350
 * unadjusted for the offset and the rate.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the internal time of the clock. Or GST_CLOCK_TIME_NONE when
Packit Service 963350
 * given invalid input.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_get_internal_time (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockTime ret;
Packit Service 963350
  GstClockClass *cclass;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  if (G_UNLIKELY (GST_OBJECT_FLAG_IS_SET (clock,
Packit Service 963350
              GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC) && !clock->priv->synced))
Packit Service 963350
    GST_CAT_WARNING_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "clock is not synchronized yet");
Packit Service 963350
Packit Service 963350
  cclass = GST_CLOCK_GET_CLASS (clock);
Packit Service 963350
Packit Service 963350
  if (G_UNLIKELY (cclass->get_internal_time == NULL))
Packit Service 963350
    goto not_supported;
Packit Service 963350
Packit Service 963350
  ret = cclass->get_internal_time (clock);
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "internal time %" GST_TIME_FORMAT,
Packit Service 963350
      GST_TIME_ARGS (ret));
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
Packit Service 963350
  /* ERRORS */
Packit Service 963350
not_supported:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "internal time not supported, return 0");
Packit Service 963350
    return G_GINT64_CONSTANT (0);
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_time:
Packit Service 963350
 * @clock: a #GstClock to query
Packit Service 963350
 *
Packit Service 963350
 * Gets the current time of the given clock. The time is always
Packit Service 963350
 * monotonically increasing and adjusted according to the current
Packit Service 963350
 * offset and rate.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the time of the clock. Or GST_CLOCK_TIME_NONE when
Packit Service 963350
 * given invalid input.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_get_time (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockTime ret;
Packit Service 963350
  gint seq;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  do {
Packit Service 963350
    /* reget the internal time when we retry to get the most current
Packit Service 963350
     * timevalue */
Packit Service 963350
    ret = gst_clock_get_internal_time (clock);
Packit Service 963350
Packit Service 963350
    seq = read_seqbegin (clock);
Packit Service 963350
    /* this will scale for rate and offset */
Packit Service 963350
    ret = gst_clock_adjust_unlocked (clock, ret);
Packit Service 963350
  } while (read_seqretry (clock, seq));
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "adjusted time %" GST_TIME_FORMAT,
Packit Service 963350
      GST_TIME_ARGS (ret));
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_set_calibration:
Packit Service 963350
 * @clock: a #GstClock to calibrate
Packit Service 963350
 * @internal: a reference internal time
Packit Service 963350
 * @external: a reference external time
Packit Service 963350
 * @rate_num: the numerator of the rate of the clock relative to its
Packit Service 963350
 *            internal time
Packit Service 963350
 * @rate_denom: the denominator of the rate of the clock
Packit Service 963350
 *
Packit Service 963350
 * Adjusts the rate and time of @clock. A rate of 1/1 is the normal speed of
Packit Service 963350
 * the clock. Values bigger than 1/1 make the clock go faster.
Packit Service 963350
 *
Packit Service 963350
 * @internal and @external are calibration parameters that arrange that
Packit Service 963350
 * gst_clock_get_time() should have been @external at internal time @internal.
Packit Service 963350
 * This internal time should not be in the future; that is, it should be less
Packit Service 963350
 * than the value of gst_clock_get_internal_time() when this function is called.
Packit Service 963350
 *
Packit Service 963350
 * Subsequent calls to gst_clock_get_time() will return clock times computed as
Packit Service 963350
 * follows:
Packit Service 963350
 *
Packit Service 963350
 * |[
Packit Service 963350
 *   time = (internal_time - internal) * rate_num / rate_denom + external
Packit Service 963350
 * ]|
Packit Service 963350
 *
Packit Service 963350
 * This formula is implemented in gst_clock_adjust_unlocked(). Of course, it
Packit Service 963350
 * tries to do the integer arithmetic as precisely as possible.
Packit Service 963350
 *
Packit Service 963350
 * Note that gst_clock_get_time() always returns increasing values so when you
Packit Service 963350
 * move the clock backwards, gst_clock_get_time() will report the previous value
Packit Service 963350
 * until the clock catches up.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_set_calibration (GstClock * clock, GstClockTime internal, GstClockTime
Packit Service 963350
    external, GstClockTime rate_num, GstClockTime rate_denom)
Packit Service 963350
{
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (GST_IS_CLOCK (clock));
Packit Service 963350
  g_return_if_fail (rate_num != GST_CLOCK_TIME_NONE);
Packit Service 963350
  g_return_if_fail (rate_denom > 0 && rate_denom != GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  write_seqlock (clock);
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "internal %" GST_TIME_FORMAT " external %" GST_TIME_FORMAT " %"
Packit Service 963350
      G_GUINT64_FORMAT "/%" G_GUINT64_FORMAT " = %f", GST_TIME_ARGS (internal),
Packit Service 963350
      GST_TIME_ARGS (external), rate_num, rate_denom,
Packit Service 963350
      gst_guint64_to_gdouble (rate_num) / gst_guint64_to_gdouble (rate_denom));
Packit Service 963350
Packit Service 963350
  priv->internal_calibration = internal;
Packit Service 963350
  priv->external_calibration = external;
Packit Service 963350
  priv->rate_numerator = rate_num;
Packit Service 963350
  priv->rate_denominator = rate_denom;
Packit Service 963350
  write_sequnlock (clock);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_calibration:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @internal: (out) (allow-none): a location to store the internal time
Packit Service 963350
 * @external: (out) (allow-none): a location to store the external time
Packit Service 963350
 * @rate_num: (out) (allow-none): a location to store the rate numerator
Packit Service 963350
 * @rate_denom: (out) (allow-none): a location to store the rate denominator
Packit Service 963350
 *
Packit Service 963350
 * Gets the internal rate and reference time of @clock. See
Packit Service 963350
 * gst_clock_set_calibration() for more information.
Packit Service 963350
 *
Packit Service 963350
 * @internal, @external, @rate_num, and @rate_denom can be left %NULL if the
Packit Service 963350
 * caller is not interested in the values.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_get_calibration (GstClock * clock, GstClockTime * internal,
Packit Service 963350
    GstClockTime * external, GstClockTime * rate_num, GstClockTime * rate_denom)
Packit Service 963350
{
Packit Service 963350
  gint seq;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (GST_IS_CLOCK (clock));
Packit Service 963350
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  do {
Packit Service 963350
    seq = read_seqbegin (clock);
Packit Service 963350
    if (rate_num)
Packit Service 963350
      *rate_num = priv->rate_numerator;
Packit Service 963350
    if (rate_denom)
Packit Service 963350
      *rate_denom = priv->rate_denominator;
Packit Service 963350
    if (external)
Packit Service 963350
      *external = priv->external_calibration;
Packit Service 963350
    if (internal)
Packit Service 963350
      *internal = priv->internal_calibration;
Packit Service 963350
  } while (read_seqretry (clock, seq));
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* will be called repeatedly to sample the master and slave clock
Packit Service 963350
 * to recalibrate the clock  */
Packit Service 963350
static gboolean
Packit Service 963350
gst_clock_slave_callback (GstClock * master, GstClockTime time,
Packit Service 963350
    GstClockID id, GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockTime stime, mtime;
Packit Service 963350
  gdouble r_squared;
Packit Service 963350
Packit Service 963350
  if (!gst_clock_is_synced (clock)) {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "Slave clock is not synced yet");
Packit Service 963350
    return TRUE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  stime = gst_clock_get_internal_time (clock);
Packit Service 963350
  mtime = gst_clock_get_time (master);
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "master %" GST_TIME_FORMAT ", slave %" GST_TIME_FORMAT,
Packit Service 963350
      GST_TIME_ARGS (mtime), GST_TIME_ARGS (stime));
Packit Service 963350
Packit Service 963350
  gst_clock_add_observation (clock, stime, mtime, &r_squared);
Packit Service 963350
Packit Service 963350
  /* FIXME, we can use the r_squared value to adjust the timeout
Packit Service 963350
   * value of the clockid */
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_set_master:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @master: (allow-none): a master #GstClock
Packit Service 963350
 *
Packit Service 963350
 * Set @master as the master clock for @clock. @clock will be automatically
Packit Service 963350
 * calibrated so that gst_clock_get_time() reports the same time as the
Packit Service 963350
 * master clock.
Packit Service 963350
 *
Packit Service 963350
 * A clock provider that slaves its clock to a master can get the current
Packit Service 963350
 * calibration values with gst_clock_get_calibration().
Packit Service 963350
 *
Packit Service 963350
 * @master can be %NULL in which case @clock will not be slaved anymore. It will
Packit Service 963350
 * however keep reporting its time adjusted with the last configured rate
Packit Service 963350
 * and time offsets.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the clock is capable of being slaved to a master clock.
Packit Service 963350
 * Trying to set a master on a clock without the
Packit Service 963350
 * #GST_CLOCK_FLAG_CAN_SET_MASTER flag will make this function return %FALSE.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_set_master (GstClock * clock, GstClock * master)
Packit Service 963350
{
Packit Service 963350
  GstClock **master_p;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), FALSE);
Packit Service 963350
  g_return_val_if_fail (master != clock, FALSE);
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  /* we always allow setting the master to NULL */
Packit Service 963350
  if (master && !GST_OBJECT_FLAG_IS_SET (clock, GST_CLOCK_FLAG_CAN_SET_MASTER))
Packit Service 963350
    goto not_supported;
Packit Service 963350
  if (master && !gst_clock_is_synced (master))
Packit Service 963350
    goto master_not_synced;
Packit Service 963350
Packit Service 963350
  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "slaving %p to master clock %p", clock, master);
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
  if (priv->clockid) {
Packit Service 963350
    gst_clock_id_unschedule (priv->clockid);
Packit Service 963350
    gst_clock_id_unref (priv->clockid);
Packit Service 963350
    priv->clockid = NULL;
Packit Service 963350
  }
Packit Service 963350
  if (master) {
Packit Service 963350
    priv->filling = TRUE;
Packit Service 963350
    priv->time_index = 0;
Packit Service 963350
    /* use the master periodic id to schedule sampling and
Packit Service 963350
     * clock calibration. */
Packit Service 963350
    priv->clockid = gst_clock_new_periodic_id (master,
Packit Service 963350
        gst_clock_get_time (master), priv->timeout);
Packit Service 963350
    gst_clock_id_wait_async (priv->clockid,
Packit Service 963350
        (GstClockCallback) gst_clock_slave_callback,
Packit Service 963350
        gst_object_ref (clock), (GDestroyNotify) gst_object_unref);
Packit Service 963350
  }
Packit Service 963350
  GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  master_p = &priv->master;
Packit Service 963350
  gst_object_replace ((GstObject **) master_p, (GstObject *) master);
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
Packit Service 963350
  /* ERRORS */
Packit Service 963350
not_supported:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
        "cannot be slaved to a master clock");
Packit Service 963350
    GST_OBJECT_UNLOCK (clock);
Packit Service 963350
    return FALSE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
master_not_synced:
Packit Service 963350
  {
Packit Service 963350
    GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, master,
Packit Service 963350
        "master clock is not synced yet");
Packit Service 963350
    GST_OBJECT_UNLOCK (clock);
Packit Service 963350
    return FALSE;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_master:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 *
Packit Service 963350
 * Get the master clock that @clock is slaved to or %NULL when the clock is
Packit Service 963350
 * not slaved to any master clock.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full) (nullable): a master #GstClock or %NULL
Packit Service 963350
 *     when this clock is not slaved to a master clock. Unref after
Packit Service 963350
 *     usage.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
GstClock *
Packit Service 963350
gst_clock_get_master (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClock *result = NULL;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), NULL);
Packit Service 963350
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  if (priv->master)
Packit Service 963350
    result = gst_object_ref (priv->master);
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  return result;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_add_observation:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @slave: a time on the slave
Packit Service 963350
 * @master: a time on the master
Packit Service 963350
 * @r_squared: (out): a pointer to hold the result
Packit Service 963350
 *
Packit Service 963350
 * The time @master of the master clock and the time @slave of the slave
Packit Service 963350
 * clock are added to the list of observations. If enough observations
Packit Service 963350
 * are available, a linear regression algorithm is run on the
Packit Service 963350
 * observations and @clock is recalibrated.
Packit Service 963350
 *
Packit Service 963350
 * If this functions returns %TRUE, @r_squared will contain the
Packit Service 963350
 * correlation coefficient of the interpolation. A value of 1.0
Packit Service 963350
 * means a perfect regression was performed. This value can
Packit Service 963350
 * be used to control the sampling frequency of the master and slave
Packit Service 963350
 * clocks.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if enough observations were added to run the
Packit Service 963350
 * regression algorithm.
Packit Service 963350
 *
Packit Service 963350
 * MT safe.
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_add_observation (GstClock * clock, GstClockTime slave,
Packit Service 963350
    GstClockTime master, gdouble * r_squared)
Packit Service 963350
{
Packit Service 963350
  GstClockTime m_num, m_denom, b, xbase;
Packit Service 963350
Packit Service 963350
  if (!gst_clock_add_observation_unapplied (clock, slave, master, r_squared,
Packit Service 963350
          &xbase, &b, &m_num, &m_denom))
Packit Service 963350
    return FALSE;
Packit Service 963350
Packit Service 963350
  /* if we have a valid regression, adjust the clock */
Packit Service 963350
  gst_clock_set_calibration (clock, xbase, b, m_num, m_denom);
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_add_observation_unapplied:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @slave: a time on the slave
Packit Service 963350
 * @master: a time on the master
Packit Service 963350
 * @r_squared: (out): a pointer to hold the result
Packit Service 963350
 * @internal: (out) (allow-none): a location to store the internal time
Packit Service 963350
 * @external: (out) (allow-none): a location to store the external time
Packit Service 963350
 * @rate_num: (out) (allow-none): a location to store the rate numerator
Packit Service 963350
 * @rate_denom: (out) (allow-none): a location to store the rate denominator
Packit Service 963350
 *
Packit Service 963350
 * Add a clock observation to the internal slaving algorithm the same as
Packit Service 963350
 * gst_clock_add_observation(), and return the result of the master clock
Packit Service 963350
 * estimation, without updating the internal calibration.
Packit Service 963350
 *
Packit Service 963350
 * The caller can then take the results and call gst_clock_set_calibration()
Packit Service 963350
 * with the values, or some modified version of them.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.6
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_add_observation_unapplied (GstClock * clock, GstClockTime slave,
Packit Service 963350
    GstClockTime master, gdouble * r_squared,
Packit Service 963350
    GstClockTime * internal, GstClockTime * external,
Packit Service 963350
    GstClockTime * rate_num, GstClockTime * rate_denom)
Packit Service 963350
{
Packit Service 963350
  GstClockTime m_num, m_denom, b, xbase;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
  guint n;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), FALSE);
Packit Service 963350
  g_return_val_if_fail (r_squared != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
Packit Service 963350
  GST_CAT_LOG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "adding observation slave %" GST_TIME_FORMAT ", master %" GST_TIME_FORMAT,
Packit Service 963350
      GST_TIME_ARGS (slave), GST_TIME_ARGS (master));
Packit Service 963350
Packit Service 963350
  priv->times[(2 * priv->time_index)] = slave;
Packit Service 963350
  priv->times[(2 * priv->time_index) + 1] = master;
Packit Service 963350
Packit Service 963350
  priv->time_index++;
Packit Service 963350
  if (G_UNLIKELY (priv->time_index == priv->window_size)) {
Packit Service 963350
    priv->filling = FALSE;
Packit Service 963350
    priv->time_index = 0;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  if (G_UNLIKELY (priv->filling && priv->time_index < priv->window_threshold))
Packit Service 963350
    goto filling;
Packit Service 963350
Packit Service 963350
  n = priv->filling ? priv->time_index : priv->window_size;
Packit Service 963350
  if (!gst_calculate_linear_regression (priv->times, priv->times_temp, n,
Packit Service 963350
          &m_num, &m_denom, &b, &xbase, r_squared))
Packit Service 963350
    goto invalid;
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  GST_CAT_LOG_OBJECT (GST_CAT_CLOCK, clock,
Packit Service 963350
      "adjusting clock to m=%" G_GUINT64_FORMAT "/%" G_GUINT64_FORMAT ", b=%"
Packit Service 963350
      G_GUINT64_FORMAT " (rsquared=%g)", m_num, m_denom, b, *r_squared);
Packit Service 963350
Packit Service 963350
  if (internal)
Packit Service 963350
    *internal = xbase;
Packit Service 963350
  if (external)
Packit Service 963350
    *external = b;
Packit Service 963350
  if (rate_num)
Packit Service 963350
    *rate_num = m_num;
Packit Service 963350
  if (rate_denom)
Packit Service 963350
    *rate_denom = m_denom;
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
Packit Service 963350
filling:
Packit Service 963350
  {
Packit Service 963350
    GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
    return FALSE;
Packit Service 963350
  }
Packit Service 963350
invalid:
Packit Service 963350
  {
Packit Service 963350
    /* no valid regression has been done, ignore the result then */
Packit Service 963350
    GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
    return FALSE;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_set_timeout:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 * @timeout: a timeout
Packit Service 963350
 *
Packit Service 963350
 * Set the amount of time, in nanoseconds, to sample master and slave
Packit Service 963350
 * clocks
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_set_timeout (GstClock * clock, GstClockTime timeout)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (GST_IS_CLOCK (clock));
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
  clock->priv->timeout = timeout;
Packit Service 963350
  GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_get_timeout:
Packit Service 963350
 * @clock: a #GstClock
Packit Service 963350
 *
Packit Service 963350
 * Get the amount of time that master and slave clocks are sampled.
Packit Service 963350
 *
Packit Service 963350
 * Returns: the interval between samples.
Packit Service 963350
 */
Packit Service 963350
GstClockTime
Packit Service 963350
gst_clock_get_timeout (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  GstClockTime result;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
  result = clock->priv->timeout;
Packit Service 963350
  GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  return result;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_set_property (GObject * object, guint prop_id,
Packit Service 963350
    const GValue * value, GParamSpec * pspec)
Packit Service 963350
{
Packit Service 963350
  GstClock *clock;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  clock = GST_CLOCK (object);
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  switch (prop_id) {
Packit Service 963350
    case PROP_WINDOW_SIZE:
Packit Service 963350
      GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
      priv->window_size = g_value_get_int (value);
Packit Service 963350
      priv->window_threshold = MIN (priv->window_threshold, priv->window_size);
Packit Service 963350
      priv->times = g_renew (GstClockTime, priv->times, 4 * priv->window_size);
Packit Service 963350
      priv->times_temp = priv->times + 2 * priv->window_size;
Packit Service 963350
      /* restart calibration */
Packit Service 963350
      priv->filling = TRUE;
Packit Service 963350
      priv->time_index = 0;
Packit Service 963350
      GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
      break;
Packit Service 963350
    case PROP_WINDOW_THRESHOLD:
Packit Service 963350
      GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
      priv->window_threshold = MIN (g_value_get_int (value), priv->window_size);
Packit Service 963350
      GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
      break;
Packit Service 963350
    case PROP_TIMEOUT:
Packit Service 963350
      gst_clock_set_timeout (clock, g_value_get_uint64 (value));
Packit Service 963350
      break;
Packit Service 963350
    default:
Packit Service 963350
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_clock_get_property (GObject * object, guint prop_id,
Packit Service 963350
    GValue * value, GParamSpec * pspec)
Packit Service 963350
{
Packit Service 963350
  GstClock *clock;
Packit Service 963350
  GstClockPrivate *priv;
Packit Service 963350
Packit Service 963350
  clock = GST_CLOCK (object);
Packit Service 963350
  priv = clock->priv;
Packit Service 963350
Packit Service 963350
  switch (prop_id) {
Packit Service 963350
    case PROP_WINDOW_SIZE:
Packit Service 963350
      GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
      g_value_set_int (value, priv->window_size);
Packit Service 963350
      GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
      break;
Packit Service 963350
    case PROP_WINDOW_THRESHOLD:
Packit Service 963350
      GST_CLOCK_SLAVE_LOCK (clock);
Packit Service 963350
      g_value_set_int (value, priv->window_threshold);
Packit Service 963350
      GST_CLOCK_SLAVE_UNLOCK (clock);
Packit Service 963350
      break;
Packit Service 963350
    case PROP_TIMEOUT:
Packit Service 963350
      g_value_set_uint64 (value, gst_clock_get_timeout (clock));
Packit Service 963350
      break;
Packit Service 963350
    default:
Packit Service 963350
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_wait_for_sync:
Packit Service 963350
 * @clock: a GstClock
Packit Service 963350
 * @timeout: timeout for waiting or %GST_CLOCK_TIME_NONE
Packit Service 963350
 *
Packit Service 963350
 * Waits until @clock is synced for reporting the current time. If @timeout
Packit Service 963350
 * is %GST_CLOCK_TIME_NONE it will wait forever, otherwise it will time out
Packit Service 963350
 * after @timeout nanoseconds.
Packit Service 963350
 *
Packit Service 963350
 * For asynchronous waiting, the GstClock::synced signal can be used.
Packit Service 963350
 *
Packit Service 963350
 * This returns immediately with TRUE if GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC
Packit Service 963350
 * is not set on the clock, or if the clock is already synced.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if waiting was successful, or %FALSE on timeout
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.6
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_wait_for_sync (GstClock * clock, GstClockTime timeout)
Packit Service 963350
{
Packit Service 963350
  gboolean timed_out = FALSE;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), FALSE);
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  if (!GST_OBJECT_FLAG_IS_SET (clock, GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC)
Packit Service 963350
      || clock->priv->synced) {
Packit Service 963350
    GST_OBJECT_UNLOCK (clock);
Packit Service 963350
    return TRUE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  if (timeout != GST_CLOCK_TIME_NONE) {
Packit Service 963350
    gint64 end_time = g_get_monotonic_time () + gst_util_uint64_scale (timeout,
Packit Service 963350
        G_TIME_SPAN_SECOND, GST_SECOND);
Packit Service 963350
Packit Service 963350
    while (!clock->priv->synced && !timed_out) {
Packit Service 963350
      timed_out =
Packit Service 963350
          !g_cond_wait_until (&clock->priv->sync_cond,
Packit Service 963350
          GST_OBJECT_GET_LOCK (clock), end_time);
Packit Service 963350
    }
Packit Service 963350
  } else {
Packit Service 963350
    timed_out = FALSE;
Packit Service 963350
    while (!clock->priv->synced) {
Packit Service 963350
      g_cond_wait (&clock->priv->sync_cond, GST_OBJECT_GET_LOCK (clock));
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
  GST_OBJECT_UNLOCK (clock);
Packit Service 963350
Packit Service 963350
  return !timed_out;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_is_synced:
Packit Service 963350
 * @clock: a GstClock
Packit Service 963350
 *
Packit Service 963350
 * Checks if the clock is currently synced.
Packit Service 963350
 *
Packit Service 963350
 * This returns if GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC is not set on the clock.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the clock is currently synced
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.6
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_clock_is_synced (GstClock * clock)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CLOCK (clock), TRUE);
Packit Service 963350
Packit Service 963350
  return !GST_OBJECT_FLAG_IS_SET (clock, GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC)
Packit Service 963350
      || clock->priv->synced;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_clock_set_synced:
Packit Service 963350
 * @clock: a GstClock
Packit Service 963350
 * @synced: if the clock is synced
Packit Service 963350
 *
Packit Service 963350
 * Sets @clock to synced and emits the GstClock::synced signal, and wakes up any
Packit Service 963350
 * thread waiting in gst_clock_wait_for_sync().
Packit Service 963350
 *
Packit Service 963350
 * This function must only be called if GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC
Packit Service 963350
 * is set on the clock, and is intended to be called by subclasses only.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.6
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_clock_set_synced (GstClock * clock, gboolean synced)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (GST_IS_CLOCK (clock));
Packit Service 963350
  g_return_if_fail (GST_OBJECT_FLAG_IS_SET (clock,
Packit Service 963350
          GST_CLOCK_FLAG_NEEDS_STARTUP_SYNC));
Packit Service 963350
Packit Service 963350
  GST_OBJECT_LOCK (clock);
Packit Service 963350
  if (clock->priv->synced != ! !synced) {
Packit Service 963350
    clock->priv->synced = ! !synced;
Packit Service 963350
    g_cond_signal (&clock->priv->sync_cond);
Packit Service 963350
    GST_OBJECT_UNLOCK (clock);
Packit Service 963350
    g_signal_emit (clock, gst_clock_signals[SIGNAL_SYNCED], 0, ! !synced);
Packit Service 963350
  } else {
Packit Service 963350
    GST_OBJECT_UNLOCK (clock);
Packit Service 963350
  }
Packit Service 963350
}