WebRtc Video Receiver(七)-基于Kalman filter模型的平滑渲染时间估计

1)前言

  • 前一篇文章分析了FrameBuffer模块对视频帧的插入原理,以及出队(送到解码队列)的机制。
  • 在出队的过程中涉及到了很多和延迟相关的信息,没有分析,诸如渲染时间的计算、帧延迟的计算、抖动的计算等都未进行相应的分析。
  • 同时经过上文的分析,在实际的测试过程中发现,在视频接收的过程中如果不出现丢帧的现象,那么从组帧到送入到FrameBuffer的缓存队列的耗时是十分小的,那么实际测试过程中的延迟究竟是怎么来的,经过上文的分析,初步得出,首先是在出队前从缓存列表中获取待解码的帧的时候会根据期望渲染时间计算延迟,这个延迟直接会作用到延迟重复任务的调度时间。
  • 其次就是在实际解码过程中不同的硬件平台能力不一样硬件解码器的原理也有区别,比如有些解码器本身就会缓存视频帧也是导致实际播放延迟的一个原因。
  • 再者就是音频和视频之间的同步也不会是导致播放延迟的因素之一。
  • 通过前面几篇文章的一系列分析,不难看出如VideoReceiveStream2模块、RtpVideoStreamReceiver2模块、FrameBuffer模块在整个工作的过程中都复用了同一个VCMTiming模块,在Call模块创建VideoReceiveStream2模块的时候被实例化,之后在其他模块中被引用。
  • 接下来按照实际函数调用栈的流程对视频接收模块涉及到的时序相关信息进行一一阐述。

2)PlayoutDelay更新

  • 顾名思义,叫做播放延迟,该值可在发送端通过RTP头扩展进行携带,如果未携带,默认值为{-1,-1}
  • 首先看其定义如下:
#common_types.h
struct PlayoutDelay {
  PlayoutDelay(int min_ms, int max_ms) : min_ms(min_ms), max_ms(max_ms) {}
  int min_ms;//最小播放延迟
  int max_ms;//做到播放延迟
  ....  
}    
#rtp_video_header.h
struct RTPVideoHeader {
  ....  
  PlayoutDelay playout_delay = {-1, -1};
  ....
}
# encoded_image.h
class RTC_EXPORT EncodedImage {
 public:
  ...  
  // When an application indicates non-zero values here, it is taken as an
  // indication that all future frames will be constrained with those limits
  // until the application indicates a change again.
  PlayoutDelay playout_delay_ = {-1, -1};
  ...
}
RtpFrameObject::RtpFrameObject(
   ......
    : first_seq_num_(first_seq_num),
      last_seq_num_(last_seq_num),
      last_packet_received_time_(last_packet_received_time),
      times_nacked_(times_nacked) {

  // Setting frame's playout delays to the same values
  // as of the first packet's.
  SetPlayoutDelay(rtp_video_header_.playout_delay);
  ...
}
  • PacketBuffer模块组帧过程中,每组帧成功后会执行发现帧的处理,此时会创建对应的RtpFrameObject
  • 而在RtpFrameObject的构造函数会通过SetPlayoutDelay函数为当前帧设置播放延迟时间,由此可看出如果发送端未扩展该RTP头的话,那么默认值为{-1,-1}。
  • 默认是未扩展的如果需要扩展需要SDP支持如下:
"http://www.webrtc.org/experiments/rtp-hdrext/playout-delay"
  • 接下来介绍,在何时会将该延迟作用到其他模块。


    WebRtc_Video_Stream_Receiver_07_01.png
  • 接上文的分析,待解码视频帧的插入驱动是由VideoReceiveStream2:: OnCompleteFrame()函数来驱动的,在插入视频帧前首先会将播放延迟信息作用到VCMTiming模块,上面有介绍VideoReceiveStream2模块、RtpVideoStreamReceiver2模块、FrameBuffer模块在整个工作的过程中都复用了同一个VCMTiming实例(同一路流)。

void VideoReceiveStream2::OnCompleteFrame(
    std::unique_ptr<video_coding::EncodedFrame> frame) {
  ....
  //拿到PlayoutDelay引用    
  const PlayoutDelay& playout_delay = frame->EncodedImage().playout_delay_;
    
  if (playout_delay.min_ms >= 0) {
    frame_minimum_playout_delay_ms_ = playout_delay.min_ms;
    UpdatePlayoutDelays();
  }

  if (playout_delay.max_ms >= 0) {
    frame_maximum_playout_delay_ms_ = playout_delay.max_ms;
    UpdatePlayoutDelays();
  }
  ....
}
  • 如果playout_delay.min_ms >= 0或者playout_delay.max_ms >= 0都会调用UpdatePlayoutDelays函数将该播放延迟作用到VCMTiming模块。
void VCMTiming::set_min_playout_delay(int min_playout_delay_ms) {
  rtc::CritScope cs(&crit_sect_);
  min_playout_delay_ms_ = min_playout_delay_ms;
}

void VCMTiming::set_max_playout_delay(int max_playout_delay_ms) {
  rtc::CritScope cs(&crit_sect_);
  max_playout_delay_ms_ = max_playout_delay_ms;
}
  • VCMTiming模块记录当前帧的min_playout_delay_ms_max_playout_delay_ms_供后续延迟估计使用。

3)RenderTimeMs设置流程

  • 回顾上文的分析,FindNextFrame函数在找待解码帧的时候会通过VCMTiming模块获取期望渲染时间。
  • 本节内容着重介绍RenderTimeMs的设置业务流程以及大致的原理,涉及到Kalman filter的实现原理在第4节内容将进行详细分析。本节涉及的大致流程如下:
    WebRtc_Video_Stream_Receiver_07_02.png
  • 从上图左侧部分可知,在当前帧插入到frame_的时候如果当前帧不是重传帧的话,会使用VCMTiming模块调用TimestampExtrapolator模块的Update()函数依据当前帧的rtp时间戳来估计当前帧的期望接收时间,并对Kalman gain进行校准,其原理将在第4节内容进行详细分析。
