webrtc DelayManager代码研读

编程入门 行业动态 更新时间:2024-10-11 13:19:18

webrtc DelayManager<a href=https://www.elefans.com/category/jswz/34/1771412.html style=代码研读"/>

webrtc DelayManager代码研读

DelayManager代码研读

DelayManager利用方直图,统计出当前的延迟并,算出现在的应提供的buffer大小。

DelayManager通过调用Update更新包的延迟统计,并计算出target_level_;

调用BufferLimits根据target_level_获取最大最小缓存(lower_limit, higher_limit);

通过最大,最小缓存生成dsp的相关指令;

absl::optional Update(uint16_t sequence_number, uint32_t timestamp, int sample_rate_hz)分析

参数:

sequence_number:接收到的rtp包的序列号。

timestamp:包的时间戳,注意这个时间戳是按照总共多少帧来给的,就是前后两个时间戳相减算出的是有多少帧,需要除以帧率才能换算成时间

sample_rate_hz:声音的采样率。

算法代码:(去除了异常处理)

absl::optional<int> DelayManager::Update(uint16_t sequence_number,uint32_t timestamp,int sample_rate_hz) {//第一次进入if (!first_packet_received_) {// Prepare for next packet arrival.packet_iat_stopwatch_ = tick_timer_->GetNewStopwatch();last_seq_no_ = sequence_number;last_timestamp_ = timestamp;first_packet_received_ = true;return absl::nullopt;}//计算两个包的间隔时长,用ms表示,这个是包的发送时的时间间隔。// Calculate timestamps per packet and derive packet length in ms.int64_t packet_len_samp =static_cast<uint32_t>(timestamp - last_timestamp_) /static_cast<uint16_t>(sequence_number - last_seq_no_);packet_len_ms =rtc::saturated_cast<int>(1000 * packet_len_samp / sample_rate_hz);bool reordered = false;absl::optional<int> relative_delay;if (packet_len_ms > 0) {// Cannot update statistics unless |packet_len_ms| is valid.//接收端真实的收到包的时间// Inter-arrival time (IAT) in integer "packet times" (rounding down). This// is the value added to the inter-arrival time histogram.int iat_ms = packet_iat_stopwatch_->ElapsedMs();//接收时间减去发送的时间,就是iat延迟。int iat_delay = iat_ms - packet_len_ms;//进行统计,将包的延迟放入delay_history_, 具体看代码UpdateDelayHistory(iat_delay, timestamp, sample_rate_hz);//统计两秒内发送端所有包的延迟relative_delay = CalculateRelativePacketArrivalDelay();//kBucketSizeMs固定为20ms,也就是方直图的index间隔为20ms一个。索引index大小100个(方直图算法请看方直图算法博客)const int index = relative_delay.value() / kBucketSizeMs;if (index < histogram_->NumBuckets()) {// Maximum delay to register is 2000 ms.histogram_->Add(index);}//通过方直图统计出当前的target_level_,就是现在的方直图统计的延迟,用Q8表示的。// Calculate new |target_level_| based on updated statistics.target_level_ = CalculateTargetLevel();//将target_level_限制在 //effective_minimum_delay_ms_ ,//maximum_delay_ms_,//max_buffer_packets_q8//所规定的范围内LimitTargetLevel();}  // End if (packet_len_ms > 0).if (enable_rtx_handling_ && reordered &&num_reordered_packets_ < kMaxReorderedPackets) {++num_reordered_packets_;return relative_delay;}num_reordered_packets_ = 0;// Prepare for next packet arrival.packet_iat_stopwatch_ = tick_timer_->GetNewStopwatch();last_seq_no_ = sequence_number;last_timestamp_ = timestamp;return relative_delay; //返回两秒内的延迟统计
}

void DelayManager::BufferLimits(int target_level, int* lower_limit, int* higher_limit)分析

依据target_level_ 获取 lower_limit, higher_limit

