| /* |
| * Copyright 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 "p2p/base/regathering_controller.h" |
| |
| namespace webrtc { |
| |
| using Config = BasicRegatheringController::Config; |
| |
| Config::Config(const absl::optional<rtc::IntervalRange>& |
| regather_on_all_networks_interval_range, |
| int regather_on_failed_networks_interval) |
| : regather_on_all_networks_interval_range( |
| regather_on_all_networks_interval_range), |
| regather_on_failed_networks_interval( |
| regather_on_failed_networks_interval) {} |
| |
| Config::Config(const Config& other) = default; |
| |
| Config::~Config() = default; |
| Config& Config::operator=(const Config& other) = default; |
| |
| BasicRegatheringController::BasicRegatheringController( |
| const Config& config, |
| cricket::IceTransportInternal* ice_transport, |
| rtc::Thread* thread) |
| : config_(config), |
| ice_transport_(ice_transport), |
| thread_(thread), |
| rand_(rtc::SystemTimeNanos()) { |
| RTC_DCHECK(ice_transport_); |
| RTC_DCHECK(thread_); |
| ice_transport_->SignalStateChanged.connect( |
| this, &BasicRegatheringController::OnIceTransportStateChanged); |
| ice_transport->SignalWritableState.connect( |
| this, &BasicRegatheringController::OnIceTransportWritableState); |
| ice_transport->SignalReceivingState.connect( |
| this, &BasicRegatheringController::OnIceTransportReceivingState); |
| ice_transport->SignalNetworkRouteChanged.connect( |
| this, &BasicRegatheringController::OnIceTransportNetworkRouteChanged); |
| } |
| |
| BasicRegatheringController::~BasicRegatheringController() = default; |
| |
| void BasicRegatheringController::Start() { |
| ScheduleRecurringRegatheringOnFailedNetworks(); |
| if (config_.regather_on_all_networks_interval_range) { |
| ScheduleRecurringRegatheringOnAllNetworks(); |
| } |
| } |
| |
| void BasicRegatheringController::SetConfig(const Config& config) { |
| bool need_cancel_on_all_networks = |
| has_recurring_schedule_on_all_networks_ && |
| (config_.regather_on_all_networks_interval_range != |
| config.regather_on_all_networks_interval_range); |
| bool need_reschedule_on_all_networks = |
| config.regather_on_all_networks_interval_range && |
| (config_.regather_on_all_networks_interval_range != |
| config.regather_on_all_networks_interval_range); |
| bool need_cancel_and_reschedule_on_failed_networks = |
| has_recurring_schedule_on_failed_networks_ && |
| (config_.regather_on_failed_networks_interval != |
| config.regather_on_failed_networks_interval); |
| config_ = config; |
| if (need_cancel_on_all_networks) { |
| CancelScheduledRecurringRegatheringOnAllNetworks(); |
| } |
| if (need_reschedule_on_all_networks) { |
| ScheduleRecurringRegatheringOnAllNetworks(); |
| } |
| if (need_cancel_and_reschedule_on_failed_networks) { |
| CancelScheduledRecurringRegatheringOnFailedNetworks(); |
| ScheduleRecurringRegatheringOnFailedNetworks(); |
| } |
| } |
| |
| void BasicRegatheringController::ScheduleRecurringRegatheringOnAllNetworks() { |
| RTC_DCHECK(config_.regather_on_all_networks_interval_range && |
| config_.regather_on_all_networks_interval_range.value().min() >= |
| 0); |
| int delay_ms = SampleRegatherAllNetworksInterval( |
| config_.regather_on_all_networks_interval_range.value()); |
| CancelScheduledRecurringRegatheringOnAllNetworks(); |
| has_recurring_schedule_on_all_networks_ = true; |
| invoker_for_all_networks_.AsyncInvokeDelayed<void>( |
| RTC_FROM_HERE, thread(), |
| rtc::Bind( |
| &BasicRegatheringController::RegatherOnAllNetworksIfDoneGathering, |
| this, true), |
| delay_ms); |
| } |
| |
| void BasicRegatheringController::RegatherOnAllNetworksIfDoneGathering( |
| bool repeated) { |
| // Only regather when the current session is in the CLEARED state (i.e., not |
| // running or stopped). It is only possible to enter this state when we gather |
| // continually, so there is an implicit check on continual gathering here. |
| if (allocator_session_ && allocator_session_->IsCleared()) { |
| allocator_session_->RegatherOnAllNetworks(); |
| } |
| if (repeated) { |
| ScheduleRecurringRegatheringOnAllNetworks(); |
| } |
| } |
| |
| void BasicRegatheringController:: |
| ScheduleRecurringRegatheringOnFailedNetworks() { |
| RTC_DCHECK(config_.regather_on_failed_networks_interval >= 0); |
| CancelScheduledRecurringRegatheringOnFailedNetworks(); |
| has_recurring_schedule_on_failed_networks_ = true; |
| invoker_for_failed_networks_.AsyncInvokeDelayed<void>( |
| RTC_FROM_HERE, thread(), |
| rtc::Bind( |
| &BasicRegatheringController::RegatherOnFailedNetworksIfDoneGathering, |
| this, true), |
| config_.regather_on_failed_networks_interval); |
| } |
| |
| void BasicRegatheringController::RegatherOnFailedNetworksIfDoneGathering( |
| bool repeated) { |
| // Only regather when the current session is in the CLEARED state (i.e., not |
| // running or stopped). It is only possible to enter this state when we gather |
| // continually, so there is an implicit check on continual gathering here. |
| if (allocator_session_ && allocator_session_->IsCleared()) { |
| allocator_session_->RegatherOnFailedNetworks(); |
| } |
| if (repeated) { |
| ScheduleRecurringRegatheringOnFailedNetworks(); |
| } |
| } |
| |
| void BasicRegatheringController:: |
| CancelScheduledRecurringRegatheringOnAllNetworks() { |
| invoker_for_all_networks_.Clear(); |
| has_recurring_schedule_on_all_networks_ = false; |
| } |
| |
| void BasicRegatheringController:: |
| CancelScheduledRecurringRegatheringOnFailedNetworks() { |
| invoker_for_failed_networks_.Clear(); |
| has_recurring_schedule_on_failed_networks_ = false; |
| } |
| |
| int BasicRegatheringController::SampleRegatherAllNetworksInterval( |
| const rtc::IntervalRange& range) { |
| return rand_.Rand(range.min(), range.max()); |
| } |
| |
| } // namespace webrtc |