int64_t FrameBuffer::FindNextFrame(int64_t now_ms) {
  ....
  //默认该函数调用到这里的时候期望渲染时间都还未赋值的    
  if (frame->RenderTime() == -1) {
      //首先调用VCMTiming获取期望渲染时间,然后将其设置到Frame中,供后续使用
      frame->SetRenderTime(timing_->RenderTimeMs(frame->Timestamp(), now_ms));
  }
  ...  
  //得出最大等待时间    
  wait_ms = timing_->MaxWaitingTime(frame->RenderTime(), now_ms);
  ....
  //取最小时间,如果在一次调度时间(未超时范围内)的话,返回wait_ms    
  wait_ms = std::min<int64_t>(wait_ms, latest_return_time_ms_ - now_ms);
    
  wait_ms = std::max<int64_t>(wait_ms, 0);  
  return wait_ms;
}    
  • 以上函数返回的wait_ms值会直接决定重复延迟队列的执行时间(也就是等多节执行),如果wait_ms等于0,则说明重复延迟队列会立即执行。优化该值趋向0会节省一定的延迟。

3.1)VCMTiming模块获取期望渲染时间

int64_t VCMTiming::RenderTimeMs(uint32_t frame_timestamp,
                                int64_t now_ms) const {
  rtc::CritScope cs(&crit_sect_);
  return RenderTimeMsInternal(frame_timestamp, now_ms);
}
//frame_timestamp为当前帧的时间戳以1/90k为单位,now_ms为当前Clock时间
int64_t VCMTiming::RenderTimeMsInternal(uint32_t frame_timestamp,
                                        int64_t now_ms) const {
  //如果min_playout_delay_ms_=0并且max_playout_delay_ms_=0则表示立即渲染
 // 不建议赋值0,若赋值0的话jitterDelay就失效了
  if (min_playout_delay_ms_ == 0 && max_playout_delay_ms_ == 0) {
    // Render as soon as possible.
    return 0;
  }
  //传入当前帧的时间戳,来得到一个平滑渲染时间,TimestampExtrapolator通过卡尔曼滤波负责期望接收时间的产生
  int64_t estimated_complete_time_ms =
      ts_extrapolator_->ExtrapolateLocalTime(frame_timestamp);
  if (estimated_complete_time_ms == -1) {
    estimated_complete_time_ms = now_ms;
  }

  // Make sure the actual delay stays in the range of |min_playout_delay_ms_|
  // and |max_playout_delay_ms_|.
  // 和min_playout_delay_ms_取最大值,min_playout_delay_ms_默认值-1,  
  int actual_delay = std::max(current_delay_ms_, min_playout_delay_ms_);
  //和max_playout_delay_ms_求最小值,max_playout_delay_ms_默认值-1  
  actual_delay = std::min(actual_delay, max_playout_delay_ms_);
  return estimated_complete_time_ms + actual_delay;
}
  • 该函数首先判断min_playout_delay_ms_max_playout_delay_ms_是否同时为0,如果同时为0则表示会理解发送到解码队列并解码后后立即渲染。
  • 若上述条件不满足,也就是说什么时候渲染依据系统框架的估计来做。
  • 首先调用TimestampExtrapolator模块的ExtrapolateLocalTime函数来估计出一个期望接收时间,最后将该时间和actual_delay实际延迟相加得到最终的期望渲染时间
  • 其中actual_delay的值是通过VCMTiming::SetJitterDelayVCMTiming::UpdateCurrentDelay两个函数来进行更新的,这两个函数在下面进行分析。
  • 最终得出视频帧的最终期望渲染时间 = 平滑渲染时间 + 当前实际播放延迟(它的原理有是什么)

