| /* |
| * Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "modules/congestion_controller/include/receive_side_congestion_controller.h" |
| |
| #include "api/units/data_rate.h" |
| #include "modules/pacing/packet_router.h" |
| #include "modules/remote_bitrate_estimator/include/bwe_defines.h" |
| #include "modules/remote_bitrate_estimator/remote_bitrate_estimator_abs_send_time.h" |
| #include "modules/remote_bitrate_estimator/remote_bitrate_estimator_single_stream.h" |
| #include "rtc_base/logging.h" |
| |
| namespace webrtc { |
| |
| namespace { |
| static const uint32_t kTimeOffsetSwitchThreshold = 30; |
| } // namespace |
| |
| ReceiveSideCongestionController::WrappingBitrateEstimator:: |
| WrappingBitrateEstimator(RemoteBitrateObserver* observer, Clock* clock) |
| : observer_(observer), |
| clock_(clock), |
| rbe_(new RemoteBitrateEstimatorSingleStream(observer_, clock_)), |
| using_absolute_send_time_(false), |
| packets_since_absolute_send_time_(0), |
| min_bitrate_bps_(congestion_controller::GetMinBitrateBps()) {} |
| |
| ReceiveSideCongestionController::WrappingBitrateEstimator:: |
| ~WrappingBitrateEstimator() = default; |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator::IncomingPacket( |
| int64_t arrival_time_ms, |
| size_t payload_size, |
| const RTPHeader& header) { |
| MutexLock lock(&mutex_); |
| PickEstimatorFromHeader(header); |
| rbe_->IncomingPacket(arrival_time_ms, payload_size, header); |
| } |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator::Process() { |
| MutexLock lock(&mutex_); |
| rbe_->Process(); |
| } |
| |
| int64_t ReceiveSideCongestionController::WrappingBitrateEstimator:: |
| TimeUntilNextProcess() { |
| MutexLock lock(&mutex_); |
| return rbe_->TimeUntilNextProcess(); |
| } |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator::OnRttUpdate( |
| int64_t avg_rtt_ms, |
| int64_t max_rtt_ms) { |
| MutexLock lock(&mutex_); |
| rbe_->OnRttUpdate(avg_rtt_ms, max_rtt_ms); |
| } |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator::RemoveStream( |
| unsigned int ssrc) { |
| MutexLock lock(&mutex_); |
| rbe_->RemoveStream(ssrc); |
| } |
| |
| bool ReceiveSideCongestionController::WrappingBitrateEstimator::LatestEstimate( |
| std::vector<unsigned int>* ssrcs, |
| unsigned int* bitrate_bps) const { |
| MutexLock lock(&mutex_); |
| return rbe_->LatestEstimate(ssrcs, bitrate_bps); |
| } |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator::SetMinBitrate( |
| int min_bitrate_bps) { |
| MutexLock lock(&mutex_); |
| rbe_->SetMinBitrate(min_bitrate_bps); |
| min_bitrate_bps_ = min_bitrate_bps; |
| } |
| |
| void ReceiveSideCongestionController::WrappingBitrateEstimator:: |
| PickEstimatorFromHeader(const RTPHeader& header) { |
| if (header.extension.hasAbsoluteSendTime) { |
| // If we see AST in header, switch RBE strategy immediately. |
| if (!using_absolute_send_time_) { |
| RTC_LOG(LS_INFO) |
| << "WrappingBitrateEstimator: Switching to absolute send time RBE."; |
| using_absolute_send_time_ = true; |
| PickEstimator(); |
| } |
| packets_since_absolute_send_time_ = 0; |
| } else { |
| // When we don't see AST, wait for a few packets before going back to TOF. |
| if (using_absolute_send_time_) { |
| ++packets_since_absolute_send_time_; |
| if (packets_since_absolute_send_time_ >= kTimeOffsetSwitchThreshold) { |
| RTC_LOG(LS_INFO) |
| << "WrappingBitrateEstimator: Switching to transmission " |
| "time offset RBE."; |
| using_absolute_send_time_ = false; |
| PickEstimator(); |
| } |
| } |
| } |
| } |
| |
| // Instantiate RBE for Time Offset or Absolute Send Time extensions. |
| void ReceiveSideCongestionController::WrappingBitrateEstimator:: |
| PickEstimator() { |
| if (using_absolute_send_time_) { |
| rbe_.reset(new RemoteBitrateEstimatorAbsSendTime(observer_, clock_)); |
| } else { |
| rbe_.reset(new RemoteBitrateEstimatorSingleStream(observer_, clock_)); |
| } |
| rbe_->SetMinBitrate(min_bitrate_bps_); |
| } |
| |
| ReceiveSideCongestionController::ReceiveSideCongestionController( |
| Clock* clock, |
| PacketRouter* packet_router) |
| : ReceiveSideCongestionController(clock, packet_router, nullptr) {} |
| |
| ReceiveSideCongestionController::ReceiveSideCongestionController( |
| Clock* clock, |
| PacketRouter* packet_router, |
| NetworkStateEstimator* network_state_estimator) |
| : remb_throttler_([](auto...) {}, clock), |
| remote_bitrate_estimator_(packet_router, clock), |
| remote_estimator_proxy_( |
| clock, |
| [packet_router]( |
| std::vector<std::unique_ptr<rtcp::RtcpPacket>> packets) { |
| packet_router->SendCombinedRtcpPacket(std::move(packets)); |
| }, |
| &field_trial_config_, |
| network_state_estimator) {} |
| |
| ReceiveSideCongestionController::ReceiveSideCongestionController( |
| Clock* clock, |
| RemoteEstimatorProxy::TransportFeedbackSender feedback_sender, |
| RembThrottler::RembSender remb_sender, |
| NetworkStateEstimator* network_state_estimator) |
| : remb_throttler_(std::move(remb_sender), clock), |
| remote_bitrate_estimator_(&remb_throttler_, clock), |
| remote_estimator_proxy_(clock, |
| std::move(feedback_sender), |
| &field_trial_config_, |
| network_state_estimator) {} |
| |
| void ReceiveSideCongestionController::OnReceivedPacket( |
| int64_t arrival_time_ms, |
| size_t payload_size, |
| const RTPHeader& header) { |
| remote_estimator_proxy_.IncomingPacket(arrival_time_ms, payload_size, header); |
| if (!header.extension.hasTransportSequenceNumber) { |
| // Receive-side BWE. |
| remote_bitrate_estimator_.IncomingPacket(arrival_time_ms, payload_size, |
| header); |
| } |
| } |
| |
| void ReceiveSideCongestionController::SetSendPeriodicFeedback( |
| bool send_periodic_feedback) { |
| remote_estimator_proxy_.SetSendPeriodicFeedback(send_periodic_feedback); |
| } |
| |
| RemoteBitrateEstimator* |
| ReceiveSideCongestionController::GetRemoteBitrateEstimator(bool send_side_bwe) { |
| if (send_side_bwe) { |
| return &remote_estimator_proxy_; |
| } else { |
| return &remote_bitrate_estimator_; |
| } |
| } |
| |
| const RemoteBitrateEstimator* |
| ReceiveSideCongestionController::GetRemoteBitrateEstimator( |
| bool send_side_bwe) const { |
| if (send_side_bwe) { |
| return &remote_estimator_proxy_; |
| } else { |
| return &remote_bitrate_estimator_; |
| } |
| } |
| |
| void ReceiveSideCongestionController::OnRttUpdate(int64_t avg_rtt_ms, |
| int64_t max_rtt_ms) { |
| remote_bitrate_estimator_.OnRttUpdate(avg_rtt_ms, max_rtt_ms); |
| } |
| |
| void ReceiveSideCongestionController::OnBitrateChanged(int bitrate_bps) { |
| remote_estimator_proxy_.OnBitrateChanged(bitrate_bps); |
| } |
| |
| int64_t ReceiveSideCongestionController::TimeUntilNextProcess() { |
| return remote_bitrate_estimator_.TimeUntilNextProcess(); |
| } |
| |
| void ReceiveSideCongestionController::Process() { |
| remote_bitrate_estimator_.Process(); |
| } |
| |
| void ReceiveSideCongestionController::SetMaxDesiredReceiveBitrate( |
| DataRate bitrate) { |
| remb_throttler_.SetMaxDesiredReceiveBitrate(bitrate); |
| } |
| |
| } // namespace webrtc |