Bootstrap

GStreamer basesink中的时钟同步

播放复杂媒体时,每个audio和video sample必须在特定时间按特定顺序播放。为此,GStreamer提供了一种同步机制,通过使用 GstClock object、buffer timestamps和SEGMENT event来实现:
(1)GstClock:GstPipeline中的所有element的全局clock,pipeline由PAUSED state转换到PLAYING state时候通过gst_element_provide_clock() 获取到一个clock,并下发给pipeline中所以element。
(2)SEGMENT event:element向下游元素发送buffer之前,必须先发送SEGMENT事件。SEGMENT event主要包括了segment的 rate, start/stop等, 其中SEGMENT 的start/stop规定了buffer的有效时间戳范围。
(3)buffer timestamps:只有timestamps处于SEGMENT 的start 和stop 之间的buffers 是可以被显示的,否则将被丢弃或者裁剪。

1 Clock running-time

一台计算机中,有多种源可以用作时间源,例如系统时间、声卡、CPU计数器等。因此,GStreamer有许多可用的GstClock实现。时钟时间不必从0或其他特定值开始,一些时钟从特定的开始日期开始计数,另一些则从上次重新启动开始计数。
在pipeline由PAUSED向PLAYING状态切换时,gst_pipeline_change_state函数会调用gst_element_provide_clock产生一个clock,并将这个clock发布给所有element,这也意味着pipeline中所有element拥有相同的clock。
在这里插入图片描述
pipeline选择clock后,clock保持在running_time,running_time表示在PLAYING state下花费的总时间,不同状态下running_time的计算如下:
(1)如果pipeline为NULL/READY,则running_time未定义。
(2)在PAUSED中,running_time为上次PAUSED时的时间。当stream第一次进入PAUSED state时,running_time为0。
(3)在PLAYING中,running_time是absolute_time和base time之间的差值。base time定义为absolute_time减去pipeline设置为PLAYING时的运行时间。
(4)seek后,running_time设置为0。
gstreamer中使用gst_clock_get_time ()来返回一个absolute-time,每次调用该函数获得的absolute-time都是一个新值,且是单调递增的。
base-time是pipeline由PAUSED向PLAYING那一瞬间的系统时间,pipeline由PAUSED向PLAYING状态切换时会调用gst_clock_get_time ()获取到当前的系统时间,并将之作为base-time,设置给所有的element, 因此同一个pipeline中所有的element具有相同的base-time。在PLAYING state时,running-time是指base-time与当前absolute-time 之间的差值。Pipeline 由PAUSED向PLAYING状态切换时获取当前系统时间的code如下:
在这里插入图片描述
获取到的当前系统时间now调整后作为base-time 设置给所有element。
在这里插入图片描述
由于pipeline中的所有element都有相同的clock和base-time,因此它们都可以根据pipeline时钟计算运行时间。
在PLAYING state时,running_time的计算方法如下:

    C.running_time = absolute_time - base_time

C.running_time是通过clock获得的running_time,该值以时钟的速率单调增加。

2 Buffer running-time

上面介绍了clock的running-time计算方法,实际上每个buffer也有一个running-time,同步前需要计算buffer的running-time。要计算buffer running-time,需要使用buffer 的timestamp和buffer前面的SEGMENT事件。首先,需要将SEGMENT事件转换为GstSegment对象,然后使用gst_SEGMENT_to_running_time()函数来计算buffer的running-time。
buffer的running-time计算前先使用以下符号定义:
 B: GstBuffer
 B.timestamp = buffer timestamp (GST_BUFFER_PTS or GST_BUFFER_DTS)
 S: SEGMENT event preceding the buffers.。
 S.start:SEGMENT事件中的开始字段,这是允许的最低时间戳。
 S.stop:SEGMENT事件中的停止字段,这是允许的上限时间戳。
 S.rate:SEGMENT事件的速率字段,定义播放速率。
 S.base:时间的基准时间,先前所有segment的总running_time。
 S.offset: S.start或S.stop的偏移量。