3.2)VCMTiming模块获取调度等待时间

  • FrameBuffer模块的FindNextFrame函数通过该函数返回一个最大等待时间,也就是说如果找到了一帧数据,但是FrameBuffer模块并不会立即将其发送到解码队列,而是要等待一段时间,再发送到解码器,该函数的作用就是得到等待多长时间发送到解码队列进行解码
int64_t VCMTiming::MaxWaitingTime(int64_t render_time_ms,
                                  int64_t now_ms) const {
  rtc::CritScope cs(&crit_sect_);
  const int64_t max_wait_time_ms =
      render_time_ms - now_ms - RequiredDecodeTimeMs() - render_delay_ms_;

  return max_wait_time_ms;
}
  • 上述代码显示最大等待时间 (何时发到解码队列)= 期望渲染时间 - 当前时间 - 解码所需要的的时间 - 渲染延迟的时间
  • 其中正常情况下期望渲染时间是根据卡尔曼滤波理论估计出来的。
  • render_delay_ms_默认为10ms,可通过VCMTiming::set_render_delay进行设置,默认在初始化阶段实例化VideoReceiveStream2模块的时候在其构造函数中有调用该函数,可通过修改webrtc::VideoReceiveStream::Config::render_delay_ms变量进行设置。
  • 经过以上的分析最大的困难之处就是平滑渲染时间estimated_complete_time_ms的估计过程,在后续将专门分析卡尔曼的原理。
  • 按照正常的流程如果卡尔曼估计出来平滑渲染时间比较大,然后解码所需要的的时间已知的情况下,那么优化就必须放在卡尔曼滤波器的身上。
  • 通过上述的分析,期望渲染时间的延迟会直接影响到FrameBuffer模块重复延迟队列的调度,也就决定了当前帧播放的延迟。
  • 用时间轴来描述当前时间,进入解码队列时间,解码延迟时间,渲染等待时间的关系,如下:


    image-20210614150723557.png

4)TimestampExtrapolator Kalman filter期望渲染时间估计

  • 本节内容从四个方面来进行介绍。
  • 首先介绍卡尔曼理论5大公式以及基于rtp时间戳和当前实际接收时间的理论模型
  • 其次基于理论模型建立状态转移方程以及观测方程
  • 再次介绍TimestampExtrapolator::ExtrapolateLocalTime函数期望接收时间的计算。
  • 最后结合TimestampExtrapolator::Update函数分析状态变量的计算原理和Kalman Ganin的更新。