// Note that |low_limit| and |higher_limit| are not assigned to
// |minimum_delay_ms_| and |maximum_delay_ms_| defined by the client of this
// class. They are computed from |target_level| in Q8 and used for decision
// making.
void DelayManager::BufferLimits(int target_level,int* lower_limit,int* higher_limit) const {if (!lower_limit || !higher_limit) {RTC_LOG_F(LS_ERROR) << "NULL pointers supplied as input";assert(false);return;}//最小的限制 target_level的3/4,当target_level的1/4大于85ms(kDecelerationTargetLevelOffsetMs==85)时,使用//target_level - kDecelerationTargetLevelOffsetMs/packet_len_ms_。// |target_level| is in Q8 already.*lower_limit = (target_level * 3) / 4;if (packet_len_ms_ > 0) {*lower_limit =std::max(*lower_limit, target_level - kDecelerationTargetLevelOffsetMs /packet_len_ms_);}//最大限制就是target_level,如果target_level的1/4小于20ms,就直接lower_limit + window_20msint window_20ms = 0x7FFF;  // Default large value for legacy bit-exactness.if (packet_len_ms_ > 0) {window_20ms = (20 << 8) / packet_len_ms_;}// |higher_limit| is equal to |target_level|, but should at// least be 20 ms higher than |lower_limit|.*higher_limit = std::max(target_level, *lower_limit + window_20ms);
}
void DelayManager::UpdateDelayHistory(int iat_delay_ms,uint32_t timestamp,int sample_rate_hz) {PacketDelay delay;delay.iat_delay_ms = iat_delay_ms;delay.timestamp = timestamp;delay_history_.push_back(delay);//kMaxHistoryMs 是固定值2000,也就是2s,timestamp是包的时间戳,又发送端打上的时间戳,delay_history_保存的是发送端2s内发送的包延迟。while (timestamp - delay_history_.front().timestamp >static_cast<uint32_t>(kMaxHistoryMs * sample_rate_hz / 1000)) {delay_history_.pop_front();}
}
统计delay_history_所有包的延迟
int DelayManager::CalculateRelativePacketArrivalDelay() const {// This effectively calculates arrival delay of a packet relative to the// packet preceding the history window. If the arrival delay ever becomes// smaller than zero, it means the reference packet is invalid, and we// move the reference.int relative_delay = 0;for (const PacketDelay& delay : delay_history_) {relative_delay += delay.iat_delay_ms;relative_delay = std::max(relative_delay, 0);}return relative_delay;
}
int DelayManager::CalculateTargetLevel() {int limit_probability = histogram_quantile_;int bucket_index = histogram_->Quantile(limit_probability);int target_level = 1;if (packet_len_ms_ > 0) {target_level += bucket_index * kBucketSizeMs / packet_len_ms_;}base_target_level_ = target_level;// Sanity check. |target_level| must be strictly positive.target_level = std::max(target_level, 1);// Scale to Q8 and assign to member variable.target_level_ = target_level << 8;return target_level_;
}
// Enforces upper and lower limits for |target_level_|. The upper limit is
// chosen to be minimum of i) 75% of |max_packets_in_buffer_|, to leave some
// headroom for natural fluctuations around the target, and ii) equivalent of
// |maximum_delay_ms_| in packets. Note that in practice, if no
// |maximum_delay_ms_| is specified, this does not have any impact, since the
// target level is far below the buffer capacity in all reasonable cases.
// The lower limit is equivalent of |effective_minimum_delay_ms_| in packets.
// We update |least_required_level_| while the above limits are applied.
// TODO(hlundin): Move this check to the buffer logistics class.
void DelayManager::LimitTargetLevel() {if (packet_len_ms_ > 0 && effective_minimum_delay_ms_ > 0) {int minimum_delay_packet_q8 =(effective_minimum_delay_ms_ << 8) / packet_len_ms_;target_level_ = std::max(target_level_, minimum_delay_packet_q8);}if (maximum_delay_ms_ > 0 && packet_len_ms_ > 0) {int maximum_delay_packet_q8 = (maximum_delay_ms_ << 8) / packet_len_ms_;target_level_ = std::min(target_level_, maximum_delay_packet_q8);}// Shift to Q8, then 75%.;int max_buffer_packets_q8 =static_cast<int>((3 * (max_packets_in_buffer_ << 8)) / 4);target_level_ = std::min(target_level_, max_buffer_packets_q8);// Sanity check, at least 1 packet (in Q8).target_level_ = std::max(target_level_, 1 << 8);
}

成员变量作用分析

packet_len_ms_ : 每一个包的长度,有30ms,20ms。

max_packets_in_buffer_ : buffer中最多包的个数,它的3/4转成ms为target_level的最大值。

packet_iat_stopwatch_ :tick_timer_->GetNewStopwatch()得到, 标记一次ticktimer的开始。

tick_timer_的用法:

​ packet_iat_stopwatch_ = tick_timer_.GetNewStopwatch();开始一次tick_timer

​ tick_timer_.Increment();比如10ms一次tick,50ms就需要调用5次,来进行计数

​ int iat_ms = packet_iat_stopwatch_->ElapsedMs();调用多少次Increment()就过了多少时间,如果调用5次,就是5*10=50ms

base_target_level_ : 是10进制的方直图计算值

target_level_ : 使用Q8表示,base_target_level_进过LimitTargetLevel限制后的值

maximum_delay_ms_ : 最大的延迟,在LimitTargetLevel对target_level_进行限制

minimum_delay_ms_ : 最小的延迟

base_minimum_delay_ms_: 基本最小延迟,需要再[0, 10000]之间

effective_minimum_delay_ms_ : minimum_delay_ms_ 和 base_minimum_delay_ms_ 中较大的值等于该值,查看void DelayManager::UpdateEffectiveMinimumDelay()中的计算方法。

更多推荐

webrtc DelayManager代码研读

本文发布于:2024-02-06 11:51:45,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1748938.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:代码   webrtc   DelayManager

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!