有效buffer是B.timestamp在S.start和S.stop之间buffer,应将该范围之外的所有其他buffer会被丢弃或剪裁。GstBuffer timestamps和前面的SEGMENT事件定义了buffer timestamps到running_time的转换如下:

    if (S.rate > 0.0)
      B.running_time = (B.timestamp - (S.start + S.offset)) / ABS (S.rate) + S.base
      =>
      B.timestamp = (B.running_time - S.base) * ABS (S.rate) + S.start + S.offset
    else
      B.running_time = ((S.stop - S.offset) - B.timestamp) / ABS (S.rate) + S.base
      =>
      B.timestamp = S.stop - S.offset - ((B.running_time - S.base) * ABS (S.rate))

B.running_time代表根据SEGMENT event和该segment中buffer的timestamp获得的running_time。
正常情况下第一个可显示的buffer的B.running_time值为0,因为B.timestamp=S.start, S.offset = 0以及S.base = 0。
对于S.rate>1.0,timestamp将按比例缩小以提高播放速率。同样,0.0和1.0之间的速率会减慢播放速度。
对于负速率,从S.stop到S.start接收时间戳,使得接收到的第一缓冲区将被转换为B.running_time为0(B.timestamp==S.stop和S.base==0)。

3 Stream time

stream time也称为stream中的position,该值介于0和媒体文件duration之间的值。stream time作用如下:
(1)回报pipeline中的POSITION查询
(2)seek events/queries中使用的位置
(3)用于同步控制器值的位置
stream time计算的时候会使用SEGMENT中的字段:
S.time:SEGMENT事件中的时间字段,是S.start的stream-time。
S.applied_rate:已应用于segment的速率。
stream time是使用buffer times和前面的SEGMENTevent计算的,如下所示:

    stream_time = (B.timestamp - S.start) * ABS (S.applied_rate) + S.time
    => B.timestamp = (stream_time - S.time) / ABS(S.applied_rate) + S.start

对于负速率,B.timestamp将从S.stop倒退到S.start,使stream-time倒退:

    stream_time = (S.stop - B.timestamp) * ABS(S.applied_rate) + S.time
    => B.timestamp = S.stop - (stream_time - S.time) / ABS(S.applied_rate)

4 Gstreamer 同步用到的各个时间的对应关系

上面介绍了clock-time、running-time,stream-time以及basetime的含义,以下是各个timelines的之间的对应关系。下图表示播放100ms样本并在50ms到100ms之间重复该部分时,pipeline中的不同时间。
在这里插入图片描述
从上图中可以看到buffer的running-time总是随着clock time单调递增。当buffer的running-time等于(clock-time) – (base-time)时,就会播放该buffer。stream time表示流中的位置,并在重复时向后跳跃。

5 stream同步

使用clock和element的base_time,可以获得一个clock的running_time:

C.running_time = absolute_time - base_time

使用buffer timestamp和先前的SEGMENT event(假定正播放速率)可以获取一个buffer的running_time:

 B.running_time = (B.timestamp - (S.start + S.offset)) / ABS (S.rate) + S.base

同步的任务是确保当clock达到相同的C.running_time时播放B.running_time的buffer。因此,以下内容必须成立:

 B.running_time = C.running_time

应播放B.running_time buffer时的absolute_time用B.sync_time表示,

 B.sync_time = B.running_time + base_time

Sink中render buffer前需要等时钟达到B.sync_time,对于多个流,这意味着具有相同running_time的buffer将同时显示,这样就达到了同步的目的。

5.1 buffer running time 计算

每一张video数据render前都会走到gst_base_sink_do_sync这个api中,该函数中首先会调用到gst_base_sink_default_get_times这个api,用来计算buffer 的start timestamp和stop timestamp。Start timestamp是buffer本身的timestamp,stop是buffer timestamp + duration。
在这里插入图片描述

Start timestamp和end timestamp计算后会经过gst_segment_clip ,变为clip的 strart position =>cstart,以及clip后的stop postion=> cstop。clip的作用就是保证cstart, cstop在segment之内。
在这里插入图片描述

然后cstart, cstop调用gst_segment_to_running_time 将这buffer 的start 和stop timestamp变为running time start 和running time stop。
在这里插入图片描述

gst_segment_to_running_time中调用了gst_segment_to_running_time_full,该api中先计算了B.timestamp - (S.start + S.offset) 这一部分:
在这里插入图片描述