4.1)TimestampExtrapolator模块Kalman模型

  • 为后续的方便分析首先介绍卡尔曼滤波的5大核心公式,本文不做理论推导


    image-20210614000341875.png
  • 根据卡尔曼滤波的5大核心公式,首先需要建立状态转移方程和观测方程

  • 为建立状态转移方程和观测方程,需要先了解一下rtp time stamp和实际时间的相应关系

  // Local time in webrtc time base.
  int64_t current_time_us = clock_->TimeInMicroseconds();
  int64_t current_time_ms = current_time_us / rtc::kNumMicrosecsPerMillisec;

  // Capture time may come from clock with an offset and drift from clock_.
  int64_t capture_ntp_time_ms = current_time_ms + delta_ntp_internal_ms_;
  // Convert NTP time, in ms, to RTP timestamp.
  const int kMsToRtpTimestamp = 90;
  uint32_t timestamp_rtp =
      kMsToRtpTimestamp * static_cast<uint32_t>(capture_ntp_time_ms);

  • webrtc 在发送每一帧视频数据前通过上述代码来设置每帧的rtp时间戳,以90KHZ为采样率,也就是每秒钟被划分成了90000个时间块,假设是60fps每秒的帧率,那么每帧的rtp时间戳理论上是相隔90000 / 60 = 1500个时间块,也就是每帧数据之间RTP时间戳的增量为1500,如果将该时间增量换算成ms数如下:
 fps = 60fps
 samplerate = 90000    
 timestampDiff(k) = rtpTimeStamp(k) - rtpTimeStamp(0)                        (4.1.1)
 timestampDiffToMs(k) = timestampDiff(k) * 1000 / samplerate                 (4.1.2)
  • timestampDiff(k)表示第(k)帧视频和第一帧视频的rtpTimeStamp之差

  • timestampDiffToMs(k)表示第(k)帧视频和第一帧视频之间所经历的毫秒数

  • 基于此,建立如下到达传输模型


    WebRtc_Video_Stream_Receiver_07_06.png
  • 以第一帧(t0,T0)作为基准,来估计接收到第k帧的期望接收时间

  • 假设上述模型没有任何误差和干扰,那么在已知_startMs的情况下,上述的传输曲线应该都是蓝色实现所示,并很容易就能得到如下计算

 t(0) = _startMs
 t(k) = timestampDiffToMs(k) + t(0)                                     (4.1.3)
  • 但事实上不上这样,传输过程有很多的不确定性诸如网络延迟抖动等,且由于采集帧率可能也具有误差,也就是sample_rate可能大于90KHZ,最终每帧数据的到达模型可能就变成了上述的红色虚线部分所示,这样timestampDiffToMs(k)就会比第(k)帧达到接收端实际所经历的时间要大或者要小,从而使得t(k)第(k)帧的接收时间变得不准,这样较为准确的t(k)应该用如下公式来描述
 t(k) = timestampDiffToMs(k) + t(0) + error(k)                          (4.1.4)
  • 其中error(k)表示传输过程中采集噪声和网络噪声以及其他噪声的总集合,如果将网络延迟导致的误差和因采集噪声所导致的延迟误差提取出来,就可以将上述公司进行如下变换
 t(k) = (timestampDiff(k) - jitterTimestamp(k))  / sampleratePermillage(k) + t(0)   (4.1.5)
  • 其中jitterTimestamp(k)就是因网络波动导致的第(k)帧和第1帧数据之间的rtpTimeStamp时延抖动
  • sampleratePermillage(k)表示第k帧的千分之采样率
  • 基于此我们可以建立如下状态转移方程,并使用卡尔曼滤波,通过迭代和更新使得jitterTimestamp(k)sampleratePermillage(k)的误差尽可能的小从而使得第(k)帧的期望接收时间更加的准确
 w(k) = w(k-1) + u(k-1)                                 P(u) ~ (0,Q)    (4.1.6)      
 w_bar(k) = [sampleratePermillage(k) jitterTimestamp(k)]^    
  • 定义目标二维向量w_bar(k)

  • u(k-1)为过程噪声服从正太分布,由于样本samplerate_permillage(k)jitterTimestamp(k)完全独立,所以其协方差矩阵Q似乎可以取0

  • 状态转移方程如果用矩阵的表示方式如下:


    image-20210614190609595.png
  • 同时建立如下观测方程

 timestampDiff(k) = t_bar(k)^ * w_bar(k) + v(k)             P(v) ~ (0,R)    (4.1.7)
 t_bar(k) = [recvTimeMsDiff(k) 1]^    
  • v(k)为测量噪声服从正太分布,其协方差矩阵为R,取值为1

  • t_bar(i)为第(k)帧观测方程系数矩阵

  • recvTimeMsDiff(k)表示第(k)帧和第一帧的本地接收时间之差

  • 观测方程如果用矩阵的表示方式如下:


    image-20210614190704946.png
  • 网络残差公式

