diff options
Diffstat (limited to 'third_party/libwebrtc/pc/jsep_transport_controller.cc')
-rw-r--r-- | third_party/libwebrtc/pc/jsep_transport_controller.cc | 1435 |
1 files changed, 1435 insertions, 0 deletions
diff --git a/third_party/libwebrtc/pc/jsep_transport_controller.cc b/third_party/libwebrtc/pc/jsep_transport_controller.cc new file mode 100644 index 0000000000..3ea24d4503 --- /dev/null +++ b/third_party/libwebrtc/pc/jsep_transport_controller.cc @@ -0,0 +1,1435 @@ +/* + * Copyright 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 "pc/jsep_transport_controller.h" + +#include <stddef.h> + +#include <functional> +#include <memory> +#include <string> +#include <type_traits> +#include <utility> + +#include "absl/algorithm/container.h" +#include "api/dtls_transport_interface.h" +#include "api/rtp_parameters.h" +#include "api/sequence_checker.h" +#include "api/transport/enums.h" +#include "media/sctp/sctp_transport_internal.h" +#include "p2p/base/dtls_transport.h" +#include "p2p/base/ice_transport_internal.h" +#include "p2p/base/p2p_constants.h" +#include "p2p/base/port.h" +#include "rtc_base/checks.h" +#include "rtc_base/logging.h" +#include "rtc_base/thread.h" +#include "rtc_base/trace_event.h" + +using webrtc::SdpType; + +namespace webrtc { + +JsepTransportController::JsepTransportController( + rtc::Thread* network_thread, + cricket::PortAllocator* port_allocator, + AsyncDnsResolverFactoryInterface* async_dns_resolver_factory, + Config config) + : network_thread_(network_thread), + port_allocator_(port_allocator), + async_dns_resolver_factory_(async_dns_resolver_factory), + transports_( + [this](const std::string& mid, cricket::JsepTransport* transport) { + return OnTransportChanged(mid, transport); + }, + [this]() { + RTC_DCHECK_RUN_ON(network_thread_); + UpdateAggregateStates_n(); + }), + config_(config), + active_reset_srtp_params_(config.active_reset_srtp_params), + bundles_(config.bundle_policy) { + // The `transport_observer` is assumed to be non-null. + RTC_DCHECK(config_.transport_observer); + RTC_DCHECK(config_.rtcp_handler); + RTC_DCHECK(config_.ice_transport_factory); + RTC_DCHECK(config_.on_dtls_handshake_error_); + RTC_DCHECK(config_.field_trials); + if (port_allocator_) { + port_allocator_->SetIceTiebreaker(ice_tiebreaker_); + } +} + +JsepTransportController::~JsepTransportController() { + // Channel destructors may try to send packets, so this needs to happen on + // the network thread. + RTC_DCHECK_RUN_ON(network_thread_); + DestroyAllJsepTransports_n(); +} + +RTCError JsepTransportController::SetLocalDescription( + SdpType type, + const cricket::SessionDescription* description) { + TRACE_EVENT0("webrtc", "JsepTransportController::SetLocalDescription"); + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall( + [=] { return SetLocalDescription(type, description); }); + } + + RTC_DCHECK_RUN_ON(network_thread_); + if (!initial_offerer_.has_value()) { + initial_offerer_.emplace(type == SdpType::kOffer); + if (*initial_offerer_) { + SetIceRole_n(cricket::ICEROLE_CONTROLLING); + } else { + SetIceRole_n(cricket::ICEROLE_CONTROLLED); + } + } + return ApplyDescription_n(/*local=*/true, type, description); +} + +RTCError JsepTransportController::SetRemoteDescription( + SdpType type, + const cricket::SessionDescription* description) { + TRACE_EVENT0("webrtc", "JsepTransportController::SetRemoteDescription"); + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall( + [=] { return SetRemoteDescription(type, description); }); + } + + RTC_DCHECK_RUN_ON(network_thread_); + return ApplyDescription_n(/*local=*/false, type, description); +} + +RtpTransportInternal* JsepTransportController::GetRtpTransport( + absl::string_view mid) const { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->rtp_transport(); +} + +DataChannelTransportInterface* JsepTransportController::GetDataChannelTransport( + const std::string& mid) const { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->data_channel_transport(); +} + +cricket::DtlsTransportInternal* JsepTransportController::GetDtlsTransport( + const std::string& mid) { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->rtp_dtls_transport(); +} + +const cricket::DtlsTransportInternal* +JsepTransportController::GetRtcpDtlsTransport(const std::string& mid) const { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->rtcp_dtls_transport(); +} + +rtc::scoped_refptr<webrtc::DtlsTransport> +JsepTransportController::LookupDtlsTransportByMid(const std::string& mid) { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->RtpDtlsTransport(); +} + +rtc::scoped_refptr<SctpTransport> JsepTransportController::GetSctpTransport( + const std::string& mid) const { + RTC_DCHECK_RUN_ON(network_thread_); + auto jsep_transport = GetJsepTransportForMid(mid); + if (!jsep_transport) { + return nullptr; + } + return jsep_transport->SctpTransport(); +} + +void JsepTransportController::SetIceConfig(const cricket::IceConfig& config) { + RTC_DCHECK_RUN_ON(network_thread_); + ice_config_ = config; + for (auto& dtls : GetDtlsTransports()) { + dtls->ice_transport()->SetIceConfig(ice_config_); + } +} + +void JsepTransportController::SetNeedsIceRestartFlag() { + RTC_DCHECK_RUN_ON(network_thread_); + for (auto& transport : transports_.Transports()) { + transport->SetNeedsIceRestartFlag(); + } +} + +bool JsepTransportController::NeedsIceRestart( + const std::string& transport_name) const { + RTC_DCHECK_RUN_ON(network_thread_); + + const cricket::JsepTransport* transport = + GetJsepTransportByName(transport_name); + if (!transport) { + return false; + } + return transport->needs_ice_restart(); +} + +absl::optional<rtc::SSLRole> JsepTransportController::GetDtlsRole( + const std::string& mid) const { + // TODO(tommi): Remove this hop. Currently it's called from the signaling + // thread during negotiations, potentially multiple times. + // WebRtcSessionDescriptionFactory::InternalCreateAnswer is one example. + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall([&] { return GetDtlsRole(mid); }); + } + + RTC_DCHECK_RUN_ON(network_thread_); + + const cricket::JsepTransport* t = GetJsepTransportForMid(mid); + if (!t) { + return absl::optional<rtc::SSLRole>(); + } + return t->GetDtlsRole(); +} + +bool JsepTransportController::SetLocalCertificate( + const rtc::scoped_refptr<rtc::RTCCertificate>& certificate) { + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall( + [&] { return SetLocalCertificate(certificate); }); + } + + RTC_DCHECK_RUN_ON(network_thread_); + + // Can't change a certificate, or set a null certificate. + if (certificate_ || !certificate) { + return false; + } + certificate_ = certificate; + + // Set certificate for JsepTransport, which verifies it matches the + // fingerprint in SDP, and DTLS transport. + // Fallback from DTLS to SDES is not supported. + for (auto& transport : transports_.Transports()) { + transport->SetLocalCertificate(certificate_); + } + for (auto& dtls : GetDtlsTransports()) { + bool set_cert_success = dtls->SetLocalCertificate(certificate_); + RTC_DCHECK(set_cert_success); + } + return true; +} + +rtc::scoped_refptr<rtc::RTCCertificate> +JsepTransportController::GetLocalCertificate( + const std::string& transport_name) const { + RTC_DCHECK_RUN_ON(network_thread_); + + const cricket::JsepTransport* t = GetJsepTransportByName(transport_name); + if (!t) { + return nullptr; + } + return t->GetLocalCertificate(); +} + +std::unique_ptr<rtc::SSLCertChain> +JsepTransportController::GetRemoteSSLCertChain( + const std::string& transport_name) const { + RTC_DCHECK_RUN_ON(network_thread_); + + // Get the certificate from the RTP transport's DTLS handshake. Should be + // identical to the RTCP transport's, since they were given the same remote + // fingerprint. + auto jsep_transport = GetJsepTransportByName(transport_name); + if (!jsep_transport) { + return nullptr; + } + auto dtls = jsep_transport->rtp_dtls_transport(); + if (!dtls) { + return nullptr; + } + + return dtls->GetRemoteSSLCertChain(); +} + +void JsepTransportController::MaybeStartGathering() { + if (!network_thread_->IsCurrent()) { + network_thread_->BlockingCall([&] { MaybeStartGathering(); }); + return; + } + + for (auto& dtls : GetDtlsTransports()) { + dtls->ice_transport()->MaybeStartGathering(); + } +} + +RTCError JsepTransportController::AddRemoteCandidates( + const std::string& transport_name, + const cricket::Candidates& candidates) { + RTC_DCHECK_RUN_ON(network_thread_); + RTC_DCHECK(VerifyCandidates(candidates).ok()); + auto jsep_transport = GetJsepTransportByName(transport_name); + if (!jsep_transport) { + RTC_LOG(LS_WARNING) << "Not adding candidate because the JsepTransport " + "doesn't exist. Ignore it."; + return RTCError::OK(); + } + return jsep_transport->AddRemoteCandidates(candidates); +} + +RTCError JsepTransportController::RemoveRemoteCandidates( + const cricket::Candidates& candidates) { + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall( + [&] { return RemoveRemoteCandidates(candidates); }); + } + + RTC_DCHECK_RUN_ON(network_thread_); + + // Verify each candidate before passing down to the transport layer. + RTCError error = VerifyCandidates(candidates); + if (!error.ok()) { + return error; + } + + std::map<std::string, cricket::Candidates> candidates_by_transport_name; + for (const cricket::Candidate& cand : candidates) { + if (!cand.transport_name().empty()) { + candidates_by_transport_name[cand.transport_name()].push_back(cand); + } else { + RTC_LOG(LS_ERROR) << "Not removing candidate because it does not have a " + "transport name set: " + << cand.ToSensitiveString(); + } + } + + for (const auto& kv : candidates_by_transport_name) { + const std::string& transport_name = kv.first; + const cricket::Candidates& candidates = kv.second; + cricket::JsepTransport* jsep_transport = + GetJsepTransportByName(transport_name); + if (!jsep_transport) { + RTC_LOG(LS_WARNING) + << "Not removing candidate because the JsepTransport doesn't exist."; + continue; + } + for (const cricket::Candidate& candidate : candidates) { + cricket::DtlsTransportInternal* dtls = + candidate.component() == cricket::ICE_CANDIDATE_COMPONENT_RTP + ? jsep_transport->rtp_dtls_transport() + : jsep_transport->rtcp_dtls_transport(); + if (dtls) { + dtls->ice_transport()->RemoveRemoteCandidate(candidate); + } + } + } + return RTCError::OK(); +} + +bool JsepTransportController::GetStats(const std::string& transport_name, + cricket::TransportStats* stats) { + RTC_DCHECK_RUN_ON(network_thread_); + + cricket::JsepTransport* transport = GetJsepTransportByName(transport_name); + if (!transport) { + return false; + } + return transport->GetStats(stats); +} + +void JsepTransportController::SetActiveResetSrtpParams( + bool active_reset_srtp_params) { + if (!network_thread_->IsCurrent()) { + network_thread_->BlockingCall( + [=] { SetActiveResetSrtpParams(active_reset_srtp_params); }); + return; + } + RTC_DCHECK_RUN_ON(network_thread_); + RTC_LOG(LS_INFO) + << "Updating the active_reset_srtp_params for JsepTransportController: " + << active_reset_srtp_params; + active_reset_srtp_params_ = active_reset_srtp_params; + for (auto& transport : transports_.Transports()) { + transport->SetActiveResetSrtpParams(active_reset_srtp_params); + } +} + +RTCError JsepTransportController::RollbackTransports() { + if (!network_thread_->IsCurrent()) { + return network_thread_->BlockingCall([=] { return RollbackTransports(); }); + } + RTC_DCHECK_RUN_ON(network_thread_); + bundles_.Rollback(); + if (!transports_.RollbackTransports()) { + LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR, + "Failed to roll back transport state."); + } + return RTCError::OK(); +} + +rtc::scoped_refptr<webrtc::IceTransportInterface> +JsepTransportController::CreateIceTransport(const std::string& transport_name, + bool rtcp) { + int component = rtcp ? cricket::ICE_CANDIDATE_COMPONENT_RTCP + : cricket::ICE_CANDIDATE_COMPONENT_RTP; + + IceTransportInit init; + init.set_port_allocator(port_allocator_); + init.set_async_dns_resolver_factory(async_dns_resolver_factory_); + init.set_event_log(config_.event_log); + init.set_field_trials(config_.field_trials); + auto transport = config_.ice_transport_factory->CreateIceTransport( + transport_name, component, std::move(init)); + RTC_DCHECK(transport); + transport->internal()->SetIceRole(ice_role_); + transport->internal()->SetIceTiebreaker(ice_tiebreaker_); + transport->internal()->SetIceConfig(ice_config_); + return transport; +} + +std::unique_ptr<cricket::DtlsTransportInternal> +JsepTransportController::CreateDtlsTransport( + const cricket::ContentInfo& content_info, + cricket::IceTransportInternal* ice) { + RTC_DCHECK_RUN_ON(network_thread_); + + std::unique_ptr<cricket::DtlsTransportInternal> dtls; + + if (config_.dtls_transport_factory) { + dtls = config_.dtls_transport_factory->CreateDtlsTransport( + ice, config_.crypto_options, config_.ssl_max_version); + } else { + dtls = std::make_unique<cricket::DtlsTransport>(ice, config_.crypto_options, + config_.event_log, + config_.ssl_max_version); + } + + RTC_DCHECK(dtls); + RTC_DCHECK_EQ(ice, dtls->ice_transport()); + + if (certificate_) { + bool set_cert_success = dtls->SetLocalCertificate(certificate_); + RTC_DCHECK(set_cert_success); + } + + // Connect to signals offered by the DTLS and ICE transport. + dtls->SignalWritableState.connect( + this, &JsepTransportController::OnTransportWritableState_n); + dtls->SignalReceivingState.connect( + this, &JsepTransportController::OnTransportReceivingState_n); + dtls->ice_transport()->SignalGatheringState.connect( + this, &JsepTransportController::OnTransportGatheringState_n); + dtls->ice_transport()->SignalCandidateGathered.connect( + this, &JsepTransportController::OnTransportCandidateGathered_n); + dtls->ice_transport()->SignalCandidateError.connect( + this, &JsepTransportController::OnTransportCandidateError_n); + dtls->ice_transport()->SignalCandidatesRemoved.connect( + this, &JsepTransportController::OnTransportCandidatesRemoved_n); + dtls->ice_transport()->SignalRoleConflict.connect( + this, &JsepTransportController::OnTransportRoleConflict_n); + dtls->ice_transport()->SignalStateChanged.connect( + this, &JsepTransportController::OnTransportStateChanged_n); + dtls->ice_transport()->SignalIceTransportStateChanged.connect( + this, &JsepTransportController::OnTransportStateChanged_n); + dtls->ice_transport()->SignalCandidatePairChanged.connect( + this, &JsepTransportController::OnTransportCandidatePairChanged_n); + + dtls->SubscribeDtlsHandshakeError( + [this](rtc::SSLHandshakeError error) { OnDtlsHandshakeError(error); }); + return dtls; +} + +std::unique_ptr<webrtc::RtpTransport> +JsepTransportController::CreateUnencryptedRtpTransport( + const std::string& transport_name, + rtc::PacketTransportInternal* rtp_packet_transport, + rtc::PacketTransportInternal* rtcp_packet_transport) { + RTC_DCHECK_RUN_ON(network_thread_); + auto unencrypted_rtp_transport = + std::make_unique<RtpTransport>(rtcp_packet_transport == nullptr); + unencrypted_rtp_transport->SetRtpPacketTransport(rtp_packet_transport); + if (rtcp_packet_transport) { + unencrypted_rtp_transport->SetRtcpPacketTransport(rtcp_packet_transport); + } + return unencrypted_rtp_transport; +} + +std::unique_ptr<webrtc::SrtpTransport> +JsepTransportController::CreateSdesTransport( + const std::string& transport_name, + cricket::DtlsTransportInternal* rtp_dtls_transport, + cricket::DtlsTransportInternal* rtcp_dtls_transport) { + RTC_DCHECK_RUN_ON(network_thread_); + auto srtp_transport = std::make_unique<webrtc::SrtpTransport>( + rtcp_dtls_transport == nullptr, *config_.field_trials); + RTC_DCHECK(rtp_dtls_transport); + srtp_transport->SetRtpPacketTransport(rtp_dtls_transport); + if (rtcp_dtls_transport) { + srtp_transport->SetRtcpPacketTransport(rtcp_dtls_transport); + } + if (config_.enable_external_auth) { + srtp_transport->EnableExternalAuth(); + } + return srtp_transport; +} + +std::unique_ptr<webrtc::DtlsSrtpTransport> +JsepTransportController::CreateDtlsSrtpTransport( + const std::string& transport_name, + cricket::DtlsTransportInternal* rtp_dtls_transport, + cricket::DtlsTransportInternal* rtcp_dtls_transport) { + RTC_DCHECK_RUN_ON(network_thread_); + auto dtls_srtp_transport = std::make_unique<webrtc::DtlsSrtpTransport>( + rtcp_dtls_transport == nullptr, *config_.field_trials); + if (config_.enable_external_auth) { + dtls_srtp_transport->EnableExternalAuth(); + } + + dtls_srtp_transport->SetDtlsTransports(rtp_dtls_transport, + rtcp_dtls_transport); + dtls_srtp_transport->SetActiveResetSrtpParams(active_reset_srtp_params_); + // Capturing this in the callback because JsepTransportController will always + // outlive the DtlsSrtpTransport. + dtls_srtp_transport->SetOnDtlsStateChange([this]() { + RTC_DCHECK_RUN_ON(this->network_thread_); + this->UpdateAggregateStates_n(); + }); + return dtls_srtp_transport; +} + +std::vector<cricket::DtlsTransportInternal*> +JsepTransportController::GetDtlsTransports() { + RTC_DCHECK_RUN_ON(network_thread_); + std::vector<cricket::DtlsTransportInternal*> dtls_transports; + for (auto jsep_transport : transports_.Transports()) { + RTC_DCHECK(jsep_transport); + if (jsep_transport->rtp_dtls_transport()) { + dtls_transports.push_back(jsep_transport->rtp_dtls_transport()); + } + + if (jsep_transport->rtcp_dtls_transport()) { + dtls_transports.push_back(jsep_transport->rtcp_dtls_transport()); + } + } + return dtls_transports; +} + +std::vector<cricket::DtlsTransportInternal*> +JsepTransportController::GetActiveDtlsTransports() { + RTC_DCHECK_RUN_ON(network_thread_); + std::vector<cricket::DtlsTransportInternal*> dtls_transports; + for (auto jsep_transport : transports_.ActiveTransports()) { + RTC_DCHECK(jsep_transport); + if (jsep_transport->rtp_dtls_transport()) { + dtls_transports.push_back(jsep_transport->rtp_dtls_transport()); + } + + if (jsep_transport->rtcp_dtls_transport()) { + dtls_transports.push_back(jsep_transport->rtcp_dtls_transport()); + } + } + return dtls_transports; +} + +RTCError JsepTransportController::ApplyDescription_n( + bool local, + SdpType type, + const cricket::SessionDescription* description) { + TRACE_EVENT0("webrtc", "JsepTransportController::ApplyDescription_n"); + RTC_DCHECK(description); + + if (local) { + local_desc_ = description; + } else { + remote_desc_ = description; + } + + RTCError error; + error = ValidateAndMaybeUpdateBundleGroups(local, type, description); + if (!error.ok()) { + return error; + } + + std::map<const cricket::ContentGroup*, std::vector<int>> + merged_encrypted_extension_ids_by_bundle; + if (!bundles_.bundle_groups().empty()) { + merged_encrypted_extension_ids_by_bundle = + MergeEncryptedHeaderExtensionIdsForBundles(description); + } + + for (const cricket::ContentInfo& content_info : description->contents()) { + // Don't create transports for rejected m-lines and bundled m-lines. + if (content_info.rejected || + !bundles_.IsFirstMidInGroup(content_info.name)) { + continue; + } + error = MaybeCreateJsepTransport(local, content_info, *description); + if (!error.ok()) { + return error; + } + } + + RTC_DCHECK(description->contents().size() == + description->transport_infos().size()); + for (size_t i = 0; i < description->contents().size(); ++i) { + const cricket::ContentInfo& content_info = description->contents()[i]; + const cricket::TransportInfo& transport_info = + description->transport_infos()[i]; + + if (content_info.rejected) { + // This may cause groups to be removed from |bundles_.bundle_groups()|. + HandleRejectedContent(content_info); + continue; + } + + const cricket::ContentGroup* established_bundle_group = + bundles_.LookupGroupByMid(content_info.name); + + // For bundle members that are not BUNDLE-tagged (not first in the group), + // configure their transport to be the same as the BUNDLE-tagged transport. + if (established_bundle_group && + content_info.name != *established_bundle_group->FirstContentName()) { + if (!HandleBundledContent(content_info, *established_bundle_group)) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "Failed to process the bundled m= section with " + "mid='" + + content_info.name + "'."); + } + continue; + } + + error = ValidateContent(content_info); + if (!error.ok()) { + return error; + } + + std::vector<int> extension_ids; + // Is BUNDLE-tagged (first in the group)? + if (established_bundle_group && + content_info.name == *established_bundle_group->FirstContentName()) { + auto it = merged_encrypted_extension_ids_by_bundle.find( + established_bundle_group); + RTC_DCHECK(it != merged_encrypted_extension_ids_by_bundle.end()); + extension_ids = it->second; + } else { + extension_ids = GetEncryptedHeaderExtensionIds(content_info); + } + + int rtp_abs_sendtime_extn_id = + GetRtpAbsSendTimeHeaderExtensionId(content_info); + + cricket::JsepTransport* transport = + GetJsepTransportForMid(content_info.name); + RTC_DCHECK(transport); + + SetIceRole_n(DetermineIceRole(transport, transport_info, type, local)); + + cricket::JsepTransportDescription jsep_description = + CreateJsepTransportDescription(content_info, transport_info, + extension_ids, rtp_abs_sendtime_extn_id); + if (local) { + error = + transport->SetLocalJsepTransportDescription(jsep_description, type); + } else { + error = + transport->SetRemoteJsepTransportDescription(jsep_description, type); + } + + if (!error.ok()) { + LOG_AND_RETURN_ERROR( + RTCErrorType::INVALID_PARAMETER, + "Failed to apply the description for m= section with mid='" + + content_info.name + "': " + error.message()); + } + } + if (type == SdpType::kAnswer) { + transports_.CommitTransports(); + bundles_.Commit(); + } + return RTCError::OK(); +} + +RTCError JsepTransportController::ValidateAndMaybeUpdateBundleGroups( + bool local, + SdpType type, + const cricket::SessionDescription* description) { + RTC_DCHECK(description); + + std::vector<const cricket::ContentGroup*> new_bundle_groups = + description->GetGroupsByName(cricket::GROUP_TYPE_BUNDLE); + // Verify `new_bundle_groups`. + std::map<std::string, const cricket::ContentGroup*> new_bundle_groups_by_mid; + for (const cricket::ContentGroup* new_bundle_group : new_bundle_groups) { + for (const std::string& content_name : new_bundle_group->content_names()) { + // The BUNDLE group must not contain a MID that is a member of a different + // BUNDLE group, or that contains the same MID multiple times. + if (new_bundle_groups_by_mid.find(content_name) != + new_bundle_groups_by_mid.end()) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "A BUNDLE group contains a MID='" + content_name + + "' that is already in a BUNDLE group."); + } + new_bundle_groups_by_mid.insert( + std::make_pair(content_name, new_bundle_group)); + // The BUNDLE group must not contain a MID that no m= section has. + if (!description->GetContentByName(content_name)) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "A BUNDLE group contains a MID='" + content_name + + "' matching no m= section."); + } + } + } + + if (type == SdpType::kOffer) { + // For an offer, we need to verify that there is not a conflicting mapping + // between existing and new bundle groups. For example, if the existing + // groups are [[1,2],[3,4]] and new are [[1,3],[2,4]] or [[1,2,3,4]], or + // vice versa. Switching things around like this requires a separate offer + // that removes the relevant sections from their group, as per RFC 8843, + // section 7.5.2. + std::map<const cricket::ContentGroup*, const cricket::ContentGroup*> + new_bundle_groups_by_existing_bundle_groups; + std::map<const cricket::ContentGroup*, const cricket::ContentGroup*> + existing_bundle_groups_by_new_bundle_groups; + for (const cricket::ContentGroup* new_bundle_group : new_bundle_groups) { + for (const std::string& mid : new_bundle_group->content_names()) { + cricket::ContentGroup* existing_bundle_group = + bundles_.LookupGroupByMid(mid); + if (!existing_bundle_group) { + continue; + } + auto it = new_bundle_groups_by_existing_bundle_groups.find( + existing_bundle_group); + if (it != new_bundle_groups_by_existing_bundle_groups.end() && + it->second != new_bundle_group) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "MID " + mid + " in the offer has changed group."); + } + new_bundle_groups_by_existing_bundle_groups.insert( + std::make_pair(existing_bundle_group, new_bundle_group)); + it = existing_bundle_groups_by_new_bundle_groups.find(new_bundle_group); + if (it != existing_bundle_groups_by_new_bundle_groups.end() && + it->second != existing_bundle_group) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "MID " + mid + " in the offer has changed group."); + } + existing_bundle_groups_by_new_bundle_groups.insert( + std::make_pair(new_bundle_group, existing_bundle_group)); + } + } + } else if (type == SdpType::kAnswer) { + std::vector<const cricket::ContentGroup*> offered_bundle_groups = + local ? remote_desc_->GetGroupsByName(cricket::GROUP_TYPE_BUNDLE) + : local_desc_->GetGroupsByName(cricket::GROUP_TYPE_BUNDLE); + + std::map<std::string, const cricket::ContentGroup*> + offered_bundle_groups_by_mid; + for (const cricket::ContentGroup* offered_bundle_group : + offered_bundle_groups) { + for (const std::string& content_name : + offered_bundle_group->content_names()) { + offered_bundle_groups_by_mid[content_name] = offered_bundle_group; + } + } + + std::map<const cricket::ContentGroup*, const cricket::ContentGroup*> + new_bundle_groups_by_offered_bundle_groups; + for (const cricket::ContentGroup* new_bundle_group : new_bundle_groups) { + if (!new_bundle_group->FirstContentName()) { + // Empty groups could be a subset of any group. + continue; + } + // The group in the answer (new_bundle_group) must have a corresponding + // group in the offer (original_group), because the answer groups may only + // be subsets of the offer groups. + auto it = offered_bundle_groups_by_mid.find( + *new_bundle_group->FirstContentName()); + if (it == offered_bundle_groups_by_mid.end()) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "A BUNDLE group was added in the answer that did not " + "exist in the offer."); + } + const cricket::ContentGroup* offered_bundle_group = it->second; + if (new_bundle_groups_by_offered_bundle_groups.find( + offered_bundle_group) != + new_bundle_groups_by_offered_bundle_groups.end()) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "A MID in the answer has changed group."); + } + new_bundle_groups_by_offered_bundle_groups.insert( + std::make_pair(offered_bundle_group, new_bundle_group)); + for (const std::string& content_name : + new_bundle_group->content_names()) { + it = offered_bundle_groups_by_mid.find(content_name); + // The BUNDLE group in answer should be a subset of offered group. + if (it == offered_bundle_groups_by_mid.end() || + it->second != offered_bundle_group) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "A BUNDLE group in answer contains a MID='" + + content_name + + "' that was not in the offered group."); + } + } + } + + for (const auto& bundle_group : bundles_.bundle_groups()) { + for (const std::string& content_name : bundle_group->content_names()) { + // An answer that removes m= sections from pre-negotiated BUNDLE group + // without rejecting it, is invalid. + auto it = new_bundle_groups_by_mid.find(content_name); + if (it == new_bundle_groups_by_mid.end()) { + auto* content_info = description->GetContentByName(content_name); + if (!content_info || !content_info->rejected) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "Answer cannot remove m= section with mid='" + + content_name + + "' from already-established BUNDLE group."); + } + } + } + } + } + + if (config_.bundle_policy == + PeerConnectionInterface::kBundlePolicyMaxBundle && + !description->HasGroup(cricket::GROUP_TYPE_BUNDLE)) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "max-bundle is used but no bundle group found."); + } + + bundles_.Update(description, type); + + for (const auto& bundle_group : bundles_.bundle_groups()) { + if (!bundle_group->FirstContentName()) + continue; + + // The first MID in a BUNDLE group is BUNDLE-tagged. + auto bundled_content = + description->GetContentByName(*bundle_group->FirstContentName()); + if (!bundled_content) { + return RTCError( + RTCErrorType::INVALID_PARAMETER, + "An m= section associated with the BUNDLE-tag doesn't exist."); + } + + // If the `bundled_content` is rejected, other contents in the bundle group + // must also be rejected. + if (bundled_content->rejected) { + for (const auto& content_name : bundle_group->content_names()) { + auto other_content = description->GetContentByName(content_name); + if (!other_content->rejected) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "The m= section with mid='" + content_name + + "' should be rejected."); + } + } + } + } + return RTCError::OK(); +} + +RTCError JsepTransportController::ValidateContent( + const cricket::ContentInfo& content_info) { + if (config_.rtcp_mux_policy == + PeerConnectionInterface::kRtcpMuxPolicyRequire && + content_info.type == cricket::MediaProtocolType::kRtp && + !content_info.media_description()->rtcp_mux()) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "The m= section with mid='" + content_info.name + + "' is invalid. RTCP-MUX is not " + "enabled when it is required."); + } + return RTCError::OK(); +} + +void JsepTransportController::HandleRejectedContent( + const cricket::ContentInfo& content_info) { + // If the content is rejected, let the + // BaseChannel/SctpTransport change the RtpTransport/DtlsTransport first, + // then destroy the cricket::JsepTransport. + cricket::ContentGroup* bundle_group = + bundles_.LookupGroupByMid(content_info.name); + if (bundle_group && !bundle_group->content_names().empty() && + content_info.name == *bundle_group->FirstContentName()) { + // Rejecting a BUNDLE group's first mid means we are rejecting the entire + // group. + for (const auto& content_name : bundle_group->content_names()) { + transports_.RemoveTransportForMid(content_name); + } + // Delete the BUNDLE group. + bundles_.DeleteGroup(bundle_group); + } else { + transports_.RemoveTransportForMid(content_info.name); + if (bundle_group) { + // Remove the rejected content from the `bundle_group`. + bundles_.DeleteMid(bundle_group, content_info.name); + } + } +} + +bool JsepTransportController::HandleBundledContent( + const cricket::ContentInfo& content_info, + const cricket::ContentGroup& bundle_group) { + TRACE_EVENT0("webrtc", "JsepTransportController::HandleBundledContent"); + RTC_DCHECK(bundle_group.FirstContentName()); + auto jsep_transport = + GetJsepTransportByName(*bundle_group.FirstContentName()); + RTC_DCHECK(jsep_transport); + // If the content is bundled, let the + // BaseChannel/SctpTransport change the RtpTransport/DtlsTransport first, + // then destroy the cricket::JsepTransport. + // TODO(bugs.webrtc.org/9719) For media transport this is far from ideal, + // because it means that we first create media transport and start + // connecting it, and then we destroy it. We will need to address it before + // video path is enabled. + return transports_.SetTransportForMid(content_info.name, jsep_transport); +} + +cricket::JsepTransportDescription +JsepTransportController::CreateJsepTransportDescription( + const cricket::ContentInfo& content_info, + const cricket::TransportInfo& transport_info, + const std::vector<int>& encrypted_extension_ids, + int rtp_abs_sendtime_extn_id) { + TRACE_EVENT0("webrtc", + "JsepTransportController::CreateJsepTransportDescription"); + const cricket::MediaContentDescription* content_desc = + content_info.media_description(); + RTC_DCHECK(content_desc); + bool rtcp_mux_enabled = content_info.type == cricket::MediaProtocolType::kSctp + ? true + : content_desc->rtcp_mux(); + + return cricket::JsepTransportDescription( + rtcp_mux_enabled, content_desc->cryptos(), encrypted_extension_ids, + rtp_abs_sendtime_extn_id, transport_info.description); +} + +std::vector<int> JsepTransportController::GetEncryptedHeaderExtensionIds( + const cricket::ContentInfo& content_info) { + const cricket::MediaContentDescription* content_desc = + content_info.media_description(); + + if (!config_.crypto_options.srtp.enable_encrypted_rtp_header_extensions) { + return std::vector<int>(); + } + + std::vector<int> encrypted_header_extension_ids; + for (const auto& extension : content_desc->rtp_header_extensions()) { + if (!extension.encrypt) { + continue; + } + if (!absl::c_linear_search(encrypted_header_extension_ids, extension.id)) { + encrypted_header_extension_ids.push_back(extension.id); + } + } + return encrypted_header_extension_ids; +} + +std::map<const cricket::ContentGroup*, std::vector<int>> +JsepTransportController::MergeEncryptedHeaderExtensionIdsForBundles( + const cricket::SessionDescription* description) { + RTC_DCHECK(description); + RTC_DCHECK(!bundles_.bundle_groups().empty()); + std::map<const cricket::ContentGroup*, std::vector<int>> + merged_encrypted_extension_ids_by_bundle; + // Union the encrypted header IDs in the group when bundle is enabled. + for (const cricket::ContentInfo& content_info : description->contents()) { + auto group = bundles_.LookupGroupByMid(content_info.name); + if (!group) + continue; + // Get or create list of IDs for the BUNDLE group. + std::vector<int>& merged_ids = + merged_encrypted_extension_ids_by_bundle[group]; + // Add IDs not already in the list. + std::vector<int> extension_ids = + GetEncryptedHeaderExtensionIds(content_info); + for (int id : extension_ids) { + if (!absl::c_linear_search(merged_ids, id)) { + merged_ids.push_back(id); + } + } + } + return merged_encrypted_extension_ids_by_bundle; +} + +int JsepTransportController::GetRtpAbsSendTimeHeaderExtensionId( + const cricket::ContentInfo& content_info) { + if (!config_.enable_external_auth) { + return -1; + } + + const cricket::MediaContentDescription* content_desc = + content_info.media_description(); + + const webrtc::RtpExtension* send_time_extension = + webrtc::RtpExtension::FindHeaderExtensionByUri( + content_desc->rtp_header_extensions(), + webrtc::RtpExtension::kAbsSendTimeUri, + config_.crypto_options.srtp.enable_encrypted_rtp_header_extensions + ? webrtc::RtpExtension::kPreferEncryptedExtension + : webrtc::RtpExtension::kDiscardEncryptedExtension); + return send_time_extension ? send_time_extension->id : -1; +} + +const cricket::JsepTransport* JsepTransportController::GetJsepTransportForMid( + const std::string& mid) const { + return transports_.GetTransportForMid(mid); +} + +cricket::JsepTransport* JsepTransportController::GetJsepTransportForMid( + const std::string& mid) { + return transports_.GetTransportForMid(mid); +} +const cricket::JsepTransport* JsepTransportController::GetJsepTransportForMid( + absl::string_view mid) const { + return transports_.GetTransportForMid(mid); +} + +cricket::JsepTransport* JsepTransportController::GetJsepTransportForMid( + absl::string_view mid) { + return transports_.GetTransportForMid(mid); +} + +const cricket::JsepTransport* JsepTransportController::GetJsepTransportByName( + const std::string& transport_name) const { + return transports_.GetTransportByName(transport_name); +} + +cricket::JsepTransport* JsepTransportController::GetJsepTransportByName( + const std::string& transport_name) { + return transports_.GetTransportByName(transport_name); +} + +RTCError JsepTransportController::MaybeCreateJsepTransport( + bool local, + const cricket::ContentInfo& content_info, + const cricket::SessionDescription& description) { + cricket::JsepTransport* transport = GetJsepTransportByName(content_info.name); + if (transport) { + return RTCError::OK(); + } + const cricket::MediaContentDescription* content_desc = + content_info.media_description(); + if (certificate_ && !content_desc->cryptos().empty()) { + return RTCError(RTCErrorType::INVALID_PARAMETER, + "SDES and DTLS-SRTP cannot be enabled at the same time."); + } + + rtc::scoped_refptr<webrtc::IceTransportInterface> ice = + CreateIceTransport(content_info.name, /*rtcp=*/false); + + std::unique_ptr<cricket::DtlsTransportInternal> rtp_dtls_transport = + CreateDtlsTransport(content_info, ice->internal()); + + std::unique_ptr<cricket::DtlsTransportInternal> rtcp_dtls_transport; + std::unique_ptr<RtpTransport> unencrypted_rtp_transport; + std::unique_ptr<SrtpTransport> sdes_transport; + std::unique_ptr<DtlsSrtpTransport> dtls_srtp_transport; + + rtc::scoped_refptr<webrtc::IceTransportInterface> rtcp_ice; + if (config_.rtcp_mux_policy != + PeerConnectionInterface::kRtcpMuxPolicyRequire && + content_info.type == cricket::MediaProtocolType::kRtp) { + rtcp_ice = CreateIceTransport(content_info.name, /*rtcp=*/true); + rtcp_dtls_transport = + CreateDtlsTransport(content_info, rtcp_ice->internal()); + } + + if (config_.disable_encryption) { + RTC_LOG(LS_INFO) + << "Creating UnencryptedRtpTransport, becayse encryption is disabled."; + unencrypted_rtp_transport = CreateUnencryptedRtpTransport( + content_info.name, rtp_dtls_transport.get(), rtcp_dtls_transport.get()); + } else if (!content_desc->cryptos().empty()) { + sdes_transport = CreateSdesTransport( + content_info.name, rtp_dtls_transport.get(), rtcp_dtls_transport.get()); + RTC_LOG(LS_INFO) << "Creating SdesTransport."; + } else { + RTC_LOG(LS_INFO) << "Creating DtlsSrtpTransport."; + dtls_srtp_transport = CreateDtlsSrtpTransport( + content_info.name, rtp_dtls_transport.get(), rtcp_dtls_transport.get()); + } + + std::unique_ptr<cricket::SctpTransportInternal> sctp_transport; + if (config_.sctp_factory) { + sctp_transport = + config_.sctp_factory->CreateSctpTransport(rtp_dtls_transport.get()); + } + + std::unique_ptr<cricket::JsepTransport> jsep_transport = + std::make_unique<cricket::JsepTransport>( + content_info.name, certificate_, std::move(ice), std::move(rtcp_ice), + std::move(unencrypted_rtp_transport), std::move(sdes_transport), + std::move(dtls_srtp_transport), std::move(rtp_dtls_transport), + std::move(rtcp_dtls_transport), std::move(sctp_transport), [&]() { + RTC_DCHECK_RUN_ON(network_thread_); + UpdateAggregateStates_n(); + }); + + jsep_transport->rtp_transport()->SignalRtcpPacketReceived.connect( + this, &JsepTransportController::OnRtcpPacketReceived_n); + + transports_.RegisterTransport(content_info.name, std::move(jsep_transport)); + UpdateAggregateStates_n(); + return RTCError::OK(); +} + +void JsepTransportController::DestroyAllJsepTransports_n() { + transports_.DestroyAllTransports(); +} + +void JsepTransportController::SetIceRole_n(cricket::IceRole ice_role) { + ice_role_ = ice_role; + auto dtls_transports = GetDtlsTransports(); + for (auto& dtls : dtls_transports) { + dtls->ice_transport()->SetIceRole(ice_role_); + } +} + +cricket::IceRole JsepTransportController::DetermineIceRole( + cricket::JsepTransport* jsep_transport, + const cricket::TransportInfo& transport_info, + SdpType type, + bool local) { + cricket::IceRole ice_role = ice_role_; + auto tdesc = transport_info.description; + if (local) { + // The initial offer side may use ICE Lite, in which case, per RFC5245 + // Section 5.1.1, the answer side should take the controlling role if it is + // in the full ICE mode. + // + // When both sides use ICE Lite, the initial offer side must take the + // controlling role, and this is the default logic implemented in + // SetLocalDescription in JsepTransportController. + if (jsep_transport->remote_description() && + jsep_transport->remote_description()->transport_desc.ice_mode == + cricket::ICEMODE_LITE && + ice_role_ == cricket::ICEROLE_CONTROLLED && + tdesc.ice_mode == cricket::ICEMODE_FULL) { + ice_role = cricket::ICEROLE_CONTROLLING; + } + } else { + // If our role is cricket::ICEROLE_CONTROLLED and the remote endpoint + // supports only ice_lite, this local endpoint should take the CONTROLLING + // role. + // TODO(deadbeef): This is a session-level attribute, so it really shouldn't + // be in a TransportDescription in the first place... + if (ice_role_ == cricket::ICEROLE_CONTROLLED && + tdesc.ice_mode == cricket::ICEMODE_LITE) { + ice_role = cricket::ICEROLE_CONTROLLING; + } + + // If we use ICE Lite and the remote endpoint uses the full implementation + // of ICE, the local endpoint must take the controlled role, and the other + // side must be the controlling role. + if (jsep_transport->local_description() && + jsep_transport->local_description()->transport_desc.ice_mode == + cricket::ICEMODE_LITE && + ice_role_ == cricket::ICEROLE_CONTROLLING && + tdesc.ice_mode == cricket::ICEMODE_FULL) { + ice_role = cricket::ICEROLE_CONTROLLED; + } + } + + return ice_role; +} + +void JsepTransportController::OnTransportWritableState_n( + rtc::PacketTransportInternal* transport) { + RTC_LOG(LS_INFO) << " Transport " << transport->transport_name() + << " writability changed to " << transport->writable() + << "."; + UpdateAggregateStates_n(); +} + +void JsepTransportController::OnTransportReceivingState_n( + rtc::PacketTransportInternal* transport) { + UpdateAggregateStates_n(); +} + +void JsepTransportController::OnTransportGatheringState_n( + cricket::IceTransportInternal* transport) { + UpdateAggregateStates_n(); +} + +void JsepTransportController::OnTransportCandidateGathered_n( + cricket::IceTransportInternal* transport, + const cricket::Candidate& candidate) { + // We should never signal peer-reflexive candidates. + if (candidate.type() == cricket::PRFLX_PORT_TYPE) { + RTC_DCHECK_NOTREACHED(); + return; + } + + signal_ice_candidates_gathered_.Send( + transport->transport_name(), std::vector<cricket::Candidate>{candidate}); +} + +void JsepTransportController::OnTransportCandidateError_n( + cricket::IceTransportInternal* transport, + const cricket::IceCandidateErrorEvent& event) { + signal_ice_candidate_error_.Send(event); +} +void JsepTransportController::OnTransportCandidatesRemoved_n( + cricket::IceTransportInternal* transport, + const cricket::Candidates& candidates) { + signal_ice_candidates_removed_.Send(candidates); +} +void JsepTransportController::OnTransportCandidatePairChanged_n( + const cricket::CandidatePairChangeEvent& event) { + signal_ice_candidate_pair_changed_.Send(event); +} + +void JsepTransportController::OnTransportRoleConflict_n( + cricket::IceTransportInternal* transport) { + // Note: since the role conflict is handled entirely on the network thread, + // we don't need to worry about role conflicts occurring on two ports at + // once. The first one encountered should immediately reverse the role. + cricket::IceRole reversed_role = (ice_role_ == cricket::ICEROLE_CONTROLLING) + ? cricket::ICEROLE_CONTROLLED + : cricket::ICEROLE_CONTROLLING; + RTC_LOG(LS_INFO) << "Got role conflict; switching to " + << (reversed_role == cricket::ICEROLE_CONTROLLING + ? "controlling" + : "controlled") + << " role."; + SetIceRole_n(reversed_role); +} + +void JsepTransportController::OnTransportStateChanged_n( + cricket::IceTransportInternal* transport) { + RTC_LOG(LS_INFO) << transport->transport_name() << " Transport " + << transport->component() + << " state changed. Check if state is complete."; + UpdateAggregateStates_n(); +} + +void JsepTransportController::UpdateAggregateStates_n() { + TRACE_EVENT0("webrtc", "JsepTransportController::UpdateAggregateStates_n"); + auto dtls_transports = GetActiveDtlsTransports(); + cricket::IceConnectionState new_connection_state = + cricket::kIceConnectionConnecting; + PeerConnectionInterface::IceConnectionState new_ice_connection_state = + PeerConnectionInterface::IceConnectionState::kIceConnectionNew; + PeerConnectionInterface::PeerConnectionState new_combined_state = + PeerConnectionInterface::PeerConnectionState::kNew; + cricket::IceGatheringState new_gathering_state = cricket::kIceGatheringNew; + bool any_failed = false; + bool all_connected = !dtls_transports.empty(); + bool all_completed = !dtls_transports.empty(); + bool any_gathering = false; + bool all_done_gathering = !dtls_transports.empty(); + + std::map<IceTransportState, int> ice_state_counts; + std::map<DtlsTransportState, int> dtls_state_counts; + + for (const auto& dtls : dtls_transports) { + any_failed = any_failed || dtls->ice_transport()->GetState() == + cricket::IceTransportState::STATE_FAILED; + all_connected = all_connected && dtls->writable(); + all_completed = + all_completed && dtls->writable() && + dtls->ice_transport()->GetState() == + cricket::IceTransportState::STATE_COMPLETED && + dtls->ice_transport()->GetIceRole() == cricket::ICEROLE_CONTROLLING && + dtls->ice_transport()->gathering_state() == + cricket::kIceGatheringComplete; + any_gathering = any_gathering || dtls->ice_transport()->gathering_state() != + cricket::kIceGatheringNew; + all_done_gathering = + all_done_gathering && dtls->ice_transport()->gathering_state() == + cricket::kIceGatheringComplete; + + dtls_state_counts[dtls->dtls_state()]++; + ice_state_counts[dtls->ice_transport()->GetIceTransportState()]++; + } + + if (any_failed) { + new_connection_state = cricket::kIceConnectionFailed; + } else if (all_completed) { + new_connection_state = cricket::kIceConnectionCompleted; + } else if (all_connected) { + new_connection_state = cricket::kIceConnectionConnected; + } + if (ice_connection_state_ != new_connection_state) { + ice_connection_state_ = new_connection_state; + + signal_ice_connection_state_.Send(new_connection_state); + } + + // Compute the current RTCIceConnectionState as described in + // https://www.w3.org/TR/webrtc/#dom-rtciceconnectionstate. + // The PeerConnection is responsible for handling the "closed" state. + int total_ice_checking = ice_state_counts[IceTransportState::kChecking]; + int total_ice_connected = ice_state_counts[IceTransportState::kConnected]; + int total_ice_completed = ice_state_counts[IceTransportState::kCompleted]; + int total_ice_failed = ice_state_counts[IceTransportState::kFailed]; + int total_ice_disconnected = + ice_state_counts[IceTransportState::kDisconnected]; + int total_ice_closed = ice_state_counts[IceTransportState::kClosed]; + int total_ice_new = ice_state_counts[IceTransportState::kNew]; + int total_ice = dtls_transports.size(); + + if (total_ice_failed > 0) { + // Any RTCIceTransports are in the "failed" state. + new_ice_connection_state = PeerConnectionInterface::kIceConnectionFailed; + } else if (total_ice_disconnected > 0) { + // None of the previous states apply and any RTCIceTransports are in the + // "disconnected" state. + new_ice_connection_state = + PeerConnectionInterface::kIceConnectionDisconnected; + } else if (total_ice_new + total_ice_closed == total_ice) { + // None of the previous states apply and all RTCIceTransports are in the + // "new" or "closed" state, or there are no transports. + new_ice_connection_state = PeerConnectionInterface::kIceConnectionNew; + } else if (total_ice_new + total_ice_checking > 0) { + // None of the previous states apply and any RTCIceTransports are in the + // "new" or "checking" state. + new_ice_connection_state = PeerConnectionInterface::kIceConnectionChecking; + } else if (total_ice_completed + total_ice_closed == total_ice || + all_completed) { + // None of the previous states apply and all RTCIceTransports are in the + // "completed" or "closed" state. + // + // TODO(https://bugs.webrtc.org/10356): The all_completed condition is added + // to mimic the behavior of the old ICE connection state, and should be + // removed once we get end-of-candidates signaling in place. + new_ice_connection_state = PeerConnectionInterface::kIceConnectionCompleted; + } else if (total_ice_connected + total_ice_completed + total_ice_closed == + total_ice) { + // None of the previous states apply and all RTCIceTransports are in the + // "connected", "completed" or "closed" state. + new_ice_connection_state = PeerConnectionInterface::kIceConnectionConnected; + } else { + RTC_DCHECK_NOTREACHED(); + } + + if (standardized_ice_connection_state_ != new_ice_connection_state) { + if (standardized_ice_connection_state_ == + PeerConnectionInterface::kIceConnectionChecking && + new_ice_connection_state == + PeerConnectionInterface::kIceConnectionCompleted) { + // Ensure that we never skip over the "connected" state. + signal_standardized_ice_connection_state_.Send( + PeerConnectionInterface::kIceConnectionConnected); + } + standardized_ice_connection_state_ = new_ice_connection_state; + signal_standardized_ice_connection_state_.Send(new_ice_connection_state); + } + + // Compute the current RTCPeerConnectionState as described in + // https://www.w3.org/TR/webrtc/#dom-rtcpeerconnectionstate. + // The PeerConnection is responsible for handling the "closed" state. + // Note that "connecting" is only a valid state for DTLS transports while + // "checking", "completed" and "disconnected" are only valid for ICE + // transports. + int total_connected = + total_ice_connected + dtls_state_counts[DtlsTransportState::kConnected]; + int total_dtls_connecting = + dtls_state_counts[DtlsTransportState::kConnecting]; + int total_failed = + total_ice_failed + dtls_state_counts[DtlsTransportState::kFailed]; + int total_closed = + total_ice_closed + dtls_state_counts[DtlsTransportState::kClosed]; + int total_new = total_ice_new + dtls_state_counts[DtlsTransportState::kNew]; + int total_transports = total_ice * 2; + + if (total_failed > 0) { + // Any of the RTCIceTransports or RTCDtlsTransports are in a "failed" state. + new_combined_state = PeerConnectionInterface::PeerConnectionState::kFailed; + } else if (total_ice_disconnected > 0) { + // None of the previous states apply and any RTCIceTransports or + // RTCDtlsTransports are in the "disconnected" state. + new_combined_state = + PeerConnectionInterface::PeerConnectionState::kDisconnected; + } else if (total_new + total_closed == total_transports) { + // None of the previous states apply and all RTCIceTransports and + // RTCDtlsTransports are in the "new" or "closed" state, or there are no + // transports. + new_combined_state = PeerConnectionInterface::PeerConnectionState::kNew; + } else if (total_new + total_dtls_connecting + total_ice_checking > 0) { + // None of the previous states apply and all RTCIceTransports or + // RTCDtlsTransports are in the "new", "connecting" or "checking" state. + new_combined_state = + PeerConnectionInterface::PeerConnectionState::kConnecting; + } else if (total_connected + total_ice_completed + total_closed == + total_transports) { + // None of the previous states apply and all RTCIceTransports and + // RTCDtlsTransports are in the "connected", "completed" or "closed" state. + new_combined_state = + PeerConnectionInterface::PeerConnectionState::kConnected; + } else { + RTC_DCHECK_NOTREACHED(); + } + + if (combined_connection_state_ != new_combined_state) { + combined_connection_state_ = new_combined_state; + signal_connection_state_.Send(new_combined_state); + } + + // Compute the gathering state. + if (dtls_transports.empty()) { + new_gathering_state = cricket::kIceGatheringNew; + } else if (all_done_gathering) { + new_gathering_state = cricket::kIceGatheringComplete; + } else if (any_gathering) { + new_gathering_state = cricket::kIceGatheringGathering; + } + if (ice_gathering_state_ != new_gathering_state) { + ice_gathering_state_ = new_gathering_state; + signal_ice_gathering_state_.Send(new_gathering_state); + } +} + +void JsepTransportController::OnRtcpPacketReceived_n( + rtc::CopyOnWriteBuffer* packet, + int64_t packet_time_us) { + RTC_DCHECK(config_.rtcp_handler); + config_.rtcp_handler(*packet, packet_time_us); +} + +void JsepTransportController::OnDtlsHandshakeError( + rtc::SSLHandshakeError error) { + config_.on_dtls_handshake_error_(error); +} + +bool JsepTransportController::OnTransportChanged( + const std::string& mid, + cricket::JsepTransport* jsep_transport) { + if (config_.transport_observer) { + if (jsep_transport) { + return config_.transport_observer->OnTransportChanged( + mid, jsep_transport->rtp_transport(), + jsep_transport->RtpDtlsTransport(), + jsep_transport->data_channel_transport()); + } else { + return config_.transport_observer->OnTransportChanged(mid, nullptr, + nullptr, nullptr); + } + } + return false; +} + +} // namespace webrtc |