然后将上面的计算结果除以ABS (S.rate) 再加上S.base:
在这里插入图片描述

与之前的提到的buffer running time的计算方法一致。

        B.running_time = (B.timestamp - (S.start + S.offset)) / ABS (S.rate) + S.base

5.2 buffer stream time 计算

cstart, cstop通过调用gst_segment_to_stream_time api获得buffer的stream time start 和stream time stop,需要注意的是stream time并不参与同步。
在这里插入图片描述

gst_segment_to_stream_time先获取segment start 和segment time:

  start = segment->start;
  time = segment->time;

然后根据stream time的计算公式获得stream time。

    stream_time = (B.timestamp - S.start) * ABS (S.applied_rate) + S.time
  /* add or subtract from segment time based on applied rate */
  if (G_LIKELY (segment->applied_rate > 0.0)) {
    if (G_LIKELY (position > start)) {
      /* bring to uncorrected position in segment */
      *stream_time = position - start;
      /* correct for applied rate if needed */
      if (G_UNLIKELY (abs_applied_rate != 1.0))
        *stream_time *= abs_applied_rate;
      /* correct for segment time */
      *stream_time += time;
      res = 1;
    } else {
      *stream_time = start - position;
      if (G_UNLIKELY (abs_applied_rate != 1.0))
        *stream_time *= abs_applied_rate;
      if (*stream_time > time) {
        *stream_time -= time;
        res = -1;
      } else {
        *stream_time = time - *stream_time;
        res = 1;
      }
    }
  

5.3 buffer running start time调整

获取到rstart(buffer running start time)后需要根据latency 和 timestamp offset调整buffer running start time。

  /* adjust for latency */
  stime = gst_base_sink_adjust_time (basesink, rstart);

引入latency主要是考虑到buffer从source推送到sink需要花费一定的时间。
timestamp offset主要让timestamp不正确的buffer也可以播放,该值可以通过set property设置给sink。设置负值,buffer将在早于其timestamp render,设置正值会延迟render。

/* with STREAM_LOCK, PREROLL_LOCK, LOCK
 * adjust a timestamp with the latency and timestamp offset. This function does
 * not adjust for the render delay. */
static GstClockTime
gst_base_sink_adjust_time (GstBaseSink * basesink, GstClockTime time)
{
  GstClockTimeDiff ts_offset;

  /* don't do anything funny with invalid timestamps */
  if (G_UNLIKELY (!GST_CLOCK_TIME_IS_VALID (time)))
    return time;

  time += basesink->priv->latency;

  /* apply offset, be careful for underflows */
  ts_offset = basesink->priv->ts_offset;
  if (ts_offset < 0) {
    ts_offset = -ts_offset;
    if (ts_offset < time)
      time -= ts_offset;
    else
      time = 0;
  } else
    time += ts_offset;

  /* subtract the render delay again, which was included in the latency */
  if (time > basesink->priv->render_delay)
    time -= basesink->priv->render_delay;
  else
    time = 0;

  return time;
}

5.4 wait clock time 到达buffer sync time

拿到了buffer running start time后就可以通过gst_base_sink_wait_clock这api来等clock running time 到达buffer running start time,该函数会一直block到clock running time 到达buffer running start time。

  /* This function will return immediately if start == -1, no clock
   * or sync is disabled with GST_CLOCK_BADTIME. */
  status = gst_base_sink_wait_clock (basesink, stime, &jitter);

gst_base_sink_wait_clock中先会将base-time加给buffer running start time,从而获得buffer的sync time(前面有介绍buffer sync time的含义以及计算方法:B.sync_time = B.running_time + base_time)。

GstClockReturn
gst_base_sink_wait_clock (GstBaseSink * sink, GstClockTime time,
    GstClockTimeDiff * jitter)
{
  GstClockReturn ret;
  GstClock *clock;
  GstClockTime base_time;

  if (G_UNLIKELY (!GST_CLOCK_TIME_IS_VALID (time)))
    goto invalid_time;

  GST_OBJECT_LOCK (sink);
  if (G_UNLIKELY (!sink->sync))
    goto no_sync;

  if (G_UNLIKELY ((clock = GST_ELEMENT_CLOCK (sink)) == NULL))
    goto no_clock;

  base_time = GST_ELEMENT_CAST (sink)->base_time;
  GST_LOG_OBJECT (sink,
      "time %" GST_TIME_FORMAT ", base_time %" GST_TIME_FORMAT,
      GST_TIME_ARGS (time), GST_TIME_ARGS (base_time));

  /* add base_time to running_time to get the time against the clock */
  time += base_time;

拿到buffer的sync time会通过调用gst_clock_entry_reinit api 将 buffer 的sync time 传递给global clock。

/* WARNING : Does not modify the refcount
 * WARNING : Do not use if a pending clock operation is happening on that entry */
static gboolean
gst_clock_entry_reinit (GstClock * clock, GstClockEntry * entry,
    GstClockTime time, GstClockTime interval, GstClockEntryType type)
{
  g_return_val_if_fail (entry->status != GST_CLOCK_BUSY, FALSE);
  g_return_val_if_fail (gst_clock_id_uses_clock ((GstClockID) entry, clock),
      FALSE);

  entry->type = type;
  entry->time = time;
  entry->interval = interval;
  entry->status = GST_CLOCK_OK;
  entry->unscheduled = FALSE;
  entry->woken_up = FALSE;

  return TRUE;
}

然后gst_base_sink_wait_clock 会调用到gst_system_clock_id_wait_jitter_unlocked这个api,该api会调用gst_clock_get_time ()获取当前clock 的absolute-time,并计算absolute-time和buffer sync time之间的diff,该函数最后会调用gst_pthread_cond_wait_until等待diff时长后返回。

/* synchronously wait on the given GstClockEntry.
 *
 * We do this by blocking on the entry specifically rather than a global
 * condition variable so that each possible thread may be woken up
 * individually. This ensures that we don't wake up possibly multiple threads
 * when unscheduling an entry.
 *
 * Entries that arrive too late are simply not waited on and a
 * GST_CLOCK_EARLY result is returned.
 *
 * This is called with the ENTRY_LOCK but not SYSTEM_CLOCK_LOCK!
 *
 * MT safe.
 */
static GstClockReturn
gst_system_clock_id_wait_jitter_unlocked (GstClock * clock,
    GstClockEntry * entry, GstClockTimeDiff * jitter, gboolean restart)
{
  GstClockTime entryt, now;
  GstClockTimeDiff diff;
  GstClockReturn status;
  gint64 mono_ts;

  status = GST_CLOCK_ENTRY_STATUS (entry);
  if (G_UNLIKELY (status == GST_CLOCK_UNSCHEDULED)) {
    return GST_CLOCK_UNSCHEDULED;
  }

  /* need to call the overridden method because we want to sync against the time
   * of the clock, whatever the subclass uses as a clock. */
  now = gst_clock_get_time (clock);
  mono_ts = g_get_monotonic_time ();

  /* get the time of the entry */
  entryt = GST_CLOCK_ENTRY_TIME (entry);

  /* the diff of the entry with the clock is the amount of time we have to
   * wait */
  diff = GST_CLOCK_DIFF (now, entryt);
  if (G_LIKELY (jitter))
    *jitter = -diff;

  GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock, "entry %p"
      " time %" GST_TIME_FORMAT
      " now %" GST_TIME_FORMAT
      " diff (time-now) %" G_GINT64_FORMAT,
      entry, GST_TIME_ARGS (entryt), GST_TIME_ARGS (now), diff);

  if (G_LIKELY (diff > CLOCK_MIN_WAIT_TIME)) {
#ifdef WAIT_DEBUGGING
    GstClockTime final;
#endif

    while (TRUE) {
      gboolean waitret;

#ifdef HAVE_CLOCK_NANOSLEEP
      if (diff <= 500 * GST_USECOND) {
        /* In order to provide more accurate wait, we will use BLOCKING
           clock_nanosleep for any deadlines at or below 500us */
        struct timespec end;
        GST_TIME_TO_TIMESPEC (mono_ts * 1000 + diff, end);
        GST_SYSTEM_CLOCK_ENTRY_UNLOCK ((GstClockEntryImpl *) entry);
        waitret =
            clock_nanosleep (CLOCK_MONOTONIC, TIMER_ABSTIME, &end, NULL) == 0;
        GST_SYSTEM_CLOCK_ENTRY_LOCK ((GstClockEntryImpl *) entry);
      } else {

        if (diff < 2 * GST_MSECOND) {
          /* For any deadline within 2ms, we first use the regular non-blocking
             wait by reducing the diff accordingly */
          diff -= 500 * GST_USECOND;
        }
#endif

        /* now wait on the entry, it either times out or the cond is signalled.
         * The status of the entry is BUSY only around the wait. */
        waitret =
            GST_SYSTEM_CLOCK_ENTRY_WAIT_UNTIL ((GstClockEntryImpl *) entry,
            mono_ts * 1000 + diff);

#ifdef HAVE_CLOCK_NANOSLEEP
      }
#endif

      /* get the new status, mark as DONE. We do this so that the unschedule
       * function knows when we left the poll and doesn't need to wakeup the
       * poll anymore. */
      status = GST_CLOCK_ENTRY_STATUS (entry);
      /* we were unscheduled, exit immediately */
      if (G_UNLIKELY (status == GST_CLOCK_UNSCHEDULED))
        break;
      if (G_UNLIKELY (status != GST_CLOCK_BUSY))
        GST_CAT_ERROR_OBJECT (GST_CAT_CLOCK, clock,
            "unexpected status %d for entry %p", status, entry);
      GST_CLOCK_ENTRY_STATUS (entry) = GST_CLOCK_DONE;

      GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
          "entry %p unlocked, status %d", entry, status);

      if (G_UNLIKELY (status == GST_CLOCK_UNSCHEDULED)) {
        goto done;
      } else {
        if (waitret) {
          /* some other id got unlocked */
          if (!restart) {
            /* this can happen if the entry got unlocked because of an async
             * entry was added to the head of the async queue. */
            GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
                "wakeup waiting for entry %p", entry);
            goto done;
          }

          GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
              "entry %p needs to be restarted", entry);
        } else {
          GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
              "entry %p unlocked after timeout", entry);
        }

        /* reschedule if gst_cond_wait_until returned early or we have to reschedule after
         * an unlock*/
        mono_ts = g_get_monotonic_time ();
        now = gst_clock_get_time (clock);
        diff = GST_CLOCK_DIFF (now, entryt);

        if (diff <= CLOCK_MIN_WAIT_TIME) {
          /* timeout, this is fine, we can report success now */
          GST_CLOCK_ENTRY_STATUS (entry) = status = GST_CLOCK_OK;
          GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
              "entry %p finished, diff %" G_GINT64_FORMAT, entry, diff);

#ifdef WAIT_DEBUGGING
          final = gst_system_clock_get_internal_time (clock);
          GST_CAT_DEBUG (GST_CAT_CLOCK, "Waited for %" G_GINT64_FORMAT
              " got %" G_GINT64_FORMAT " diff %" G_GINT64_FORMAT
              " %g target-offset %" G_GINT64_FORMAT " %g", entryt, now,
              now - entryt,
              (double) (GstClockTimeDiff) (now - entryt) / GST_SECOND,
              (final - target),
              ((double) (GstClockTimeDiff) (final - target)) / GST_SECOND);
#endif
          goto done;
        } else {
          GST_CAT_DEBUG_OBJECT (GST_CAT_CLOCK, clock,
              "entry %p restart, diff %" G_GINT64_FORMAT, entry, diff);
          /* we are going to poll again, set status back to busy */
          GST_CLOCK_ENTRY_STATUS (entry) = GST_CLOCK_BUSY;
        }
      }
    }
  } else {
    /* we are right on time or too late */
    if (G_UNLIKELY (diff == 0)) {
      GST_CLOCK_ENTRY_STATUS (entry) = status = GST_CLOCK_OK;
    } else {
      GST_CLOCK_ENTRY_STATUS (entry) = status = GST_CLOCK_EARLY;
    }
  }
done:
  return status;
}

gst_base_sink_do_sync函数执行完毕后会调用render函数来render frame,gst_base_sink_do_sync主要是等待clock time 到到达buffer的 sync time(或者说clock的running_time到达buffer的running_time),以此来实现stream的同步。

6 参考

1 Clocks and synchronization in GStreamer
2 Synchronisation (gstreamer.freedesktop.org)

;