residual(k) = timestampDiff(k) - t_bar(k)^ * w_hat(k-1)                 (4.1.8)
  • 网络残差体现了噪声的大小,使用第(k)帧的观测值 - 第(k-1)的估计计算值

4.2)TimestampExtrapolator模块计算期望接收时间

int64_t TimestampExtrapolator::ExtrapolateLocalTime(uint32_t timestamp90khz) {
  ReadLockScoped rl(*_rwLock);
  int64_t localTimeMs = 0;
  CheckForWrapArounds(timestamp90khz);
  double unwrapped_ts90khz =
      static_cast<double>(timestamp90khz) +
      _wrapArounds * ((static_cast<int64_t>(1) << 32) - 1);
  if (_packetCount == 0) {
    localTimeMs = -1;
  } else if (_packetCount < _startUpFilterDelayInPackets) {
    localTimeMs =
        _prevMs +
        static_cast<int64_t>(
            static_cast<double>(unwrapped_ts90khz - _prevUnwrappedTimestamp) /
                90.0 +
            0.5);
  } else {
    if (_w[0] < 1e-3) {
      localTimeMs = _startMs;
    } else {
      double timestampDiff =
          unwrapped_ts90khz - static_cast<double>(_firstTimestamp);
      localTimeMs = static_cast<int64_t>(static_cast<double>(_startMs) +
                                         (timestampDiff - _w[1]) / _w[0] + 0.5);
    }
  }
  return localTimeMs;
}
  • 在3.1节中有介绍到获取当前帧的期望渲染时间 = 期望接收时间 + 实际延迟时间
  • timestampDiff = (第k帧的rtp时间戳 - 第一帧时间戳 )
  • localTimeMs = (timestampDiff - _w[1]) / _w[0] + (第一帧的实际接收时间),使用timestampDiff (发送端决定的) - 由于网络延迟或波动造成的延迟所对应的rtp时间戳的大小得出第k帧和第一帧最优的时间戳只差
  • _w[0] = sampleratePermillage(k)表示第(k)帧的千分之采样率
  • _w[1] = jitterTimestamp(k)表示第(k)帧的抖动rtp timestamp 时延

4.3)TimestampExtrapolator模块Kalman预测及校正

  • 对于未接收到的未丢过包的视频帧,每帧数据插入到FrameBuffer缓存后,通过当前帧的rtp 时间戳以及接收时间来更新TimestampExtrapolator的卡尔曼滤波器,进行迭代和校正
