| /* |
| * Copyright (c) 2018 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 "call/degraded_call.h" |
| |
| #include <memory> |
| #include <utility> |
| |
| #include "absl/strings/string_view.h" |
| #include "api/sequence_checker.h" |
| #include "modules/rtp_rtcp/source/rtp_util.h" |
| #include "rtc_base/thread.h" |
| |
| namespace webrtc { |
| |
| DegradedCall::FakeNetworkPipeOnTaskQueue::FakeNetworkPipeOnTaskQueue( |
| TaskQueueBase* task_queue, |
| rtc::scoped_refptr<PendingTaskSafetyFlag> call_alive, |
| Clock* clock, |
| std::unique_ptr<NetworkBehaviorInterface> network_behavior) |
| : clock_(clock), |
| task_queue_(task_queue), |
| call_alive_(std::move(call_alive)), |
| pipe_(clock, std::move(network_behavior)) {} |
| |
| void DegradedCall::FakeNetworkPipeOnTaskQueue::SendRtp( |
| rtc::ArrayView<const uint8_t> packet, |
| const PacketOptions& options, |
| Transport* transport) { |
| pipe_.SendRtp(packet, options, transport); |
| Process(); |
| } |
| |
| void DegradedCall::FakeNetworkPipeOnTaskQueue::SendRtcp( |
| rtc::ArrayView<const uint8_t> packet, |
| Transport* transport) { |
| pipe_.SendRtcp(packet, transport); |
| Process(); |
| } |
| |
| void DegradedCall::FakeNetworkPipeOnTaskQueue::AddActiveTransport( |
| Transport* transport) { |
| pipe_.AddActiveTransport(transport); |
| } |
| |
| void DegradedCall::FakeNetworkPipeOnTaskQueue::RemoveActiveTransport( |
| Transport* transport) { |
| pipe_.RemoveActiveTransport(transport); |
| } |
| |
| bool DegradedCall::FakeNetworkPipeOnTaskQueue::Process() { |
| pipe_.Process(); |
| auto time_to_next = pipe_.TimeUntilNextProcess(); |
| if (!time_to_next) { |
| // Packet was probably sent immediately. |
| return false; |
| } |
| |
| task_queue_->PostTask(SafeTask(call_alive_, [this, time_to_next] { |
| RTC_DCHECK_RUN_ON(task_queue_); |
| int64_t next_process_time = *time_to_next + clock_->TimeInMilliseconds(); |
| if (!next_process_ms_ || next_process_time < *next_process_ms_) { |
| next_process_ms_ = next_process_time; |
| task_queue_->PostDelayedHighPrecisionTask( |
| SafeTask(call_alive_, |
| [this] { |
| RTC_DCHECK_RUN_ON(task_queue_); |
| if (!Process()) { |
| next_process_ms_.reset(); |
| } |
| }), |
| TimeDelta::Millis(*time_to_next)); |
| } |
| })); |
| |
| return true; |
| } |
| |
| DegradedCall::FakeNetworkPipeTransportAdapter::FakeNetworkPipeTransportAdapter( |
| FakeNetworkPipeOnTaskQueue* fake_network, |
| Call* call, |
| Clock* clock, |
| Transport* real_transport) |
| : network_pipe_(fake_network), |
| call_(call), |
| clock_(clock), |
| real_transport_(real_transport) { |
| network_pipe_->AddActiveTransport(real_transport); |
| } |
| |
| DegradedCall::FakeNetworkPipeTransportAdapter:: |
| ~FakeNetworkPipeTransportAdapter() { |
| network_pipe_->RemoveActiveTransport(real_transport_); |
| } |
| |
| bool DegradedCall::FakeNetworkPipeTransportAdapter::SendRtp( |
| rtc::ArrayView<const uint8_t> packet, |
| const PacketOptions& options) { |
| // A call here comes from the RTP stack (probably pacer). We intercept it and |
| // put it in the fake network pipe instead, but report to Call that is has |
| // been sent, so that the bandwidth estimator sees the delay we add. |
| network_pipe_->SendRtp(packet, options, real_transport_); |
| if (options.packet_id != -1) { |
| rtc::SentPacket sent_packet; |
| sent_packet.packet_id = options.packet_id; |
| sent_packet.send_time_ms = clock_->TimeInMilliseconds(); |
| sent_packet.info.included_in_feedback = options.included_in_feedback; |
| sent_packet.info.included_in_allocation = options.included_in_allocation; |
| sent_packet.info.packet_size_bytes = packet.size(); |
| sent_packet.info.packet_type = rtc::PacketType::kData; |
| call_->OnSentPacket(sent_packet); |
| } |
| return true; |
| } |
| |
| bool DegradedCall::FakeNetworkPipeTransportAdapter::SendRtcp( |
| rtc::ArrayView<const uint8_t> packet) { |
| network_pipe_->SendRtcp(packet, real_transport_); |
| return true; |
| } |
| |
| DegradedCall::DegradedCall( |
| std::unique_ptr<Call> call, |
| const std::vector<TimeScopedNetworkConfig>& send_configs, |
| const std::vector<TimeScopedNetworkConfig>& receive_configs) |
| : clock_(Clock::GetRealTimeClock()), |
| call_(std::move(call)), |
| call_alive_(PendingTaskSafetyFlag::CreateDetached()), |
| send_config_index_(0), |
| send_configs_(send_configs), |
| send_simulated_network_(nullptr), |
| receive_config_index_(0), |
| receive_configs_(receive_configs) { |
| if (!receive_configs_.empty()) { |
| auto network = std::make_unique<SimulatedNetwork>(receive_configs_[0]); |
| receive_simulated_network_ = network.get(); |
| receive_pipe_ = |
| std::make_unique<webrtc::FakeNetworkPipe>(clock_, std::move(network)); |
| receive_pipe_->SetReceiver(call_->Receiver()); |
| if (receive_configs_.size() > 1) { |
| call_->network_thread()->PostDelayedTask( |
| SafeTask(call_alive_, [this] { UpdateReceiveNetworkConfig(); }), |
| receive_configs_[0].duration); |
| } |
| } |
| if (!send_configs_.empty()) { |
| auto network = std::make_unique<SimulatedNetwork>(send_configs_[0]); |
| send_simulated_network_ = network.get(); |
| send_pipe_ = std::make_unique<FakeNetworkPipeOnTaskQueue>( |
| call_->network_thread(), call_alive_, clock_, std::move(network)); |
| if (send_configs_.size() > 1) { |
| call_->network_thread()->PostDelayedTask( |
| SafeTask(call_alive_, [this] { UpdateSendNetworkConfig(); }), |
| send_configs_[0].duration); |
| } |
| } |
| } |
| |
| DegradedCall::~DegradedCall() { |
| RTC_DCHECK_RUN_ON(call_->worker_thread()); |
| // Thread synchronization is required to call `SetNotAlive`. |
| // Otherwise, when the `DegradedCall` object is destroyed but |
| // `SetNotAlive` has not yet been called, |
| // another Closure guarded by `call_alive_` may be called. |
| // TODO(https://crbug.com/webrtc/12649): Remove this block-invoke. |
| static_cast<rtc::Thread*>(call_->network_thread()) |
| ->BlockingCall( |
| [flag = std::move(call_alive_)]() mutable { flag->SetNotAlive(); }); |
| } |
| |
| AudioSendStream* DegradedCall::CreateAudioSendStream( |
| const AudioSendStream::Config& config) { |
| if (!send_configs_.empty()) { |
| auto transport_adapter = std::make_unique<FakeNetworkPipeTransportAdapter>( |
| send_pipe_.get(), call_.get(), clock_, config.send_transport); |
| AudioSendStream::Config degrade_config = config; |
| degrade_config.send_transport = transport_adapter.get(); |
| AudioSendStream* send_stream = call_->CreateAudioSendStream(degrade_config); |
| if (send_stream) { |
| audio_send_transport_adapters_[send_stream] = |
| std::move(transport_adapter); |
| } |
| return send_stream; |
| } |
| return call_->CreateAudioSendStream(config); |
| } |
| |
| void DegradedCall::DestroyAudioSendStream(AudioSendStream* send_stream) { |
| call_->DestroyAudioSendStream(send_stream); |
| audio_send_transport_adapters_.erase(send_stream); |
| } |
| |
| AudioReceiveStreamInterface* DegradedCall::CreateAudioReceiveStream( |
| const AudioReceiveStreamInterface::Config& config) { |
| return call_->CreateAudioReceiveStream(config); |
| } |
| |
| void DegradedCall::DestroyAudioReceiveStream( |
| AudioReceiveStreamInterface* receive_stream) { |
| call_->DestroyAudioReceiveStream(receive_stream); |
| } |
| |
| VideoSendStream* DegradedCall::CreateVideoSendStream( |
| VideoSendStream::Config config, |
| VideoEncoderConfig encoder_config) { |
| std::unique_ptr<FakeNetworkPipeTransportAdapter> transport_adapter; |
| if (!send_configs_.empty()) { |
| transport_adapter = std::make_unique<FakeNetworkPipeTransportAdapter>( |
| send_pipe_.get(), call_.get(), clock_, config.send_transport); |
| config.send_transport = transport_adapter.get(); |
| } |
| VideoSendStream* send_stream = call_->CreateVideoSendStream( |
| std::move(config), std::move(encoder_config)); |
| if (send_stream && transport_adapter) { |
| video_send_transport_adapters_[send_stream] = std::move(transport_adapter); |
| } |
| return send_stream; |
| } |
| |
| VideoSendStream* DegradedCall::CreateVideoSendStream( |
| VideoSendStream::Config config, |
| VideoEncoderConfig encoder_config, |
| std::unique_ptr<FecController> fec_controller) { |
| std::unique_ptr<FakeNetworkPipeTransportAdapter> transport_adapter; |
| if (!send_configs_.empty()) { |
| transport_adapter = std::make_unique<FakeNetworkPipeTransportAdapter>( |
| send_pipe_.get(), call_.get(), clock_, config.send_transport); |
| config.send_transport = transport_adapter.get(); |
| } |
| VideoSendStream* send_stream = call_->CreateVideoSendStream( |
| std::move(config), std::move(encoder_config), std::move(fec_controller)); |
| if (send_stream && transport_adapter) { |
| video_send_transport_adapters_[send_stream] = std::move(transport_adapter); |
| } |
| return send_stream; |
| } |
| |
| void DegradedCall::DestroyVideoSendStream(VideoSendStream* send_stream) { |
| call_->DestroyVideoSendStream(send_stream); |
| video_send_transport_adapters_.erase(send_stream); |
| } |
| |
| VideoReceiveStreamInterface* DegradedCall::CreateVideoReceiveStream( |
| VideoReceiveStreamInterface::Config configuration) { |
| return call_->CreateVideoReceiveStream(std::move(configuration)); |
| } |
| |
| void DegradedCall::DestroyVideoReceiveStream( |
| VideoReceiveStreamInterface* receive_stream) { |
| call_->DestroyVideoReceiveStream(receive_stream); |
| } |
| |
| FlexfecReceiveStream* DegradedCall::CreateFlexfecReceiveStream( |
| const FlexfecReceiveStream::Config config) { |
| return call_->CreateFlexfecReceiveStream(std::move(config)); |
| } |
| |
| void DegradedCall::DestroyFlexfecReceiveStream( |
| FlexfecReceiveStream* receive_stream) { |
| call_->DestroyFlexfecReceiveStream(receive_stream); |
| } |
| |
| void DegradedCall::AddAdaptationResource( |
| rtc::scoped_refptr<Resource> resource) { |
| call_->AddAdaptationResource(std::move(resource)); |
| } |
| |
| PacketReceiver* DegradedCall::Receiver() { |
| if (!receive_configs_.empty()) { |
| return this; |
| } |
| return call_->Receiver(); |
| } |
| |
| RtpTransportControllerSendInterface* |
| DegradedCall::GetTransportControllerSend() { |
| return call_->GetTransportControllerSend(); |
| } |
| |
| Call::Stats DegradedCall::GetStats() const { |
| return call_->GetStats(); |
| } |
| |
| const FieldTrialsView& DegradedCall::trials() const { |
| return call_->trials(); |
| } |
| |
| TaskQueueBase* DegradedCall::network_thread() const { |
| return call_->network_thread(); |
| } |
| |
| TaskQueueBase* DegradedCall::worker_thread() const { |
| return call_->worker_thread(); |
| } |
| |
| void DegradedCall::SignalChannelNetworkState(MediaType media, |
| NetworkState state) { |
| call_->SignalChannelNetworkState(media, state); |
| } |
| |
| void DegradedCall::OnAudioTransportOverheadChanged( |
| int transport_overhead_per_packet) { |
| call_->OnAudioTransportOverheadChanged(transport_overhead_per_packet); |
| } |
| |
| void DegradedCall::OnLocalSsrcUpdated(AudioReceiveStreamInterface& stream, |
| uint32_t local_ssrc) { |
| call_->OnLocalSsrcUpdated(stream, local_ssrc); |
| } |
| |
| void DegradedCall::OnLocalSsrcUpdated(VideoReceiveStreamInterface& stream, |
| uint32_t local_ssrc) { |
| call_->OnLocalSsrcUpdated(stream, local_ssrc); |
| } |
| |
| void DegradedCall::OnLocalSsrcUpdated(FlexfecReceiveStream& stream, |
| uint32_t local_ssrc) { |
| call_->OnLocalSsrcUpdated(stream, local_ssrc); |
| } |
| |
| void DegradedCall::OnUpdateSyncGroup(AudioReceiveStreamInterface& stream, |
| absl::string_view sync_group) { |
| call_->OnUpdateSyncGroup(stream, sync_group); |
| } |
| |
| void DegradedCall::OnSentPacket(const rtc::SentPacket& sent_packet) { |
| if (!send_configs_.empty()) { |
| // If we have a degraded send-transport, we have already notified call |
| // about the supposed network send time. Discard the actual network send |
| // time in order to properly fool the BWE. |
| return; |
| } |
| call_->OnSentPacket(sent_packet); |
| } |
| |
| void DegradedCall::DeliverRtpPacket( |
| MediaType media_type, |
| RtpPacketReceived packet, |
| OnUndemuxablePacketHandler undemuxable_packet_handler) { |
| RTC_DCHECK_RUN_ON(&received_packet_sequence_checker_); |
| receive_pipe_->DeliverRtpPacket(media_type, std::move(packet), |
| std::move(undemuxable_packet_handler)); |
| receive_pipe_->Process(); |
| } |
| |
| void DegradedCall::DeliverRtcpPacket(rtc::CopyOnWriteBuffer packet) { |
| RTC_DCHECK_RUN_ON(&received_packet_sequence_checker_); |
| receive_pipe_->DeliverRtcpPacket(std::move(packet)); |
| receive_pipe_->Process(); |
| } |
| |
| void DegradedCall::SetClientBitratePreferences( |
| const webrtc::BitrateSettings& preferences) { |
| call_->SetClientBitratePreferences(preferences); |
| } |
| |
| void DegradedCall::UpdateSendNetworkConfig() { |
| send_config_index_ = (send_config_index_ + 1) % send_configs_.size(); |
| send_simulated_network_->SetConfig(send_configs_[send_config_index_]); |
| call_->network_thread()->PostDelayedTask( |
| SafeTask(call_alive_, [this] { UpdateSendNetworkConfig(); }), |
| send_configs_[send_config_index_].duration); |
| } |
| |
| void DegradedCall::UpdateReceiveNetworkConfig() { |
| receive_config_index_ = (receive_config_index_ + 1) % receive_configs_.size(); |
| receive_simulated_network_->SetConfig( |
| receive_configs_[receive_config_index_]); |
| call_->network_thread()->PostDelayedTask( |
| SafeTask(call_alive_, [this] { UpdateReceiveNetworkConfig(); }), |
| receive_configs_[receive_config_index_].duration); |
| } |
| } // namespace webrtc |