//参数tMs为当前帧实际接收时间
//参数ts90khz为当前帧的rtp时间戳
void TimestampExtrapolator::Update(int64_t tMs, uint32_t ts90khz) {
   _rwLock->AcquireLockExclusive();
  //1)第一帧初始赋值  
  if (tMs - _prevMs > 10e3) {//第一帧或者10秒钟内未收到任何完整的帧则重置
    // Ten seconds without a complete frame.
    // Reset the extrapolator
    _rwLock->ReleaseLockExclusive();
    Reset(tMs);
    _rwLock->AcquireLockExclusive();
  } else {
    _prevMs = tMs;
  } 
    
  //2)根据当前帧的本地接收时间计算detalRecvTimeMs(k)  
  // Remove offset to prevent badly scaled matrices
  // 将当前帧接收时间 - 第一帧的接收时间得当前帧和第一帧的本地接收时间差
  // 此处记为detalRecvTimeMs =  tMs - _startMs  
  int64_t recvTimeMsDiff = tMs - _startMs;

  CheckForWrapArounds(ts90khz);

  int64_t unwrapped_ts90khz =
      static_cast<int64_t>(ts90khz) +
      _wrapArounds * ((static_cast<int64_t>(1) << 32) - 1);

  if (_firstAfterReset) {//重置后赋值初值
    // Make an initial guess of the offset,
    // should be almost correct since tMs - _startMs
    // should about zero at this time.
    _w[1] = -_w[0] * tMs;
    _firstTimestamp = unwrapped_ts90khz;
    _firstAfterReset = false;
  }
  /*3)使用上一次最优估计计算网络残差为计算验估计做准备,对应5大核心公式的公式(4) 以及4.1.8
      用当前帧真实的rtp时间戳 - 第一帧的时间戳 - detalRecvTimeMs * _w[0] - _w[1]
      detalRecvTimeMs * _w[0](上一次的最优采样率) 得出detalRtpTimeStamp 
  */  
  double residual = (static_cast<double>(unwrapped_ts90khz) - _firstTimestamp) -
                    static_cast<double>(recvTimeMsDiff) * _w[0] - _w[1];
    
  if (DelayChangeDetection(residual) &&
      _packetCount >= _startUpFilterDelayInPackets) {
    // A sudden change of average network delay has been detected.
    // Force the filter to adjust its offset parameter by changing
    // the offset uncertainty. Don't do this during startup.
    _pP[1][1] = _pP11;
  }

  if (_prevUnwrappedTimestamp >= 0 &&
      unwrapped_ts90khz < _prevUnwrappedTimestamp) {
    // Drop reordered frames.
    _rwLock->ReleaseLockExclusive();
    return;
  }

  // T = [t(k) 1]';
  // that = T'*w;
  // K = P*T/(lambda + T'*P*T);
  // 4)计算卡尔曼增益  
  double K[2];
  // 对应5大公式,公式3中的分子部分  
  K[0] = _pP[0][0] * recvTimeMsDiff + _pP[0][1];
  K[1] = _pP[1][0] * recvTimeMsDiff + _pP[1][1];
  // 对应5大公式,公式3中的分母部分   
  double TPT = _lambda + recvTimeMsDiff * K[0] + K[1];
  K[0] /= TPT;
  K[1] /= TPT;
  //5) 根据最优卡尔曼因子进行校正,计算后验估计值  
  // w = w + K*(ts(k) - that);
  _w[0] = _w[0] + K[0] * residual;
  _w[1] = _w[1] + K[1] * residual;
  //6)更新误差协方差  
  // P = 1/lambda*(P - K*T'*P);
  double p00 =
      1 / _ * (_pP[0][0] - (K[0] * recvTimeMsDiff * _pP[0][0] + K[0] * _pP[1][0]));
  double p01 =
      1 / _lambda * (_pP[0][1] - (K[0] * recvTimeMsDiff * _pP[0][1] + K[0] * _pP[1][1]));
  _pP[1][0] =
      1 / _lambda * (_pP[1][0] - (K[1] * recvTimeMsDiff * _pP[0][0] + K[1] * _pP[1][0]));
  _pP[1][1] =
      1 / _lambda * (_pP[1][1] - (K[1] * recvTimeMsDiff * _pP[0][1] + K[1] * _pP[1][1]));
  _pP[0][0] = p00;
  _pP[0][1] = p01;
  _prevUnwrappedTimestamp = unwrapped_ts90khz;
  if (_packetCount < _startUpFilterDelayInPackets) {
    _packetCount++;
  }
  _rwLock->ReleaseLockExclusive();
}
  • 以上首先是根据图WebRtc_Video_Stream_Receiver_07_05中的公式(3)计算卡尔曼增量因子,计算的过程中先求其分子部分,然后再求分母部分
  • 以上观测噪声的协方差R等于lambda,默认为1。
  • 到此就能得出一个期望的接收时间,根据期望接收时间就能得出一个期望渲染时间从而得到期望渲染时间
expectRenderTime = expectRecvTime + actual_delay
  • 接下来开始分析actual_delay是如何求取的?
  • actual_delay指的是前面已解码的帧的实际的延迟时间

5)计算期望渲染时间

  • 首先再回顾FrameBuffer模块取帧和将帧发送到解码队列的流程如下图

    image-20210614175550223.png

  • 解码任务队列,每次轮询通过FrameBuffer::FindNextFrame()找到一帧完整的帧,并为其设置期望最优渲染时间,然后再通过MaxWaitingTime函数返回一个延迟时间,告知重复任务队列,让该任务等待wait_ms之后开始执行,将该帧数据发送到解码任务队列进行解码,解码完毕后又重新轮询新的帧

  • 其中在计算渲染时间的时候有用到上一帧数据的平滑actual_delay,也就是current_delay_ms_,基于此开始分析current_delay_ms_的计算过程

void VCMTiming::SetJitterDelay(int jitter_delay_ms) {
  rtc::CritScope cs(&crit_sect_);
  if (jitter_delay_ms != jitter_delay_ms_) {
    jitter_delay_ms_ = jitter_delay_ms;
    // When in initial state, set current delay to minimum delay.
    if (current_delay_ms_ == 0) {
      current_delay_ms_ = jitter_delay_ms_;
    }
  }
}
  • jitter_delay_ms是在每一帧数据送到解码队列之前使用VCMJitterEstimator模块估计出来的,对于第一帧数据current_delay_ms_的值就等于该帧的jitter_delay_ms_
void VCMTiming::UpdateCurrentDelay(int64_t render_time_ms,
                                   int64_t actual_decode_time_ms) {
  rtc::CritScope cs(&crit_sect_);
  uint32_t target_delay_ms = TargetDelayInternal();//目标延迟
  //计算实际延迟  
  int64_t delayed_ms =
      actual_decode_time_ms -
      (render_time_ms - RequiredDecodeTimeMs() - render_delay_ms_);
  if (delayed_ms < 0) {
    return;
  }
  if (current_delay_ms_ + delayed_ms <= target_delay_ms) {
    current_delay_ms_ += delayed_ms;
  } else {
    current_delay_ms_ = target_delay_ms;
  }
}
int VCMTiming::TargetDelayInternal() const {
  //计算出目标延迟 =  jitter_delay_ms_ + 解码耗时 +  渲染延迟
  return std::max(min_playout_delay_ms_,
                  jitter_delay_ms_ + RequiredDecodeTimeMs() + render_delay_ms_);
}
  • 以上函数的作用就是在当前帧顺利发送到解码队列之后尽量确保current_delay_ms_的值逼近jitter_delay_ms_ + RequiredDecodeTimeMs() + render_delay_ms_
  • 然后在后续帧获取期望渲染时间的时候,尽量保证所有帧的延迟间隔趋向于平滑
  current_delay_ms_ = 帧间抖动延时 + 解码耗时 + 渲染延迟 
  • 最后再回顾3.1)节中的VCMTiming模块获取期望渲染时间
expectRenderTime = expectRecvTime + actual_delay
                 = 期望接收时间 + 帧间抖动延时 + 解码耗时 + 渲染延迟 
  • 已知默认情况下渲染延迟一般都是默认值10ms

  • 解码耗时依据硬解的性能而定,一般都会比较均匀

  • 然而期望接收时间和帧间抖动延时都会因为网络的千变万化以及发送端的各种不确定性存在一定的波动

  • 其实在调试中发现不管用不用卡尔曼滤波,对于期望接收时间,如果网络不是很差,基本上估计出来的曲线和实际曲线是基本一致的,当然也有可能是我测试的条件比较好


    image-20210614185054732.png
  • 其中蓝色曲线直接试用timestampDiff / 90 + _startMs所得,当我设置接收端5%丢包的时候两条曲线完全重合了

6)总结

  • 通过本文的分析,首先明确webrtc视频接收过程中的逻辑处理是十分复杂的,同时通过两大卡尔曼滤波对每帧的期望接收和每帧的帧间抖动进行滤波处理
  • 其中对帧间抖动的滤波可以使得每帧数据进入解码队列的时机变得相对平滑,这样可以有效的缓解因网络丢包等情况导致视频的卡顿等问题
  • 而期望渲染时间的估计使得渲染过程趋向于平滑,同时也明确本文中使用的卡尔曼滤波的作用是为每帧产生一个最优的接收时间,改时间最终决定了该帧的渲染时间,这对音视频的同步是非常有作用的
  • 同时通过学习本文深入学习卡尔曼滤波的应用场景
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容